It is to be used to enable HDR by allowing userpace to create and pass
3D LUTs to kernel and hardware.

new drm_colorop_type: DRM_COLOROP_3D_LUT.

Signed-off-by: Alex Hung <alex.h...@amd.com>
---
v8:
 - Fix typo in subject (Simon Ser)
 - Update documentation for DRM_COLOROP_3D_LUT (Simon Ser)
 - Delete empty lines (Simon Ser)

v7:
 - Simplify 3D LUT by removing lut_3d_modes and related functions (Simon Ser)

 drivers/gpu/drm/drm_atomic.c      |  6 +++
 drivers/gpu/drm/drm_atomic_uapi.c |  6 +++
 drivers/gpu/drm/drm_colorop.c     | 72 +++++++++++++++++++++++++++++++
 include/drm/drm_colorop.h         | 21 +++++++++
 include/uapi/drm/drm_mode.h       | 33 ++++++++++++++
 5 files changed, 138 insertions(+)

diff --git a/drivers/gpu/drm/drm_atomic.c b/drivers/gpu/drm/drm_atomic.c
index 0efb0ead204a..ef47a06344f3 100644
--- a/drivers/gpu/drm/drm_atomic.c
+++ b/drivers/gpu/drm/drm_atomic.c
@@ -806,6 +806,12 @@ static void drm_atomic_colorop_print_state(struct 
drm_printer *p,
        case DRM_COLOROP_MULTIPLIER:
                drm_printf(p, "\tmultiplier=%llu\n", state->multiplier);
                break;
+       case DRM_COLOROP_3D_LUT:
+               drm_printf(p, "\tsize=%d\n", colorop->lut_size);
+               drm_printf(p, "\tinterpolation=%s\n",
+                          
drm_get_colorop_lut3d_interpolation_name(colorop->lut3d_interpolation));
+               drm_printf(p, "\tdata blob id=%d\n", state->data ? 
state->data->base.id : 0);
+               break;
        default:
                break;
        }
