root/drivers/gpu/drm/arm/display/komeda/komeda_format_caps.c
// SPDX-License-Identifier: GPL-2.0
/*
 * (C) COPYRIGHT 2018 ARM Limited. All rights reserved.
 * Author: James.Qian.Wang <james.qian.wang@arm.com>
 *
 */

#include <linux/slab.h>
#include "komeda_format_caps.h"
#include "malidp_utils.h"

const struct komeda_format_caps *
komeda_get_format_caps(struct komeda_format_caps_table *table,
                       u32 fourcc, u64 modifier)
{
        const struct komeda_format_caps *caps;
        u64 afbc_features = modifier & ~(AFBC_FORMAT_MOD_BLOCK_SIZE_MASK);
        u32 afbc_layout = modifier & AFBC_FORMAT_MOD_BLOCK_SIZE_MASK;
        int id;

        for (id = 0; id < table->n_formats; id++) {
                caps = &table->format_caps[id];

                if (fourcc != caps->fourcc)
                        continue;

                if ((modifier == 0ULL) && (caps->supported_afbc_layouts == 0))
                        return caps;

                if (has_bits(afbc_features, caps->supported_afbc_features) &&
                    has_bit(afbc_layout, caps->supported_afbc_layouts))
                        return caps;
        }

        return NULL;
}

u32 komeda_get_afbc_format_bpp(const struct drm_format_info *info, u64 modifier)
{
        u32 bpp;

        switch (info->format) {
        case DRM_FORMAT_YUV420_8BIT:
                bpp = 12;
                break;
        case DRM_FORMAT_YUV420_10BIT:
                bpp = 15;
                break;
        default:
                bpp = info->cpp[0] * 8;
                break;
        }

        return bpp;
}

/* Two assumptions
 * 1. RGB always has YTR
 * 2. Tiled RGB always has SC
 */
u64 komeda_supported_modifiers[] = {
        /* AFBC_16x16 + features: YUV+RGB both */
        AFBC_16x16(0),
        /* SPARSE */
        AFBC_16x16(_SPARSE),
        /* YTR + (SPARSE) */
        AFBC_16x16(_YTR | _SPARSE),
        AFBC_16x16(_YTR),
        /* SPLIT + SPARSE + YTR RGB only */
        /* split mode is only allowed for sparse mode */
        AFBC_16x16(_SPLIT | _SPARSE | _YTR),
        /* TILED + (SPARSE) */
        /* TILED YUV format only */
        AFBC_16x16(_TILED | _SPARSE),
        AFBC_16x16(_TILED),
        /* TILED + SC + (SPLIT+SPARSE | SPARSE) + (YTR) */
        AFBC_16x16(_TILED | _SC | _SPLIT | _SPARSE | _YTR),
        AFBC_16x16(_TILED | _SC | _SPARSE | _YTR),
        AFBC_16x16(_TILED | _SC | _YTR),
        /* AFBC_32x8 + features: which are RGB formats only */
        /* YTR + (SPARSE) */
        AFBC_32x8(_YTR | _SPARSE),
        AFBC_32x8(_YTR),
        /* SPLIT + SPARSE + (YTR) */
        /* split mode is only allowed for sparse mode */
        AFBC_32x8(_SPLIT | _SPARSE | _YTR),
        /* TILED + SC + (SPLIT+SPARSE | SPARSE) + YTR */
        AFBC_32x8(_TILED | _SC | _SPLIT | _SPARSE | _YTR),
        AFBC_32x8(_TILED | _SC | _SPARSE | _YTR),
        AFBC_32x8(_TILED | _SC | _YTR),
        DRM_FORMAT_MOD_LINEAR,
        DRM_FORMAT_MOD_INVALID
};

bool komeda_format_mod_supported(struct komeda_format_caps_table *table,
                                 u32 layer_type, u32 fourcc, u64 modifier,
                                 u32 rot)
{
        const struct komeda_format_caps *caps;

        caps = komeda_get_format_caps(table, fourcc, modifier);
        if (!caps)
                return false;

        if (!(caps->supported_layer_types & layer_type))
                return false;

        if (table->format_mod_supported)
                return table->format_mod_supported(caps, layer_type, modifier,
                                                   rot);

        return true;
}

u32 *komeda_get_layer_fourcc_list(struct komeda_format_caps_table *table,
                                  u32 layer_type, u32 *n_fmts)
{
        const struct komeda_format_caps *cap;
        u32 *fmts;
        int i, j, n = 0;

        fmts = kcalloc(table->n_formats, sizeof(u32), GFP_KERNEL);
        if (!fmts)
                return NULL;

        for (i = 0; i < table->n_formats; i++) {
                cap = &table->format_caps[i];
                if (!(layer_type & cap->supported_layer_types) ||
                    (cap->fourcc == 0))
                        continue;

                /* one fourcc may has two caps items in table (afbc/none-afbc),
                 * so check the existing list to avoid adding a duplicated one.
                 */
                for (j = n - 1; j >= 0; j--)
                        if (fmts[j] == cap->fourcc)
                                break;

                if (j < 0)
                        fmts[n++] = cap->fourcc;
        }

        if (n_fmts)
                *n_fmts = n;

        return fmts;
}

void komeda_put_fourcc_list(u32 *fourcc_list)
{
        kfree(fourcc_list);
}