| /* SPDX-License-Identifier: GPL-2.0+ */ |
| |
| #ifndef _VKMS_CONFIG_H_ |
| #define _VKMS_CONFIG_H_ |
| |
| #include <linux/list.h> |
| #include <linux/types.h> |
| #include <linux/xarray.h> |
| |
| #include "vkms_drv.h" |
| |
| /** |
| * struct vkms_config - General configuration for VKMS driver |
| * |
| * @dev_name: Name of the device |
| * @planes: List of planes configured for the device |
| * @crtcs: List of CRTCs configured for the device |
| * @encoders: List of encoders configured for the device |
| * @connectors: List of connectors configured for the device |
| * @dev: Used to store the current VKMS device. Only set when the device is instantiated. |
| */ |
| struct vkms_config { |
| const char *dev_name; |
| struct list_head planes; |
| struct list_head crtcs; |
| struct list_head encoders; |
| struct list_head connectors; |
| struct vkms_device *dev; |
| }; |
| |
| /** |
| * struct vkms_config_plane |
| * |
| * @link: Link to the others planes in vkms_config |
| * @config: The vkms_config this plane belongs to |
| * @type: Type of the plane. The creator of configuration needs to ensures that |
| * at least one primary plane is present. |
| * @possible_crtcs: Array of CRTCs that can be used with this plane |
| * @plane: Internal usage. This pointer should never be considered as valid. |
| * It can be used to store a temporary reference to a VKMS plane during |
| * device creation. This pointer is not managed by the configuration and |
| * must be managed by other means. |
| */ |
| struct vkms_config_plane { |
| struct list_head link; |
| struct vkms_config *config; |
| |
| enum drm_plane_type type; |
| struct xarray possible_crtcs; |
| |
| /* Internal usage */ |
| struct vkms_plane *plane; |
| }; |
| |
| /** |
| * struct vkms_config_crtc |
| * |
| * @link: Link to the others CRTCs in vkms_config |
| * @config: The vkms_config this CRTC belongs to |
| * @writeback: If true, a writeback buffer can be attached to the CRTC |
| * @crtc: Internal usage. This pointer should never be considered as valid. |
| * It can be used to store a temporary reference to a VKMS CRTC during |
| * device creation. This pointer is not managed by the configuration and |
| * must be managed by other means. |
| */ |
| struct vkms_config_crtc { |
| struct list_head link; |
| struct vkms_config *config; |
| |
| bool writeback; |
| |
| /* Internal usage */ |
| struct vkms_output *crtc; |
| }; |
| |
| /** |
| * struct vkms_config_encoder |
| * |
| * @link: Link to the others encoders in vkms_config |
| * @config: The vkms_config this CRTC belongs to |
| * @possible_crtcs: Array of CRTCs that can be used with this encoder |
| * @encoder: Internal usage. This pointer should never be considered as valid. |
| * It can be used to store a temporary reference to a VKMS encoder |
| * during device creation. This pointer is not managed by the |
| * configuration and must be managed by other means. |
| */ |
| struct vkms_config_encoder { |
| struct list_head link; |
| struct vkms_config *config; |
| |
| struct xarray possible_crtcs; |
| |
| /* Internal usage */ |
| struct drm_encoder *encoder; |
| }; |
| |
| /** |
| * struct vkms_config_connector |
| * |
| * @link: Link to the others connector in vkms_config |
| * @config: The vkms_config this connector belongs to |
| * @possible_encoders: Array of encoders that can be used with this connector |
| * @connector: Internal usage. This pointer should never be considered as valid. |
| * It can be used to store a temporary reference to a VKMS connector |
| * during device creation. This pointer is not managed by the |
| * configuration and must be managed by other means. |
| */ |
| struct vkms_config_connector { |
| struct list_head link; |
| struct vkms_config *config; |
| |
| struct xarray possible_encoders; |
| |
| /* Internal usage */ |
| struct vkms_connector *connector; |
| }; |
| |
| /** |
| * vkms_config_for_each_plane - Iterate over the vkms_config planes |
| * @config: &struct vkms_config pointer |
| * @plane_cfg: &struct vkms_config_plane pointer used as cursor |
| */ |
| #define vkms_config_for_each_plane(config, plane_cfg) \ |
| list_for_each_entry((plane_cfg), &(config)->planes, link) |
| |
| /** |
| * vkms_config_for_each_crtc - Iterate over the vkms_config CRTCs |
| * @config: &struct vkms_config pointer |
| * @crtc_cfg: &struct vkms_config_crtc pointer used as cursor |
| */ |
| #define vkms_config_for_each_crtc(config, crtc_cfg) \ |
| list_for_each_entry((crtc_cfg), &(config)->crtcs, link) |
| |
| /** |
| * vkms_config_for_each_encoder - Iterate over the vkms_config encoders |
| * @config: &struct vkms_config pointer |
| * @encoder_cfg: &struct vkms_config_encoder pointer used as cursor |
| */ |
| #define vkms_config_for_each_encoder(config, encoder_cfg) \ |
| list_for_each_entry((encoder_cfg), &(config)->encoders, link) |
| |
| /** |
| * vkms_config_for_each_connector - Iterate over the vkms_config connectors |
| * @config: &struct vkms_config pointer |
| * @connector_cfg: &struct vkms_config_connector pointer used as cursor |
| */ |
| #define vkms_config_for_each_connector(config, connector_cfg) \ |
| list_for_each_entry((connector_cfg), &(config)->connectors, link) |
| |
| /** |
| * vkms_config_plane_for_each_possible_crtc - Iterate over the vkms_config_plane |
| * possible CRTCs |
| * @plane_cfg: &struct vkms_config_plane pointer |
| * @idx: Index of the cursor |
| * @possible_crtc: &struct vkms_config_crtc pointer used as cursor |
| */ |
| #define vkms_config_plane_for_each_possible_crtc(plane_cfg, idx, possible_crtc) \ |
| xa_for_each(&(plane_cfg)->possible_crtcs, idx, (possible_crtc)) |
| |
| /** |
| * vkms_config_encoder_for_each_possible_crtc - Iterate over the |
| * vkms_config_encoder possible CRTCs |
| * @encoder_cfg: &struct vkms_config_encoder pointer |
| * @idx: Index of the cursor |
| * @possible_crtc: &struct vkms_config_crtc pointer used as cursor |
| */ |
| #define vkms_config_encoder_for_each_possible_crtc(encoder_cfg, idx, possible_crtc) \ |
| xa_for_each(&(encoder_cfg)->possible_crtcs, idx, (possible_crtc)) |
| |
| /** |
| * vkms_config_connector_for_each_possible_encoder - Iterate over the |
| * vkms_config_connector possible encoders |
| * @connector_cfg: &struct vkms_config_connector pointer |
| * @idx: Index of the cursor |
| * @possible_encoder: &struct vkms_config_encoder pointer used as cursor |
| */ |
| #define vkms_config_connector_for_each_possible_encoder(connector_cfg, idx, possible_encoder) \ |
| xa_for_each(&(connector_cfg)->possible_encoders, idx, (possible_encoder)) |
| |
| /** |
| * vkms_config_create() - Create a new VKMS configuration |
| * @dev_name: Name of the device |
| * |
| * Returns: |
| * The new vkms_config or an error. Call vkms_config_destroy() to free the |
| * returned configuration. |
| */ |
| struct vkms_config *vkms_config_create(const char *dev_name); |
| |
| /** |
| * vkms_config_default_create() - Create the configuration for the default device |
| * @enable_cursor: Create or not a cursor plane |
| * @enable_writeback: Create or not a writeback connector |
| * @enable_overlay: Create or not overlay planes |
| * |
| * Returns: |
| * The default vkms_config or an error. Call vkms_config_destroy() to free the |
| * returned configuration. |
| */ |
| struct vkms_config *vkms_config_default_create(bool enable_cursor, |
| bool enable_writeback, |
| bool enable_overlay); |
| |
| /** |
| * vkms_config_destroy() - Free a VKMS configuration |
| * @config: vkms_config to free |
| */ |
| void vkms_config_destroy(struct vkms_config *config); |
| |
| /** |
| * vkms_config_get_device_name() - Return the name of the device |
| * @config: Configuration to get the device name from |
| * |
| * Returns: |
| * The device name. Only valid while @config is valid. |
| */ |
| static inline const char * |
| vkms_config_get_device_name(struct vkms_config *config) |
| { |
| return config->dev_name; |
| } |
| |
| /** |
| * vkms_config_get_num_crtcs() - Return the number of CRTCs in the configuration |
| * @config: Configuration to get the number of CRTCs from |
| */ |
| static inline size_t vkms_config_get_num_crtcs(struct vkms_config *config) |
| { |
| return list_count_nodes(&config->crtcs); |
| } |
| |
| /** |
| * vkms_config_is_valid() - Validate a configuration |
| * @config: Configuration to validate |
| * |
| * Returns: |
| * Whether the configuration is valid or not. |
| * For example, a configuration without primary planes is not valid. |
| */ |
| bool vkms_config_is_valid(const struct vkms_config *config); |
| |
| /** |
| * vkms_config_register_debugfs() - Register a debugfs file to show the device's |
| * configuration |
| * @vkms_device: Device to register |
| */ |
| void vkms_config_register_debugfs(struct vkms_device *vkms_device); |
| |
| /** |
| * vkms_config_create_plane() - Add a new plane configuration |
| * @config: Configuration to add the plane to |
| * |
| * Returns: |
| * The new plane configuration or an error. Call vkms_config_destroy_plane() to |
| * free the returned plane configuration. |
| */ |
| struct vkms_config_plane *vkms_config_create_plane(struct vkms_config *config); |
| |
| /** |
| * vkms_config_destroy_plane() - Remove and free a plane configuration |
| * @plane_cfg: Plane configuration to destroy |
| */ |
| void vkms_config_destroy_plane(struct vkms_config_plane *plane_cfg); |
| |
| /** |
| * vkms_config_plane_type() - Return the plane type |
| * @plane_cfg: Plane to get the type from |
| */ |
| static inline enum drm_plane_type |
| vkms_config_plane_get_type(struct vkms_config_plane *plane_cfg) |
| { |
| return plane_cfg->type; |
| } |
| |
| /** |
| * vkms_config_plane_set_type() - Set the plane type |
| * @plane_cfg: Plane to set the type to |
| * @type: New plane type |
| */ |
| static inline void |
| vkms_config_plane_set_type(struct vkms_config_plane *plane_cfg, |
| enum drm_plane_type type) |
| { |
| plane_cfg->type = type; |
| } |
| |
| /** |
| * vkms_config_plane_attach_crtc - Attach a plane to a CRTC |
| * @plane_cfg: Plane to attach |
| * @crtc_cfg: CRTC to attach @plane_cfg to |
| */ |
| int __must_check vkms_config_plane_attach_crtc(struct vkms_config_plane *plane_cfg, |
| struct vkms_config_crtc *crtc_cfg); |
| |
| /** |
| * vkms_config_plane_detach_crtc - Detach a plane from a CRTC |
| * @plane_cfg: Plane to detach |
| * @crtc_cfg: CRTC to detach @plane_cfg from |
| */ |
| void vkms_config_plane_detach_crtc(struct vkms_config_plane *plane_cfg, |
| struct vkms_config_crtc *crtc_cfg); |
| |
| /** |
| * vkms_config_create_crtc() - Add a new CRTC configuration |
| * @config: Configuration to add the CRTC to |
| * |
| * Returns: |
| * The new CRTC configuration or an error. Call vkms_config_destroy_crtc() to |
| * free the returned CRTC configuration. |
| */ |
| struct vkms_config_crtc *vkms_config_create_crtc(struct vkms_config *config); |
| |
| /** |
| * vkms_config_destroy_crtc() - Remove and free a CRTC configuration |
| * @config: Configuration to remove the CRTC from |
| * @crtc_cfg: CRTC configuration to destroy |
| */ |
| void vkms_config_destroy_crtc(struct vkms_config *config, |
| struct vkms_config_crtc *crtc_cfg); |
| |
| /** |
| * vkms_config_crtc_get_writeback() - If a writeback connector will be created |
| * @crtc_cfg: CRTC with or without a writeback connector |
| */ |
| static inline bool |
| vkms_config_crtc_get_writeback(struct vkms_config_crtc *crtc_cfg) |
| { |
| return crtc_cfg->writeback; |
| } |
| |
| /** |
| * vkms_config_crtc_set_writeback() - If a writeback connector will be created |
| * @crtc_cfg: Target CRTC |
| * @writeback: Enable or disable the writeback connector |
| */ |
| static inline void |
| vkms_config_crtc_set_writeback(struct vkms_config_crtc *crtc_cfg, |
| bool writeback) |
| { |
| crtc_cfg->writeback = writeback; |
| } |
| |
| /** |
| * vkms_config_crtc_primary_plane() - Return the primary plane for a CRTC |
| * @config: Configuration containing the CRTC |
| * @crtc_config: Target CRTC |
| * |
| * Note that, if multiple primary planes are found, the first one is returned. |
| * In this case, the configuration will be invalid. See vkms_config_is_valid(). |
| * |
| * Returns: |
| * The primary plane or NULL if none is assigned yet. |
| */ |
| struct vkms_config_plane *vkms_config_crtc_primary_plane(const struct vkms_config *config, |
| struct vkms_config_crtc *crtc_cfg); |
| |
| /** |
| * vkms_config_crtc_cursor_plane() - Return the cursor plane for a CRTC |
| * @config: Configuration containing the CRTC |
| * @crtc_config: Target CRTC |
| * |
| * Note that, if multiple cursor planes are found, the first one is returned. |
| * In this case, the configuration will be invalid. See vkms_config_is_valid(). |
| * |
| * Returns: |
| * The cursor plane or NULL if none is assigned yet. |
| */ |
| struct vkms_config_plane *vkms_config_crtc_cursor_plane(const struct vkms_config *config, |
| struct vkms_config_crtc *crtc_cfg); |
| |
| /** |
| * vkms_config_create_encoder() - Add a new encoder configuration |
| * @config: Configuration to add the encoder to |
| * |
| * Returns: |
| * The new encoder configuration or an error. Call vkms_config_destroy_encoder() |
| * to free the returned encoder configuration. |
| */ |
| struct vkms_config_encoder *vkms_config_create_encoder(struct vkms_config *config); |
| |
| /** |
| * vkms_config_destroy_encoder() - Remove and free a encoder configuration |
| * @config: Configuration to remove the encoder from |
| * @encoder_cfg: Encoder configuration to destroy |
| */ |
| void vkms_config_destroy_encoder(struct vkms_config *config, |
| struct vkms_config_encoder *encoder_cfg); |
| |
| /** |
| * vkms_config_encoder_attach_crtc - Attach a encoder to a CRTC |
| * @encoder_cfg: Encoder to attach |
| * @crtc_cfg: CRTC to attach @encoder_cfg to |
| */ |
| int __must_check vkms_config_encoder_attach_crtc(struct vkms_config_encoder *encoder_cfg, |
| struct vkms_config_crtc *crtc_cfg); |
| |
| /** |
| * vkms_config_encoder_detach_crtc - Detach a encoder from a CRTC |
| * @encoder_cfg: Encoder to detach |
| * @crtc_cfg: CRTC to detach @encoder_cfg from |
| */ |
| void vkms_config_encoder_detach_crtc(struct vkms_config_encoder *encoder_cfg, |
| struct vkms_config_crtc *crtc_cfg); |
| |
| /** |
| * vkms_config_create_connector() - Add a new connector configuration |
| * @config: Configuration to add the connector to |
| * |
| * Returns: |
| * The new connector configuration or an error. Call |
| * vkms_config_destroy_connector() to free the returned connector configuration. |
| */ |
| struct vkms_config_connector *vkms_config_create_connector(struct vkms_config *config); |
| |
| /** |
| * vkms_config_destroy_connector() - Remove and free a connector configuration |
| * @connector_cfg: Connector configuration to destroy |
| */ |
| void vkms_config_destroy_connector(struct vkms_config_connector *connector_cfg); |
| |
| /** |
| * vkms_config_connector_attach_encoder - Attach a connector to an encoder |
| * @connector_cfg: Connector to attach |
| * @encoder_cfg: Encoder to attach @connector_cfg to |
| */ |
| int __must_check vkms_config_connector_attach_encoder(struct vkms_config_connector *connector_cfg, |
| struct vkms_config_encoder *encoder_cfg); |
| |
| /** |
| * vkms_config_connector_detach_encoder - Detach a connector from an encoder |
| * @connector_cfg: Connector to detach |
| * @encoder_cfg: Encoder to detach @connector_cfg from |
| */ |
| void vkms_config_connector_detach_encoder(struct vkms_config_connector *connector_cfg, |
| struct vkms_config_encoder *encoder_cfg); |
| |
| #endif /* _VKMS_CONFIG_H_ */ |