diff --git a/drivers/gpu/drm/drm_atomic_uapi.c 
b/drivers/gpu/drm/drm_atomic_uapi.c
index 947c18e8bf9b..d5d464b4d0f6 100644
--- a/drivers/gpu/drm/drm_atomic_uapi.c
+++ b/drivers/gpu/drm/drm_atomic_uapi.c
@@ -719,6 +719,10 @@ static int drm_atomic_color_set_data_property(struct 
drm_colorop *colorop,
        case DRM_COLOROP_CTM_3X4:
                size = sizeof(struct drm_color_ctm_3x4);
                break;
+       case DRM_COLOROP_3D_LUT:
+               size = colorop->lut_size * colorop->lut_size * 
colorop->lut_size *
+                      sizeof(struct drm_color_lut);
+               break;
        default:
                /* should never get here */
                return -EINVAL;
@@ -771,6 +775,8 @@ drm_atomic_colorop_get_property(struct drm_colorop *colorop,
                *val = state->multiplier;
        } else if (property == colorop->lut_size_property) {
                *val = colorop->lut_size;
+       } else if (property == colorop->lut3d_interpolation_property) {
+               *val = colorop->lut3d_interpolation;
        } else if (property == colorop->data_property) {
                *val = (state->data) ? state->data->base.id : 0;
        } else {
diff --git a/drivers/gpu/drm/drm_colorop.c b/drivers/gpu/drm/drm_colorop.c
index e03706e7179b..224c6be237d2 100644
--- a/drivers/gpu/drm/drm_colorop.c
+++ b/drivers/gpu/drm/drm_colorop.c
@@ -67,6 +67,7 @@ static const struct drm_prop_enum_list 
drm_colorop_type_enum_list[] = {
        { DRM_COLOROP_1D_LUT, "1D LUT" },
        { DRM_COLOROP_CTM_3X4, "3x4 Matrix"},
        { DRM_COLOROP_MULTIPLIER, "Multiplier"},
+       { DRM_COLOROP_3D_LUT, "3D LUT"},
 };
 
 static const char * const colorop_curve_1d_type_names[] = {
@@ -82,6 +83,11 @@ static const struct drm_prop_enum_list 
drm_colorop_lut1d_interpolation_list[] =
        { DRM_COLOROP_LUT1D_INTERPOLATION_LINEAR, "Linear" },
 };
 
+
+static const struct drm_prop_enum_list drm_colorop_lut3d_interpolation_list[] 
= {
+       { DRM_COLOROP_LUT3D_INTERPOLATION_TETRAHEDRAL, "Tetrahedral" },
+};
+
 /* Init Helpers */
 
 static int drm_colorop_init(struct drm_device *dev, struct drm_colorop 
*colorop,
@@ -349,6 +355,51 @@ int drm_colorop_mult_init(struct drm_device *dev, struct 
drm_colorop *colorop,
 }
 EXPORT_SYMBOL(drm_colorop_mult_init);
 
+int drm_colorop_3dlut_init(struct drm_device *dev, struct drm_colorop *colorop,
+                          struct drm_plane *plane,
+                          uint32_t lut_size,
+                          enum drm_colorop_lut3d_interpolation_type 
interpolation,
+                          bool allow_bypass)
+{
+       struct drm_property *prop;
+       int ret;
+
+       ret = drm_colorop_init(dev, colorop, plane, DRM_COLOROP_3D_LUT, 
allow_bypass);
+       if (ret)
+               return ret;
+
+       /* LUT size */
+       prop = drm_property_create_range(dev, DRM_MODE_PROP_IMMUTABLE  | 
DRM_MODE_PROP_ATOMIC,
+                                        "SIZE", 0, UINT_MAX);
+       if (!prop)
+               return -ENOMEM;
+
+       colorop->lut_size_property = prop;
+       drm_object_attach_property(&colorop->base, colorop->lut_size_property, 
lut_size);
+       colorop->lut_size = lut_size;
+
+       /* interpolation */
+       prop = drm_property_create_enum(dev, DRM_MODE_PROP_IMMUTABLE, 
"LUT3D_INTERPOLATION",
+                                       drm_colorop_lut3d_interpolation_list,
+                                       
ARRAY_SIZE(drm_colorop_lut3d_interpolation_list));
+       if (!prop)
+               return -ENOMEM;
+
+       colorop->lut3d_interpolation_property = prop;
+       drm_object_attach_property(&colorop->base, prop, interpolation);
+       colorop->lut3d_interpolation = interpolation;
+
+       /* data */
+       ret = drm_colorop_create_data_prop(dev, colorop);
+       if (ret)
+               return ret;
+
+       drm_colorop_reset(colorop);
+
+       return 0;
+}
+EXPORT_SYMBOL(drm_colorop_3dlut_init);
+
 static void __drm_atomic_helper_colorop_duplicate_state(struct drm_colorop 
*colorop,
                                                        struct 
drm_colorop_state *state)
 {
@@ -441,7 +492,13 @@ static const char * const colorop_type_name[] = {
        [DRM_COLOROP_1D_LUT] = "1D LUT",
        [DRM_COLOROP_CTM_3X4] = "3x4 Matrix",
        [DRM_COLOROP_MULTIPLIER] = "Multiplier",
+       [DRM_COLOROP_3D_LUT] = "3D LUT",
 };
+
+static const char * const colorop_lu3d_interpolation_name[] = {
+       [DRM_COLOROP_LUT3D_INTERPOLATION_TETRAHEDRAL] = "Tetrahedral",
+};
+
 static const char * const colorop_lut1d_interpolation_name[] = {
        [DRM_COLOROP_LUT1D_INTERPOLATION_LINEAR] = "Linear",
 };
@@ -477,6 +534,21 @@ const char *drm_get_colorop_lut1d_interpolation_name(enum 
drm_colorop_lut1d_inte
        return colorop_lut1d_interpolation_name[type];
 }
 
+/**
+ * drm_get_colorop_lut3d_interpolation_name - return a string for 
interpolation type
+ * @type: interpolation type to compute name of
+ *
+ * In contrast to the other drm_get_*_name functions this one here returns a
+ * const pointer and hence is threadsafe.
+ */
+const char *drm_get_colorop_lut3d_interpolation_name(enum 
drm_colorop_lut3d_interpolation_type type)
+{
+       if (WARN_ON(type >= ARRAY_SIZE(colorop_lu3d_interpolation_name)))
+               return "unknown";
+
+       return colorop_lu3d_interpolation_name[type];
+}
+
 /**
  * drm_colorop_set_next_property - sets the next pointer
  * @colorop: drm colorop
diff --git a/include/drm/drm_colorop.h b/include/drm/drm_colorop.h
index c89d5eb44856..e999d5ceb8a5 100644
--- a/include/drm/drm_colorop.h
+++ b/include/drm/drm_colorop.h
@@ -281,6 +281,14 @@ struct drm_colorop {
         */
        enum drm_colorop_lut1d_interpolation_type lut1d_interpolation;
 
+       /**
+        * @lut3d_interpolation:
+        *
+        * Read-only
+        * Interpolation for DRM_COLOROP_3D_LUT
+        */
+       enum drm_colorop_lut3d_interpolation_type lut3d_interpolation;
+
        /**
         * @lut1d_interpolation_property:
         *
@@ -309,6 +317,13 @@ struct drm_colorop {
         */
        struct drm_property *lut_size_property;
 
+       /**
+        * @lut3d_interpolation_property:
+        *
+        * Read-only property for DRM_COLOROP_3D_LUT interpolation
+        */
+       struct drm_property *lut3d_interpolation_property;
+
        /**
         * @data_property:
         *
@@ -362,6 +377,11 @@ int drm_colorop_ctm_3x4_init(struct drm_device *dev, 
struct drm_colorop *colorop
                             struct drm_plane *plane, bool allow_bypass);
 int drm_colorop_mult_init(struct drm_device *dev, struct drm_colorop *colorop,
                              struct drm_plane *plane, bool allow_bypass);
+int drm_colorop_3dlut_init(struct drm_device *dev, struct drm_colorop *colorop,
+                          struct drm_plane *plane,
+                          uint32_t lut_size,
+                          enum drm_colorop_lut3d_interpolation_type 
interpolation,
+                          bool allow_bypass);
 
 struct drm_colorop_state *
 drm_atomic_helper_colorop_duplicate_state(struct drm_colorop *colorop);
@@ -412,6 +432,7 @@ const char *drm_get_colorop_type_name(enum drm_colorop_type 
type);
  */
 const char *drm_get_colorop_curve_1d_type_name(enum drm_colorop_curve_1d_type 
type);
 const char *drm_get_colorop_lut1d_interpolation_name(enum 
drm_colorop_lut1d_interpolation_type type);
+const char *drm_get_colorop_lut3d_interpolation_name(enum 
drm_colorop_lut3d_interpolation_type type);
 
 void drm_colorop_set_next_property(struct drm_colorop *colorop, struct 
drm_colorop *next);
 
diff --git a/include/uapi/drm/drm_mode.h b/include/uapi/drm/drm_mode.h
index d76c8ffe5408..88fafbdeb2a2 100644
--- a/include/uapi/drm/drm_mode.h
+++ b/include/uapi/drm/drm_mode.h
@@ -930,6 +930,39 @@ enum drm_colorop_type {
         * property.
         */
        DRM_COLOROP_MULTIPLIER,
+
+       /**
+        * @DRM_COLOROP_3D_LUT:
+        *
+        * enum string "3D LUT"
+        *
+        * A 3D LUT of &drm_color_lut entries,
+        * packed into a blob via the DATA property. The driver's expected
+        * LUT size is advertised via the SIZE property, i.e., a 3D LUT with
+        * 17x17x17 entries will have SIZE set to 17.
+        *
+        * The DATA blob is a 3D array of struct drm_color_lut with dimension
+        * length of "lut_size".
+        * The LUT elements are traversed like so:
+        *
+        *   for R in range 0..n
+        *     for G in range 0..n
+        *       for B in range 0..n
+        *         color = lut3d[R][G][B]
+        */
+       DRM_COLOROP_3D_LUT,
+};
+
+/**
+ * enum drm_colorop_lut3d_interpolation_type - type of 3DLUT interpolation
+ */
+enum drm_colorop_lut3d_interpolation_type {
+       /**
+        * @DRM_COLOROP_LUT3D_INTERPOLATION_TETRAHEDRAL:
+        *
+        * Tetrahedral 3DLUT interpolation
+        */
+       DRM_COLOROP_LUT3D_INTERPOLATION_TETRAHEDRAL,
 };
 
 /**
-- 
2.43.0

Reply via email to