drm/sysfs: Send out uevent when connector->force changes

To avoid even more code duplication punt this all to the probe worker,
which needs some slight adjustment to also generate a uevent when the
status has changed to due connector->force.

v2: Instead of running the output_poll_work (which is kinda the wrong
thing and a layering violation since it's an internal of the probe
helpers), or calling ->detect (which is again a layering violation
since it's used only by probe helpers) just call the official
->fill_modes function, like a GET_CONNECTOR ioctl call.

v3: Restore the accidentally removed forced-probe for echo "detect" >
force.

Cc: Chris Wilson <chris@chris-wilson.co.uk>
Reported-by: Chris Wilson <chris@chris-wilson.co.uk>
Signed-off-by: Daniel Vetter <daniel.vetter@intel.com>
Link: http://patchwork.freedesktop.org/patch/msgid/1447951610-12622-22-git-send-email-daniel.vetter@ffwll.ch
Reviewed-by: Chris Wilson <chris@chris-wilson.co.uk>
Signed-off-by: Daniel Vetter <daniel.vetter@ffwll.ch>
This commit is contained in:
Daniel Vetter 2015-11-19 17:46:50 +01:00
Родитель 9744bf41f1
Коммит ed293f7754
2 изменённых файлов: 36 добавлений и 48 удалений

Просмотреть файл

@ -147,6 +147,8 @@ static int drm_helper_probe_single_connector_modes_merge_bits(struct drm_connect
list_for_each_entry(mode, &connector->modes, head) list_for_each_entry(mode, &connector->modes, head)
mode->status = MODE_UNVERIFIED; mode->status = MODE_UNVERIFIED;
old_status = connector->status;
if (connector->force) { if (connector->force) {
if (connector->force == DRM_FORCE_ON || if (connector->force == DRM_FORCE_ON ||
connector->force == DRM_FORCE_ON_DIGITAL) connector->force == DRM_FORCE_ON_DIGITAL)
@ -156,33 +158,31 @@ static int drm_helper_probe_single_connector_modes_merge_bits(struct drm_connect
if (connector->funcs->force) if (connector->funcs->force)
connector->funcs->force(connector); connector->funcs->force(connector);
} else { } else {
old_status = connector->status;
connector->status = connector->funcs->detect(connector, true); connector->status = connector->funcs->detect(connector, true);
}
/*
* Normally either the driver's hpd code or the poll loop should
* pick up any changes and fire the hotplug event. But if
* userspace sneaks in a probe, we might miss a change. Hence
* check here, and if anything changed start the hotplug code.
*/
if (old_status != connector->status) {
DRM_DEBUG_KMS("[CONNECTOR:%d:%s] status updated from %d to %d\n",
connector->base.id,
connector->name,
old_status, connector->status);
/* /*
* Normally either the driver's hpd code or the poll loop should * The hotplug event code might call into the fb
* pick up any changes and fire the hotplug event. But if * helpers, and so expects that we do not hold any
* userspace sneaks in a probe, we might miss a change. Hence * locks. Fire up the poll struct instead, it will
* check here, and if anything changed start the hotplug code. * disable itself again.
*/ */
if (old_status != connector->status) { dev->mode_config.delayed_event = true;
DRM_DEBUG_KMS("[CONNECTOR:%d:%s] status updated from %d to %d\n", if (dev->mode_config.poll_enabled)
connector->base.id, schedule_delayed_work(&dev->mode_config.output_poll_work,
connector->name, 0);
old_status, connector->status);
/*
* The hotplug event code might call into the fb
* helpers, and so expects that we do not hold any
* locks. Fire up the poll struct instead, it will
* disable itself again.
*/
dev->mode_config.delayed_event = true;
if (dev->mode_config.poll_enabled)
schedule_delayed_work(&dev->mode_config.output_poll_work,
0);
}
} }
/* Re-enable polling in case the global poll config changed. */ /* Re-enable polling in case the global poll config changed. */

Просмотреть файл

@ -167,47 +167,35 @@ static ssize_t status_store(struct device *device,
{ {
struct drm_connector *connector = to_drm_connector(device); struct drm_connector *connector = to_drm_connector(device);
struct drm_device *dev = connector->dev; struct drm_device *dev = connector->dev;
enum drm_connector_status old_status; enum drm_connector_force old_force;
int ret; int ret;
ret = mutex_lock_interruptible(&dev->mode_config.mutex); ret = mutex_lock_interruptible(&dev->mode_config.mutex);
if (ret) if (ret)
return ret; return ret;
old_status = connector->status; old_force = connector->force;
if (sysfs_streq(buf, "detect")) { if (sysfs_streq(buf, "detect"))
connector->force = 0; connector->force = 0;
connector->status = connector->funcs->detect(connector, true); else if (sysfs_streq(buf, "on"))
} else if (sysfs_streq(buf, "on")) {
connector->force = DRM_FORCE_ON; connector->force = DRM_FORCE_ON;
} else if (sysfs_streq(buf, "on-digital")) { else if (sysfs_streq(buf, "on-digital"))
connector->force = DRM_FORCE_ON_DIGITAL; connector->force = DRM_FORCE_ON_DIGITAL;
} else if (sysfs_streq(buf, "off")) { else if (sysfs_streq(buf, "off"))
connector->force = DRM_FORCE_OFF; connector->force = DRM_FORCE_OFF;
} else else
ret = -EINVAL; ret = -EINVAL;
if (ret == 0 && connector->force) { if (old_force != connector->force || !connector->force) {
if (connector->force == DRM_FORCE_ON || DRM_DEBUG_KMS("[CONNECTOR:%d:%s] force updated from %d to %d or reprobing\n",
connector->force == DRM_FORCE_ON_DIGITAL)
connector->status = connector_status_connected;
else
connector->status = connector_status_disconnected;
if (connector->funcs->force)
connector->funcs->force(connector);
}
if (old_status != connector->status) {
DRM_DEBUG_KMS("[CONNECTOR:%d:%s] status updated from %d to %d\n",
connector->base.id, connector->base.id,
connector->name, connector->name,
old_status, connector->status); old_force, connector->force);
dev->mode_config.delayed_event = true; connector->funcs->fill_modes(connector,
if (dev->mode_config.poll_enabled) dev->mode_config.max_width,
schedule_delayed_work(&dev->mode_config.output_poll_work, dev->mode_config.max_height);
0);
} }
mutex_unlock(&dev->mode_config.mutex); mutex_unlock(&dev->mode_config.mutex);