[PATCH 001/001] drivers/gpu/radeon: NULL pointer deference workaround

Christian König christian.koenig at amd.com
Thu Sep 8 07:14:23 UTC 2016


Am 07.09.2016 um 19:38 schrieb Mark Fortescue:
>
> On an LV-683 (AMD Dual-core G-T56N) Mini-ITX board, I get a Kernel 
> Oops because Connector 0 (LCD Panel interface) does not have DDC.

I'm not an expert on this, but that is really odd cause even LCD Panels 
should have a DDC interface.

>
> Ubuntu 16.04 LTS Kernel (4.4 series):
>
> ...
> [ 8.262990] [drm] ib test on ring 5 succeeded
> [ 8.288897] [drm] Radeon Display Connectors
> [ 8.293175] [drm] Connector 0:
> [ 8.296252] [drm] DP-1

Especially since the BIOS claims that this is a displayport connector 
and there is no physical way to have a DP without an DDC as far as I know.

Please open a bug report on FDO and attach you BIOS image.

Alex can probably take a look when he's back from vacation.

Regards,
Christian.

> [ 8.298791] [drm] Encoders:
> [ 8.301770] [drm] DFP1: INTERNAL_UNIPHY
> [ 8.305973] [drm] Connector 1:
> [ 8.309043] [drm] DP-2
> [ 8.311598] [drm] HPD2
> [ 8.314169] [drm] DDC: 0x6440 0x6440 0x6444 0x6444 0x6448 0x6448 
> 0x644c 0x644c
> [ 8.321609] [drm] Encoders:
> [ 8.324589] [drm] DFP2: INTERNAL_UNIPHY
> [ 8.328793] [drm] Connector 2:
> [ 8.331856] [drm] VGA-1
> [ 8.342947] [drm] DDC: 0x64d8 0x64d8 0x64dc 0x64dc 0x64e0 0x64e0 
> 0x64e4 0x64e4
> [ 8.350341] [drm] Encoders:
> [ 8.353310] [drm] CRT1: INTERNAL_KLDSCP_DAC1
> [ 8.358195] BUG: unable to handle kernel NULL pointer dereference at 
> 0000000000000409
> [ 8.409733] [<ffffffffc02024da>] radeon_dp_getsinktype+0x1a/0x30 [radeon]
> [ 8.416805] PGD 0
> [ 8.418841] Oops: 0000 [#1] SMP
> ...
>
> This patch prevents Kernel failures due to a connector not having a 
> DDC interface by changing the code so that ddc_bus is always checked 
> before use.
> The problem was first identified using the uBuntu MATE 14.04 LTS (3.16 
> series kernels) but not dealt with at that time. On attempting to 
> install uBuntu MATE 16.04 LTS (4.4 series kernels), it became clear 
> that using various workarounds to allow the issue to be ignored were 
> not viable so more effort was put in to sorting the issue resulting in 
> this patch. See https://bugs.launchpad.net/bugs/1587885 for more details.
>
> Signed-off-by: Mark Fortescue <mark at thurning-instruments.co.uk>
> Tested-by: Mark Fortescue <mark at thurning-instruments.co.uk>
>
> ---
>
> Looks like Thunderbird may have made a mess of the patch when pasting 
> the contents into the mail message - my alternate mail client (pine) 
> also has strict line length handling and trashes non-MIME encoded 
> patches.
>
> This may not be the correct approach to solving the issue but it is 
> clean in that it ensures that ddc_bus is never used when NULL 
> regardless of how the code ended up at the point of use.
>
> If it helps with back porting, I have patches for the uBuntu 14.04 LTS 
> [3.13 series], uBuntu MATE 14.04 LTS [3.16 series] and uBuntu 16.04 
> LTS [4.4 series] kernels.
>
> Test Hardware:
> Commell LV-683 Mini-ITX with onboard AMD Dual-core G-T56N
> 4G Ram, 2x1TB Disk, HANNS-G HC194D 1280x1024 LCD (VGA).
> 4.8.0-rc5 with patch boots without error.
>
>   drivers/gpu/drm/radeon/atombios_dp.c       |   60 ++++++++++++-------
>   drivers/gpu/drm/radeon/radeon_connectors.c |   46 +++++++-------
>   drivers/gpu/drm/radeon/radeon_dp_mst.c     |    9 ++
>   drivers/gpu/drm/radeon/radeon_i2c.c        |    3
>   4 files changed, 73 insertions(+), 45 deletions(-)
>
> Patch:
> diff --git a/drivers/gpu/drm/radeon/atombios_dp.c 
> b/drivers/gpu/drm/radeon/atombios_dp.c
> index cead089a..98b3c0e 100644
> --- a/drivers/gpu/drm/radeon/atombios_dp.c
> +++ b/drivers/gpu/drm/radeon/atombios_dp.c
> @@ -232,6 +232,9 @@ void radeon_dp_aux_init(struct radeon_connector 
> *radeon_connector)
>       struct radeon_device *rdev = dev->dev_private;
>       int ret;
>
> +    if (!radeon_connector->ddc_bus)
> +        return;
> +
>       radeon_connector->ddc_bus->rec.hpd = radeon_connector->hpd.hpd;
>       radeon_connector->ddc_bus->aux.dev = radeon_connector->base.kdev;
>       if (ASIC_IS_DCE5(rdev)) {
> @@ -364,6 +367,9 @@ u8 radeon_dp_getsinktype(struct radeon_connector 
> *radeon_connector)
>       struct drm_device *dev = radeon_connector->base.dev;
>       struct radeon_device *rdev = dev->dev_private;
>
> +    if (!radeon_connector->ddc_bus)
> +        return 0;
> +
>       return radeon_dp_encoder_service(rdev, 
> ATOM_DP_ACTION_GET_SINK_TYPE, 0,
> radeon_connector->ddc_bus->rec.i2c_id, 0);
>   }
> @@ -376,6 +382,9 @@ static void radeon_dp_probe_oui(struct 
> radeon_connector *radeon_connector)
>       if (!(dig_connector->dpcd[DP_DOWN_STREAM_PORT_COUNT] & 
> DP_OUI_SUPPORT))
>           return;
>
> +    if (!radeon_connector->ddc_bus)
> +        return;
> +
>       if (drm_dp_dpcd_read(&radeon_connector->ddc_bus->aux, 
> DP_SINK_OUI, buf, 3) == 3)
>           DRM_DEBUG_KMS("Sink OUI: %02hx%02hx%02hx\n",
>                     buf[0], buf[1], buf[2]);
> @@ -391,6 +400,9 @@ bool radeon_dp_getdpcd(struct radeon_connector 
> *radeon_connector)
>       u8 msg[DP_DPCD_SIZE];
>       int ret, i;
>
> +    if (!radeon_connector->ddc_bus)
> +        return false;
> +
>       for (i = 0; i < 7; i++) {
>           ret = drm_dp_dpcd_read(&radeon_connector->ddc_bus->aux, 
> DP_DPCD_REV, msg,
>                          DP_DPCD_SIZE);
> @@ -428,24 +440,26 @@ int radeon_dp_get_panel_mode(struct drm_encoder 
> *encoder,
>
>       dig_connector = radeon_connector->con_priv;
>
> -    if (dp_bridge != ENCODER_OBJECT_ID_NONE) {
> -        /* DP bridge chips */
> -        if (drm_dp_dpcd_readb(&radeon_connector->ddc_bus->aux,
> -                      DP_EDP_CONFIGURATION_CAP, &tmp) == 1) {
> -            if (tmp & 1)
> -                panel_mode = DP_PANEL_MODE_INTERNAL_DP2_MODE;
> -            else if ((dp_bridge == ENCODER_OBJECT_ID_NUTMEG) ||
> -                 (dp_bridge == ENCODER_OBJECT_ID_TRAVIS))
> -                panel_mode = DP_PANEL_MODE_INTERNAL_DP1_MODE;
> -            else
> -                panel_mode = DP_PANEL_MODE_EXTERNAL_DP_MODE;
> -        }
> -    } else if (connector->connector_type == DRM_MODE_CONNECTOR_eDP) {
> -        /* eDP */
> -        if (drm_dp_dpcd_readb(&radeon_connector->ddc_bus->aux,
> -                      DP_EDP_CONFIGURATION_CAP, &tmp) == 1) {
> -            if (tmp & 1)
> -                panel_mode = DP_PANEL_MODE_INTERNAL_DP2_MODE;
> +    if (radeon_connector->ddc_bus) {
> +        if (dp_bridge != ENCODER_OBJECT_ID_NONE) {
> +            /* DP bridge chips */
> +            if (drm_dp_dpcd_readb(&radeon_connector->ddc_bus->aux,
> +                          DP_EDP_CONFIGURATION_CAP, &tmp) == 1) {
> +                if (tmp & 1)
> +                    panel_mode = DP_PANEL_MODE_INTERNAL_DP2_MODE;
> +                else if ((dp_bridge == ENCODER_OBJECT_ID_NUTMEG) ||
> +                     (dp_bridge == ENCODER_OBJECT_ID_TRAVIS))
> +                    panel_mode = DP_PANEL_MODE_INTERNAL_DP1_MODE;
> +                else
> +                    panel_mode = DP_PANEL_MODE_EXTERNAL_DP_MODE;
> +            }
> +        } else if (connector->connector_type == 
> DRM_MODE_CONNECTOR_eDP) {
> +            /* eDP */
> +            if (drm_dp_dpcd_readb(&radeon_connector->ddc_bus->aux,
> +                          DP_EDP_CONFIGURATION_CAP, &tmp) == 1) {
> +                if (tmp & 1)
> +                    panel_mode = DP_PANEL_MODE_INTERNAL_DP2_MODE;
> +            }
>           }
>       }
>
> @@ -511,6 +525,9 @@ bool radeon_dp_needs_link_train(struct 
> radeon_connector *radeon_connector)
>       u8 link_status[DP_LINK_STATUS_SIZE];
>       struct radeon_connector_atom_dig *dig = radeon_connector->con_priv;
>
> +    if (!radeon_connector->ddc_bus)
> +        return false;
> +
>       if 
> (drm_dp_dpcd_read_link_status(&radeon_connector->ddc_bus->aux, 
> link_status)
>           <= 0)
>           return false;
> @@ -531,7 +548,7 @@ void radeon_dp_set_rx_power_state(struct 
> drm_connector *connector,
>       dig_connector = radeon_connector->con_priv;
>
>       /* power up/down the sink */
> -    if (dig_connector->dpcd[0] >= 0x11) {
> +    if (radeon_connector->ddc_bus && dig_connector->dpcd[0] >= 0x11) {
> drm_dp_dpcd_writeb(&radeon_connector->ddc_bus->aux,
>                      DP_SET_POWER, power_state);
>           usleep_range(1000, 2000);
> @@ -834,7 +851,8 @@ void radeon_dp_link_train(struct drm_encoder 
> *encoder,
>       else
>           dp_info.enc_id |= ATOM_DP_CONFIG_LINK_A;
>
> -    if (drm_dp_dpcd_readb(&radeon_connector->ddc_bus->aux, 
> DP_MAX_LANE_COUNT, &tmp)
> +    if (radeon_connector->ddc_bus &&
> + drm_dp_dpcd_readb(&radeon_connector->ddc_bus->aux, 
> DP_MAX_LANE_COUNT, &tmp)
>           == 1) {
>           if (ASIC_IS_DCE5(rdev) && (tmp & DP_TPS3_SUPPORTED))
>               dp_info.tp3_supported = true;
> @@ -850,7 +868,7 @@ void radeon_dp_link_train(struct drm_encoder 
> *encoder,
>       dp_info.connector = connector;
>       dp_info.dp_lane_count = dig_connector->dp_lane_count;
>       dp_info.dp_clock = dig_connector->dp_clock;
> -    dp_info.aux = &radeon_connector->ddc_bus->aux;
> +    dp_info.aux = radeon_connector->ddc_bus ? 
> &radeon_connector->ddc_bus->aux : 0;
>
>       if (radeon_dp_link_train_init(&dp_info))
>           goto done;
> diff --git a/drivers/gpu/drm/radeon/radeon_connectors.c 
> b/drivers/gpu/drm/radeon/radeon_connectors.c
> index b79f3b0..cec30c9 100644
> --- a/drivers/gpu/drm/radeon/radeon_connectors.c
> +++ b/drivers/gpu/drm/radeon/radeon_connectors.c
> @@ -328,31 +328,32 @@ static void radeon_connector_get_edid(struct 
> drm_connector *connector)
>       if (radeon_connector->router.ddc_valid)
>           radeon_router_select_ddc_port(radeon_connector);
>
> -    if ((radeon_connector_encoder_get_dp_bridge_encoder_id(connector) !=
> -         ENCODER_OBJECT_ID_NONE) &&
> -        radeon_connector->ddc_bus->has_aux) {
> -        radeon_connector->edid = drm_get_edid(connector,
> - &radeon_connector->ddc_bus->aux.ddc);
> -    } else if ((connector->connector_type == 
> DRM_MODE_CONNECTOR_DisplayPort) ||
> -           (connector->connector_type == DRM_MODE_CONNECTOR_eDP)) {
> -        struct radeon_connector_atom_dig *dig = 
> radeon_connector->con_priv;
> -
> -        if ((dig->dp_sink_type == CONNECTOR_OBJECT_ID_DISPLAYPORT ||
> -             dig->dp_sink_type == CONNECTOR_OBJECT_ID_eDP) &&
> -            radeon_connector->ddc_bus->has_aux)
> -            radeon_connector->edid = 
> drm_get_edid(&radeon_connector->base,
> +    if (radeon_connector->ddc_bus) {
> +        if 
> ((radeon_connector_encoder_get_dp_bridge_encoder_id(connector) !=
> +             ENCODER_OBJECT_ID_NONE) &&
> +            radeon_connector->ddc_bus->has_aux) {
> +            radeon_connector->edid = drm_get_edid(connector,
> &radeon_connector->ddc_bus->aux.ddc);
> -        else if (radeon_connector->ddc_bus)
> +        } else if ((connector->connector_type == 
> DRM_MODE_CONNECTOR_DisplayPort) ||
> +               (connector->connector_type == DRM_MODE_CONNECTOR_eDP)) {
> +            struct radeon_connector_atom_dig *dig = 
> radeon_connector->con_priv;
> +
> +            if ((dig->dp_sink_type == CONNECTOR_OBJECT_ID_DISPLAYPORT ||
> +                 dig->dp_sink_type == CONNECTOR_OBJECT_ID_eDP) &&
> +                radeon_connector->ddc_bus->has_aux)
> +                radeon_connector->edid = 
> drm_get_edid(&radeon_connector->base,
> + &radeon_connector->ddc_bus->aux.ddc);
> +            else
> +                radeon_connector->edid = 
> drm_get_edid(&radeon_connector->base,
> + &radeon_connector->ddc_bus->adapter);
> +        } else if (vga_switcheroo_handler_flags() & 
> VGA_SWITCHEROO_CAN_SWITCH_DDC &&
> +               connector->connector_type == DRM_MODE_CONNECTOR_LVDS) {
> +            radeon_connector->edid = 
> drm_get_edid_switcheroo(&radeon_connector->base,
> + &radeon_connector->ddc_bus->adapter);
> +        } else {
>               radeon_connector->edid = 
> drm_get_edid(&radeon_connector->base,
> &radeon_connector->ddc_bus->adapter);
> -    } else if (vga_switcheroo_handler_flags() & 
> VGA_SWITCHEROO_CAN_SWITCH_DDC &&
> -           connector->connector_type == DRM_MODE_CONNECTOR_LVDS &&
> -           radeon_connector->ddc_bus) {
> -        radeon_connector->edid = 
> drm_get_edid_switcheroo(&radeon_connector->base,
> - &radeon_connector->ddc_bus->adapter);
> -    } else if (radeon_connector->ddc_bus) {
> -        radeon_connector->edid = drm_get_edid(&radeon_connector->base,
> - &radeon_connector->ddc_bus->adapter);
> +        }
>       }
>
>       if (!radeon_connector->edid) {
> @@ -1312,6 +1313,7 @@ radeon_dvi_detect(struct drm_connector 
> *connector, bool force)
>                           continue;
>                       list_radeon_connector = 
> to_radeon_connector(list_connector);
>                       if (list_radeon_connector->shared_ddc &&
> +                        radeon_connector->ddc_bus &&
> (list_radeon_connector->ddc_bus->rec.i2c_id ==
> radeon_connector->ddc_bus->rec.i2c_id)) {
>                           /* cases where both connectors are digital */
> diff --git a/drivers/gpu/drm/radeon/radeon_dp_mst.c 
> b/drivers/gpu/drm/radeon/radeon_dp_mst.c
> index de504ea..89e91f3 100644
> --- a/drivers/gpu/drm/radeon/radeon_dp_mst.c
> +++ b/drivers/gpu/drm/radeon/radeon_dp_mst.c
> @@ -661,7 +661,7 @@ radeon_dp_mst_init(struct radeon_connector 
> *radeon_connector)
>   {
>       struct drm_device *dev = radeon_connector->base.dev;
>
> -    if (!radeon_connector->ddc_bus->has_aux)
> +    if (!radeon_connector->ddc_bus || 
> !radeon_connector->ddc_bus->has_aux)
>           return 0;
>
>       radeon_connector->mst_mgr.cbs = &mst_cbs;
> @@ -688,6 +688,9 @@ radeon_dp_mst_probe(struct radeon_connector 
> *radeon_connector)
>       if (dig_connector->dpcd[DP_DPCD_REV] < 0x12)
>           return 0;
>
> +    if (!radeon_connector->ddc_bus || 
> !radeon_connector->ddc_bus->has_aux)
> +        return 0;
> +
>       ret = drm_dp_dpcd_read(&radeon_connector->ddc_bus->aux, 
> DP_MSTM_CAP, msg,
>                      1);
>       if (ret) {
> @@ -711,7 +714,9 @@ radeon_dp_mst_check_status(struct radeon_connector 
> *radeon_connector)
>       struct radeon_connector_atom_dig *dig_connector = 
> radeon_connector->con_priv;
>       int retry;
>
> -    if (dig_connector->is_mst) {
> +    if (dig_connector->is_mst &&
> +        radeon_connector->ddc_bus &&
> +        radeon_connector->ddc_bus->has_aux) {
>           u8 esi[16] = { 0 };
>           int dret;
>           int ret = 0;
> diff --git a/drivers/gpu/drm/radeon/radeon_i2c.c 
> b/drivers/gpu/drm/radeon/radeon_i2c.c
> index 9590bcd..c18f7ba 100644
> --- a/drivers/gpu/drm/radeon/radeon_i2c.c
> +++ b/drivers/gpu/drm/radeon/radeon_i2c.c
> @@ -63,6 +63,9 @@ bool radeon_ddc_probe(struct radeon_connector 
> *radeon_connector, bool use_aux)
>       if (radeon_connector->router.ddc_valid)
>           radeon_router_select_ddc_port(radeon_connector);
>
> +    if (!radeon_connector->ddc_bus)
> +        return false;
> +
>       if (use_aux) {
>           ret = i2c_transfer(&radeon_connector->ddc_bus->aux.ddc, 
> msgs, 2);
>       } else {
> _______________________________________________
> dri-devel mailing list
> dri-devel at lists.freedesktop.org
> https://lists.freedesktop.org/mailman/listinfo/dri-devel




More information about the dri-devel mailing list