| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649 |
- // SPDX-License-Identifier: GPL-2.0+
- #include <linux/slab.h>
- #include <drm/drm_print.h>
- #include <drm/drm_debugfs.h>
- #include <kunit/visibility.h>
- #include "vkms_config.h"
- struct vkms_config *vkms_config_create(const char *dev_name)
- {
- struct vkms_config *config;
- config = kzalloc_obj(*config);
- if (!config)
- return ERR_PTR(-ENOMEM);
- config->dev_name = kstrdup_const(dev_name, GFP_KERNEL);
- if (!config->dev_name) {
- kfree(config);
- return ERR_PTR(-ENOMEM);
- }
- INIT_LIST_HEAD(&config->planes);
- INIT_LIST_HEAD(&config->crtcs);
- INIT_LIST_HEAD(&config->encoders);
- INIT_LIST_HEAD(&config->connectors);
- return config;
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_create);
- struct vkms_config *vkms_config_default_create(bool enable_cursor,
- bool enable_writeback,
- bool enable_overlay,
- bool enable_plane_pipeline)
- {
- struct vkms_config *config;
- struct vkms_config_plane *plane_cfg;
- struct vkms_config_crtc *crtc_cfg;
- struct vkms_config_encoder *encoder_cfg;
- struct vkms_config_connector *connector_cfg;
- int n;
- config = vkms_config_create(DEFAULT_DEVICE_NAME);
- if (IS_ERR(config))
- return config;
- plane_cfg = vkms_config_create_plane(config);
- if (IS_ERR(plane_cfg))
- goto err_alloc;
- vkms_config_plane_set_type(plane_cfg, DRM_PLANE_TYPE_PRIMARY);
- crtc_cfg = vkms_config_create_crtc(config);
- if (IS_ERR(crtc_cfg))
- goto err_alloc;
- vkms_config_crtc_set_writeback(crtc_cfg, enable_writeback);
- if (vkms_config_plane_attach_crtc(plane_cfg, crtc_cfg))
- goto err_alloc;
- vkms_config_plane_set_default_pipeline(plane_cfg, enable_plane_pipeline);
- if (enable_overlay) {
- for (n = 0; n < NUM_OVERLAY_PLANES; n++) {
- plane_cfg = vkms_config_create_plane(config);
- if (IS_ERR(plane_cfg))
- goto err_alloc;
- vkms_config_plane_set_type(plane_cfg,
- DRM_PLANE_TYPE_OVERLAY);
- vkms_config_plane_set_default_pipeline(plane_cfg, enable_plane_pipeline);
- if (vkms_config_plane_attach_crtc(plane_cfg, crtc_cfg))
- goto err_alloc;
- }
- }
- if (enable_cursor) {
- plane_cfg = vkms_config_create_plane(config);
- if (IS_ERR(plane_cfg))
- goto err_alloc;
- vkms_config_plane_set_type(plane_cfg, DRM_PLANE_TYPE_CURSOR);
- vkms_config_plane_set_default_pipeline(plane_cfg, enable_plane_pipeline);
- if (vkms_config_plane_attach_crtc(plane_cfg, crtc_cfg))
- goto err_alloc;
- }
- encoder_cfg = vkms_config_create_encoder(config);
- if (IS_ERR(encoder_cfg))
- goto err_alloc;
- if (vkms_config_encoder_attach_crtc(encoder_cfg, crtc_cfg))
- goto err_alloc;
- connector_cfg = vkms_config_create_connector(config);
- if (IS_ERR(connector_cfg))
- goto err_alloc;
- if (vkms_config_connector_attach_encoder(connector_cfg, encoder_cfg))
- goto err_alloc;
- return config;
- err_alloc:
- vkms_config_destroy(config);
- return ERR_PTR(-ENOMEM);
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_default_create);
- void vkms_config_destroy(struct vkms_config *config)
- {
- struct vkms_config_plane *plane_cfg, *plane_tmp;
- struct vkms_config_crtc *crtc_cfg, *crtc_tmp;
- struct vkms_config_encoder *encoder_cfg, *encoder_tmp;
- struct vkms_config_connector *connector_cfg, *connector_tmp;
- list_for_each_entry_safe(plane_cfg, plane_tmp, &config->planes, link)
- vkms_config_destroy_plane(plane_cfg);
- list_for_each_entry_safe(crtc_cfg, crtc_tmp, &config->crtcs, link)
- vkms_config_destroy_crtc(config, crtc_cfg);
- list_for_each_entry_safe(encoder_cfg, encoder_tmp, &config->encoders, link)
- vkms_config_destroy_encoder(config, encoder_cfg);
- list_for_each_entry_safe(connector_cfg, connector_tmp, &config->connectors, link)
- vkms_config_destroy_connector(connector_cfg);
- kfree_const(config->dev_name);
- kfree(config);
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_destroy);
- static bool valid_plane_number(const struct vkms_config *config)
- {
- struct drm_device *dev = config->dev ? &config->dev->drm : NULL;
- size_t n_planes;
- n_planes = list_count_nodes((struct list_head *)&config->planes);
- if (n_planes <= 0 || n_planes >= 32) {
- drm_info(dev, "The number of planes must be between 1 and 31\n");
- return false;
- }
- return true;
- }
- static bool valid_planes_for_crtc(const struct vkms_config *config,
- struct vkms_config_crtc *crtc_cfg)
- {
- struct drm_device *dev = config->dev ? &config->dev->drm : NULL;
- struct vkms_config_plane *plane_cfg;
- bool has_primary_plane = false;
- bool has_cursor_plane = false;
- vkms_config_for_each_plane(config, plane_cfg) {
- struct vkms_config_crtc *possible_crtc;
- unsigned long idx = 0;
- enum drm_plane_type type;
- type = vkms_config_plane_get_type(plane_cfg);
- vkms_config_plane_for_each_possible_crtc(plane_cfg, idx, possible_crtc) {
- if (possible_crtc != crtc_cfg)
- continue;
- if (type == DRM_PLANE_TYPE_PRIMARY) {
- if (has_primary_plane) {
- drm_info(dev, "Multiple primary planes\n");
- return false;
- }
- has_primary_plane = true;
- } else if (type == DRM_PLANE_TYPE_CURSOR) {
- if (has_cursor_plane) {
- drm_info(dev, "Multiple cursor planes\n");
- return false;
- }
- has_cursor_plane = true;
- }
- }
- }
- if (!has_primary_plane) {
- drm_info(dev, "Primary plane not found\n");
- return false;
- }
- return true;
- }
- static bool valid_plane_possible_crtcs(const struct vkms_config *config)
- {
- struct drm_device *dev = config->dev ? &config->dev->drm : NULL;
- struct vkms_config_plane *plane_cfg;
- vkms_config_for_each_plane(config, plane_cfg) {
- if (xa_empty(&plane_cfg->possible_crtcs)) {
- drm_info(dev, "All planes must have at least one possible CRTC\n");
- return false;
- }
- }
- return true;
- }
- static bool valid_crtc_number(const struct vkms_config *config)
- {
- struct drm_device *dev = config->dev ? &config->dev->drm : NULL;
- size_t n_crtcs;
- n_crtcs = list_count_nodes((struct list_head *)&config->crtcs);
- if (n_crtcs <= 0 || n_crtcs >= 32) {
- drm_info(dev, "The number of CRTCs must be between 1 and 31\n");
- return false;
- }
- return true;
- }
- static bool valid_encoder_number(const struct vkms_config *config)
- {
- struct drm_device *dev = config->dev ? &config->dev->drm : NULL;
- size_t n_encoders;
- n_encoders = list_count_nodes((struct list_head *)&config->encoders);
- if (n_encoders <= 0 || n_encoders >= 32) {
- drm_info(dev, "The number of encoders must be between 1 and 31\n");
- return false;
- }
- return true;
- }
- static bool valid_encoder_possible_crtcs(const struct vkms_config *config)
- {
- struct drm_device *dev = config->dev ? &config->dev->drm : NULL;
- struct vkms_config_crtc *crtc_cfg;
- struct vkms_config_encoder *encoder_cfg;
- vkms_config_for_each_encoder(config, encoder_cfg) {
- if (xa_empty(&encoder_cfg->possible_crtcs)) {
- drm_info(dev, "All encoders must have at least one possible CRTC\n");
- return false;
- }
- }
- vkms_config_for_each_crtc(config, crtc_cfg) {
- bool crtc_has_encoder = false;
- vkms_config_for_each_encoder(config, encoder_cfg) {
- struct vkms_config_crtc *possible_crtc;
- unsigned long idx = 0;
- vkms_config_encoder_for_each_possible_crtc(encoder_cfg,
- idx, possible_crtc) {
- if (possible_crtc == crtc_cfg)
- crtc_has_encoder = true;
- }
- }
- if (!crtc_has_encoder) {
- drm_info(dev, "All CRTCs must have at least one possible encoder\n");
- return false;
- }
- }
- return true;
- }
- static bool valid_connector_number(const struct vkms_config *config)
- {
- struct drm_device *dev = config->dev ? &config->dev->drm : NULL;
- size_t n_connectors;
- n_connectors = list_count_nodes((struct list_head *)&config->connectors);
- if (n_connectors <= 0 || n_connectors >= 32) {
- drm_info(dev, "The number of connectors must be between 1 and 31\n");
- return false;
- }
- return true;
- }
- static bool valid_connector_possible_encoders(const struct vkms_config *config)
- {
- struct drm_device *dev = config->dev ? &config->dev->drm : NULL;
- struct vkms_config_connector *connector_cfg;
- vkms_config_for_each_connector(config, connector_cfg) {
- if (xa_empty(&connector_cfg->possible_encoders)) {
- drm_info(dev,
- "All connectors must have at least one possible encoder\n");
- return false;
- }
- }
- return true;
- }
- bool vkms_config_is_valid(const struct vkms_config *config)
- {
- struct vkms_config_crtc *crtc_cfg;
- if (!valid_plane_number(config))
- return false;
- if (!valid_crtc_number(config))
- return false;
- if (!valid_encoder_number(config))
- return false;
- if (!valid_connector_number(config))
- return false;
- if (!valid_plane_possible_crtcs(config))
- return false;
- vkms_config_for_each_crtc(config, crtc_cfg) {
- if (!valid_planes_for_crtc(config, crtc_cfg))
- return false;
- }
- if (!valid_encoder_possible_crtcs(config))
- return false;
- if (!valid_connector_possible_encoders(config))
- return false;
- return true;
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_is_valid);
- static int vkms_config_show(struct seq_file *m, void *data)
- {
- struct drm_debugfs_entry *entry = m->private;
- struct drm_device *dev = entry->dev;
- struct vkms_device *vkmsdev = drm_device_to_vkms_device(dev);
- const char *dev_name;
- struct vkms_config_plane *plane_cfg;
- struct vkms_config_crtc *crtc_cfg;
- struct vkms_config_encoder *encoder_cfg;
- struct vkms_config_connector *connector_cfg;
- dev_name = vkms_config_get_device_name((struct vkms_config *)vkmsdev->config);
- seq_printf(m, "dev_name=%s\n", dev_name);
- vkms_config_for_each_plane(vkmsdev->config, plane_cfg) {
- seq_puts(m, "plane:\n");
- seq_printf(m, "\ttype=%d\n",
- vkms_config_plane_get_type(plane_cfg));
- }
- vkms_config_for_each_crtc(vkmsdev->config, crtc_cfg) {
- seq_puts(m, "crtc:\n");
- seq_printf(m, "\twriteback=%d\n",
- vkms_config_crtc_get_writeback(crtc_cfg));
- }
- vkms_config_for_each_encoder(vkmsdev->config, encoder_cfg)
- seq_puts(m, "encoder\n");
- vkms_config_for_each_connector(vkmsdev->config, connector_cfg) {
- seq_puts(m, "connector:\n");
- seq_printf(m, "\tstatus=%d\n",
- vkms_config_connector_get_status(connector_cfg));
- }
- return 0;
- }
- static const struct drm_debugfs_info vkms_config_debugfs_list[] = {
- { "vkms_config", vkms_config_show, 0 },
- };
- void vkms_config_register_debugfs(struct vkms_device *vkms_device)
- {
- drm_debugfs_add_files(&vkms_device->drm, vkms_config_debugfs_list,
- ARRAY_SIZE(vkms_config_debugfs_list));
- }
- struct vkms_config_plane *vkms_config_create_plane(struct vkms_config *config)
- {
- struct vkms_config_plane *plane_cfg;
- plane_cfg = kzalloc_obj(*plane_cfg);
- if (!plane_cfg)
- return ERR_PTR(-ENOMEM);
- plane_cfg->config = config;
- plane_cfg->default_pipeline = false;
- vkms_config_plane_set_type(plane_cfg, DRM_PLANE_TYPE_OVERLAY);
- xa_init_flags(&plane_cfg->possible_crtcs, XA_FLAGS_ALLOC);
- list_add_tail(&plane_cfg->link, &config->planes);
- return plane_cfg;
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_create_plane);
- void vkms_config_destroy_plane(struct vkms_config_plane *plane_cfg)
- {
- xa_destroy(&plane_cfg->possible_crtcs);
- list_del(&plane_cfg->link);
- kfree(plane_cfg);
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_destroy_plane);
- int __must_check vkms_config_plane_attach_crtc(struct vkms_config_plane *plane_cfg,
- struct vkms_config_crtc *crtc_cfg)
- {
- struct vkms_config_crtc *possible_crtc;
- unsigned long idx = 0;
- u32 crtc_idx = 0;
- if (plane_cfg->config != crtc_cfg->config)
- return -EINVAL;
- vkms_config_plane_for_each_possible_crtc(plane_cfg, idx, possible_crtc) {
- if (possible_crtc == crtc_cfg)
- return -EEXIST;
- }
- return xa_alloc(&plane_cfg->possible_crtcs, &crtc_idx, crtc_cfg,
- xa_limit_32b, GFP_KERNEL);
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_plane_attach_crtc);
- void vkms_config_plane_detach_crtc(struct vkms_config_plane *plane_cfg,
- struct vkms_config_crtc *crtc_cfg)
- {
- struct vkms_config_crtc *possible_crtc;
- unsigned long idx = 0;
- vkms_config_plane_for_each_possible_crtc(plane_cfg, idx, possible_crtc) {
- if (possible_crtc == crtc_cfg)
- xa_erase(&plane_cfg->possible_crtcs, idx);
- }
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_plane_detach_crtc);
- struct vkms_config_crtc *vkms_config_create_crtc(struct vkms_config *config)
- {
- struct vkms_config_crtc *crtc_cfg;
- crtc_cfg = kzalloc_obj(*crtc_cfg);
- if (!crtc_cfg)
- return ERR_PTR(-ENOMEM);
- crtc_cfg->config = config;
- vkms_config_crtc_set_writeback(crtc_cfg, false);
- list_add_tail(&crtc_cfg->link, &config->crtcs);
- return crtc_cfg;
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_create_crtc);
- void vkms_config_destroy_crtc(struct vkms_config *config,
- struct vkms_config_crtc *crtc_cfg)
- {
- struct vkms_config_plane *plane_cfg;
- struct vkms_config_encoder *encoder_cfg;
- vkms_config_for_each_plane(config, plane_cfg)
- vkms_config_plane_detach_crtc(plane_cfg, crtc_cfg);
- vkms_config_for_each_encoder(config, encoder_cfg)
- vkms_config_encoder_detach_crtc(encoder_cfg, crtc_cfg);
- list_del(&crtc_cfg->link);
- kfree(crtc_cfg);
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_destroy_crtc);
- /**
- * vkms_config_crtc_get_plane() - Return the first attached plane to a CRTC with
- * the specific type
- * @config: Configuration containing the CRTC and the plane
- * @crtc_cfg: Only find planes attached to this CRTC
- * @type: Plane type to search
- *
- * Returns:
- * The first plane found attached to @crtc_cfg with the type @type.
- */
- static struct vkms_config_plane *vkms_config_crtc_get_plane(const struct vkms_config *config,
- struct vkms_config_crtc *crtc_cfg,
- enum drm_plane_type type)
- {
- struct vkms_config_plane *plane_cfg;
- struct vkms_config_crtc *possible_crtc;
- enum drm_plane_type current_type;
- unsigned long idx = 0;
- vkms_config_for_each_plane(config, plane_cfg) {
- current_type = vkms_config_plane_get_type(plane_cfg);
- vkms_config_plane_for_each_possible_crtc(plane_cfg, idx, possible_crtc) {
- if (possible_crtc == crtc_cfg && current_type == type)
- return plane_cfg;
- }
- }
- return NULL;
- }
- struct vkms_config_plane *vkms_config_crtc_primary_plane(const struct vkms_config *config,
- struct vkms_config_crtc *crtc_cfg)
- {
- return vkms_config_crtc_get_plane(config, crtc_cfg, DRM_PLANE_TYPE_PRIMARY);
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_crtc_primary_plane);
- struct vkms_config_plane *vkms_config_crtc_cursor_plane(const struct vkms_config *config,
- struct vkms_config_crtc *crtc_cfg)
- {
- return vkms_config_crtc_get_plane(config, crtc_cfg, DRM_PLANE_TYPE_CURSOR);
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_crtc_cursor_plane);
- struct vkms_config_encoder *vkms_config_create_encoder(struct vkms_config *config)
- {
- struct vkms_config_encoder *encoder_cfg;
- encoder_cfg = kzalloc_obj(*encoder_cfg);
- if (!encoder_cfg)
- return ERR_PTR(-ENOMEM);
- encoder_cfg->config = config;
- xa_init_flags(&encoder_cfg->possible_crtcs, XA_FLAGS_ALLOC);
- list_add_tail(&encoder_cfg->link, &config->encoders);
- return encoder_cfg;
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_create_encoder);
- void vkms_config_destroy_encoder(struct vkms_config *config,
- struct vkms_config_encoder *encoder_cfg)
- {
- struct vkms_config_connector *connector_cfg;
- vkms_config_for_each_connector(config, connector_cfg)
- vkms_config_connector_detach_encoder(connector_cfg, encoder_cfg);
- xa_destroy(&encoder_cfg->possible_crtcs);
- list_del(&encoder_cfg->link);
- kfree(encoder_cfg);
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_destroy_encoder);
- int __must_check vkms_config_encoder_attach_crtc(struct vkms_config_encoder *encoder_cfg,
- struct vkms_config_crtc *crtc_cfg)
- {
- struct vkms_config_crtc *possible_crtc;
- unsigned long idx = 0;
- u32 crtc_idx = 0;
- if (encoder_cfg->config != crtc_cfg->config)
- return -EINVAL;
- vkms_config_encoder_for_each_possible_crtc(encoder_cfg, idx, possible_crtc) {
- if (possible_crtc == crtc_cfg)
- return -EEXIST;
- }
- return xa_alloc(&encoder_cfg->possible_crtcs, &crtc_idx, crtc_cfg,
- xa_limit_32b, GFP_KERNEL);
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_encoder_attach_crtc);
- void vkms_config_encoder_detach_crtc(struct vkms_config_encoder *encoder_cfg,
- struct vkms_config_crtc *crtc_cfg)
- {
- struct vkms_config_crtc *possible_crtc;
- unsigned long idx = 0;
- vkms_config_encoder_for_each_possible_crtc(encoder_cfg, idx, possible_crtc) {
- if (possible_crtc == crtc_cfg)
- xa_erase(&encoder_cfg->possible_crtcs, idx);
- }
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_encoder_detach_crtc);
- struct vkms_config_connector *vkms_config_create_connector(struct vkms_config *config)
- {
- struct vkms_config_connector *connector_cfg;
- connector_cfg = kzalloc_obj(*connector_cfg);
- if (!connector_cfg)
- return ERR_PTR(-ENOMEM);
- connector_cfg->config = config;
- connector_cfg->status = connector_status_connected;
- xa_init_flags(&connector_cfg->possible_encoders, XA_FLAGS_ALLOC);
- list_add_tail(&connector_cfg->link, &config->connectors);
- return connector_cfg;
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_create_connector);
- void vkms_config_destroy_connector(struct vkms_config_connector *connector_cfg)
- {
- xa_destroy(&connector_cfg->possible_encoders);
- list_del(&connector_cfg->link);
- kfree(connector_cfg);
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_destroy_connector);
- int __must_check vkms_config_connector_attach_encoder(struct vkms_config_connector *connector_cfg,
- struct vkms_config_encoder *encoder_cfg)
- {
- struct vkms_config_encoder *possible_encoder;
- unsigned long idx = 0;
- u32 encoder_idx = 0;
- if (connector_cfg->config != encoder_cfg->config)
- return -EINVAL;
- vkms_config_connector_for_each_possible_encoder(connector_cfg, idx,
- possible_encoder) {
- if (possible_encoder == encoder_cfg)
- return -EEXIST;
- }
- return xa_alloc(&connector_cfg->possible_encoders, &encoder_idx,
- encoder_cfg, xa_limit_32b, GFP_KERNEL);
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_connector_attach_encoder);
- void vkms_config_connector_detach_encoder(struct vkms_config_connector *connector_cfg,
- struct vkms_config_encoder *encoder_cfg)
- {
- struct vkms_config_encoder *possible_encoder;
- unsigned long idx = 0;
- vkms_config_connector_for_each_possible_encoder(connector_cfg, idx,
- possible_encoder) {
- if (possible_encoder == encoder_cfg)
- xa_erase(&connector_cfg->possible_encoders, idx);
- }
- }
- EXPORT_SYMBOL_IF_KUNIT(vkms_config_connector_detach_encoder);
|