linux/drivers/staging/media/hantro/hantro_postproc.c
<<
>>
Prefs
   1// SPDX-License-Identifier: GPL-2.0
   2/*
   3 * Hantro G1 post-processor support
   4 *
   5 * Copyright (C) 2019 Collabora, Ltd.
   6 */
   7
   8#include <linux/dma-mapping.h>
   9#include <linux/types.h>
  10
  11#include "hantro.h"
  12#include "hantro_hw.h"
  13#include "hantro_g1_regs.h"
  14
  15#define HANTRO_PP_REG_WRITE(vpu, reg_name, val) \
  16{ \
  17        hantro_reg_write(vpu, \
  18                         &(vpu)->variant->postproc_regs->reg_name, \
  19                         val); \
  20}
  21
  22#define HANTRO_PP_REG_WRITE_S(vpu, reg_name, val) \
  23{ \
  24        hantro_reg_write_s(vpu, \
  25                           &(vpu)->variant->postproc_regs->reg_name, \
  26                           val); \
  27}
  28
  29#define VPU_PP_IN_YUYV                  0x0
  30#define VPU_PP_IN_NV12                  0x1
  31#define VPU_PP_IN_YUV420                0x2
  32#define VPU_PP_IN_YUV240_TILED          0x5
  33#define VPU_PP_OUT_RGB                  0x0
  34#define VPU_PP_OUT_YUYV                 0x3
  35
  36const struct hantro_postproc_regs hantro_g1_postproc_regs = {
  37        .pipeline_en = {G1_REG_PP_INTERRUPT, 1, 0x1},
  38        .max_burst = {G1_REG_PP_DEV_CONFIG, 0, 0x1f},
  39        .clk_gate = {G1_REG_PP_DEV_CONFIG, 1, 0x1},
  40        .out_swap32 = {G1_REG_PP_DEV_CONFIG, 5, 0x1},
  41        .out_endian = {G1_REG_PP_DEV_CONFIG, 6, 0x1},
  42        .out_luma_base = {G1_REG_PP_OUT_LUMA_BASE, 0, 0xffffffff},
  43        .input_width = {G1_REG_PP_INPUT_SIZE, 0, 0x1ff},
  44        .input_height = {G1_REG_PP_INPUT_SIZE, 9, 0x1ff},
  45        .output_width = {G1_REG_PP_CONTROL, 4, 0x7ff},
  46        .output_height = {G1_REG_PP_CONTROL, 15, 0x7ff},
  47        .input_fmt = {G1_REG_PP_CONTROL, 29, 0x7},
  48        .output_fmt = {G1_REG_PP_CONTROL, 26, 0x7},
  49        .orig_width = {G1_REG_PP_MASK1_ORIG_WIDTH, 23, 0x1ff},
  50        .display_width = {G1_REG_PP_DISPLAY_WIDTH, 0, 0xfff},
  51};
  52
  53void hantro_postproc_enable(struct hantro_ctx *ctx)
  54{
  55        struct hantro_dev *vpu = ctx->dev;
  56        struct vb2_v4l2_buffer *dst_buf;
  57        u32 src_pp_fmt, dst_pp_fmt;
  58        dma_addr_t dst_dma;
  59
  60        if (!vpu->variant->postproc_regs)
  61                return;
  62
  63        /* Turn on pipeline mode. Must be done first. */
  64        HANTRO_PP_REG_WRITE_S(vpu, pipeline_en, 0x1);
  65
  66        src_pp_fmt = VPU_PP_IN_NV12;
  67
  68        switch (ctx->vpu_dst_fmt->fourcc) {
  69        case V4L2_PIX_FMT_YUYV:
  70                dst_pp_fmt = VPU_PP_OUT_YUYV;
  71                break;
  72        default:
  73                WARN(1, "output format %d not supported by the post-processor, this wasn't expected.",
  74                     ctx->vpu_dst_fmt->fourcc);
  75                dst_pp_fmt = 0;
  76                break;
  77        }
  78
  79        dst_buf = v4l2_m2m_next_dst_buf(ctx->fh.m2m_ctx);
  80        dst_dma = vb2_dma_contig_plane_dma_addr(&dst_buf->vb2_buf, 0);
  81
  82        HANTRO_PP_REG_WRITE(vpu, clk_gate, 0x1);
  83        HANTRO_PP_REG_WRITE(vpu, out_endian, 0x1);
  84        HANTRO_PP_REG_WRITE(vpu, out_swap32, 0x1);
  85        HANTRO_PP_REG_WRITE(vpu, max_burst, 16);
  86        HANTRO_PP_REG_WRITE(vpu, out_luma_base, dst_dma);
  87        HANTRO_PP_REG_WRITE(vpu, input_width, MB_WIDTH(ctx->dst_fmt.width));
  88        HANTRO_PP_REG_WRITE(vpu, input_height, MB_HEIGHT(ctx->dst_fmt.height));
  89        HANTRO_PP_REG_WRITE(vpu, input_fmt, src_pp_fmt);
  90        HANTRO_PP_REG_WRITE(vpu, output_fmt, dst_pp_fmt);
  91        HANTRO_PP_REG_WRITE(vpu, output_width, ctx->dst_fmt.width);
  92        HANTRO_PP_REG_WRITE(vpu, output_height, ctx->dst_fmt.height);
  93        HANTRO_PP_REG_WRITE(vpu, orig_width, MB_WIDTH(ctx->dst_fmt.width));
  94        HANTRO_PP_REG_WRITE(vpu, display_width, ctx->dst_fmt.width);
  95}
  96
  97void hantro_postproc_free(struct hantro_ctx *ctx)
  98{
  99        struct hantro_dev *vpu = ctx->dev;
 100        unsigned int i;
 101
 102        for (i = 0; i < VB2_MAX_FRAME; ++i) {
 103                struct hantro_aux_buf *priv = &ctx->postproc.dec_q[i];
 104
 105                if (priv->cpu) {
 106                        dma_free_attrs(vpu->dev, priv->size, priv->cpu,
 107                                       priv->dma, priv->attrs);
 108                        priv->cpu = NULL;
 109                }
 110        }
 111}
 112
 113int hantro_postproc_alloc(struct hantro_ctx *ctx)
 114{
 115        struct hantro_dev *vpu = ctx->dev;
 116        struct v4l2_m2m_ctx *m2m_ctx = ctx->fh.m2m_ctx;
 117        struct vb2_queue *cap_queue = &m2m_ctx->cap_q_ctx.q;
 118        unsigned int num_buffers = cap_queue->num_buffers;
 119        unsigned int i, buf_size;
 120
 121        buf_size = ctx->dst_fmt.plane_fmt[0].sizeimage +
 122                   hantro_h264_mv_size(ctx->dst_fmt.width,
 123                                       ctx->dst_fmt.height);
 124
 125        for (i = 0; i < num_buffers; ++i) {
 126                struct hantro_aux_buf *priv = &ctx->postproc.dec_q[i];
 127
 128                /*
 129                 * The buffers on this queue are meant as intermediate
 130                 * buffers for the decoder, so no mapping is needed.
 131                 */
 132                priv->attrs = DMA_ATTR_NO_KERNEL_MAPPING;
 133                priv->cpu = dma_alloc_attrs(vpu->dev, buf_size, &priv->dma,
 134                                            GFP_KERNEL, priv->attrs);
 135                if (!priv->cpu)
 136                        return -ENOMEM;
 137                priv->size = buf_size;
 138        }
 139        return 0;
 140}
 141
 142void hantro_postproc_disable(struct hantro_ctx *ctx)
 143{
 144        struct hantro_dev *vpu = ctx->dev;
 145
 146        if (!vpu->variant->postproc_regs)
 147                return;
 148
 149        HANTRO_PP_REG_WRITE_S(vpu, pipeline_en, 0x0);
 150}
 151