summaryrefslogtreecommitdiff
path: root/src/atombios_output.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/atombios_output.c')
-rw-r--r--src/atombios_output.c81
1 files changed, 60 insertions, 21 deletions
diff --git a/src/atombios_output.c b/src/atombios_output.c
index 024fa4c5..fc6c1ec1 100644
--- a/src/atombios_output.c
+++ b/src/atombios_output.c
@@ -595,7 +595,7 @@ atombios_output_dig_encoder_setup(xf86OutputPtr output, int action)
/* doesn't really matter which dig encoder we pick as long as it's
* not already in use
*/
- if (radeon_output->linkb)
+ if (dig_block)
index = GetIndexIntoMasterTable(COMMAND, DIG2EncoderControl);
else
index = GetIndexIntoMasterTable(COMMAND, DIG1EncoderControl);
@@ -779,7 +779,7 @@ atombios_output_dig_transmitter_setup(xf86OutputPtr output, int action, uint8_t
/* doesn't really matter which dig encoder we pick as long as it's
* not already in use
*/
- if (radeon_output->linkb)
+ if (radeon_output->dig_block)
disp_data.v1.ucConfig |= ATOM_TRANSMITTER_CONFIG_DIG2_ENCODER;
else
disp_data.v1.ucConfig |= ATOM_TRANSMITTER_CONFIG_DIG1_ENCODER;
@@ -1489,7 +1489,7 @@ atombios_set_output_crtc_source(xf86OutputPtr output)
/* doesn't really matter which dig encoder we pick as long as it's
* not already in use
*/
- if (radeon_output->linkb)
+ if (radeon_output->dig_block)
crtc_src_param2.ucEncoderID = ASIC_INT_DIG2_ENCODER_ID;
else
crtc_src_param2.ucEncoderID = ASIC_INT_DIG1_ENCODER_ID;
@@ -1581,20 +1581,67 @@ atombios_apply_output_quirks(xf86OutputPtr output, DisplayModePtr mode)
}
}
+static void
+atombios_pick_dig_block(xf86OutputPtr output)
+{
+ xf86CrtcConfigPtr xf86_config = XF86_CRTC_CONFIG_PTR(output->scrn);
+ RADEONOutputPrivatePtr radeon_output = output->driver_private;
+ RADEONInfoPtr info = RADEONPTR(output->scrn);
+ Bool is_lvtma = FALSE;
+ int i, mode;
+ uint32_t dig_enc_use_mask = 0;
+
+ /* non digital encoders don't need a dig block */
+ mode = atombios_get_encoder_mode(output);
+ if (mode == ATOM_ENCODER_MODE_CRT ||
+ mode == ATOM_ENCODER_MODE_TV ||
+ mode == ATOM_ENCODER_MODE_CV)
+ return;
+
+ if (IS_DCE32_VARIANT) {
+ RADEONCrtcPrivatePtr radeon_crtc = output->crtc->driver_private;
+ radeon_output->dig_block = radeon_crtc->crtc_id;
+ return;
+ }
+
+ for (i = 0; i < xf86_config->num_output; i++) {
+ xf86OutputPtr test = xf86_config->output[i];
+ radeon_encoder_ptr radeon_encoder = radeon_get_encoder(test);
+ RADEONOutputPrivatePtr radeon_test = test->driver_private;
+
+ if (!radeon_encoder || !test->crtc)
+ continue;
+
+ if (output == test && radeon_encoder->encoder_id == ENCODER_OBJECT_ID_INTERNAL_KLDSCP_LVTMA)
+ is_lvtma = TRUE;
+ if (output != test && (radeon_test->dig_block >= 0))
+ dig_enc_use_mask |= (1 << radeon_test->dig_block);
+
+ }
+ if (is_lvtma) {
+ if (dig_enc_use_mask & 0x2)
+ ErrorF("Need digital encoder 2 for LVTMA and it isn't free - stealing\n");
+ radeon_output->dig_block = 1;
+ return;
+ }
+ if (!(dig_enc_use_mask & 1))
+ radeon_output->dig_block = 0;
+ else
+ radeon_output->dig_block = 1;
+}
void
atombios_output_mode_set(xf86OutputPtr output,
DisplayModePtr mode,
DisplayModePtr adjusted_mode)
{
RADEONOutputPrivatePtr radeon_output = output->driver_private;
- RADEONCrtcPrivatePtr radeon_crtc = output->crtc->driver_private;
radeon_encoder_ptr radeon_encoder = radeon_get_encoder(output);
RADEONInfoPtr info = RADEONPTR(output->scrn);
if (radeon_encoder == NULL)
return;
radeon_output->pixel_clock = adjusted_mode->Clock;
- radeon_output->dig_block = radeon_crtc->crtc_id;
+ atombios_pick_dig_block(output);
atombios_output_overscan_setup(output, mode, adjusted_mode);
atombios_output_scaler_setup(output);
atombios_set_output_crtc_source(output);
@@ -1869,24 +1916,16 @@ atombios_dac_detect(xf86OutputPtr output)
static inline int atom_dp_get_encoder_id(xf86OutputPtr output)
{
- RADEONInfoPtr info = RADEONPTR(output->scrn);
RADEONOutputPrivatePtr radeon_output = output->driver_private;
int ret = 0;
- if (IS_DCE32_VARIANT) {
- if (radeon_output->dig_block)
- ret |= ATOM_DP_CONFIG_DIG2_ENCODER;
- else
- ret |= ATOM_DP_CONFIG_DIG1_ENCODER;
- if (radeon_output->linkb)
- ret |= ATOM_DP_CONFIG_LINK_B;
- else
- ret |= ATOM_DP_CONFIG_LINK_A;
- } else {
- if (radeon_output->linkb)
- ret |= ATOM_DP_CONFIG_DIG2_ENCODER | ATOM_DP_CONFIG_LINK_B;
- else
- ret |= ATOM_DP_CONFIG_DIG1_ENCODER | ATOM_DP_CONFIG_LINK_A;
- }
+ if (radeon_output->dig_block)
+ ret |= ATOM_DP_CONFIG_DIG2_ENCODER;
+ else
+ ret |= ATOM_DP_CONFIG_DIG1_ENCODER;
+ if (radeon_output->linkb)
+ ret |= ATOM_DP_CONFIG_LINK_B;
+ else
+ ret |= ATOM_DP_CONFIG_LINK_A;
return ret;
}