diff options
author | Chris Wilson <chris@chris-wilson.co.uk> | 2014-05-08 06:45:43 +0100 |
---|---|---|
committer | Chris Wilson <chris@chris-wilson.co.uk> | 2014-05-08 13:48:43 +0100 |
commit | e78a00c5da922024ac64270fae0797d98d1d6e57 (patch) | |
tree | d040920103a73f509891b067db5271194c21ee8e /src/sna | |
parent | a93157587a82c699e7df76fb482c49569f0686b9 (diff) |
sna: Add an xorg.conf option for removing unused outputs
Clients are not yet ready for outputs that disappear at runtime, often
unexpectedly dieing with asynchronous BadOutput errors. A simple
workaround is to not remove any output just yet - but allow users to opt
in via xorg.conf, i.e.
Section "Device"
Option "DeleteUnusedDP12Displays" "true"
EndSection
The name chosen is to be consistent with the nvidia driver, which did
MST first and encountered all of these issues first, and has also been
adopted for -modesetting. If this is combined with persistent output
naming (i.e. based on DisplayPort branch topology), the number of
outputs presented to the user should be static.
Instead of removing the outputs, we have to mark them detached instead
to avoid throwing errors from the kernel.
Based on the patch for -modesetting by Dave Airlie.
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Diffstat (limited to 'src/sna')
-rw-r--r-- | src/sna/sna.h | 1 | ||||
-rw-r--r-- | src/sna/sna_display.c | 38 | ||||
-rw-r--r-- | src/sna/sna_driver.c | 3 |
3 files changed, 34 insertions, 8 deletions
diff --git a/src/sna/sna.h b/src/sna/sna.h index 75a0e36c..3a45924f 100644 --- a/src/sna/sna.h +++ b/src/sna/sna.h @@ -244,6 +244,7 @@ struct sna { #define SNA_IS_HOSTED 0x80 #define SNA_PERFORMANCE 0x100 #define SNA_POWERSAVE 0x200 +#define SNA_REMOVE_OUTPUTS 0x400 #define SNA_REDISCOVER 0x40000000 #define SNA_REPROBE 0x80000000 diff --git a/src/sna/sna_display.c b/src/sna/sna_display.c index af71f116..51b4db67 100644 --- a/src/sna/sna_display.c +++ b/src/sna/sna_display.c @@ -172,6 +172,8 @@ static inline struct sna_output *to_sna_output(xf86OutputPtr output) static inline int to_connector_id(xf86OutputPtr output) { + assert(to_sna_output(output)); + assert(to_sna_output(output)->id); return to_sna_output(output)->id; } @@ -1904,7 +1906,10 @@ sna_output_detect(xf86OutputPtr output) struct sna_output *sna_output = output->driver_private; union compat_mode_get_connector compat_conn; - DBG(("%s(%s)\n", __FUNCTION__, output->name)); + DBG(("%s(%s:%d)\n", __FUNCTION__, output->name, sna_output->id)); + + if (!sna_output->id) + return XF86OutputStatusDisconnected; VG_CLEAR(compat_conn); compat_conn.conn.connector_id = sna_output->id; @@ -2134,7 +2139,8 @@ sna_output_get_modes(xf86OutputPtr output) DisplayModePtr Modes = NULL, Mode, current = NULL; int i; - DBG(("%s(%s)\n", __FUNCTION__, output->name)); + DBG(("%s(%s:%d)\n", __FUNCTION__, output->name, sna_output->id)); + assert(sna_output->id); sna_output_attach_edid(output); @@ -2245,7 +2251,7 @@ sna_output_dpms_backlight(xf86OutputPtr output, int oldmode, int mode) if (!sna_output->backlight.iface) return; - DBG(("%s(%s) -- %d -> %d\n", __FUNCTION__, output->name, oldmode, mode)); + DBG(("%s(%s:%d) -- %d -> %d\n", __FUNCTION__, output->name, sna_output->id, oldmode, mode)); if (mode == DPMSModeOn) { /* If we're going from off->on we may need to turn on the backlight. */ @@ -2266,11 +2272,14 @@ sna_output_dpms(xf86OutputPtr output, int dpms) struct sna *sna = to_sna(output->scrn); struct sna_output *sna_output = output->driver_private; - DBG(("%s(%s): dpms=%d (current: %d), active? %d\n", - __FUNCTION__, output->name, + DBG(("%s(%s:%d): dpms=%d (current: %d), active? %d\n", + __FUNCTION__, output->name, sna_output->id, dpms, sna_output->dpms_mode, output->crtc != NULL)); + if (!sna_output->id) + return; + if (sna_output->dpms_mode == dpms) return; @@ -2465,6 +2474,9 @@ sna_output_set_property(xf86OutputPtr output, Atom property, return TRUE; } + if (!sna_output->id) + return TRUE; + for (i = 0; i < sna_output->num_props; i++) { struct sna_property *p = &sna_output->props[i]; @@ -2725,8 +2737,13 @@ sna_mode_compute_possible_outputs(struct sna *sna) assert(sna_output); - output->possible_clones = sna_output->possible_encoders; - encoder_mask[i] = sna_output->attached_encoders; + if (sna_output->id) { + output->possible_clones = sna_output->possible_encoders; + encoder_mask[i] = sna_output->attached_encoders; + } else { + output->possible_clones = 0; + encoder_mask[i] = 0; + } } /* Convert from encoder numbering to output numbering */ @@ -3045,7 +3062,12 @@ void sna_mode_discover(struct sna *sna) for (i = 0; i < sna->mode.num_real_output; i++) { xf86OutputPtr output = config->output[i]; if (to_sna_output(output)->serial != serial) { - sna_output_del(output); i--; + if (sna->flags & SNA_REMOVE_OUTPUTS) { + sna_output_del(output); i--; + } else { + to_sna_output(output)->id = 0; + output->crtc = NULL; + } changed = true; } } diff --git a/src/sna/sna_driver.c b/src/sna/sna_driver.c index f1a30dcb..58e5cbc9 100644 --- a/src/sna/sna_driver.c +++ b/src/sna/sna_driver.c @@ -564,6 +564,9 @@ static Bool sna_pre_init(ScrnInfoPtr scrn, int flags) if (xf86ReturnOptValBool(sna->Options, OPTION_TILING_FB, FALSE)) sna->tiling &= ~SNA_TILING_FB; + if (xf86ReturnOptValBool(sna->Options, OPTION_DELETE_DP12, FALSE)) + sna->flags |= SNA_REMOVE_OUTPUTS; + if (!xf86ReturnOptValBool(sna->Options, OPTION_SWAPBUFFERS_WAIT, TRUE)) sna->flags |= SNA_NO_WAIT; DBG(("%s: swapbuffer wait? %s\n", __FUNCTION__, sna->flags & SNA_NO_WAIT ? "disabled" : "enabled")); |