radeon_hd: Start work on proper DP link training
* The AtomBIOS timeout fix has made my DP bridge stop working * The current DisplayPort code is a little lacking on DP link training... I think thats the cause. * This puts the first steps towards DP training in place. * I plan on trying to make some of this DP stuff common accelerant stuff after it works.
This commit is contained in:
parent
6f7c0aadbe
commit
83e3a8ea50
@ -136,6 +136,7 @@
|
||||
#define DP_INTERLANE_ALIGN_DONE (1 << 0)
|
||||
#define DP_DOWNSTREAM_PORT_STATUS_CHANGED (1 << 6)
|
||||
#define DP_LINK_STATUS_UPDATED (1 << 7)
|
||||
#define DP_LINK_STATUS_SIZE 6
|
||||
|
||||
#define DP_SINK_STATUS 0x205
|
||||
|
||||
|
@ -14,6 +14,7 @@
|
||||
#include <edid.h>
|
||||
|
||||
#include "atom.h"
|
||||
#include "displayport_reg.h"
|
||||
#include "encoder.h"
|
||||
#include "mode.h"
|
||||
#include "pll.h"
|
||||
@ -126,11 +127,22 @@ typedef struct {
|
||||
|
||||
typedef struct {
|
||||
bool valid;
|
||||
uint32 connectorIndex;
|
||||
|
||||
uint32 auxPin; // normally GPIO pin on GPU
|
||||
|
||||
uint8 config[8]; // DP configuration data
|
||||
uint8 sinkType;
|
||||
uint8 clock;
|
||||
int laneCount;
|
||||
|
||||
bool trainingUseEncoder;
|
||||
|
||||
uint8 trainingAttempts;
|
||||
uint8 trainingSet[4];
|
||||
int trainingReadInterval;
|
||||
uint8 linkStatus[DP_LINK_STATUS_SIZE];
|
||||
|
||||
bool eDPOn;
|
||||
} dp_info;
|
||||
|
||||
|
@ -11,7 +11,6 @@
|
||||
|
||||
#include <Debug.h>
|
||||
|
||||
#include "accelerant.h"
|
||||
#include "accelerant_protos.h"
|
||||
#include "connector.h"
|
||||
#include "mode.h"
|
||||
@ -403,12 +402,15 @@ dp_setup_connectors()
|
||||
}
|
||||
|
||||
uint32 gpioID = gConnector[index]->gpioID;
|
||||
uint32 hwPin = gGPIOInfo[gpioID]->hwPin;
|
||||
|
||||
uint32 auxPin = gGPIOInfo[gpioID]->hwPin;
|
||||
gDPInfo[index]->auxPin = auxPin;
|
||||
gDPInfo[index]->connectorIndex = index;
|
||||
|
||||
uint8 auxMessage[25];
|
||||
int result;
|
||||
|
||||
result = dp_aux_read(hwPin, DP_DPCD_REV, auxMessage, 8, 0);
|
||||
result = dp_aux_read(auxPin, DP_DPCD_REV, auxMessage, 8, 0);
|
||||
if (result > 0) {
|
||||
gDPInfo[index]->valid = true;
|
||||
memcpy(gDPInfo[index]->config, auxMessage, 8);
|
||||
@ -419,14 +421,255 @@ dp_setup_connectors()
|
||||
}
|
||||
|
||||
|
||||
static bool
|
||||
dp_get_link_status(dp_info* dp)
|
||||
{
|
||||
int result = dp_aux_read(dp->auxPin, DP_LANE0_1_STATUS,
|
||||
dp->linkStatus, DP_LINK_STATUS_SIZE, 100);
|
||||
|
||||
if (result <= 0) {
|
||||
ERROR("%s: DisplayPort link status failed\n", __func__);
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
|
||||
static uint8
|
||||
dp_get_lane_status(dp_info* dp, int lane)
|
||||
{
|
||||
int i = DP_LANE0_1_STATUS + (lane >> 1);
|
||||
int s = (lane & 1) * 4;
|
||||
uint8 l = dp->linkStatus[i - DP_LANE0_1_STATUS];
|
||||
return (l >> s) & 0xf;
|
||||
}
|
||||
|
||||
|
||||
static bool
|
||||
dp_clock_recovery_ok(dp_info* dp)
|
||||
{
|
||||
int lane;
|
||||
uint8 laneStatus;
|
||||
|
||||
for (lane = 0; lane < dp->laneCount; lane++) {
|
||||
laneStatus = dp_get_lane_status(dp, lane);
|
||||
if ((laneStatus & DP_LANE_CR_DONE) == 0)
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
|
||||
static void
|
||||
dp_update_vs_emph(dp_info* dp)
|
||||
{
|
||||
// Set initial vs and emph on source
|
||||
transmitter_dig_setup(dp->connectorIndex, dp->clock, 0, dp->trainingSet[0],
|
||||
ATOM_TRANSMITTER_ACTION_SETUP_VSEMPH);
|
||||
|
||||
// Set vs and emph on the sink
|
||||
dp_aux_write(dp->auxPin, DP_TRAINING_LANE0_SET,
|
||||
dp->trainingSet, dp->laneCount, 0);
|
||||
}
|
||||
|
||||
|
||||
static uint8
|
||||
dp_get_adjust_request_voltage(dp_info* dp, int lane)
|
||||
{
|
||||
int i = DP_ADJUST_REQUEST_LANE0_1 + (lane >> 1);
|
||||
int s = (((lane & 1) != 0) ? DP_ADJUST_VOLTAGE_SWING_LANE1_SHIFT
|
||||
: DP_ADJUST_VOLTAGE_SWING_LANE0_SHIFT);
|
||||
uint8 l = dp->linkStatus[i - DP_LANE0_1_STATUS];
|
||||
|
||||
return ((l >> s) & 0x3) << DP_TRAIN_VOLTAGE_SWING_SHIFT;
|
||||
}
|
||||
|
||||
|
||||
static uint8
|
||||
dp_get_adjust_request_pre_emphasis(dp_info* dp, int lane)
|
||||
{
|
||||
int i = DP_ADJUST_REQUEST_LANE0_1 + (lane >> 1);
|
||||
int s = (((lane & 1) != 0) ? DP_ADJUST_PRE_EMPHASIS_LANE1_SHIFT
|
||||
: DP_ADJUST_PRE_EMPHASIS_LANE0_SHIFT);
|
||||
uint8 l = dp->linkStatus[i - DP_LANE0_1_STATUS];
|
||||
|
||||
return ((l >> s) & 0x3) << DP_TRAIN_PRE_EMPHASIS_SHIFT;
|
||||
}
|
||||
|
||||
|
||||
#define DP_VOLTAGE_MAX DP_TRAIN_VOLTAGE_SWING_1200
|
||||
#define DP_PRE_EMPHASIS_MAX DP_TRAIN_PRE_EMPHASIS_9_5
|
||||
|
||||
|
||||
static void
|
||||
dp_get_adjust_train(dp_info* dp)
|
||||
{
|
||||
TRACE("%s\n", __func__);
|
||||
|
||||
const char* voltageNames[] = {
|
||||
"0.4V", "0.6V", "0.8V", "1.2V"
|
||||
};
|
||||
const char* preEmphasisNames[] = {
|
||||
"0dB", "3.5dB", "6dB", "9.5dB"
|
||||
};
|
||||
|
||||
uint8 voltage = 0;
|
||||
uint8 preEmphasis = 0;
|
||||
int lane;
|
||||
|
||||
for (lane = 0; lane < dp->laneCount; lane++) {
|
||||
uint8 laneVoltage = dp_get_adjust_request_voltage(dp, lane);
|
||||
uint8 lanePreEmphasis = dp_get_adjust_request_pre_emphasis(dp, lane);
|
||||
|
||||
TRACE("%s: Requested %s at %s for lane %d\n", __func__,
|
||||
preEmphasisNames[lanePreEmphasis >> DP_TRAIN_PRE_EMPHASIS_SHIFT],
|
||||
voltageNames[laneVoltage >> DP_TRAIN_VOLTAGE_SWING_SHIFT],
|
||||
lane);
|
||||
|
||||
if (laneVoltage > voltage)
|
||||
voltage = laneVoltage;
|
||||
if (lanePreEmphasis > preEmphasis)
|
||||
preEmphasis = lanePreEmphasis;
|
||||
}
|
||||
|
||||
if (voltage >= DP_VOLTAGE_MAX)
|
||||
voltage |= DP_TRAIN_MAX_SWING_REACHED;
|
||||
|
||||
if (preEmphasis >= DP_PRE_EMPHASIS_MAX)
|
||||
preEmphasis |= DP_TRAIN_MAX_PRE_EMPHASIS_REACHED;
|
||||
|
||||
for (lane = 0; lane < 4; lane++)
|
||||
dp->trainingSet[lane] = voltage | preEmphasis;
|
||||
}
|
||||
|
||||
|
||||
static void
|
||||
dp_set_tp(dp_info* dp, int trainingPattern)
|
||||
{
|
||||
TRACE("%s\n", __func__);
|
||||
|
||||
radeon_shared_info &info = *gInfo->shared_info;
|
||||
|
||||
int rawTrainingPattern = 0;
|
||||
|
||||
/* set training pattern on the source */
|
||||
if (info.dceMajor >= 4 || !dp->trainingUseEncoder) {
|
||||
switch (trainingPattern) {
|
||||
case DP_TRAINING_PATTERN_1:
|
||||
rawTrainingPattern = ATOM_ENCODER_CMD_DP_LINK_TRAINING_PATTERN1;
|
||||
break;
|
||||
case DP_TRAINING_PATTERN_2:
|
||||
rawTrainingPattern = ATOM_ENCODER_CMD_DP_LINK_TRAINING_PATTERN2;
|
||||
break;
|
||||
case DP_TRAINING_PATTERN_3:
|
||||
rawTrainingPattern = ATOM_ENCODER_CMD_DP_LINK_TRAINING_PATTERN3;
|
||||
break;
|
||||
}
|
||||
// TODO: PixelClock 0 ok?
|
||||
encoder_dig_setup(dp->connectorIndex, 0, rawTrainingPattern);
|
||||
} else {
|
||||
ERROR("%s: TODO: dp_encoder_service\n", __func__);
|
||||
return;
|
||||
#if 0
|
||||
switch (trainingPattern) {
|
||||
case DP_TRAINING_PATTERN_1:
|
||||
rawTrainingPattern = 0;
|
||||
break;
|
||||
case DP_TRAINING_PATTERN_2:
|
||||
rawTrainingPattern = 1;
|
||||
break;
|
||||
}
|
||||
radeon_dp_encoder_service(dp_info->rdev,
|
||||
ATOM_DP_ACTION_TRAINING_PATTERN_SEL, dp_info->dp_clock,
|
||||
dp_info->enc_id, rawTrainingPattern);
|
||||
#endif
|
||||
}
|
||||
|
||||
// Enable training pattern on the sink
|
||||
dpcd_reg_write(dp->auxPin, DP_TRAINING_PATTERN_SET, trainingPattern);
|
||||
}
|
||||
|
||||
|
||||
status_t
|
||||
dp_link_train_cr(dp_info* dp)
|
||||
{
|
||||
TRACE("%s\n", __func__);
|
||||
|
||||
// Display Port Clock Recovery Training
|
||||
|
||||
bool clockRecovery = false;
|
||||
uint8 voltage = 0xff;
|
||||
int lane;
|
||||
|
||||
dp_set_tp(dp, DP_TRAINING_PATTERN_1);
|
||||
memset(dp->trainingSet, 0, 4);
|
||||
dp_update_vs_emph(dp);
|
||||
|
||||
while (1) {
|
||||
if (dp->trainingReadInterval == 0)
|
||||
snooze(100);
|
||||
else
|
||||
snooze(1000 * 4 * dp->trainingReadInterval);
|
||||
|
||||
if (!dp_get_link_status(dp))
|
||||
break;
|
||||
|
||||
if (dp_clock_recovery_ok(dp)) {
|
||||
clockRecovery = true;
|
||||
break;
|
||||
}
|
||||
|
||||
for (lane = 0; lane < dp->laneCount; lane++) {
|
||||
if ((dp->trainingSet[lane] & DP_TRAIN_MAX_SWING_REACHED) == 0)
|
||||
break;
|
||||
}
|
||||
|
||||
if (lane == dp->laneCount) {
|
||||
ERROR("%s: clock recovery reached max voltage\n", __func__);
|
||||
break;
|
||||
}
|
||||
|
||||
if ((dp->trainingSet[0] & DP_TRAIN_VOLTAGE_SWING_MASK) == voltage) {
|
||||
dp->trainingAttempts++;
|
||||
if (dp->trainingAttempts >= 5) {
|
||||
ERROR("%s: clock recovery tried 5 times\n", __func__);
|
||||
break;
|
||||
}
|
||||
} else
|
||||
dp->trainingAttempts = 0;
|
||||
|
||||
voltage = dp->trainingSet[0] & DP_TRAIN_VOLTAGE_SWING_MASK;
|
||||
|
||||
// Compute new trainingSet as requested by sink
|
||||
dp_get_adjust_train(dp);
|
||||
|
||||
dp_update_vs_emph(dp);
|
||||
}
|
||||
|
||||
if (!clockRecovery) {
|
||||
ERROR("%s: clock recovery failed\n", __func__);
|
||||
return B_ERROR;
|
||||
}
|
||||
|
||||
TRACE("%s: clock recovery at voltage %d pre-emphasis %d\n",
|
||||
__func__, dp->trainingSet[0] & DP_TRAIN_VOLTAGE_SWING_MASK,
|
||||
(dp->trainingSet[0] & DP_TRAIN_PRE_EMPHASIS_MASK)
|
||||
>> DP_TRAIN_PRE_EMPHASIS_SHIFT);
|
||||
return B_OK;
|
||||
}
|
||||
|
||||
|
||||
status_t
|
||||
dp_link_train(uint8 crtcID, display_mode* mode)
|
||||
{
|
||||
TRACE("%s\n", __func__);
|
||||
|
||||
uint32 connectorIndex = gDisplay[crtcID]->connectorIndex;
|
||||
if (gDPInfo[connectorIndex]->valid != true) {
|
||||
ERROR("%s: started on non-DisplayPort connector #%" B_PRIu32 "\n",
|
||||
dp_info* dp = gDPInfo[connectorIndex];
|
||||
|
||||
if (dp->valid != true) {
|
||||
ERROR("%s: started on invalid DisplayPort connector #%" B_PRIu32 "\n",
|
||||
__func__, connectorIndex);
|
||||
return B_ERROR;
|
||||
}
|
||||
@ -436,13 +679,13 @@ dp_link_train(uint8 crtcID, display_mode* mode)
|
||||
uint8 tableMajor;
|
||||
uint8 tableMinor;
|
||||
|
||||
bool dpUseEncoder = true;
|
||||
dp->trainingUseEncoder = true;
|
||||
if (atom_parse_cmd_header(gAtomContext, index, &tableMajor, &tableMinor)
|
||||
== B_OK) {
|
||||
if (tableMinor > 1) {
|
||||
// The AtomBIOS DPEncoderService greater then 1.1 can't program the
|
||||
// training pattern properly.
|
||||
dpUseEncoder = false;
|
||||
dp->trainingUseEncoder = false;
|
||||
}
|
||||
}
|
||||
|
||||
@ -461,22 +704,24 @@ dp_link_train(uint8 crtcID, display_mode* mode)
|
||||
else
|
||||
dpEncoderID |= ATOM_DP_CONFIG_LINK_A;
|
||||
|
||||
//uint8 dpReadInterval = dpcd_reg_read(hwPin, DP_TRAINING_AUX_RD_INTERVAL);
|
||||
dp->trainingReadInterval
|
||||
= dpcd_reg_read(hwPin, DP_TRAINING_AUX_RD_INTERVAL);
|
||||
|
||||
uint8 sandbox = dpcd_reg_read(hwPin, DP_MAX_LANE_COUNT);
|
||||
|
||||
radeon_shared_info &info = *gInfo->shared_info;
|
||||
bool dpTPS3Supported = false;
|
||||
if (info.dceMajor >= 5 && (sandbox & DP_TPS3_SUPPORTED) != 0)
|
||||
dpTPS3Supported = true;
|
||||
//bool dpTPS3Supported = false;
|
||||
//if (info.dceMajor >= 5 && (sandbox & DP_TPS3_SUPPORTED) != 0)
|
||||
// dpTPS3Supported = true;
|
||||
|
||||
// DisplayPort training initialization
|
||||
// *** DisplayPort link training initialization
|
||||
|
||||
// Power up the DP sink
|
||||
if (gDPInfo[connectorIndex]->config[0] >= 0x11)
|
||||
if (dp->config[0] >= 0x11)
|
||||
dpcd_reg_write(hwPin, DP_SET_POWER, DP_SET_POWER_D0);
|
||||
|
||||
// Possibly enable downspread on the sink
|
||||
if ((gDPInfo[connectorIndex]->config[3] & 0x1) != 0)
|
||||
if ((dp->config[3] & 0x1) != 0)
|
||||
dpcd_reg_write(hwPin, DP_DOWNSPREAD_CTRL, DP_SPREAD_AMP_0_5);
|
||||
else
|
||||
dpcd_reg_write(hwPin, DP_DOWNSPREAD_CTRL, 0);
|
||||
@ -484,16 +729,16 @@ dp_link_train(uint8 crtcID, display_mode* mode)
|
||||
encoder_dig_setup(connectorIndex, mode->timing.pixel_clock,
|
||||
ATOM_ENCODER_CMD_SETUP_PANEL_MODE);
|
||||
|
||||
if (gDPInfo[connectorIndex]->config[0] >= 0x11)
|
||||
if (dp->config[0] >= 0x11)
|
||||
sandbox |= DP_LANE_COUNT_ENHANCED_FRAME_EN;
|
||||
dpcd_reg_write(hwPin, DP_LANE_COUNT_SET, sandbox);
|
||||
|
||||
// Set the link rate on the DP sink
|
||||
sandbox = dp_get_link_clock_encode(gDPInfo[connectorIndex]->clock);
|
||||
sandbox = dp_get_link_clock_encode(dp->clock);
|
||||
dpcd_reg_write(hwPin, DP_LINK_BW_SET, sandbox);
|
||||
|
||||
// Start link training on source
|
||||
if (info.dceMajor >= 4 || !dpUseEncoder) {
|
||||
if (info.dceMajor >= 4 || !dp->trainingUseEncoder) {
|
||||
encoder_dig_setup(connectorIndex, mode->timing.pixel_clock,
|
||||
ATOM_ENCODER_CMD_DP_LINK_TRAINING_START);
|
||||
} else {
|
||||
@ -501,21 +746,23 @@ dp_link_train(uint8 crtcID, display_mode* mode)
|
||||
__func__);
|
||||
}
|
||||
|
||||
/* disable the training pattern on the sink */
|
||||
// Disable the training pattern on the sink
|
||||
dpcd_reg_write(hwPin, DP_TRAINING_PATTERN_SET,
|
||||
DP_TRAINING_PATTERN_DISABLE);
|
||||
|
||||
// TODO: dp_link_train_cr
|
||||
dp_link_train_cr(dp);
|
||||
// TODO: dp_link_train_ce
|
||||
|
||||
|
||||
// *** DisplayPort link training finish
|
||||
snooze(400);
|
||||
|
||||
/* disable the training pattern on the sink */
|
||||
// Disable the training pattern on the sink
|
||||
dpcd_reg_write(hwPin, DP_TRAINING_PATTERN_SET,
|
||||
DP_TRAINING_PATTERN_DISABLE);
|
||||
|
||||
/* disable the training pattern on the source */
|
||||
if (info.dceMajor >= 4 || !dpUseEncoder) {
|
||||
// Disable the training pattern on the source
|
||||
if (info.dceMajor >= 4 || !dp->trainingUseEncoder) {
|
||||
encoder_dig_setup(connectorIndex, mode->timing.pixel_clock,
|
||||
ATOM_ENCODER_CMD_DP_LINK_TRAINING_COMPLETE);
|
||||
} else {
|
||||
|
@ -13,6 +13,7 @@
|
||||
#include <stdint.h>
|
||||
#include <SupportDefs.h>
|
||||
|
||||
#include "accelerant.h"
|
||||
#include "displayport_reg.h"
|
||||
|
||||
|
||||
@ -32,6 +33,7 @@ uint32 dp_get_link_clock_decode(uint32 dpLinkClock);
|
||||
|
||||
void dp_setup_connectors();
|
||||
status_t dp_link_train(uint8 crtcID, display_mode* mode);
|
||||
status_t dp_link_train_cr(dp_info* dp);
|
||||
|
||||
|
||||
#endif /* RADEON_HD_DISPLAYPORT_H */
|
||||
|
@ -1069,8 +1069,12 @@ transmitter_dig_setup(uint32 connectorIndex, uint32 pixelClock,
|
||||
index = GetIndexIntoMasterTable(COMMAND, LVTMATransmitterControl);
|
||||
break;
|
||||
default:
|
||||
ERROR("%s: called on non-dig encoder!\n", __func__);
|
||||
return B_ERROR;
|
||||
// Multiple encoders can be wired to a single connector
|
||||
// An example is UNIPHY -> DP -> TRAVIS -> LVDS
|
||||
ERROR("%s: BUG: guessing UNIPHY as this isn't a dig encoder!\n",
|
||||
__func__);
|
||||
index = GetIndexIntoMasterTable(COMMAND, UNIPHYTransmitterControl);
|
||||
break;
|
||||
}
|
||||
|
||||
if (index < 0) {
|
||||
|
Loading…
Reference in New Issue
Block a user