drm/nouveau/dp: move core link training calls to common code
Signed-off-by: Ben Skeggs <bskeggs@redhat.com>
This commit is contained in:
parent
b6e4ad200a
commit
6c8e4633d3
|
@ -30,6 +30,8 @@
|
||||||
#include "nouveau_encoder.h"
|
#include "nouveau_encoder.h"
|
||||||
#include "nouveau_crtc.h"
|
#include "nouveau_crtc.h"
|
||||||
|
|
||||||
|
#include <core/class.h>
|
||||||
|
|
||||||
#include <subdev/gpio.h>
|
#include <subdev/gpio.h>
|
||||||
#include <subdev/i2c.h>
|
#include <subdev/i2c.h>
|
||||||
|
|
||||||
|
@ -83,7 +85,7 @@ nouveau_dp_bios_data(struct drm_device *dev, struct dcb_output *dcb, u8 **entry)
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
struct dp_state {
|
struct dp_state {
|
||||||
struct nouveau_i2c_port *auxch;
|
struct nouveau_i2c_port *auxch;
|
||||||
struct dp_train_func *func;
|
struct nouveau_object *core;
|
||||||
struct dcb_output *dcb;
|
struct dcb_output *dcb;
|
||||||
int crtc;
|
int crtc;
|
||||||
u8 *dpcd;
|
u8 *dpcd;
|
||||||
|
@ -97,13 +99,20 @@ static void
|
||||||
dp_set_link_config(struct drm_device *dev, struct dp_state *dp)
|
dp_set_link_config(struct drm_device *dev, struct dp_state *dp)
|
||||||
{
|
{
|
||||||
struct nouveau_drm *drm = nouveau_drm(dev);
|
struct nouveau_drm *drm = nouveau_drm(dev);
|
||||||
|
struct dcb_output *dcb = dp->dcb;
|
||||||
|
const u32 or = ffs(dcb->or) - 1, link = !(dcb->sorconf.link & 1);
|
||||||
|
const u32 moff = (dp->crtc << 3) | (link << 2) | or;
|
||||||
u8 sink[2];
|
u8 sink[2];
|
||||||
|
u32 data;
|
||||||
|
|
||||||
NV_DEBUG(drm, "%d lanes at %d KB/s\n", dp->link_nr, dp->link_bw);
|
NV_DEBUG(drm, "%d lanes at %d KB/s\n", dp->link_nr, dp->link_bw);
|
||||||
|
|
||||||
/* set desired link configuration on the source */
|
/* set desired link configuration on the source */
|
||||||
dp->func->link_set(dev, dp->dcb, dp->crtc, dp->link_nr, dp->link_bw,
|
data = ((dp->link_bw / 27000) << 8) | dp->link_nr;
|
||||||
dp->dpcd[2] & DP_ENHANCED_FRAME_CAP);
|
if (dp->dpcd[2] & DP_ENHANCED_FRAME_CAP)
|
||||||
|
data |= NV94_DISP_SOR_DP_LNKCTL_FRAME_ENH;
|
||||||
|
|
||||||
|
nv_call(dp->core, NV94_DISP_SOR_DP_LNKCTL + moff, data);
|
||||||
|
|
||||||
/* inform the sink of the new configuration */
|
/* inform the sink of the new configuration */
|
||||||
sink[0] = dp->link_bw / 27000;
|
sink[0] = dp->link_bw / 27000;
|
||||||
|
@ -118,11 +127,14 @@ static void
|
||||||
dp_set_training_pattern(struct drm_device *dev, struct dp_state *dp, u8 pattern)
|
dp_set_training_pattern(struct drm_device *dev, struct dp_state *dp, u8 pattern)
|
||||||
{
|
{
|
||||||
struct nouveau_drm *drm = nouveau_drm(dev);
|
struct nouveau_drm *drm = nouveau_drm(dev);
|
||||||
|
struct dcb_output *dcb = dp->dcb;
|
||||||
|
const u32 or = ffs(dcb->or) - 1, link = !(dcb->sorconf.link & 1);
|
||||||
|
const u32 moff = (dp->crtc << 3) | (link << 2) | or;
|
||||||
u8 sink_tp;
|
u8 sink_tp;
|
||||||
|
|
||||||
NV_DEBUG(drm, "training pattern %d\n", pattern);
|
NV_DEBUG(drm, "training pattern %d\n", pattern);
|
||||||
|
|
||||||
dp->func->train_set(dev, dp->dcb, pattern);
|
nv_call(dp->core, NV94_DISP_SOR_DP_TRAIN + moff, pattern);
|
||||||
|
|
||||||
nv_rdaux(dp->auxch, DP_TRAINING_PATTERN_SET, &sink_tp, 1);
|
nv_rdaux(dp->auxch, DP_TRAINING_PATTERN_SET, &sink_tp, 1);
|
||||||
sink_tp &= ~DP_TRAINING_PATTERN_MASK;
|
sink_tp &= ~DP_TRAINING_PATTERN_MASK;
|
||||||
|
@ -134,6 +146,9 @@ static int
|
||||||
dp_link_train_commit(struct drm_device *dev, struct dp_state *dp)
|
dp_link_train_commit(struct drm_device *dev, struct dp_state *dp)
|
||||||
{
|
{
|
||||||
struct nouveau_drm *drm = nouveau_drm(dev);
|
struct nouveau_drm *drm = nouveau_drm(dev);
|
||||||
|
struct dcb_output *dcb = dp->dcb;
|
||||||
|
const u32 or = ffs(dcb->or) - 1, link = !(dcb->sorconf.link & 1);
|
||||||
|
const u32 moff = (dp->crtc << 3) | (link << 2) | or;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
for (i = 0; i < dp->link_nr; i++) {
|
for (i = 0; i < dp->link_nr; i++) {
|
||||||
|
@ -148,7 +163,8 @@ dp_link_train_commit(struct drm_device *dev, struct dp_state *dp)
|
||||||
dp->conf[i] |= DP_TRAIN_MAX_PRE_EMPHASIS_REACHED;
|
dp->conf[i] |= DP_TRAIN_MAX_PRE_EMPHASIS_REACHED;
|
||||||
|
|
||||||
NV_DEBUG(drm, "config lane %d %02x\n", i, dp->conf[i]);
|
NV_DEBUG(drm, "config lane %d %02x\n", i, dp->conf[i]);
|
||||||
dp->func->train_adj(dev, dp->dcb, i, lvsw, lpre);
|
|
||||||
|
nv_call(dp->core, NV94_DISP_SOR_DP_DRVCTL(i) + moff, (lvsw << 8) | lpre);
|
||||||
}
|
}
|
||||||
|
|
||||||
return nv_wraux(dp->auxch, DP_TRAINING_LANE0_SET, dp->conf, 4);
|
return nv_wraux(dp->auxch, DP_TRAINING_LANE0_SET, dp->conf, 4);
|
||||||
|
@ -286,7 +302,7 @@ dp_link_train_fini(struct drm_device *dev, struct dp_state *dp)
|
||||||
|
|
||||||
static bool
|
static bool
|
||||||
nouveau_dp_link_train(struct drm_encoder *encoder, u32 datarate,
|
nouveau_dp_link_train(struct drm_encoder *encoder, u32 datarate,
|
||||||
struct dp_train_func *func)
|
struct nouveau_object *core)
|
||||||
{
|
{
|
||||||
struct nouveau_encoder *nv_encoder = nouveau_encoder(encoder);
|
struct nouveau_encoder *nv_encoder = nouveau_encoder(encoder);
|
||||||
struct nouveau_crtc *nv_crtc = nouveau_crtc(encoder->crtc);
|
struct nouveau_crtc *nv_crtc = nouveau_crtc(encoder->crtc);
|
||||||
|
@ -304,7 +320,7 @@ nouveau_dp_link_train(struct drm_encoder *encoder, u32 datarate,
|
||||||
if (!dp.auxch)
|
if (!dp.auxch)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
dp.func = func;
|
dp.core = core;
|
||||||
dp.dcb = nv_encoder->dcb;
|
dp.dcb = nv_encoder->dcb;
|
||||||
dp.crtc = nv_crtc->index;
|
dp.crtc = nv_crtc->index;
|
||||||
dp.dpcd = nv_encoder->dp.dpcd;
|
dp.dpcd = nv_encoder->dp.dpcd;
|
||||||
|
@ -365,7 +381,7 @@ nouveau_dp_link_train(struct drm_encoder *encoder, u32 datarate,
|
||||||
|
|
||||||
void
|
void
|
||||||
nouveau_dp_dpms(struct drm_encoder *encoder, int mode, u32 datarate,
|
nouveau_dp_dpms(struct drm_encoder *encoder, int mode, u32 datarate,
|
||||||
struct dp_train_func *func)
|
struct nouveau_object *core)
|
||||||
{
|
{
|
||||||
struct nouveau_encoder *nv_encoder = nouveau_encoder(encoder);
|
struct nouveau_encoder *nv_encoder = nouveau_encoder(encoder);
|
||||||
struct nouveau_drm *drm = nouveau_drm(encoder->dev);
|
struct nouveau_drm *drm = nouveau_drm(encoder->dev);
|
||||||
|
@ -385,7 +401,7 @@ nouveau_dp_dpms(struct drm_encoder *encoder, int mode, u32 datarate,
|
||||||
nv_wraux(auxch, DP_SET_POWER, &status, 1);
|
nv_wraux(auxch, DP_SET_POWER, &status, 1);
|
||||||
|
|
||||||
if (mode == DRM_MODE_DPMS_ON)
|
if (mode == DRM_MODE_DPMS_ON)
|
||||||
nouveau_dp_link_train(encoder, datarate, func);
|
nouveau_dp_link_train(encoder, datarate, core);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
|
|
@ -93,7 +93,7 @@ get_slave_funcs(struct drm_encoder *enc)
|
||||||
/* nouveau_dp.c */
|
/* nouveau_dp.c */
|
||||||
bool nouveau_dp_detect(struct drm_encoder *);
|
bool nouveau_dp_detect(struct drm_encoder *);
|
||||||
void nouveau_dp_dpms(struct drm_encoder *, int mode, u32 datarate,
|
void nouveau_dp_dpms(struct drm_encoder *, int mode, u32 datarate,
|
||||||
struct dp_train_func *);
|
struct nouveau_object *);
|
||||||
u8 *nouveau_dp_bios_data(struct drm_device *, struct dcb_output *, u8 **);
|
u8 *nouveau_dp_bios_data(struct drm_device *, struct dcb_output *, u8 **);
|
||||||
|
|
||||||
struct nouveau_connector *
|
struct nouveau_connector *
|
||||||
|
|
|
@ -40,39 +40,6 @@
|
||||||
|
|
||||||
#include <subdev/timer.h>
|
#include <subdev/timer.h>
|
||||||
|
|
||||||
static void
|
|
||||||
nv50_sor_dp_train_set(struct drm_device *dev, struct dcb_output *dcb, u8 pattern)
|
|
||||||
{
|
|
||||||
struct nv50_display *disp = nv50_display(dev);
|
|
||||||
const u32 or = ffs(dcb->or) - 1, link = !(dcb->sorconf.link & 1);
|
|
||||||
const u32 moff = (link << 2) | or;
|
|
||||||
nv_call(disp->core, NV94_DISP_SOR_DP_TRAIN + moff, pattern);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
nv50_sor_dp_train_adj(struct drm_device *dev, struct dcb_output *dcb,
|
|
||||||
u8 lane, u8 swing, u8 preem)
|
|
||||||
{
|
|
||||||
struct nv50_display *disp = nv50_display(dev);
|
|
||||||
const u32 or = ffs(dcb->or) - 1, link = !(dcb->sorconf.link & 1);
|
|
||||||
const u32 moff = (link << 2) | or;
|
|
||||||
const u32 data = (swing << 8) | preem;
|
|
||||||
nv_call(disp->core, NV94_DISP_SOR_DP_DRVCTL(lane) + moff, data);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
nv50_sor_dp_link_set(struct drm_device *dev, struct dcb_output *dcb, int crtc,
|
|
||||||
int link_nr, u32 link_bw, bool enhframe)
|
|
||||||
{
|
|
||||||
struct nv50_display *disp = nv50_display(dev);
|
|
||||||
const u32 or = ffs(dcb->or) - 1, link = !(dcb->sorconf.link & 1);
|
|
||||||
const u32 moff = (crtc << 3) | (link << 2) | or;
|
|
||||||
u32 data = ((link_bw / 27000) << 8) | link_nr;
|
|
||||||
if (enhframe)
|
|
||||||
data |= NV94_DISP_SOR_DP_LNKCTL_FRAME_ENH;
|
|
||||||
nv_call(disp->core, NV94_DISP_SOR_DP_LNKCTL + moff, data);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
nv50_sor_disconnect(struct drm_encoder *encoder)
|
nv50_sor_disconnect(struct drm_encoder *encoder)
|
||||||
{
|
{
|
||||||
|
@ -133,15 +100,8 @@ nv50_sor_dpms(struct drm_encoder *encoder, int mode)
|
||||||
|
|
||||||
nv_call(priv->core, NV50_DISP_SOR_PWR + or, (mode == DRM_MODE_DPMS_ON));
|
nv_call(priv->core, NV50_DISP_SOR_PWR + or, (mode == DRM_MODE_DPMS_ON));
|
||||||
|
|
||||||
if (nv_encoder->dcb->type == DCB_OUTPUT_DP) {
|
if (nv_encoder->dcb->type == DCB_OUTPUT_DP)
|
||||||
struct dp_train_func func = {
|
nouveau_dp_dpms(encoder, mode, nv_encoder->dp.datarate, priv->core);
|
||||||
.link_set = nv50_sor_dp_link_set,
|
|
||||||
.train_set = nv50_sor_dp_train_set,
|
|
||||||
.train_adj = nv50_sor_dp_train_adj
|
|
||||||
};
|
|
||||||
|
|
||||||
nouveau_dp_dpms(encoder, mode, nv_encoder->dp.datarate, &func);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
|
|
@ -1305,39 +1305,6 @@ nvd0_hdmi_disconnect(struct drm_encoder *encoder)
|
||||||
/******************************************************************************
|
/******************************************************************************
|
||||||
* SOR
|
* SOR
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
static void
|
|
||||||
nvd0_sor_dp_train_set(struct drm_device *dev, struct dcb_output *dcb, u8 pattern)
|
|
||||||
{
|
|
||||||
struct nvd0_disp *disp = nvd0_disp(dev);
|
|
||||||
const u32 or = ffs(dcb->or) - 1, link = !(dcb->sorconf.link & 1);
|
|
||||||
const u32 moff = (link << 2) | or;
|
|
||||||
nv_call(disp->core, NV94_DISP_SOR_DP_TRAIN + moff, pattern);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
nvd0_sor_dp_train_adj(struct drm_device *dev, struct dcb_output *dcb,
|
|
||||||
u8 lane, u8 swing, u8 preem)
|
|
||||||
{
|
|
||||||
struct nvd0_disp *disp = nvd0_disp(dev);
|
|
||||||
const u32 or = ffs(dcb->or) - 1, link = !(dcb->sorconf.link & 1);
|
|
||||||
const u32 moff = (link << 2) | or;
|
|
||||||
const u32 data = (swing << 8) | preem;
|
|
||||||
nv_call(disp->core, NV94_DISP_SOR_DP_DRVCTL(lane) + moff, data);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
nvd0_sor_dp_link_set(struct drm_device *dev, struct dcb_output *dcb, int crtc,
|
|
||||||
int link_nr, u32 link_bw, bool enhframe)
|
|
||||||
{
|
|
||||||
struct nvd0_disp *disp = nvd0_disp(dev);
|
|
||||||
const u32 or = ffs(dcb->or) - 1, link = !(dcb->sorconf.link & 1);
|
|
||||||
const u32 moff = (crtc << 3) | (link << 2) | or;
|
|
||||||
u32 data = ((link_bw / 27000) << 8) | link_nr;
|
|
||||||
if (enhframe)
|
|
||||||
data |= NV94_DISP_SOR_DP_LNKCTL_FRAME_ENH;
|
|
||||||
nv_call(disp->core, NV94_DISP_SOR_DP_LNKCTL + moff, data);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
nvd0_sor_dpms(struct drm_encoder *encoder, int mode)
|
nvd0_sor_dpms(struct drm_encoder *encoder, int mode)
|
||||||
{
|
{
|
||||||
|
@ -1365,15 +1332,8 @@ nvd0_sor_dpms(struct drm_encoder *encoder, int mode)
|
||||||
|
|
||||||
nv_call(disp->core, NV50_DISP_SOR_PWR + or, (mode == DRM_MODE_DPMS_ON));
|
nv_call(disp->core, NV50_DISP_SOR_PWR + or, (mode == DRM_MODE_DPMS_ON));
|
||||||
|
|
||||||
if (nv_encoder->dcb->type == DCB_OUTPUT_DP) {
|
if (nv_encoder->dcb->type == DCB_OUTPUT_DP)
|
||||||
struct dp_train_func func = {
|
nouveau_dp_dpms(encoder, mode, nv_encoder->dp.datarate, disp->core);
|
||||||
.link_set = nvd0_sor_dp_link_set,
|
|
||||||
.train_set = nvd0_sor_dp_train_set,
|
|
||||||
.train_adj = nvd0_sor_dp_train_adj
|
|
||||||
};
|
|
||||||
|
|
||||||
nouveau_dp_dpms(encoder, mode, nv_encoder->dp.datarate, &func);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool
|
static bool
|
||||||
|
|
Loading…
Reference in New Issue