/* * Copyright (C) 2013-2019 Red Hat * Copyright (C) 2018 DisplayLink (UK) Ltd. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of the * License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA * 02111-1307, USA. */ #include "config.h" #include "backends/native/meta-kms-plane-private.h" #include #include #include "backends/meta-monitor-transform.h" #include "backends/native/meta-kms-crtc.h" #include "backends/native/meta-kms-impl-device.h" #include "backends/native/meta-kms-update-private.h" typedef struct _MetaKmsPlanePropTable { MetaKmsProp props[META_KMS_PLANE_N_PROPS]; MetaKmsEnum rotation_bitmask[META_KMS_PLANE_ROTATION_BIT_N_PROPS]; } MetaKmsPlanePropTable; struct _MetaKmsPlane { GObject parent; MetaKmsPlaneType type; gboolean is_fake; uint32_t id; uint32_t possible_crtcs; uint32_t rotation_map[META_MONITOR_N_TRANSFORMS]; uint32_t all_hw_transforms; /* * primary plane's supported formats and maybe modifiers * key: GUINT_TO_POINTER (format) * value: owned GArray* (uint64_t modifier), or NULL */ GHashTable *formats_modifiers; MetaKmsPlanePropTable prop_table; MetaKmsDevice *device; }; G_DEFINE_TYPE (MetaKmsPlane, meta_kms_plane, G_TYPE_OBJECT) MetaKmsDevice * meta_kms_plane_get_device (MetaKmsPlane *plane) { return plane->device; } uint32_t meta_kms_plane_get_id (MetaKmsPlane *plane) { g_return_val_if_fail (!plane->is_fake, 0); return plane->id; } MetaKmsPlaneType meta_kms_plane_get_plane_type (MetaKmsPlane *plane) { return plane->type; } uint32_t meta_kms_plane_get_prop_id (MetaKmsPlane *plane, MetaKmsPlaneProp prop) { return plane->prop_table.props[prop].prop_id; } const char * meta_kms_plane_get_prop_name (MetaKmsPlane *plane, MetaKmsPlaneProp prop) { return plane->prop_table.props[prop].name; } MetaKmsPropType meta_kms_plane_get_prop_internal_type (MetaKmsPlane *plane, MetaKmsPlaneProp prop) { return plane->prop_table.props[prop].internal_type; } void meta_kms_plane_update_set_rotation (MetaKmsPlane *plane, MetaKmsPlaneAssignment *plane_assignment, MetaMonitorTransform transform) { g_return_if_fail (meta_kms_plane_is_transform_handled (plane, transform)); meta_kms_plane_assignment_set_rotation (plane_assignment, plane->rotation_map[transform]); } gboolean meta_kms_plane_is_transform_handled (MetaKmsPlane *plane, MetaMonitorTransform transform) { switch (transform) { case META_MONITOR_TRANSFORM_NORMAL: case META_MONITOR_TRANSFORM_180: case META_MONITOR_TRANSFORM_FLIPPED: case META_MONITOR_TRANSFORM_FLIPPED_180: break; case META_MONITOR_TRANSFORM_90: case META_MONITOR_TRANSFORM_270: case META_MONITOR_TRANSFORM_FLIPPED_90: case META_MONITOR_TRANSFORM_FLIPPED_270: /* * Blacklist these transforms as testing shows that they don't work * anyway, e.g. due to the wrong buffer modifiers. They might as well be * less optimal due to the complexity dealing with rotation at scan-out, * potentially resulting in higher power consumption. */ return FALSE; } return plane->all_hw_transforms & (1 << transform); } GArray * meta_kms_plane_get_modifiers_for_format (MetaKmsPlane *plane, uint32_t format) { return g_hash_table_lookup (plane->formats_modifiers, GUINT_TO_POINTER (format)); } GArray * meta_kms_plane_copy_drm_format_list (MetaKmsPlane *plane) { GArray *formats; GHashTableIter it; gpointer key; unsigned int n_formats_modifiers; n_formats_modifiers = g_hash_table_size (plane->formats_modifiers); formats = g_array_sized_new (FALSE, FALSE, sizeof (uint32_t), n_formats_modifiers); g_hash_table_iter_init (&it, plane->formats_modifiers); while (g_hash_table_iter_next (&it, &key, NULL)) { uint32_t drm_format = GPOINTER_TO_UINT (key); g_array_append_val (formats, drm_format); } return formats; } gboolean meta_kms_plane_is_format_supported (MetaKmsPlane *plane, uint32_t drm_format) { return g_hash_table_lookup_extended (plane->formats_modifiers, GUINT_TO_POINTER (drm_format), NULL, NULL); } gboolean meta_kms_plane_is_usable_with (MetaKmsPlane *plane, MetaKmsCrtc *crtc) { return !!(plane->possible_crtcs & (1 << meta_kms_crtc_get_idx (crtc))); } static void parse_rotations (MetaKmsImplDevice *impl_device, MetaKmsProp *prop, drmModePropertyPtr drm_prop, uint64_t drm_prop_value, gpointer user_data) { MetaKmsPlane *plane = user_data; int i; for (i = 0; i < drm_prop->count_enums; i++) { MetaMonitorTransform transform = -1; if (strcmp (drm_prop->enums[i].name, "rotate-0") == 0) transform = META_MONITOR_TRANSFORM_NORMAL; else if (strcmp (drm_prop->enums[i].name, "rotate-90") == 0) transform = META_MONITOR_TRANSFORM_90; else if (strcmp (drm_prop->enums[i].name, "rotate-180") == 0) transform = META_MONITOR_TRANSFORM_180; else if (strcmp (drm_prop->enums[i].name, "rotate-270") == 0) transform = META_MONITOR_TRANSFORM_270; if (transform != -1) { plane->all_hw_transforms |= 1 << transform; plane->rotation_map[transform] = 1 << drm_prop->enums[i].value; } } } static inline uint32_t * drm_formats_ptr (struct drm_format_modifier_blob *blob) { return (uint32_t *) (((char *) blob) + blob->formats_offset); } static inline struct drm_format_modifier * drm_modifiers_ptr (struct drm_format_modifier_blob *blob) { return (struct drm_format_modifier *) (((char *) blob) + blob->modifiers_offset); } static void free_modifier_array (GArray *array) { if (!array) return; g_array_free (array, TRUE); } static void parse_formats (MetaKmsImplDevice *impl_device, MetaKmsProp *prop, drmModePropertyPtr drm_prop, uint64_t drm_prop_value, gpointer user_data) { MetaKmsPlane *plane = user_data; uint64_t blob_id; int fd; drmModePropertyBlobPtr blob; struct drm_format_modifier_blob *blob_fmt; uint32_t *formats; struct drm_format_modifier *drm_modifiers; unsigned int fmt_i, mod_i; g_return_if_fail (g_hash_table_size (plane->formats_modifiers) == 0); blob_id = drm_prop_value; if (blob_id == 0) return; fd = meta_kms_impl_device_get_fd (impl_device); blob = drmModeGetPropertyBlob (fd, blob_id); if (!blob) return; if (blob->length < sizeof (struct drm_format_modifier_blob)) { drmModeFreePropertyBlob (blob); return; } blob_fmt = blob->data; formats = drm_formats_ptr (blob_fmt); drm_modifiers = drm_modifiers_ptr (blob_fmt); for (fmt_i = 0; fmt_i < blob_fmt->count_formats; fmt_i++) { GArray *modifiers = g_array_new (FALSE, FALSE, sizeof (uint64_t)); for (mod_i = 0; mod_i < blob_fmt->count_modifiers; mod_i++) { struct drm_format_modifier *drm_modifier = &drm_modifiers[mod_i]; /* * The modifier advertisement blob is partitioned into groups of * 64 formats. */ if (fmt_i < drm_modifier->offset || fmt_i > drm_modifier->offset + 63) continue; if (!(drm_modifier->formats & (1 << (fmt_i - drm_modifier->offset)))) continue; g_array_append_val (modifiers, drm_modifier->modifier); } if (modifiers->len == 0) { free_modifier_array (modifiers); modifiers = NULL; } g_hash_table_insert (plane->formats_modifiers, GUINT_TO_POINTER (formats[fmt_i]), modifiers); } drmModeFreePropertyBlob (blob); } static void set_formats_from_array (MetaKmsPlane *plane, const uint32_t *formats, size_t n_formats) { size_t i; for (i = 0; i < n_formats; i++) { g_hash_table_insert (plane->formats_modifiers, GUINT_TO_POINTER (formats[i]), NULL); } } /* * In case the DRM driver does not expose a format list for the * primary plane (does not support universal planes nor * IN_FORMATS property), hardcode something that is probably supported. */ static const uint32_t drm_default_formats[] = { /* The format everything should always support by convention */ DRM_FORMAT_XRGB8888, #if G_BYTE_ORDER == G_LITTLE_ENDIAN /* OpenGL GL_RGBA, GL_UNSIGNED_BYTE format, hopefully supported */ DRM_FORMAT_XBGR8888 #endif }; static void init_legacy_formats (MetaKmsPlane *plane, MetaKmsImplDevice *impl_device, drmModePlane *drm_plane, drmModeObjectProperties *drm_plane_props) { if (g_hash_table_size (plane->formats_modifiers) == 0) { set_formats_from_array (plane, drm_plane->formats, drm_plane->count_formats); } /* final formats fallback to something hardcoded */ if (g_hash_table_size (plane->formats_modifiers) == 0) { set_formats_from_array (plane, drm_default_formats, G_N_ELEMENTS (drm_default_formats)); } } static void init_properties (MetaKmsPlane *plane, MetaKmsImplDevice *impl_device, drmModePlane *drm_plane, drmModeObjectProperties *drm_plane_props) { MetaKmsPlanePropTable *prop_table = &plane->prop_table; *prop_table = (MetaKmsPlanePropTable) { .props = { [META_KMS_PLANE_PROP_TYPE] = { .name = "type", .type = DRM_MODE_PROP_ENUM, }, [META_KMS_PLANE_PROP_ROTATION] = { .name = "rotation", .type = DRM_MODE_PROP_BITMASK, .enum_values = prop_table->rotation_bitmask, .num_enum_values = META_KMS_PLANE_ROTATION_BIT_N_PROPS, .parse = parse_rotations, }, [META_KMS_PLANE_PROP_IN_FORMATS] = { .name = "IN_FORMATS", .type = DRM_MODE_PROP_BLOB, .parse = parse_formats, }, [META_KMS_PLANE_PROP_SRC_X] = { .name = "SRC_X", .type = DRM_MODE_PROP_RANGE, .internal_type = META_KMS_PROP_TYPE_FIXED_16, }, [META_KMS_PLANE_PROP_SRC_Y] = { .name = "SRC_Y", .type = DRM_MODE_PROP_RANGE, .internal_type = META_KMS_PROP_TYPE_FIXED_16, }, [META_KMS_PLANE_PROP_SRC_W] = { .name = "SRC_W", .type = DRM_MODE_PROP_RANGE, .internal_type = META_KMS_PROP_TYPE_FIXED_16, }, [META_KMS_PLANE_PROP_SRC_H] = { .name = "SRC_H", .type = DRM_MODE_PROP_RANGE, .internal_type = META_KMS_PROP_TYPE_FIXED_16, }, [META_KMS_PLANE_PROP_CRTC_X] = { .name = "CRTC_X", .type = DRM_MODE_PROP_SIGNED_RANGE, }, [META_KMS_PLANE_PROP_CRTC_Y] = { .name = "CRTC_Y", .type = DRM_MODE_PROP_SIGNED_RANGE, }, [META_KMS_PLANE_PROP_CRTC_W] = { .name = "CRTC_W", .type = DRM_MODE_PROP_RANGE, }, [META_KMS_PLANE_PROP_CRTC_H] = { .name = "CRTC_H", .type = DRM_MODE_PROP_RANGE, }, [META_KMS_PLANE_PROP_FB_ID] = { .name = "FB_ID", .type = DRM_MODE_PROP_OBJECT, }, [META_KMS_PLANE_PROP_CRTC_ID] = { .name = "CRTC_ID", .type = DRM_MODE_PROP_OBJECT, }, [META_KMS_PLANE_PROP_FB_DAMAGE_CLIPS_ID] = { .name = "FB_DAMAGE_CLIPS", .type = DRM_MODE_PROP_BLOB, }, }, .rotation_bitmask = { [META_KMS_PLANE_ROTATION_BIT_ROTATE_0] = { .name = "rotate-0", .bitmask = META_KMS_PLANE_ROTATION_ROTATE_0, }, [META_KMS_PLANE_ROTATION_BIT_ROTATE_90] = { .name = "rotate-90", .bitmask = META_KMS_PLANE_ROTATION_ROTATE_90, }, [META_KMS_PLANE_ROTATION_BIT_ROTATE_180] = { .name = "rotate-180", .bitmask = META_KMS_PLANE_ROTATION_ROTATE_180, }, [META_KMS_PLANE_ROTATION_BIT_ROTATE_270] = { .name = "rotate-270", .bitmask = META_KMS_PLANE_ROTATION_ROTATE_270, }, [META_KMS_PLANE_ROTATION_BIT_REFLECT_X] = { .name = "reflect-x", .bitmask = META_KMS_PLANE_ROTATION_REFLECT_X, }, [META_KMS_PLANE_ROTATION_BIT_REFLECT_Y] = { .name = "reflect-y", .bitmask = META_KMS_PLANE_ROTATION_REFLECT_Y, }, }, }; meta_kms_impl_device_init_prop_table (impl_device, drm_plane_props->props, drm_plane_props->prop_values, drm_plane_props->count_props, plane->prop_table.props, META_KMS_PLANE_N_PROPS, plane); } MetaKmsPlane * meta_kms_plane_new (MetaKmsPlaneType type, MetaKmsImplDevice *impl_device, drmModePlane *drm_plane, drmModeObjectProperties *drm_plane_props) { MetaKmsPlane *plane; plane = g_object_new (META_TYPE_KMS_PLANE, NULL); plane->type = type; plane->id = drm_plane->plane_id; plane->possible_crtcs = drm_plane->possible_crtcs; plane->device = meta_kms_impl_device_get_device (impl_device); init_properties (plane, impl_device, drm_plane, drm_plane_props); init_legacy_formats (plane, impl_device, drm_plane, drm_plane_props); return plane; } MetaKmsPlane * meta_kms_plane_new_fake (MetaKmsPlaneType type, MetaKmsCrtc *crtc) { MetaKmsPlane *plane; static const uint32_t fake_plane_drm_formats[] = { DRM_FORMAT_XRGB8888, DRM_FORMAT_ARGB8888, #if G_BYTE_ORDER == G_LITTLE_ENDIAN /* OpenGL GL_RGBA, GL_UNSIGNED_BYTE format, hopefully supported */ DRM_FORMAT_XBGR8888, DRM_FORMAT_ABGR8888 #endif }; plane = g_object_new (META_TYPE_KMS_PLANE, NULL); plane->type = type; plane->is_fake = TRUE; plane->possible_crtcs = 1 << meta_kms_crtc_get_idx (crtc); plane->device = meta_kms_crtc_get_device (crtc); set_formats_from_array (plane, fake_plane_drm_formats, G_N_ELEMENTS (fake_plane_drm_formats)); return plane; } static void meta_kms_plane_finalize (GObject *object) { MetaKmsPlane *plane = META_KMS_PLANE (object); g_hash_table_destroy (plane->formats_modifiers); G_OBJECT_CLASS (meta_kms_plane_parent_class)->finalize (object); } static void meta_kms_plane_init (MetaKmsPlane *plane) { plane->formats_modifiers = g_hash_table_new_full (g_direct_hash, g_direct_equal, NULL, (GDestroyNotify) free_modifier_array); } static void meta_kms_plane_class_init (MetaKmsPlaneClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->finalize = meta_kms_plane_finalize; }