// SPDX-License-Identifier: GPL-2.0 /* * (C) COPYRIGHT 2018 ARM Limited. All rights reserved. * Author: James.Qian.Wang * */ #include #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); }