| ========================= |
| Kernel Mode Setting (KMS) |
| ========================= |
| |
| Drivers must initialize the mode setting core by calling |
| :c:func:`drm_mode_config_init()` on the DRM device. The function |
| initializes the :c:type:`struct drm_device <drm_device>` |
| mode_config field and never fails. Once done, mode configuration must |
| be setup by initializing the following fields. |
| |
| - int min_width, min_height; int max_width, max_height; |
| Minimum and maximum width and height of the frame buffers in pixel |
| units. |
| |
| - struct drm_mode_config_funcs \*funcs; |
| Mode setting functions. |
| |
| Overview |
| ======== |
| |
| .. kernel-render:: DOT |
| :alt: KMS Display Pipeline |
| :caption: KMS Display Pipeline Overview |
| |
| digraph "KMS" { |
| node [shape=box] |
| |
| subgraph cluster_static { |
| style=dashed |
| label="Static Objects" |
| |
| node [bgcolor=grey style=filled] |
| "drm_plane A" -> "drm_crtc" |
| "drm_plane B" -> "drm_crtc" |
| "drm_crtc" -> "drm_encoder A" |
| "drm_crtc" -> "drm_encoder B" |
| } |
| |
| subgraph cluster_user_created { |
| style=dashed |
| label="Userspace-Created" |
| |
| node [shape=oval] |
| "drm_framebuffer 1" -> "drm_plane A" |
| "drm_framebuffer 2" -> "drm_plane B" |
| } |
| |
| subgraph cluster_connector { |
| style=dashed |
| label="Hotpluggable" |
| |
| "drm_encoder A" -> "drm_connector A" |
| "drm_encoder B" -> "drm_connector B" |
| } |
| } |
| |
| The basic object structure KMS presents to userspace is fairly simple. |
| Framebuffers (represented by :c:type:`struct drm_framebuffer <drm_framebuffer>`, |
| see `Frame Buffer Abstraction`_) feed into planes. One or more (or even no) |
| planes feed their pixel data into a CRTC (represented by :c:type:`struct |
| drm_crtc <drm_crtc>`, see `CRTC Abstraction`_) for blending. The precise |
| blending step is explained in more detail in `Plane Composition Properties`_ and |
| related chapters. |
| |
| For the output routing the first step is encoders (represented by |
| :c:type:`struct drm_encoder <drm_encoder>`, see `Encoder Abstraction`_). Those |
| are really just internal artifacts of the helper libraries used to implement KMS |
| drivers. Besides that they make it unecessarily more complicated for userspace |
| to figure out which connections between a CRTC and a connector are possible, and |
| what kind of cloning is supported, they serve no purpose in the userspace API. |
| Unfortunately encoders have been exposed to userspace, hence can't remove them |
| at this point. Futhermore the exposed restrictions are often wrongly set by |
| drivers, and in many cases not powerful enough to express the real restrictions. |
| A CRTC can be connected to multiple encoders, and for an active CRTC there must |
| be at least one encoder. |
| |
| The final, and real, endpoint in the display chain is the connector (represented |
| by :c:type:`struct drm_connector <drm_connector>`, see `Connector |
| Abstraction`_). Connectors can have different possible encoders, but the kernel |
| driver selects which encoder to use for each connector. The use case is DVI, |
| which could switch between an analog and a digital encoder. Encoders can also |
| drive multiple different connectors. There is exactly one active connector for |
| every active encoder. |
| |
| Internally the output pipeline is a bit more complex and matches today's |
| hardware more closely: |
| |
| .. kernel-render:: DOT |
| :alt: KMS Output Pipeline |
| :caption: KMS Output Pipeline |
| |
| digraph "Output Pipeline" { |
| node [shape=box] |
| |
| subgraph { |
| "drm_crtc" [bgcolor=grey style=filled] |
| } |
| |
| subgraph cluster_internal { |
| style=dashed |
| label="Internal Pipeline" |
| { |
| node [bgcolor=grey style=filled] |
| "drm_encoder A"; |
| "drm_encoder B"; |
| "drm_encoder C"; |
| } |
| |
| { |
| node [bgcolor=grey style=filled] |
| "drm_encoder B" -> "drm_bridge B" |
| "drm_encoder C" -> "drm_bridge C1" |
| "drm_bridge C1" -> "drm_bridge C2"; |
| } |
| } |
| |
| "drm_crtc" -> "drm_encoder A" |
| "drm_crtc" -> "drm_encoder B" |
| "drm_crtc" -> "drm_encoder C" |
| |
| |
| subgraph cluster_output { |
| style=dashed |
| label="Outputs" |
| |
| "drm_encoder A" -> "drm_connector A"; |
| "drm_bridge B" -> "drm_connector B"; |
| "drm_bridge C2" -> "drm_connector C"; |
| |
| "drm_panel" |
| } |
| } |
| |
| Internally two additional helper objects come into play. First, to be able to |
| share code for encoders (sometimes on the same SoC, sometimes off-chip) one or |
| more :ref:`drm_bridges` (represented by :c:type:`struct drm_bridge |
| <drm_bridge>`) can be linked to an encoder. This link is static and cannot be |
| changed, which means the cross-bar (if there is any) needs to be mapped between |
| the CRTC and any encoders. Often for drivers with bridges there's no code left |
| at the encoder level. Atomic drivers can leave out all the encoder callbacks to |
| essentially only leave a dummy routing object behind, which is needed for |
| backwards compatibility since encoders are exposed to userspace. |
| |
| The second object is for panels, represented by :c:type:`struct drm_panel |
| <drm_panel>`, see :ref:`drm_panel_helper`. Panels do not have a fixed binding |
| point, but are generally linked to the driver private structure that embeds |
| :c:type:`struct drm_connector <drm_connector>`. |
| |
| Note that currently the bridge chaining and interactions with connectors and |
| panels are still in-flux and not really fully sorted out yet. |
| |
| KMS Core Structures and Functions |
| ================================= |
| |
| .. kernel-doc:: include/drm/drm_mode_config.h |
| :internal: |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_mode_config.c |
| :export: |
| |
| Modeset Base Object Abstraction |
| =============================== |
| |
| .. kernel-render:: DOT |
| :alt: Mode Objects and Properties |
| :caption: Mode Objects and Properties |
| |
| digraph { |
| node [shape=box] |
| |
| "drm_property A" -> "drm_mode_object A" |
| "drm_property A" -> "drm_mode_object B" |
| "drm_property B" -> "drm_mode_object A" |
| } |
| |
| The base structure for all KMS objects is :c:type:`struct drm_mode_object |
| <drm_mode_object>`. One of the base services it provides is tracking properties, |
| which are especially important for the atomic IOCTL (see `Atomic Mode |
| Setting`_). The somewhat surprising part here is that properties are not |
| directly instantiated on each object, but free-standing mode objects themselves, |
| represented by :c:type:`struct drm_property <drm_property>`, which only specify |
| the type and value range of a property. Any given property can be attached |
| multiple times to different objects using :c:func:`drm_object_attach_property() |
| <drm_object_attach_property>`. |
| |
| .. kernel-doc:: include/drm/drm_mode_object.h |
| :internal: |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_mode_object.c |
| :export: |
| |
| Atomic Mode Setting |
| =================== |
| |
| |
| .. kernel-render:: DOT |
| :alt: Mode Objects and Properties |
| :caption: Mode Objects and Properties |
| |
| digraph { |
| node [shape=box] |
| |
| subgraph cluster_state { |
| style=dashed |
| label="Free-standing state" |
| |
| "drm_atomic_state" -> "duplicated drm_plane_state A" |
| "drm_atomic_state" -> "duplicated drm_plane_state B" |
| "drm_atomic_state" -> "duplicated drm_crtc_state" |
| "drm_atomic_state" -> "duplicated drm_connector_state" |
| "drm_atomic_state" -> "duplicated driver private state" |
| } |
| |
| subgraph cluster_current { |
| style=dashed |
| label="Current state" |
| |
| "drm_device" -> "drm_plane A" |
| "drm_device" -> "drm_plane B" |
| "drm_device" -> "drm_crtc" |
| "drm_device" -> "drm_connector" |
| "drm_device" -> "driver private object" |
| |
| "drm_plane A" -> "drm_plane_state A" |
| "drm_plane B" -> "drm_plane_state B" |
| "drm_crtc" -> "drm_crtc_state" |
| "drm_connector" -> "drm_connector_state" |
| "driver private object" -> "driver private state" |
| } |
| |
| "drm_atomic_state" -> "drm_device" [label="atomic_commit"] |
| "duplicated drm_plane_state A" -> "drm_device"[style=invis] |
| } |
| |
| Atomic provides transactional modeset (including planes) updates, but a |
| bit differently from the usual transactional approach of try-commit and |
| rollback: |
| |
| - Firstly, no hardware changes are allowed when the commit would fail. This |
| allows us to implement the DRM_MODE_ATOMIC_TEST_ONLY mode, which allows |
| userspace to explore whether certain configurations would work or not. |
| |
| - This would still allow setting and rollback of just the software state, |
| simplifying conversion of existing drivers. But auditing drivers for |
| correctness of the atomic_check code becomes really hard with that: Rolling |
| back changes in data structures all over the place is hard to get right. |
| |
| - Lastly, for backwards compatibility and to support all use-cases, atomic |
| updates need to be incremental and be able to execute in parallel. Hardware |
| doesn't always allow it, but where possible plane updates on different CRTCs |
| should not interfere, and not get stalled due to output routing changing on |
| different CRTCs. |
| |
| Taken all together there's two consequences for the atomic design: |
| |
| - The overall state is split up into per-object state structures: |
| :c:type:`struct drm_plane_state <drm_plane_state>` for planes, :c:type:`struct |
| drm_crtc_state <drm_crtc_state>` for CRTCs and :c:type:`struct |
| drm_connector_state <drm_connector_state>` for connectors. These are the only |
| objects with userspace-visible and settable state. For internal state drivers |
| can subclass these structures through embeddeding, or add entirely new state |
| structures for their globally shared hardware functions. |
| |
| - An atomic update is assembled and validated as an entirely free-standing pile |
| of structures within the :c:type:`drm_atomic_state <drm_atomic_state>` |
| container. Again drivers can subclass that container for their own state |
| structure tracking needs. Only when a state is committed is it applied to the |
| driver and modeset objects. This way rolling back an update boils down to |
| releasing memory and unreferencing objects like framebuffers. |
| |
| Read on in this chapter, and also in :ref:`drm_atomic_helper` for more detailed |
| coverage of specific topics. |
| |
| Atomic Mode Setting Function Reference |
| -------------------------------------- |
| |
| .. kernel-doc:: include/drm/drm_atomic.h |
| :internal: |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_atomic.c |
| :export: |
| |
| CRTC Abstraction |
| ================ |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_crtc.c |
| :doc: overview |
| |
| CRTC Functions Reference |
| -------------------------------- |
| |
| .. kernel-doc:: include/drm/drm_crtc.h |
| :internal: |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_crtc.c |
| :export: |
| |
| Frame Buffer Abstraction |
| ======================== |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_framebuffer.c |
| :doc: overview |
| |
| Frame Buffer Functions Reference |
| -------------------------------- |
| |
| .. kernel-doc:: include/drm/drm_framebuffer.h |
| :internal: |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_framebuffer.c |
| :export: |
| |
| DRM Format Handling |
| =================== |
| |
| .. kernel-doc:: include/drm/drm_fourcc.h |
| :internal: |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_fourcc.c |
| :export: |
| |
| Dumb Buffer Objects |
| =================== |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_dumb_buffers.c |
| :doc: overview |
| |
| Plane Abstraction |
| ================= |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_plane.c |
| :doc: overview |
| |
| Plane Functions Reference |
| ------------------------- |
| |
| .. kernel-doc:: include/drm/drm_plane.h |
| :internal: |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_plane.c |
| :export: |
| |
| Display Modes Function Reference |
| ================================ |
| |
| .. kernel-doc:: include/drm/drm_modes.h |
| :internal: |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_modes.c |
| :export: |
| |
| Connector Abstraction |
| ===================== |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_connector.c |
| :doc: overview |
| |
| Connector Functions Reference |
| ----------------------------- |
| |
| .. kernel-doc:: include/drm/drm_connector.h |
| :internal: |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_connector.c |
| :export: |
| |
| Encoder Abstraction |
| =================== |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_encoder.c |
| :doc: overview |
| |
| Encoder Functions Reference |
| --------------------------- |
| |
| .. kernel-doc:: include/drm/drm_encoder.h |
| :internal: |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_encoder.c |
| :export: |
| |
| KMS Initialization and Cleanup |
| ============================== |
| |
| A KMS device is abstracted and exposed as a set of planes, CRTCs, |
| encoders and connectors. KMS drivers must thus create and initialize all |
| those objects at load time after initializing mode setting. |
| |
| CRTCs (:c:type:`struct drm_crtc <drm_crtc>`) |
| -------------------------------------------- |
| |
| A CRTC is an abstraction representing a part of the chip that contains a |
| pointer to a scanout buffer. Therefore, the number of CRTCs available |
| determines how many independent scanout buffers can be active at any |
| given time. The CRTC structure contains several fields to support this: |
| a pointer to some video memory (abstracted as a frame buffer object), a |
| display mode, and an (x, y) offset into the video memory to support |
| panning or configurations where one piece of video memory spans multiple |
| CRTCs. |
| |
| CRTC Initialization |
| ~~~~~~~~~~~~~~~~~~~ |
| |
| A KMS device must create and register at least one struct |
| :c:type:`struct drm_crtc <drm_crtc>` instance. The instance is |
| allocated and zeroed by the driver, possibly as part of a larger |
| structure, and registered with a call to :c:func:`drm_crtc_init()` |
| with a pointer to CRTC functions. |
| |
| |
| Cleanup |
| ------- |
| |
| The DRM core manages its objects' lifetime. When an object is not needed |
| anymore the core calls its destroy function, which must clean up and |
| free every resource allocated for the object. Every |
| :c:func:`drm_\*_init()` call must be matched with a corresponding |
| :c:func:`drm_\*_cleanup()` call to cleanup CRTCs |
| (:c:func:`drm_crtc_cleanup()`), planes |
| (:c:func:`drm_plane_cleanup()`), encoders |
| (:c:func:`drm_encoder_cleanup()`) and connectors |
| (:c:func:`drm_connector_cleanup()`). Furthermore, connectors that |
| have been added to sysfs must be removed by a call to |
| :c:func:`drm_connector_unregister()` before calling |
| :c:func:`drm_connector_cleanup()`. |
| |
| Connectors state change detection must be cleanup up with a call to |
| :c:func:`drm_kms_helper_poll_fini()`. |
| |
| Output discovery and initialization example |
| ------------------------------------------- |
| |
| .. code-block:: c |
| |
| void intel_crt_init(struct drm_device *dev) |
| { |
| struct drm_connector *connector; |
| struct intel_output *intel_output; |
| |
| intel_output = kzalloc(sizeof(struct intel_output), GFP_KERNEL); |
| if (!intel_output) |
| return; |
| |
| connector = &intel_output->base; |
| drm_connector_init(dev, &intel_output->base, |
| &intel_crt_connector_funcs, DRM_MODE_CONNECTOR_VGA); |
| |
| drm_encoder_init(dev, &intel_output->enc, &intel_crt_enc_funcs, |
| DRM_MODE_ENCODER_DAC); |
| |
| drm_mode_connector_attach_encoder(&intel_output->base, |
| &intel_output->enc); |
| |
| /* Set up the DDC bus. */ |
| intel_output->ddc_bus = intel_i2c_create(dev, GPIOA, "CRTDDC_A"); |
| if (!intel_output->ddc_bus) { |
| dev_printk(KERN_ERR, &dev->pdev->dev, "DDC bus registration " |
| "failed.\n"); |
| return; |
| } |
| |
| intel_output->type = INTEL_OUTPUT_ANALOG; |
| connector->interlace_allowed = 0; |
| connector->doublescan_allowed = 0; |
| |
| drm_encoder_helper_add(&intel_output->enc, &intel_crt_helper_funcs); |
| drm_connector_helper_add(connector, &intel_crt_connector_helper_funcs); |
| |
| drm_connector_register(connector); |
| } |
| |
| In the example above (taken from the i915 driver), a CRTC, connector and |
| encoder combination is created. A device-specific i2c bus is also |
| created for fetching EDID data and performing monitor detection. Once |
| the process is complete, the new connector is registered with sysfs to |
| make its properties available to applications. |
| |
| KMS Locking |
| =========== |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_modeset_lock.c |
| :doc: kms locking |
| |
| .. kernel-doc:: include/drm/drm_modeset_lock.h |
| :internal: |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_modeset_lock.c |
| :export: |
| |
| KMS Properties |
| ============== |
| |
| Property Types and Blob Property Support |
| ---------------------------------------- |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_property.c |
| :doc: overview |
| |
| .. kernel-doc:: include/drm/drm_property.h |
| :internal: |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_property.c |
| :export: |
| |
| Standard Connector Properties |
| ----------------------------- |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_connector.c |
| :doc: standard connector properties |
| |
| Plane Composition Properties |
| ---------------------------- |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_blend.c |
| :doc: overview |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_blend.c |
| :export: |
| |
| Color Management Properties |
| --------------------------- |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_color_mgmt.c |
| :doc: overview |
| |
| .. kernel-doc:: include/drm/drm_color_mgmt.h |
| :internal: |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_color_mgmt.c |
| :export: |
| |
| Tile Group Property |
| ------------------- |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_connector.c |
| :doc: Tile group |
| |
| Explicit Fencing Properties |
| --------------------------- |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_atomic.c |
| :doc: explicit fencing properties |
| |
| Existing KMS Properties |
| ----------------------- |
| |
| The following table gives description of drm properties exposed by |
| various modules/drivers. |
| |
| .. csv-table:: |
| :header-rows: 1 |
| :file: kms-properties.csv |
| |
| Vertical Blanking |
| ================= |
| |
| Vertical blanking plays a major role in graphics rendering. To achieve |
| tear-free display, users must synchronize page flips and/or rendering to |
| vertical blanking. The DRM API offers ioctls to perform page flips |
| synchronized to vertical blanking and wait for vertical blanking. |
| |
| The DRM core handles most of the vertical blanking management logic, |
| which involves filtering out spurious interrupts, keeping race-free |
| blanking counters, coping with counter wrap-around and resets and |
| keeping use counts. It relies on the driver to generate vertical |
| blanking interrupts and optionally provide a hardware vertical blanking |
| counter. Drivers must implement the following operations. |
| |
| - int (\*enable_vblank) (struct drm_device \*dev, int crtc); void |
| (\*disable_vblank) (struct drm_device \*dev, int crtc); |
| Enable or disable vertical blanking interrupts for the given CRTC. |
| |
| - u32 (\*get_vblank_counter) (struct drm_device \*dev, int crtc); |
| Retrieve the value of the vertical blanking counter for the given |
| CRTC. If the hardware maintains a vertical blanking counter its value |
| should be returned. Otherwise drivers can use the |
| :c:func:`drm_vblank_count()` helper function to handle this |
| operation. |
| |
| Drivers must initialize the vertical blanking handling core with a call |
| to :c:func:`drm_vblank_init()` in their load operation. |
| |
| Vertical blanking interrupts can be enabled by the DRM core or by |
| drivers themselves (for instance to handle page flipping operations). |
| The DRM core maintains a vertical blanking use count to ensure that the |
| interrupts are not disabled while a user still needs them. To increment |
| the use count, drivers call :c:func:`drm_vblank_get()`. Upon |
| return vertical blanking interrupts are guaranteed to be enabled. |
| |
| To decrement the use count drivers call |
| :c:func:`drm_vblank_put()`. Only when the use count drops to zero |
| will the DRM core disable the vertical blanking interrupts after a delay |
| by scheduling a timer. The delay is accessible through the |
| vblankoffdelay module parameter or the ``drm_vblank_offdelay`` global |
| variable and expressed in milliseconds. Its default value is 5000 ms. |
| Zero means never disable, and a negative value means disable |
| immediately. Drivers may override the behaviour by setting the |
| :c:type:`struct drm_device <drm_device>` |
| vblank_disable_immediate flag, which when set causes vblank interrupts |
| to be disabled immediately regardless of the drm_vblank_offdelay |
| value. The flag should only be set if there's a properly working |
| hardware vblank counter present. |
| |
| When a vertical blanking interrupt occurs drivers only need to call the |
| :c:func:`drm_handle_vblank()` function to account for the |
| interrupt. |
| |
| Resources allocated by :c:func:`drm_vblank_init()` must be freed |
| with a call to :c:func:`drm_vblank_cleanup()` in the driver unload |
| operation handler. |
| |
| Vertical Blanking and Interrupt Handling Functions Reference |
| ------------------------------------------------------------ |
| |
| .. kernel-doc:: include/drm/drm_vblank.h |
| :internal: |
| |
| .. kernel-doc:: drivers/gpu/drm/drm_vblank.c |
| :export: |