lineage_kernel_xcoverpro/drivers/gpu/exynos/g2d/g2d_command.c

1070 lines
30 KiB
C
Executable File

/*
* linux/drivers/gpu/exynos/g2d/g2d_command.c
*
* Copyright (C) 2017 Samsung Electronics Co., Ltd.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* version 2 as published by the Free Software Foundation.
*
* 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.
*/
#include <linux/kernel.h>
#include <linux/uaccess.h>
#include <linux/device.h>
#include <video/exynos_hdr_tunables.h>
#include "g2d.h"
#include "g2d_task.h"
#include "g2d_uapi.h"
#include "g2d_command.h"
#include "g2d_regs.h"
#include "g2d_format.h"
#define layer_crop_width(layer) \
(((layer)->commands[G2DSFR_IMG_RIGHT].value) - \
((layer)->commands[G2DSFR_IMG_LEFT].value))
#define layer_crop_height(layer) \
(((layer)->commands[G2DSFR_IMG_BOTTOM].value) - \
((layer)->commands[G2DSFR_IMG_TOP].value))
#define layer_width(layer) ((layer)->commands[G2DSFR_IMG_WIDTH].value)
#define layer_height(layer) ((layer)->commands[G2DSFR_IMG_HEIGHT].value)
#define layer_pixelcount(layer) (layer_width(layer) * layer_height(layer))
/*
* Number of registers of coefficients of conversion functions
* - 4 sets of color space conversion for sources
* - 1 set of color space conversion for destination
* - 2 sets of HDR coefficients
* - 2 matrices of EOTF
* - 2 matrices of Gamut mapping
* - 2 matrices Tone mappings
* - 2 set HDR Coefficient
* - 2 matrices of SET0/1 HDR Coefficient
*/
#define MAX_EXTRA_REGS (9 * 5 + (65 + 9 + 33) * 2 + 160 * 2)
enum {
TASK_REG_SOFT_RESET,
TASK_REG_SECURE_MODE,
TASK_REG_LAYER_UPDATE,
TASK_REG_COUNT
};
/*
* G2D_SECURE_LAYER_REG and G2D_LAYER_UPDATE_REG are updated in
* g2d_prepare_source() and g2d_prepare_target().
*/
static struct g2d_reg g2d_setup_commands[TASK_REG_COUNT] = {
{G2D_SOFT_RESET_REG, 0x00000004}, /* CoreSFRClear */
{G2D_SECURE_MODE_REG, 0x00000000},
{G2D_LAYER_UPDATE_REG, 0x00000000},
};
void g2d_init_commands(struct g2d_task *task)
{
memcpy(page_address(task->cmd_page),
&g2d_setup_commands, sizeof(g2d_setup_commands));
task->sec.cmd_count = ARRAY_SIZE(g2d_setup_commands);
}
static void g2d_set_taskctl_commands(struct g2d_task *task)
{
struct g2d_reg *regs = (struct g2d_reg *)page_address(task->cmd_page);
struct g2d_layer *layer;
u32 rot = 0;
u32 n_rot = 0;
u32 size = 0; /* Size doesn't cause overflow */
u32 width = layer_width(&task->target);
u32 height = layer_height(&task->target);
int i;
for (i = 0; i < task->num_source; i++) {
layer = &task->source[i];
size = layer_crop_width(layer) * layer_crop_height(layer);
if (layer->commands[G2DSFR_SRC_ROTATE].value & 1)
rot += size;
else
n_rot += size;
}
if (rot > n_rot) {
u32 mode = task->target.commands[G2DSFR_IMG_COLORMODE].value;
regs[task->sec.cmd_count].offset = G2D_TILE_DIRECTION_ORDER_REG;
regs[task->sec.cmd_count].value = G2D_TILE_DIRECTION_VERTICAL;
if (!IS_HWFC(task->flags) &&
(IS_YUV420(mode) || IS_YUV422_2P(mode)))
regs[task->sec.cmd_count].value |=
G2D_TILE_DIRECTION_ZORDER;
task->sec.cmd_count++;
}
/*
* Divide the entire destination in half by verital or horizontal,
* and let the H/W work in parallel.
* split index is half the width or height divided by 16
*/
regs[task->sec.cmd_count].offset = G2D_DST_SPLIT_TILE_IDX_REG;
regs[task->sec.cmd_count].value = (height > width) ?
((height / 2) >> 4) :
((width / 2) >> 4) | G2D_DST_SPLIT_TILE_IDX_VFLAG;
task->sec.cmd_count++;
}
static void g2d_set_hwfc_commands(struct g2d_task *task)
{
struct g2d_reg *regs = (struct g2d_reg *)page_address(task->cmd_page);
regs[task->sec.cmd_count].offset = G2D_HWFC_CAPTURE_IDX_REG;
regs[task->sec.cmd_count].value = IS_HWFC(task->flags) ?
G2D_HWFC_CAPTURE_HWFC_JOB : 0;
regs[task->sec.cmd_count].value |= task->sec.job_id;
task->sec.cmd_count++;
}
static void g2d_set_start_commands(struct g2d_task *task)
{
bool self_prot = task->g2d_dev->caps & G2D_DEVICE_CAPS_SELF_PROTECTION;
struct g2d_reg *regs = page_address(task->cmd_page);
if (!self_prot && IS_ENABLED(CONFIG_EXYNOS_CONTENT_PATH_PROTECTION))
return;
/*
* Number of commands should be multiple of 8.
* If it is not, then pad dummy commands with no side effect.
*/
while ((task->sec.cmd_count & 7) != 0) {
regs[task->sec.cmd_count].offset = G2D_LAYER_UPDATE_REG;
regs[task->sec.cmd_count].value =
regs[TASK_REG_LAYER_UPDATE].value;
task->sec.cmd_count++;
}
}
void g2d_complete_commands(struct g2d_task *task)
{
/* 832 is the total number of the G2D registers */
/* Tuned tone mapping LUT (66 entries) might be specified */
BUG_ON(task->sec.cmd_count > 896);
g2d_set_taskctl_commands(task);
g2d_set_hwfc_commands(task);
g2d_set_start_commands(task);
}
static const struct g2d_fmt g2d_formats_common[] = {
{
.name = "ARGB8888",
.fmtvalue = G2D_FMT_ARGB8888, /* [31:0] ARGB */
.bpp = { 32 },
.num_planes = 1,
}, {
.name = "ABGR8888",
.fmtvalue = G2D_FMT_ABGR8888, /* [31:0] ABGR */
.bpp = { 32 },
.num_planes = 1,
}, {
.name = "XBGR8888",
.fmtvalue = G2D_FMT_XBGR8888, /* [31:0] XBGR */
.bpp = { 32 },
.num_planes = 1,
}, {
.name = "XRGB8888",
.fmtvalue = G2D_FMT_XRGB8888, /* [31:0] XBGR */
.bpp = { 32 },
.num_planes = 1,
}, {
.name = "RGB888",
.fmtvalue = G2D_FMT_RGB888, /* [23:0] RGB */
.bpp = { 24 },
.num_planes = 1,
}, {
.name = "ARGB4444",
.fmtvalue = G2D_FMT_ARGB4444, /* [15:0] ARGB */
.bpp = { 16 },
.num_planes = 1,
}, {
.name = "ARGB1555",
.fmtvalue = G2D_FMT_ARGB1555, /* [15:0] ARGB */
.bpp = { 16 },
.num_planes = 1,
}, {
.name = "RGB565",
.fmtvalue = G2D_FMT_RGB565, /* [15:0] RGB */
.bpp = { 16 },
.num_planes = 1,
}, {
.name = "BGR565",
.fmtvalue = G2D_FMT_BGR565, /* [15:0] BGR */
.bpp = { 16 },
.num_planes = 1,
}, {
.name = "NV12",
.fmtvalue = G2D_FMT_NV12,
.bpp = { 8, 4 },
.num_planes = 2,
}, {
.name = "NV21",
.fmtvalue = G2D_FMT_NV21,
.bpp = { 8, 4 },
.num_planes = 2,
}, {
.name = "YUYV",
.fmtvalue = G2D_FMT_YUYV,
.bpp = { 16 },
.num_planes = 1,
}, {
.name = "YVYU",
.fmtvalue = G2D_FMT_YVYU,
.bpp = { 16 },
.num_planes = 1,
}, {
.name = "UYVY",
.fmtvalue = G2D_FMT_UYVY,
.bpp = { 16 },
.num_planes = 1,
}, {
.name = "VYUY",
.fmtvalue = G2D_FMT_VYUY,
.bpp = { 16 },
.num_planes = 1,
}, {
.name = "NV16",
.fmtvalue = G2D_FMT_NV16,
.bpp = { 8, 8 },
.num_planes = 2,
}, {
.name = "NV61",
.fmtvalue = G2D_FMT_NV61,
.bpp = { 8, 8 },
.num_planes = 2,
}, {
.name = "ABGR2101010",
.fmtvalue = G2D_FMT_ABGR2101010, /* [31:0] ABGR */
.bpp = { 32 },
.num_planes = 1,
}, {
.name = "XBGR2101010",
.fmtvalue = G2D_FMT_XBGR2101010,
.bpp = { 32 },
.num_planes = 1,
}
};
static const struct g2d_fmt g2d_formats_9810[] = {
{
.name = "NV12_8+2",
.fmtvalue = G2D_FMT_NV12_82_9810,
.bpp = { 8, 2, 4, 1 },
.num_planes = 4,
}, {
.name = "NV21_8+2",
.fmtvalue = G2D_FMT_NV21_82_9810,
.bpp = { 8, 2, 4, 1 },
.num_planes = 4,
}, {
.name = "NV12_P010",
.fmtvalue = G2D_FMT_NV12_P010_9810,
.bpp = { 16, 8},
.num_planes = 2,
}, {
.name = "NV21_P010",
.fmtvalue = G2D_FMT_NV21_P010_9810,
.bpp = { 16, 8},
.num_planes = 2,
},
};
static const struct g2d_fmt g2d_formats_9820[] = {
{
.name = "NV12_8+2",
.fmtvalue = G2D_FMT_NV12_82_9820,
.bpp = { 8, 2, 4, 1 },
.num_planes = 4,
}, {
.name = "NV21_8+2",
.fmtvalue = G2D_FMT_NV21_82_9820,
.bpp = { 8, 2, 4, 1 },
.num_planes = 4,
}, {
.name = "NV12_P010",
.fmtvalue = G2D_FMT_NV12_P010_9820,
.bpp = { 16, 8},
.num_planes = 2,
}, {
.name = "NV21_P010",
.fmtvalue = G2D_FMT_NV21_P010_9820,
.bpp = { 16, 8},
.num_planes = 2,
}, {
.name = "NV16_P210",
.fmtvalue = G2D_FMT_NV16_P210_9820,
.bpp = { 16, 16},
.num_planes = 2,
}, {
.name = "NV61_P210",
.fmtvalue = G2D_FMT_NV61_P210_9820,
.bpp = { 16, 16},
.num_planes = 2,
},
};
const struct g2d_fmt *g2d_find_format(u32 fmtval, unsigned long devcaps)
{
size_t i;
for (i = 0; i < ARRAY_SIZE(g2d_formats_common); i++)
if (g2d_formats_common[i].fmtvalue == G2D_IMGFMT(fmtval))
return &g2d_formats_common[i];
if (!(devcaps & G2D_DEVICE_CAPS_YUV_BITDEPTH)) {
for (i = 0; i < ARRAY_SIZE(g2d_formats_9810); i++)
if (g2d_formats_9810[i].fmtvalue == G2D_IMGFMT(fmtval))
return &g2d_formats_9810[i];
} else {
for (i = 0; i < ARRAY_SIZE(g2d_formats_9820); i++)
if (g2d_formats_9820[i].fmtvalue == G2D_IMGFMT(fmtval))
return &g2d_formats_9820[i];
}
return NULL;
}
/*
* Buffer stride alignment and padding restriction of MFC
* YCbCr420 semi-planar 8+2 layout:
* Y8 -> Y2 -> C8 -> C2
* 8 bit segments:
* - width stride: 16 bytes
* - height stride: 16 pixels
* - padding 256 bytes
* 2 bit segments:
* - width stride: 16 bytes
* - height stride: 16 pixels
* - padding: 64 bytes
*/
#define MFC_PAD_SIZE 256
#define MFC_2B_PAD_SIZE (MFC_PAD_SIZE / 4)
#define MFC_ALIGN(v) ALIGN(v, 16)
#define NV12_MFC_Y_PAYLOAD(w, h) (MFC_ALIGN(w) * MFC_ALIGN(h))
#define NV12_MFC_Y_PAYLOAD_PAD(w, h) (NV12_MFC_Y_PAYLOAD(w, h) + MFC_PAD_SIZE)
#define NV12_MFC_C_PAYLOAD(w, h) (MFC_ALIGN(w) * (h) / 2)
#define NV12_MFC_C_PAYLOAD_ALIGNED(w, h) (NV12_MFC_Y_PAYLOAD(w, h) / 2)
#define NV12_MFC_C_PAYLOAD_PAD(w, h) (NV12_MFC_C_PAYLOAD_ALIGNED(w, h) + \
MFC_PAD_SIZE)
#define NV12_MFC_PAYLOAD(w, h) (NV12_MFC_Y_PAYLOAD_PAD(w, h) + \
NV12_MFC_C_PAYLOAD(w, h))
#define NV12_82_MFC_Y_PAYLOAD(w, h) (NV12_MFC_Y_PAYLOAD_PAD(w, h) + \
MFC_ALIGN((w) / 4) * (h))
#define NV12_82_MFC_C_PAYLOAD(w, h) (NV12_MFC_C_PAYLOAD_PAD(w, h) + \
MFC_ALIGN((w) / 4) * (h) / 2)
#define NV12_82_MFC_PAYLOAD(w, h) (NV12_MFC_Y_PAYLOAD_PAD(w, h) + \
MFC_ALIGN((w) / 4) * MFC_ALIGN(h) + \
MFC_2B_PAD_SIZE + \
NV12_82_MFC_C_PAYLOAD(w, h))
#define NV12_82_MFC_2Y_PAYLOAD(w, h) (MFC_ALIGN((w) / 4) * MFC_ALIGN(h))
#define NV12_MFC_CBASE(base, w, h) (base + NV12_MFC_Y_PAYLOAD_PAD(w, h))
#define YUV82_BASE_ALIGNED(addr, idx) IS_ALIGNED((addr), 32 >> (idx & 1))
#define YUV82_BASE_ALIGN(addr, idx) ALIGN((addr), 32 >> (idx & 1))
static unsigned char src_base_reg_offset[4] = {0x1C, 0x80, 0x64, 0x68};
static unsigned char src_base_reg_offset_yuv82[4] = {0x1C, 0x64, 0x80, 0x68};
static unsigned char dst_base_reg_offset[4] = {0x00, 0x50, 0x30, 0x34};
#define BASE_REG_OFFSET(base, offsets, buf_idx) ((base) + (offsets)[(buf_idx)])
#define AFBC_HEADER_SIZE(cmd) \
((ALIGN((cmd)[G2DSFR_IMG_WIDTH].value, 16) / 16) * \
(ALIGN((cmd)[G2DSFR_IMG_HEIGHT].value, 16) / 16) * 16)
size_t g2d_get_payload_index(struct g2d_reg cmd[], const struct g2d_fmt *fmt,
unsigned int idx, unsigned int buffer_count,
unsigned long caps)
{
bool yuv82 = IS_YUV_82(fmt->fmtvalue,
caps & G2D_DEVICE_CAPS_YUV_BITDEPTH);
BUG_ON(!IS_YUV(cmd[G2DSFR_IMG_COLORMODE].value));
if (yuv82 && (buffer_count == 2)) {
/* YCbCr420 8+2 semi-planar in two buffers */
/* regard G2D_LAYERFLAG_MFC_STRIDE is set */
u32 width = cmd[G2DSFR_IMG_WIDTH].value;
u32 height = cmd[G2DSFR_IMG_BOTTOM].value;
return (idx == 0) ? NV12_82_MFC_Y_PAYLOAD(width, height)
: NV12_82_MFC_C_PAYLOAD(width, height);
}
return ((cmd[G2DSFR_IMG_WIDTH].value * fmt->bpp[idx]) / 8) *
cmd[G2DSFR_IMG_BOTTOM].value;
}
size_t g2d_get_payload(struct g2d_reg cmd[], const struct g2d_fmt *fmt,
u32 flags, unsigned long cap)
{
size_t payload = 0;
u32 mode = cmd[G2DSFR_IMG_COLORMODE].value;
u32 width = cmd[G2DSFR_IMG_WIDTH].value;
u32 height = cmd[G2DSFR_IMG_BOTTOM].value;
size_t pixcount = width * height;
bool yuv82 = IS_YUV_82(mode, cap & G2D_DEVICE_CAPS_YUV_BITDEPTH);
if (yuv82) {
if (!(flags & G2D_LAYERFLAG_MFC_STRIDE)) {
/*
* constraints of base addresses of NV12/21 8+2
* 32 byte aligned: 8bit of Y and CbCr
* 16 byte aligned: 2bit of Y and CbCr
*/
payload += ALIGN((pixcount * fmt->bpp[0]) / 8, 16);
payload += ALIGN((pixcount * fmt->bpp[1]) / 8, 32);
payload += ALIGN((pixcount * fmt->bpp[2]) / 8, 16);
payload += (pixcount * fmt->bpp[3]) / 8;
} else {
payload += NV12_82_MFC_PAYLOAD(width, height);
}
} else if (IS_YUV(mode)) {
if (!!(flags & G2D_LAYERFLAG_MFC_STRIDE) && IS_YUV420(mode)) {
payload += NV12_MFC_PAYLOAD(width, height);
} else {
unsigned int i;
for (i = 0; i < fmt->num_planes; i++)
payload += (pixcount * fmt->bpp[i]) / 8;
}
} else if (IS_AFBC(mode)) {
payload = AFBC_HEADER_SIZE(cmd) + (pixcount * fmt->bpp[0]) / 8;
} else {
payload = cmd[G2DSFR_IMG_STRIDE].value *
cmd[G2DSFR_IMG_BOTTOM].value;
}
return payload;
}
static bool check_width_height(u32 value)
{
return (value > 0) && (value <= G2D_MAX_SIZE);
}
/* 8bpp(grayscale) format is not supported */
static bool check_srccolor_mode(u32 value)
{
u32 fmt = ((value) & G2D_DATAFMT_MASK) >> G2D_DATAFMT_SHIFT;
if ((fmt > 14) || (fmt == 6) || (fmt == 7) || (fmt == 9))
return false;
if (IS_YUV(value) && (value & G2D_DATAFORMAT_AFBC))
return false;
return true;
}
static bool check_dstcolor_mode(u32 value)
{
u32 fmt = ((value) & G2D_DATAFMT_MASK) >> G2D_DATAFMT_SHIFT;
/* src + YCbCr420 3p, - YCbCr420 2p 8.2 */
if ((fmt > 14) || (fmt == 13) || (fmt == 6) || (fmt == 7))
return false;
/* AFBC and UORDER shoult not be set together */
if ((value & (G2D_DATAFORMAT_AFBC | G2D_DATAFORMAT_UORDER)) ==
(G2D_DATAFORMAT_AFBC | G2D_DATAFORMAT_UORDER))
return false;
if (IS_YUV(value) &&
(value & (G2D_DATAFORMAT_AFBC | G2D_DATAFORMAT_UORDER)))
return false;
return true;
}
static bool check_blend_mode(u32 value)
{
int i = 0;
for (i = 0; i < 2; i++) { /* for each source and destination */
if ((value & 0xF) > 7) /* Coeff */
return false;
value >>= 4;
if ((value & 0x3) > 2) /* CoeffSA */
return false;
value >>= 2;
if ((value & 0x3) > 2) /* CoeffDA */
return false;
value >>= 2;
}
return true;
}
static bool check_scale_control(u32 value)
{
return value != 3;
}
struct command_checker {
const char *cmdname;
u32 offset;
u32 mask;
bool (*checker)(u32 value);
};
static struct command_checker source_command_checker[G2DSFR_SRC_FIELD_COUNT] = {
{"STRIDE", 0x0020, 0x0001FFFF, NULL,},
{"COLORMODE", 0x0028, 0x331FFFFF, check_srccolor_mode,},
{"LEFT", 0x002C, 0x00001FFF, NULL,},
{"TOP", 0x0030, 0x00001FFF, NULL,},
{"RIGHT", 0x0034, 0x00003FFF, check_width_height,},
{"BOTTOM", 0x0038, 0x00003FFF, check_width_height,},
{"WIDTH", 0x0070, 0x00003FFF, check_width_height,},
{"HEIGHT", 0x0074, 0x00003FFF, check_width_height,},
{"COMMAND", 0x0000, 0x03100003, NULL,},
{"SELECT", 0x0004, 0x00000001, NULL,},
{"ROTATE", 0x0008, 0x00000031, NULL,},
{"DSTLEFT", 0x000C, 0x00001FFF, NULL,},
{"DSTTOP", 0x0010, 0x00001FFF, NULL,},
{"DSTRIGHT", 0x0014, 0x00003FFF, check_width_height,},
{"DSTBOTTOM", 0x0018, 0x00003FFF, check_width_height,},
{"SCALECONTROL", 0x048, 0x00000003, check_scale_control,},
{"XSCALE", 0x004C, 0x3FFFFFFF, NULL,},
{"YSCALE", 0x0050, 0x3FFFFFFF, NULL,},
{"XPHASE", 0x0054, 0x0000FFFF, NULL,},
{"YPHASE", 0x0058, 0x0000FFFF, NULL,},
{"COLOR", 0x005C, 0xFFFFFFFF, NULL,},
{"ALPHA", 0x0060, 0xFFFFFFFF, NULL,},
{"BLEND", 0x003C, 0x0035FFFF, check_blend_mode,},
{"YCBCRMODE", 0x0088, 0x10000017, NULL,},
{"HDRMODE", 0x0090, 0x000011B3, NULL,},
};
static struct command_checker target_command_checker[G2DSFR_DST_FIELD_COUNT] = {
/* BASE OFFSET: 0x0120 */
{"STRIDE", 0x0004, 0x0001FFFF, NULL,},
{"COLORMODE", 0x000C, 0x333FFFFF, check_dstcolor_mode,},
{"LEFT", 0x0010, 0x00001FFF, NULL,},
{"TOP", 0x0014, 0x00001FFF, NULL,},
{"RIGHT", 0x0018, 0x00003FFF, check_width_height,},
{"BOTTOM", 0x001C, 0x00003FFF, check_width_height,},
{"WIDTH", 0x0040, 0x00003FFF, check_width_height,},
{"HEIGHT", 0x0044, 0x00003FFF, check_width_height,},
/* TODO: check csc */
{"YCBCRMODE", 0x0058, 0x0000F714, NULL,},
};
static u16 extra_cmd_range[][2] = { /* {first, last} */
{0x2000, 0x208C}, /* SRC CSC Coefficients */
{0x2100, 0x2120}, /* DST CSC Coefficients */
{0x3000, 0x3100}, /* HDR EOTF Coefficients */
{0x3200, 0x3300}, /* Degamma Coefficients */
{0x3400, 0x3420}, /* HDR Gamut Mapping Coefficients */
{0x3500, 0x3520}, /* Degamma 2.2 Coefficients */
{0x3600, 0x3680}, /* HDR Tone Mapping Coefficients */
{0x3700, 0x3780}, /* Degamma Tone Mapping Coefficients */
{0x5000, 0x5B8C}, /* SET 0,1,2 Coefficients */
};
#define TARGET_OFFSET 0x120
#define LAYER_OFFSET(idx) ((2 + (idx)) << 8)
static int g2d_copy_commands(struct g2d_device *g2d_dev, int index,
struct g2d_reg regs[], __u32 cmd[],
struct command_checker checker[],
unsigned int num_cmds)
{
unsigned int base = (index < 0) ? TARGET_OFFSET : LAYER_OFFSET(index);
int i;
for (i = 0; i < num_cmds; i++) {
if (((cmd[i] & ~checker[i].mask) != 0) ||
(checker[i].checker &&
!checker[i].checker(cmd[i]))) {
perrfndev(g2d_dev, "Invalid %s[%d] SFR '%s' value %#x",
(index < 0) ? "target" : "source",
index, checker[i].cmdname, cmd[i]);
perrdev(g2d_dev, "mask %#x", checker[i].mask);
return -EINVAL;
}
regs[i].offset = base + checker[i].offset;
regs[i].value = cmd[i];
}
return num_cmds;
}
static bool g2d_validate_image_dimension(struct g2d_device *g2d_dev,
u32 width, u32 height,
u32 left, u32 top,
u32 right, u32 bottom)
{
if ((left >= right) || (top >= bottom) ||
(width < (right - left)) || (height < (bottom - top))) {
perrfndev(g2d_dev, "Invalid dimension [%ux%u, %ux%u) / %ux%u",
left, top, right, bottom, width, height);
return false;
}
return true;
}
#define IS_EVEN(value) (((value) & 1) == 0)
static bool g2d_validate_image_format(struct g2d_device *g2d_dev,
struct g2d_task *task, struct g2d_reg commands[], bool dst)
{
bool yuvbitdepth = !!(g2d_dev->caps & G2D_DEVICE_CAPS_YUV_BITDEPTH);
u32 stride = commands[G2DSFR_IMG_STRIDE].value;
u32 mode = commands[G2DSFR_IMG_COLORMODE].value;
u32 width = commands[G2DSFR_IMG_WIDTH].value;
u32 height = commands[G2DSFR_IMG_HEIGHT].value;
u32 left = commands[G2DSFR_IMG_LEFT].value;
u32 right = commands[G2DSFR_IMG_RIGHT].value;
u32 top = commands[G2DSFR_IMG_TOP].value;
u32 bottom = commands[G2DSFR_IMG_BOTTOM].value;
u32 Bpp = 0;
const struct g2d_fmt *fmt;
if (IS_AFBC(mode) && !dst) {
width++;
height++;
}
if (!g2d_validate_image_dimension(g2d_dev, width, height,
left, top, right, bottom))
return false;
fmt = g2d_find_format(mode, g2d_dev->caps);
if (fmt == NULL) {
perrfndev(g2d_dev, "Color mode %#x is not supported", mode);
return false;
}
Bpp = fmt->bpp[0] / 8;
if (stride) {
int err = 0;
if (IS_AFBC(mode) || IS_YUV(mode))
err |= 1 << 1;
if (IS_UORDER(mode) & (stride != ALIGN(width * Bpp, 16)))
err |= 1 << 2;
if (stride < (width * Bpp))
err |= 1 << 4;
if (stride > (G2D_MAX_SIZE * Bpp))
err |= 1 << 5;
if (err) {
perrfndev(g2d_dev,
"wrong(%#x) stride %u mode %#x size %ux%u",
err, stride, mode, width, height);
return false;
}
} else if (IS_YUV(mode)) {
/* TODO: Y8 handling if required */
if (!IS_EVEN(width) || (!IS_YUV422(mode) && !IS_EVEN(height)))
goto err_align;
} else if (!IS_AFBC(mode)) {
perrfndev(g2d_dev, "Non AFBC RGB requires valid stride");
return false;
}
if (IS_HWFC(task->flags)) {
if ((dst && IS_AFBC(mode)) || IS_UORDER(mode)) {
perrfndev(g2d_dev, "Invalid HWFC format with %s",
IS_AFBC(mode) ? "AFBC" : "UORDER");
return false;
}
if (dst &&
((width != right - left) || (height != bottom - top)))
goto err_align;
}
if (!dst) {
if (IS_AFBC(mode) && (!IS_AFBC_WIDTH_ALIGNED(width) ||
!IS_AFBC_HEIGHT_ALIGNED(height)))
goto err_align;
if (IS_YUV_82(mode, yuvbitdepth) && !IS_ALIGNED(width, 64))
goto err_align;
return true;
}
width = commands[G2DSFR_IMG_LEFT].value |
commands[G2DSFR_IMG_RIGHT].value;
height = commands[G2DSFR_IMG_TOP].value |
commands[G2DSFR_IMG_BOTTOM].value;
if (IS_AFBC(mode) && !IS_AFBC_WIDTH_ALIGNED(width | height))
goto err_align;
if (IS_YUV(mode)) {
/*
* DST clip region has the alignment restrictions
* accroding to the chroma subsampling
*/
if (!IS_EVEN(width) || (!IS_YUV422(mode) && !IS_EVEN(height)))
goto err_align;
}
return true;
err_align:
perrfndev(g2d_dev,
"Unaligned size %ux%u or crop [%ux%u, %ux%u) for %s %s %s",
commands[G2DSFR_IMG_WIDTH].value,
commands[G2DSFR_IMG_HEIGHT].value,
commands[G2DSFR_IMG_LEFT].value,
commands[G2DSFR_IMG_TOP].value,
commands[G2DSFR_IMG_RIGHT].value,
commands[G2DSFR_IMG_BOTTOM].value,
IS_AFBC(mode) ? "AFBC" :
IS_YUV422(mode) ? "YUV422" : "YUV20",
IS_YUV_82(mode, yuvbitdepth) ? "8+2" : "",
IS_HWFC(task->flags) ? "HWFC" : "");
return false;
}
bool g2d_validate_source_commands(struct g2d_device *g2d_dev,
struct g2d_task *task,
unsigned int i, struct g2d_layer *source,
struct g2d_layer *target)
{
u32 colormode = source->commands[G2DSFR_IMG_COLORMODE].value;
u32 width, height;
if (!g2d_validate_image_format(
g2d_dev, task, source->commands, false)) {
perrfndev(g2d_dev, "Failed to validate source[%d] commands", i);
return false;
}
if (((source->flags & G2D_LAYERFLAG_COLORFILL) != 0) &&
!IS_RGB(colormode)) {
perrfndev(g2d_dev,
"Image type should be RGB for Solid color layer:");
perrdev(g2d_dev, "\tindex: %d, flags %#x, colormode: %#010x",
i, source->flags, colormode);
return false;
}
width = target->commands[G2DSFR_IMG_WIDTH].value;
height = target->commands[G2DSFR_IMG_HEIGHT].value;
if (IS_AFBC(colormode)) {
width++;
height++;
}
if (!g2d_validate_image_dimension(g2d_dev, width, height,
source->commands[G2DSFR_SRC_DSTLEFT].value,
source->commands[G2DSFR_SRC_DSTTOP].value,
source->commands[G2DSFR_SRC_DSTRIGHT].value,
source->commands[G2DSFR_SRC_DSTBOTTOM].value)) {
perrfndev(g2d_dev, "Window of source[%d] floods the target", i);
return false;
}
return true;
}
bool g2d_validate_target_commands(struct g2d_device *g2d_dev,
struct g2d_task *task)
{
if (!g2d_validate_image_format(g2d_dev, task,
task->target.commands, true)) {
perrfndev(g2d_dev, "Failed to validate target commands");
return false;
}
return true;
}
static bool g2d_validate_extra_command(struct g2d_device *g2d_dev,
struct g2d_reg extra[],
unsigned int num_regs)
{
unsigned int n;
size_t i;
/*
* TODO: NxM loop ==> total 2008 comparison are required in maximum:
* Consider if we can make it simple with a single range [0x2000, 4000)
*/
for (n = 0; n < num_regs; n++) {
for (i = 0; i < ARRAY_SIZE(extra_cmd_range); i++) {
if ((extra[n].offset >= extra_cmd_range[i][0]) &&
(extra[n].offset <= extra_cmd_range[i][1]))
break;
}
if (i == ARRAY_SIZE(extra_cmd_range)) {
perrfndev(g2d_dev, "wrong offset %#x @ extra cmd[%d]",
extra[n].offset, n);
return false;
}
}
return true;
}
int g2d_import_commands(struct g2d_device *g2d_dev, struct g2d_task *task,
struct g2d_task_data *data, unsigned int num_sources)
{
struct g2d_reg *cmdaddr = page_address(task->cmd_page);
struct g2d_commands *cmds = &data->commands;
u32 tm_tuned_lut[NR_TM_LUT_VALUES];
unsigned int i;
int copied;
if (cmds->num_extra_regs > MAX_EXTRA_REGS) {
perrfndev(g2d_dev, "Too many coefficient reigsters %d",
cmds->num_extra_regs);
return -EINVAL;
}
cmdaddr += task->sec.cmd_count;
copied = g2d_copy_commands(g2d_dev, -1, cmdaddr, cmds->target,
target_command_checker, G2DSFR_DST_FIELD_COUNT);
if (copied < 0)
return -EINVAL;
task->target.commands = cmdaddr;
cmdaddr += copied;
task->sec.cmd_count += copied;
for (i = 0; i < num_sources; i++) {
u32 srccmds[G2DSFR_SRC_FIELD_COUNT];
if (copy_from_user(srccmds, cmds->source[i], sizeof(srccmds))) {
perrfndev(g2d_dev, "Failed to get src[%d] commands", i);
return -EFAULT;
}
copied = g2d_copy_commands(g2d_dev, i, cmdaddr, srccmds,
source_command_checker, G2DSFR_SRC_FIELD_COUNT);
if (copied < 0)
return -EINVAL;
task->source[i].commands = cmdaddr;
cmdaddr += copied;
task->sec.cmd_count += copied;
}
if (copy_from_user(cmdaddr, cmds->extra,
cmds->num_extra_regs * sizeof(struct g2d_reg))) {
perrfndev(g2d_dev, "Failed to get %u extra commands",
cmds->num_extra_regs);
return -EFAULT;
}
if (!g2d_validate_extra_command(g2d_dev, cmdaddr, cmds->num_extra_regs))
return -EINVAL;
task->sec.cmd_count += cmds->num_extra_regs;
/* overwrite if TM LUT values are specified: consumes 66 entries */
if (exynos_hdr_get_tm_lut(tm_tuned_lut)) {
u32 base;
/* offsets of TM LUT values: 0x3600, 0x3700 */
cmdaddr += cmds->num_extra_regs;
for (base = 0x3600; base < 0x3800; base += 0x100) {
for (i = 0; i < NR_TM_LUT_VALUES; i++) {
cmdaddr->offset =
(unsigned long)(base + i * sizeof(u32));
cmdaddr->value = tm_tuned_lut[i];
cmdaddr++;
task->sec.cmd_count++;
}
}
}
return 0;
}
static unsigned int g2d_set_image_buffer(struct g2d_task *task,
struct g2d_layer *layer, u32 colormode,
unsigned char offsets[], u32 base)
{
const struct g2d_fmt *fmt = g2d_find_format(colormode,
task->g2d_dev->caps);
struct g2d_reg *reg = (struct g2d_reg *)page_address(task->cmd_page);
unsigned int cmd_count = task->sec.cmd_count;
u32 width = layer_width(layer);
u32 height = layer_height(layer);
unsigned int i;
if (fmt->num_planes == 4) {
unsigned int nbufs = min_t(unsigned int,
layer->num_buffers, fmt->num_planes);
for (i = 0; i < nbufs; i++) {
if (!YUV82_BASE_ALIGNED(layer->buffer[i].dma_addr, i)) {
perrfndev(task->g2d_dev,
"addr[%d] not aligned for YUV8+2", i);
return 0;
}
}
} else if (!IS_ALIGNED(layer->buffer[0].dma_addr, 4)) {
perrfndev(task->g2d_dev, "Plane 0 address isn't aligned by 4.");
return 0;
}
for (i = 0; i < layer->num_buffers; i++) {
reg[cmd_count].offset = BASE_REG_OFFSET(base, offsets, i);
reg[cmd_count].value = layer->buffer[i].dma_addr;
cmd_count++;
}
if (layer->num_buffers == fmt->num_planes)
return cmd_count;
/* address of plane 0 is set in the above for() */
if (fmt->num_planes == 2) {
/* YCbCr semi-planar in a single buffer */
reg[cmd_count].offset = BASE_REG_OFFSET(base, offsets, 1);
if (!!(layer->flags & G2D_LAYERFLAG_MFC_STRIDE) &&
IS_YUV420(fmt->fmtvalue)) {
reg[cmd_count].value =
NV12_MFC_CBASE(layer->buffer[0].dma_addr,
width, height);
} else {
reg[cmd_count].value = layer_pixelcount(layer);
reg[cmd_count].value *= fmt->bpp[0] / 8;
reg[cmd_count].value += layer->buffer[0].dma_addr;
reg[cmd_count].value = ALIGN(reg[cmd_count].value, 2);
}
cmd_count++;
return cmd_count;
}
BUG_ON(fmt->num_planes != 4);
if ((layer->num_buffers == 1) &&
!(layer->flags & G2D_LAYERFLAG_MFC_STRIDE)) {
dma_addr_t addr = layer->buffer[0].dma_addr;
/* YCbCr semi-planar 8+2 in a single buffer */
for (i = 1; i < 4; i++) {
addr += (layer_pixelcount(layer) * fmt->bpp[i - 1]) / 8;
addr = YUV82_BASE_ALIGN(addr, i);
reg[cmd_count].value = addr;
reg[cmd_count].offset =
BASE_REG_OFFSET(base, offsets, i);
cmd_count++;
}
return cmd_count;
}
/* G2D_LAYERFLAG_MFC_STRIDE is set */
reg[cmd_count].offset = BASE_REG_OFFSET(base, offsets, 1);
reg[cmd_count].value = layer->buffer[0].dma_addr +
NV12_MFC_Y_PAYLOAD_PAD(width, height);
cmd_count++;
reg[cmd_count].offset = BASE_REG_OFFSET(base, offsets, 2);
if (layer->num_buffers == 2)
reg[cmd_count].value = layer->buffer[1].dma_addr;
else
reg[cmd_count].value = reg[cmd_count - 1].value +
NV12_82_MFC_2Y_PAYLOAD(width, height) +
MFC_2B_PAD_SIZE;
cmd_count++;
reg[cmd_count].offset = BASE_REG_OFFSET(base, offsets, 3);
reg[cmd_count].value = reg[cmd_count - 1].value +
NV12_MFC_C_PAYLOAD_PAD(width, height);
cmd_count++;
return cmd_count;
}
static unsigned int g2d_set_afbc_buffer(struct g2d_task *task,
struct g2d_layer *layer,
u32 base_offset)
{
struct g2d_reg *reg = (struct g2d_reg *)page_address(task->cmd_page);
u32 align = (base_offset == TARGET_OFFSET) ? 64 : 16;
unsigned char *reg_offset = (base_offset == TARGET_OFFSET) ?
dst_base_reg_offset : src_base_reg_offset;
if (!IS_ALIGNED(layer->buffer[0].dma_addr, align)) {
perrfndev(task->g2d_dev, "AFBC base %#llx is not aligned by %u",
layer->buffer[0].dma_addr, align);
return 0;
}
reg[task->sec.cmd_count].offset = base_offset + reg_offset[2];
reg[task->sec.cmd_count].value = layer->buffer[0].dma_addr;
reg[task->sec.cmd_count + 1].offset = base_offset + reg_offset[3];
if (base_offset == TARGET_OFFSET)
reg[task->sec.cmd_count + 1].value =
ALIGN(AFBC_HEADER_SIZE(layer->commands)
+ layer->buffer[0].dma_addr, align);
else
reg[task->sec.cmd_count + 1].value = layer->buffer[0].dma_addr;
return task->sec.cmd_count + 2;
}
bool g2d_prepare_source(struct g2d_task *task,
struct g2d_layer *layer, int index)
{
struct g2d_reg *reg = (struct g2d_reg *)page_address(task->cmd_page);
u32 colormode = layer->commands[G2DSFR_IMG_COLORMODE].value;
bool yuv82;
unsigned char *offsets;
yuv82 = IS_YUV_82(colormode,
task->g2d_dev->caps & G2D_DEVICE_CAPS_YUV_BITDEPTH);
offsets = yuv82 ? src_base_reg_offset_yuv82 : src_base_reg_offset;
reg[TASK_REG_LAYER_UPDATE].value |= 1 << index;
if ((layer->flags & G2D_LAYERFLAG_COLORFILL) != 0)
return true;
task->sec.cmd_count = ((colormode & G2D_DATAFORMAT_AFBC) != 0)
? g2d_set_afbc_buffer(task, layer, LAYER_OFFSET(index))
: g2d_set_image_buffer(task, layer, colormode,
offsets, LAYER_OFFSET(index));
/*
* It is alright to set task->cmd_count to 0
* because this task is to be discarded.
*/
return task->sec.cmd_count != 0;
}
bool g2d_prepare_target(struct g2d_task *task)
{
u32 colormode = task->target.commands[G2DSFR_IMG_COLORMODE].value;
task->sec.cmd_count = ((colormode & G2D_DATAFORMAT_AFBC) != 0)
? g2d_set_afbc_buffer(task, &task->target,
TARGET_OFFSET)
: g2d_set_image_buffer(task, &task->target, colormode,
dst_base_reg_offset, TARGET_OFFSET);
return task->sec.cmd_count != 0;
}