sync with OpenBSD -current
This commit is contained in:
parent
ee68147dcd
commit
1cefe29c7e
1651 changed files with 283292 additions and 68089 deletions
|
@ -36,7 +36,6 @@
|
|||
#include <drm/drm_client.h>
|
||||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_fb_helper.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_modeset_helper_vtables.h>
|
||||
#include <drm/drm_print.h>
|
||||
|
@ -223,7 +222,66 @@ drm_connector_mode_valid(struct drm_connector *connector,
|
|||
return ret;
|
||||
}
|
||||
|
||||
static void drm_kms_helper_disable_hpd(struct drm_device *dev)
|
||||
{
|
||||
struct drm_connector *connector;
|
||||
struct drm_connector_list_iter conn_iter;
|
||||
|
||||
drm_connector_list_iter_begin(dev, &conn_iter);
|
||||
drm_for_each_connector_iter(connector, &conn_iter) {
|
||||
const struct drm_connector_helper_funcs *funcs =
|
||||
connector->helper_private;
|
||||
|
||||
if (funcs && funcs->disable_hpd)
|
||||
funcs->disable_hpd(connector);
|
||||
}
|
||||
drm_connector_list_iter_end(&conn_iter);
|
||||
}
|
||||
|
||||
static bool drm_kms_helper_enable_hpd(struct drm_device *dev)
|
||||
{
|
||||
bool poll = false;
|
||||
struct drm_connector *connector;
|
||||
struct drm_connector_list_iter conn_iter;
|
||||
|
||||
drm_connector_list_iter_begin(dev, &conn_iter);
|
||||
drm_for_each_connector_iter(connector, &conn_iter) {
|
||||
const struct drm_connector_helper_funcs *funcs =
|
||||
connector->helper_private;
|
||||
|
||||
if (funcs && funcs->enable_hpd)
|
||||
funcs->enable_hpd(connector);
|
||||
|
||||
if (connector->polled & (DRM_CONNECTOR_POLL_CONNECT |
|
||||
DRM_CONNECTOR_POLL_DISCONNECT))
|
||||
poll = true;
|
||||
}
|
||||
drm_connector_list_iter_end(&conn_iter);
|
||||
|
||||
return poll;
|
||||
}
|
||||
|
||||
#define DRM_OUTPUT_POLL_PERIOD (10*HZ)
|
||||
static void reschedule_output_poll_work(struct drm_device *dev)
|
||||
{
|
||||
unsigned long delay = DRM_OUTPUT_POLL_PERIOD;
|
||||
|
||||
if (dev->mode_config.delayed_event)
|
||||
/*
|
||||
* FIXME:
|
||||
*
|
||||
* Use short (1s) delay to handle the initial delayed event.
|
||||
* This delay should not be needed, but Optimus/nouveau will
|
||||
* fail in a mysterious way if the delayed event is handled as
|
||||
* soon as possible like it is done in
|
||||
* drm_helper_probe_single_connector_modes() in case the poll
|
||||
* was enabled before.
|
||||
*/
|
||||
delay = HZ;
|
||||
|
||||
schedule_delayed_work(&dev->mode_config.output_poll_work, delay);
|
||||
}
|
||||
|
||||
/**
|
||||
* drm_kms_helper_poll_enable - re-enable output polling.
|
||||
* @dev: drm_device
|
||||
|
@ -241,42 +299,41 @@ drm_connector_mode_valid(struct drm_connector *connector,
|
|||
*/
|
||||
void drm_kms_helper_poll_enable(struct drm_device *dev)
|
||||
{
|
||||
bool poll = false;
|
||||
struct drm_connector *connector;
|
||||
struct drm_connector_list_iter conn_iter;
|
||||
unsigned long delay = DRM_OUTPUT_POLL_PERIOD;
|
||||
|
||||
if (!dev->mode_config.poll_enabled || !drm_kms_helper_poll)
|
||||
if (!dev->mode_config.poll_enabled || !drm_kms_helper_poll ||
|
||||
dev->mode_config.poll_running)
|
||||
return;
|
||||
|
||||
drm_connector_list_iter_begin(dev, &conn_iter);
|
||||
drm_for_each_connector_iter(connector, &conn_iter) {
|
||||
if (connector->polled & (DRM_CONNECTOR_POLL_CONNECT |
|
||||
DRM_CONNECTOR_POLL_DISCONNECT))
|
||||
poll = true;
|
||||
}
|
||||
drm_connector_list_iter_end(&conn_iter);
|
||||
if (drm_kms_helper_enable_hpd(dev) ||
|
||||
dev->mode_config.delayed_event)
|
||||
reschedule_output_poll_work(dev);
|
||||
|
||||
if (dev->mode_config.delayed_event) {
|
||||
/*
|
||||
* FIXME:
|
||||
*
|
||||
* Use short (1s) delay to handle the initial delayed event.
|
||||
* This delay should not be needed, but Optimus/nouveau will
|
||||
* fail in a mysterious way if the delayed event is handled as
|
||||
* soon as possible like it is done in
|
||||
* drm_helper_probe_single_connector_modes() in case the poll
|
||||
* was enabled before.
|
||||
*/
|
||||
poll = true;
|
||||
delay = HZ;
|
||||
}
|
||||
|
||||
if (poll)
|
||||
schedule_delayed_work(&dev->mode_config.output_poll_work, delay);
|
||||
dev->mode_config.poll_running = true;
|
||||
}
|
||||
EXPORT_SYMBOL(drm_kms_helper_poll_enable);
|
||||
|
||||
/**
|
||||
* drm_kms_helper_poll_reschedule - reschedule the output polling work
|
||||
* @dev: drm_device
|
||||
*
|
||||
* This function reschedules the output polling work, after polling for a
|
||||
* connector has been enabled.
|
||||
*
|
||||
* Drivers must call this helper after enabling polling for a connector by
|
||||
* setting %DRM_CONNECTOR_POLL_CONNECT / %DRM_CONNECTOR_POLL_DISCONNECT flags
|
||||
* in drm_connector::polled. Note that after disabling polling by clearing these
|
||||
* flags for a connector will stop the output polling work automatically if
|
||||
* the polling is disabled for all other connectors as well.
|
||||
*
|
||||
* The function can be called only after polling has been enabled by calling
|
||||
* drm_kms_helper_poll_init() / drm_kms_helper_poll_enable().
|
||||
*/
|
||||
void drm_kms_helper_poll_reschedule(struct drm_device *dev)
|
||||
{
|
||||
if (dev->mode_config.poll_running)
|
||||
reschedule_output_poll_work(dev);
|
||||
}
|
||||
EXPORT_SYMBOL(drm_kms_helper_poll_reschedule);
|
||||
|
||||
static enum drm_connector_status
|
||||
drm_helper_probe_detect_ctx(struct drm_connector *connector, bool force)
|
||||
{
|
||||
|
@ -367,7 +424,7 @@ static int drm_helper_probe_get_modes(struct drm_connector *connector)
|
|||
* override/firmware EDID.
|
||||
*/
|
||||
if (count == 0 && connector->status == connector_status_connected)
|
||||
count = drm_add_override_edid_modes(connector);
|
||||
count = drm_edid_override_connector_update(connector);
|
||||
|
||||
return count;
|
||||
}
|
||||
|
@ -563,10 +620,7 @@ retry:
|
|||
}
|
||||
|
||||
/* Re-enable polling in case the global poll config changed. */
|
||||
if (drm_kms_helper_poll != dev->mode_config.poll_running)
|
||||
drm_kms_helper_poll_enable(dev);
|
||||
|
||||
dev->mode_config.poll_running = drm_kms_helper_poll;
|
||||
drm_kms_helper_poll_enable(dev);
|
||||
|
||||
if (connector->status == connector_status_disconnected) {
|
||||
DRM_DEBUG_KMS("[CONNECTOR:%d:%s] disconnected\n",
|
||||
|
@ -706,8 +760,11 @@ static void output_poll_execute(struct work_struct *work)
|
|||
changed = dev->mode_config.delayed_event;
|
||||
dev->mode_config.delayed_event = false;
|
||||
|
||||
if (!drm_kms_helper_poll)
|
||||
if (!drm_kms_helper_poll && dev->mode_config.poll_running) {
|
||||
drm_kms_helper_disable_hpd(dev);
|
||||
dev->mode_config.poll_running = false;
|
||||
goto out;
|
||||
}
|
||||
|
||||
if (!mutex_trylock(&dev->mode_config.mutex)) {
|
||||
repoll = true;
|
||||
|
@ -823,9 +880,12 @@ EXPORT_SYMBOL(drm_kms_helper_is_poll_worker);
|
|||
*/
|
||||
void drm_kms_helper_poll_disable(struct drm_device *dev)
|
||||
{
|
||||
if (!dev->mode_config.poll_enabled)
|
||||
return;
|
||||
if (dev->mode_config.poll_running)
|
||||
drm_kms_helper_disable_hpd(dev);
|
||||
|
||||
cancel_delayed_work_sync(&dev->mode_config.output_poll_work);
|
||||
|
||||
dev->mode_config.poll_running = false;
|
||||
}
|
||||
EXPORT_SYMBOL(drm_kms_helper_poll_disable);
|
||||
|
||||
|
@ -866,8 +926,9 @@ void drm_kms_helper_poll_fini(struct drm_device *dev)
|
|||
if (!dev->mode_config.poll_enabled)
|
||||
return;
|
||||
|
||||
drm_kms_helper_poll_disable(dev);
|
||||
|
||||
dev->mode_config.poll_enabled = false;
|
||||
cancel_delayed_work_sync(&dev->mode_config.output_poll_work);
|
||||
}
|
||||
EXPORT_SYMBOL(drm_kms_helper_poll_fini);
|
||||
|
||||
|
@ -1144,10 +1205,94 @@ int drm_connector_helper_get_modes(struct drm_connector *connector)
|
|||
* EDID. Otherwise, if the EDID is NULL, clear the connector
|
||||
* information.
|
||||
*/
|
||||
count = drm_edid_connector_update(connector, drm_edid);
|
||||
drm_edid_connector_update(connector, drm_edid);
|
||||
|
||||
count = drm_edid_connector_add_modes(connector);
|
||||
|
||||
drm_edid_free(drm_edid);
|
||||
|
||||
return count;
|
||||
}
|
||||
EXPORT_SYMBOL(drm_connector_helper_get_modes);
|
||||
|
||||
/**
|
||||
* drm_connector_helper_tv_get_modes - Fills the modes availables to a TV connector
|
||||
* @connector: The connector
|
||||
*
|
||||
* Fills the available modes for a TV connector based on the supported
|
||||
* TV modes, and the default mode expressed by the kernel command line.
|
||||
*
|
||||
* This can be used as the default TV connector helper .get_modes() hook
|
||||
* if the driver does not need any special processing.
|
||||
*
|
||||
* Returns:
|
||||
* The number of modes added to the connector.
|
||||
*/
|
||||
int drm_connector_helper_tv_get_modes(struct drm_connector *connector)
|
||||
{
|
||||
struct drm_device *dev = connector->dev;
|
||||
struct drm_property *tv_mode_property =
|
||||
dev->mode_config.tv_mode_property;
|
||||
struct drm_cmdline_mode *cmdline = &connector->cmdline_mode;
|
||||
unsigned int ntsc_modes = BIT(DRM_MODE_TV_MODE_NTSC) |
|
||||
BIT(DRM_MODE_TV_MODE_NTSC_443) |
|
||||
BIT(DRM_MODE_TV_MODE_NTSC_J) |
|
||||
BIT(DRM_MODE_TV_MODE_PAL_M);
|
||||
unsigned int pal_modes = BIT(DRM_MODE_TV_MODE_PAL) |
|
||||
BIT(DRM_MODE_TV_MODE_PAL_N) |
|
||||
BIT(DRM_MODE_TV_MODE_SECAM);
|
||||
unsigned int tv_modes[2] = { UINT_MAX, UINT_MAX };
|
||||
unsigned int i, supported_tv_modes = 0;
|
||||
|
||||
if (!tv_mode_property)
|
||||
return 0;
|
||||
|
||||
for (i = 0; i < tv_mode_property->num_values; i++)
|
||||
supported_tv_modes |= BIT(tv_mode_property->values[i]);
|
||||
|
||||
if ((supported_tv_modes & ntsc_modes) &&
|
||||
(supported_tv_modes & pal_modes)) {
|
||||
uint64_t default_mode;
|
||||
|
||||
if (drm_object_property_get_default_value(&connector->base,
|
||||
tv_mode_property,
|
||||
&default_mode))
|
||||
return 0;
|
||||
|
||||
if (cmdline->tv_mode_specified)
|
||||
default_mode = cmdline->tv_mode;
|
||||
|
||||
if (BIT(default_mode) & ntsc_modes) {
|
||||
tv_modes[0] = DRM_MODE_TV_MODE_NTSC;
|
||||
tv_modes[1] = DRM_MODE_TV_MODE_PAL;
|
||||
} else {
|
||||
tv_modes[0] = DRM_MODE_TV_MODE_PAL;
|
||||
tv_modes[1] = DRM_MODE_TV_MODE_NTSC;
|
||||
}
|
||||
} else if (supported_tv_modes & ntsc_modes) {
|
||||
tv_modes[0] = DRM_MODE_TV_MODE_NTSC;
|
||||
} else if (supported_tv_modes & pal_modes) {
|
||||
tv_modes[0] = DRM_MODE_TV_MODE_PAL;
|
||||
} else {
|
||||
return 0;
|
||||
}
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(tv_modes); i++) {
|
||||
struct drm_display_mode *mode;
|
||||
|
||||
if (tv_modes[i] == DRM_MODE_TV_MODE_NTSC)
|
||||
mode = drm_mode_analog_ntsc_480i(dev);
|
||||
else if (tv_modes[i] == DRM_MODE_TV_MODE_PAL)
|
||||
mode = drm_mode_analog_pal_576i(dev);
|
||||
else
|
||||
break;
|
||||
if (!mode)
|
||||
return i;
|
||||
if (!i)
|
||||
mode->type |= DRM_MODE_TYPE_PREFERRED;
|
||||
drm_mode_probed_add(connector, mode);
|
||||
}
|
||||
|
||||
return i;
|
||||
}
|
||||
EXPORT_SYMBOL(drm_connector_helper_tv_get_modes);
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue