The retina MacBook Pro uses an eDP panel and a gmux controller to switch the panel between its two GPUs. Unfortunately it seems that it cannot switch the AUX channel separately from the main link. But we can emulate switching of DDC/AUX in software by using the active client as a proxy to talk to the panel. Proxying of the AUX channel is facilitated by way of Thierry Reding's awesome struct drm_dp_aux abstraction (cf. c197db75ff5c, "drm/dp: Add AUX channel infrastructure"). However, as regards usage of struct drm_dp_aux, nouveau is the odd man out: A struct drm_dp_aux is defined as part of struct nouveau_connector but never used. Instead, the AUX channel is accessed directly with nv_rdaux() and nv_wraux(), even in the DRM part of the driver. To enable proxying in nouveau, inject a pointer to the struct drm_dp_aux from the DRM part of the driver into the struct nvkm_i2c_port. Modify nv_rdaux() to try drm_dp_dpcd_read() first. If that fails, fall back to accessing the AUX channel directly. Enclose in #if IS_ENABLED(CONFIG_DRM _KMS_HELPER) to keep the NVKM part of the driver portable and free of DRM symbols. Obviously this is a bit of a kludge but it seems there's no elegant way short of factoring all the AUX communication in dport.c / outpdp.c out and into the DRM part of the driver (plus the AUX initialization in VBIOS). When the driver first initializes the output with nvkm_output_dp_init(), the pointer to the struct drm_dp_aux is not yet injected into the struct nvkm_i2c_port. Thus, if the panel is not switched to the Nvidia GPU, the dpcd attribute of struct nvkm_output_dp can't be filled and the link doesn't get trained. Make up for this by checking the link training status in nouveau_dp_detect() and calling nvkm_output_dp_detect() if the link hasn't been trained yet. Modify link training itself so that it does not fail when writing to DPCD if the value to be written is identical with what's already configured in DPCD. (Proxying is currently read only for safety reasons.) Bugzilla: https://bugzilla.kernel.org/show_bug.cgi?id=88861 Bugzilla: https://bugs.freedesktop.org/show_bug.cgi?id=61115 Tested-by: Paul Hordiienko <pvt.gord@xxxxxxxxx> [MBP 6,2 2010 intel ILK + nvidia GT216 pre-retina] Tested-by: William Brown <william@xxxxxxxxxxxxxxxx> [MBP 8,2 2011 intel SNB + amd turks pre-retina] Tested-by: Lukas Wunner <lukas@xxxxxxxxx> [MBP 9,1 2012 intel IVB + nvidia GK107 pre-retina] Tested-by: Bruno Bierbaumer <bruno@xxxxxxxxxxxxxx> [MBP 11,3 2013 intel HSW + nvidia GK107 retina -- work in progress] Signed-off-by: Lukas Wunner <lukas@xxxxxxxxx> --- drivers/gpu/drm/nouveau/include/nvkm/subdev/i2c.h | 1 + drivers/gpu/drm/nouveau/nouveau_connector.c | 4 ++-- drivers/gpu/drm/nouveau/nouveau_dp.c | 20 +++++++++++++++++++ drivers/gpu/drm/nouveau/nvkm/engine/disp/dport.c | 6 +++++- drivers/gpu/drm/nouveau/nvkm/engine/disp/outpdp.c | 2 +- drivers/gpu/drm/nouveau/nvkm/engine/disp/outpdp.h | 1 + drivers/gpu/drm/nouveau/nvkm/subdev/i2c/aux.c | 24 +++++++++++++++++++++++ 7 files changed, 54 insertions(+), 4 deletions(-) diff --git a/drivers/gpu/drm/nouveau/include/nvkm/subdev/i2c.h b/drivers/gpu/drm/nouveau/include/nvkm/subdev/i2c.h index a2e3373..9fa95fb 100644 --- a/drivers/gpu/drm/nouveau/include/nvkm/subdev/i2c.h +++ b/drivers/gpu/drm/nouveau/include/nvkm/subdev/i2c.h @@ -37,6 +37,7 @@ struct nvkm_i2c_port { struct list_head head; u8 index; int aux; + void *drm_dp_aux; const struct nvkm_i2c_func *func; }; diff --git a/drivers/gpu/drm/nouveau/nouveau_connector.c b/drivers/gpu/drm/nouveau/nouveau_connector.c index 1e5224f..159df7f 100644 --- a/drivers/gpu/drm/nouveau/nouveau_connector.c +++ b/drivers/gpu/drm/nouveau/nouveau_connector.c @@ -144,8 +144,8 @@ nouveau_connector_ddc_detect(struct drm_connector *connector) nv_encoder = nouveau_encoder(encoder); if (nv_encoder->dcb->type == DCB_OUTPUT_DP) { - int ret = nouveau_dp_detect(nv_encoder); - if (ret == 0) + nv_encoder->i2c->drm_dp_aux = &nv_connector->aux; + if (nouveau_dp_detect(nv_encoder) == 0) break; } else if (nv_encoder->i2c) { diff --git a/drivers/gpu/drm/nouveau/nouveau_dp.c b/drivers/gpu/drm/nouveau/nouveau_dp.c index c3ef30b..317d6b1 100644 --- a/drivers/gpu/drm/nouveau/nouveau_dp.c +++ b/drivers/gpu/drm/nouveau/nouveau_dp.c @@ -30,6 +30,25 @@ #include "nouveau_encoder.h" #include "nouveau_crtc.h" +#include <engine/disp.h> +#include <engine/disp/outpdp.h> + +static void +nouveau_dp_check_link_training(struct nouveau_encoder *nv_encoder) +{ + struct nvkm_disp *disp = nvkm_disp(nv_encoder->i2c); + struct nvkm_output *outp; + struct nvkm_output_dp *outpdp; + + list_for_each_entry(outp, &disp->outp, head) + if (outp->info.index == nv_encoder->dcb->index) + break; + + outpdp = (struct nvkm_output_dp *)outp; + if (!atomic_read(&outpdp->lt.done)) + nvkm_output_dp_detect(outpdp); +} + static void nouveau_dp_probe_oui(struct drm_device *dev, struct nvkm_i2c_port *auxch, u8 *dpcd) @@ -85,5 +104,6 @@ nouveau_dp_detect(struct nouveau_encoder *nv_encoder) nv_encoder->dp.link_nr, nv_encoder->dp.link_bw); nouveau_dp_probe_oui(dev, auxch, dpcd); + nouveau_dp_check_link_training(nv_encoder); return 0; } diff --git a/drivers/gpu/drm/nouveau/nvkm/engine/disp/dport.c b/drivers/gpu/drm/nouveau/nvkm/engine/disp/dport.c index 6834766..5257e4c 100644 --- a/drivers/gpu/drm/nouveau/nvkm/engine/disp/dport.c +++ b/drivers/gpu/drm/nouveau/nvkm/engine/disp/dport.c @@ -61,7 +61,7 @@ dp_set_link_config(struct dp_state *dp) .execute = 1, }; u32 lnkcmp; - u8 sink[2]; + u8 sink[2], sink_rd[2]; int ret; DBG("%d lanes at %d KB/s\n", dp->link_nr, dp->link_bw); @@ -98,6 +98,10 @@ dp_set_link_config(struct dp_state *dp) if (outp->dpcd[DPCD_RC02] & DPCD_RC02_ENHANCED_FRAME_CAP) sink[1] |= DPCD_LC01_ENHANCED_FRAME_EN; + if (nv_rdaux(outp->base.edid, DPCD_LC00_LINK_BW_SET, sink_rd, 2) == 0 && + memcmp(sink, sink_rd, 2) == 0) + return 0; + return nv_wraux(outp->base.edid, DPCD_LC00_LINK_BW_SET, sink, 2); } diff --git a/drivers/gpu/drm/nouveau/nvkm/engine/disp/outpdp.c b/drivers/gpu/drm/nouveau/nvkm/engine/disp/outpdp.c index 0bde0fa..b95373b 100644 --- a/drivers/gpu/drm/nouveau/nvkm/engine/disp/outpdp.c +++ b/drivers/gpu/drm/nouveau/nvkm/engine/disp/outpdp.c @@ -122,7 +122,7 @@ nvkm_output_dp_enable(struct nvkm_output_dp *outp, bool present) } } -static void +void nvkm_output_dp_detect(struct nvkm_output_dp *outp) { struct nvkm_i2c_port *port = outp->base.edid; diff --git a/drivers/gpu/drm/nouveau/nvkm/engine/disp/outpdp.h b/drivers/gpu/drm/nouveau/nvkm/engine/disp/outpdp.h index 70c77ae..0bd4dcb 100644 --- a/drivers/gpu/drm/nouveau/nvkm/engine/disp/outpdp.h +++ b/drivers/gpu/drm/nouveau/nvkm/engine/disp/outpdp.h @@ -58,4 +58,5 @@ struct nvkm_output_dp_impl { }; int nvkm_output_dp_train(struct nvkm_output *, u32 rate, bool wait); +void nvkm_output_dp_detect(struct nvkm_output_dp *); #endif diff --git a/drivers/gpu/drm/nouveau/nvkm/subdev/i2c/aux.c b/drivers/gpu/drm/nouveau/nvkm/subdev/i2c/aux.c index 1c18860..16ec3cc 100644 --- a/drivers/gpu/drm/nouveau/nvkm/subdev/i2c/aux.c +++ b/drivers/gpu/drm/nouveau/nvkm/subdev/i2c/aux.c @@ -23,10 +23,34 @@ */ #include "priv.h" +#if IS_ENABLED(CONFIG_DRM_KMS_HELPER) +#include <drm/drm_dp_helper.h> +#endif + +static int +drm_rdaux(struct nvkm_i2c_port *port, u32 addr, u8 *data, u8 size) +{ +#if IS_ENABLED(CONFIG_DRM_KMS_HELPER) + if (port->drm_dp_aux) { + nv_debug(port, "Try reading DPCD with KMS helper: addr=0x%x size=%d\n", + addr, size); + return !(drm_dp_dpcd_read(port->drm_dp_aux, addr, data, size) + == size); + } +#endif + return -ENODEV; +} + int nv_rdaux(struct nvkm_i2c_port *port, u32 addr, u8 *data, u8 size) { struct nvkm_i2c *i2c = nvkm_i2c(port); + + if (drm_rdaux(port, addr, data, size) == 0) + return 0; + + nv_debug(port, "Try reading DPCD directly: addr=0x%x size=%d\n", + addr, size); if (port->func->aux) { int ret = i2c->acquire(port, 0); if (ret == 0) { -- 1.8.5.2 (Apple Git-48) _______________________________________________ dri-devel mailing list dri-devel@xxxxxxxxxxxxxxxxxxxxx http://lists.freedesktop.org/mailman/listinfo/dri-devel