summaryrefslogtreecommitdiff
path: root/src/sna
diff options
context:
space:
mode:
authorChris Wilson <chris@chris-wilson.co.uk>2014-05-08 06:45:43 +0100
committerChris Wilson <chris@chris-wilson.co.uk>2014-05-08 13:48:43 +0100
commite78a00c5da922024ac64270fae0797d98d1d6e57 (patch)
treed040920103a73f509891b067db5271194c21ee8e /src/sna
parenta93157587a82c699e7df76fb482c49569f0686b9 (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.h1
-rw-r--r--src/sna/sna_display.c38
-rw-r--r--src/sna/sna_driver.c3
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"));