1*4882a593Smuzhiyun /* SPDX-License-Identifier: GPL-2.0-or-later */ 2*4882a593Smuzhiyun /* 3*4882a593Smuzhiyun * TW5864 driver - common header file 4*4882a593Smuzhiyun * 5*4882a593Smuzhiyun * Copyright (C) 2016 Bluecherry, LLC <maintainers@bluecherrydvr.com> 6*4882a593Smuzhiyun */ 7*4882a593Smuzhiyun 8*4882a593Smuzhiyun #include <linux/pci.h> 9*4882a593Smuzhiyun #include <linux/videodev2.h> 10*4882a593Smuzhiyun #include <linux/notifier.h> 11*4882a593Smuzhiyun #include <linux/delay.h> 12*4882a593Smuzhiyun #include <linux/mutex.h> 13*4882a593Smuzhiyun #include <linux/io.h> 14*4882a593Smuzhiyun #include <linux/interrupt.h> 15*4882a593Smuzhiyun 16*4882a593Smuzhiyun #include <media/v4l2-common.h> 17*4882a593Smuzhiyun #include <media/v4l2-ioctl.h> 18*4882a593Smuzhiyun #include <media/v4l2-ctrls.h> 19*4882a593Smuzhiyun #include <media/v4l2-device.h> 20*4882a593Smuzhiyun #include <media/videobuf2-dma-sg.h> 21*4882a593Smuzhiyun 22*4882a593Smuzhiyun #include "tw5864-reg.h" 23*4882a593Smuzhiyun 24*4882a593Smuzhiyun #define PCI_DEVICE_ID_TECHWELL_5864 0x5864 25*4882a593Smuzhiyun 26*4882a593Smuzhiyun #define TW5864_NORMS V4L2_STD_ALL 27*4882a593Smuzhiyun 28*4882a593Smuzhiyun /* ----------------------------------------------------------- */ 29*4882a593Smuzhiyun /* card configuration */ 30*4882a593Smuzhiyun 31*4882a593Smuzhiyun #define TW5864_INPUTS 4 32*4882a593Smuzhiyun 33*4882a593Smuzhiyun /* The TW5864 uses 192 (16x12) detection cells in full screen for motion 34*4882a593Smuzhiyun * detection. Each detection cell is composed of 44 pixels and 20 lines for 35*4882a593Smuzhiyun * NTSC and 24 lines for PAL. 36*4882a593Smuzhiyun */ 37*4882a593Smuzhiyun #define MD_CELLS_HOR 16 38*4882a593Smuzhiyun #define MD_CELLS_VERT 12 39*4882a593Smuzhiyun #define MD_CELLS (MD_CELLS_HOR * MD_CELLS_VERT) 40*4882a593Smuzhiyun 41*4882a593Smuzhiyun #define H264_VLC_BUF_SIZE 0x80000 42*4882a593Smuzhiyun #define H264_MV_BUF_SIZE 0x2000 /* device writes 5396 bytes */ 43*4882a593Smuzhiyun #define QP_VALUE 28 44*4882a593Smuzhiyun #define MAX_GOP_SIZE 255 45*4882a593Smuzhiyun #define GOP_SIZE MAX_GOP_SIZE 46*4882a593Smuzhiyun 47*4882a593Smuzhiyun enum resolution { 48*4882a593Smuzhiyun D1 = 1, 49*4882a593Smuzhiyun HD1 = 2, /* half d1 - 360x(240|288) */ 50*4882a593Smuzhiyun CIF = 3, 51*4882a593Smuzhiyun QCIF = 4, 52*4882a593Smuzhiyun }; 53*4882a593Smuzhiyun 54*4882a593Smuzhiyun /* ----------------------------------------------------------- */ 55*4882a593Smuzhiyun /* device / file handle status */ 56*4882a593Smuzhiyun 57*4882a593Smuzhiyun struct tw5864_dev; /* forward delclaration */ 58*4882a593Smuzhiyun 59*4882a593Smuzhiyun /* buffer for one video/vbi/ts frame */ 60*4882a593Smuzhiyun struct tw5864_buf { 61*4882a593Smuzhiyun struct vb2_v4l2_buffer vb; 62*4882a593Smuzhiyun struct list_head list; 63*4882a593Smuzhiyun 64*4882a593Smuzhiyun unsigned int size; 65*4882a593Smuzhiyun }; 66*4882a593Smuzhiyun 67*4882a593Smuzhiyun struct tw5864_dma_buf { 68*4882a593Smuzhiyun void *addr; 69*4882a593Smuzhiyun dma_addr_t dma_addr; 70*4882a593Smuzhiyun }; 71*4882a593Smuzhiyun 72*4882a593Smuzhiyun enum tw5864_vid_std { 73*4882a593Smuzhiyun STD_NTSC = 0, /* NTSC (M) */ 74*4882a593Smuzhiyun STD_PAL = 1, /* PAL (B, D, G, H, I) */ 75*4882a593Smuzhiyun STD_SECAM = 2, /* SECAM */ 76*4882a593Smuzhiyun STD_NTSC443 = 3, /* NTSC4.43 */ 77*4882a593Smuzhiyun STD_PAL_M = 4, /* PAL (M) */ 78*4882a593Smuzhiyun STD_PAL_CN = 5, /* PAL (CN) */ 79*4882a593Smuzhiyun STD_PAL_60 = 6, /* PAL 60 */ 80*4882a593Smuzhiyun STD_INVALID = 7, 81*4882a593Smuzhiyun STD_AUTO = 7, 82*4882a593Smuzhiyun }; 83*4882a593Smuzhiyun 84*4882a593Smuzhiyun struct tw5864_input { 85*4882a593Smuzhiyun int nr; /* input number */ 86*4882a593Smuzhiyun struct tw5864_dev *root; 87*4882a593Smuzhiyun struct mutex lock; /* used for vidq and vdev */ 88*4882a593Smuzhiyun spinlock_t slock; /* used for sync between ISR, tasklet & V4L2 API */ 89*4882a593Smuzhiyun struct video_device vdev; 90*4882a593Smuzhiyun struct v4l2_ctrl_handler hdl; 91*4882a593Smuzhiyun struct vb2_queue vidq; 92*4882a593Smuzhiyun struct list_head active; 93*4882a593Smuzhiyun enum resolution resolution; 94*4882a593Smuzhiyun unsigned int width, height; 95*4882a593Smuzhiyun unsigned int frame_seqno; 96*4882a593Smuzhiyun unsigned int frame_gop_seqno; 97*4882a593Smuzhiyun unsigned int h264_idr_pic_id; 98*4882a593Smuzhiyun int enabled; 99*4882a593Smuzhiyun enum tw5864_vid_std std; 100*4882a593Smuzhiyun v4l2_std_id v4l2_std; 101*4882a593Smuzhiyun int tail_nb_bits; 102*4882a593Smuzhiyun u8 tail; 103*4882a593Smuzhiyun u8 *buf_cur_ptr; 104*4882a593Smuzhiyun int buf_cur_space_left; 105*4882a593Smuzhiyun 106*4882a593Smuzhiyun u32 reg_interlacing; 107*4882a593Smuzhiyun u32 reg_vlc; 108*4882a593Smuzhiyun u32 reg_dsp_codec; 109*4882a593Smuzhiyun u32 reg_dsp; 110*4882a593Smuzhiyun u32 reg_emu; 111*4882a593Smuzhiyun u32 reg_dsp_qp; 112*4882a593Smuzhiyun u32 reg_dsp_ref_mvp_lambda; 113*4882a593Smuzhiyun u32 reg_dsp_i4x4_weight; 114*4882a593Smuzhiyun u32 buf_id; 115*4882a593Smuzhiyun 116*4882a593Smuzhiyun struct tw5864_buf *vb; 117*4882a593Smuzhiyun 118*4882a593Smuzhiyun struct v4l2_ctrl *md_threshold_grid_ctrl; 119*4882a593Smuzhiyun u16 md_threshold_grid_values[12 * 16]; 120*4882a593Smuzhiyun int qp; 121*4882a593Smuzhiyun int gop; 122*4882a593Smuzhiyun 123*4882a593Smuzhiyun /* 124*4882a593Smuzhiyun * In (1/MAX_FPS) units. 125*4882a593Smuzhiyun * For max FPS (default), set to 1. 126*4882a593Smuzhiyun * For 1 FPS, set to e.g. 32. 127*4882a593Smuzhiyun */ 128*4882a593Smuzhiyun int frame_interval; 129*4882a593Smuzhiyun unsigned long new_frame_deadline; 130*4882a593Smuzhiyun }; 131*4882a593Smuzhiyun 132*4882a593Smuzhiyun struct tw5864_h264_frame { 133*4882a593Smuzhiyun struct tw5864_dma_buf vlc; 134*4882a593Smuzhiyun struct tw5864_dma_buf mv; 135*4882a593Smuzhiyun int vlc_len; 136*4882a593Smuzhiyun u32 checksum; 137*4882a593Smuzhiyun struct tw5864_input *input; 138*4882a593Smuzhiyun u64 timestamp; 139*4882a593Smuzhiyun unsigned int seqno; 140*4882a593Smuzhiyun unsigned int gop_seqno; 141*4882a593Smuzhiyun }; 142*4882a593Smuzhiyun 143*4882a593Smuzhiyun /* global device status */ 144*4882a593Smuzhiyun struct tw5864_dev { 145*4882a593Smuzhiyun spinlock_t slock; /* used for sync between ISR, tasklet & V4L2 API */ 146*4882a593Smuzhiyun struct v4l2_device v4l2_dev; 147*4882a593Smuzhiyun struct tw5864_input inputs[TW5864_INPUTS]; 148*4882a593Smuzhiyun #define H264_BUF_CNT 4 149*4882a593Smuzhiyun struct tw5864_h264_frame h264_buf[H264_BUF_CNT]; 150*4882a593Smuzhiyun int h264_buf_r_index; 151*4882a593Smuzhiyun int h264_buf_w_index; 152*4882a593Smuzhiyun 153*4882a593Smuzhiyun struct tasklet_struct tasklet; 154*4882a593Smuzhiyun 155*4882a593Smuzhiyun int encoder_busy; 156*4882a593Smuzhiyun /* Input number to check next for ready raw picture (in RR fashion) */ 157*4882a593Smuzhiyun int next_input; 158*4882a593Smuzhiyun 159*4882a593Smuzhiyun /* pci i/o */ 160*4882a593Smuzhiyun char name[64]; 161*4882a593Smuzhiyun struct pci_dev *pci; 162*4882a593Smuzhiyun void __iomem *mmio; 163*4882a593Smuzhiyun u32 irqmask; 164*4882a593Smuzhiyun }; 165*4882a593Smuzhiyun 166*4882a593Smuzhiyun #define tw_readl(reg) readl(dev->mmio + reg) 167*4882a593Smuzhiyun #define tw_mask_readl(reg, mask) \ 168*4882a593Smuzhiyun (tw_readl(reg) & (mask)) 169*4882a593Smuzhiyun #define tw_mask_shift_readl(reg, mask, shift) \ 170*4882a593Smuzhiyun (tw_mask_readl((reg), ((mask) << (shift))) >> (shift)) 171*4882a593Smuzhiyun 172*4882a593Smuzhiyun #define tw_writel(reg, value) writel((value), dev->mmio + reg) 173*4882a593Smuzhiyun #define tw_mask_writel(reg, mask, value) \ 174*4882a593Smuzhiyun tw_writel(reg, (tw_readl(reg) & ~(mask)) | ((value) & (mask))) 175*4882a593Smuzhiyun #define tw_mask_shift_writel(reg, mask, shift, value) \ 176*4882a593Smuzhiyun tw_mask_writel((reg), ((mask) << (shift)), ((value) << (shift))) 177*4882a593Smuzhiyun 178*4882a593Smuzhiyun #define tw_setl(reg, bit) tw_writel((reg), tw_readl(reg) | (bit)) 179*4882a593Smuzhiyun #define tw_clearl(reg, bit) tw_writel((reg), tw_readl(reg) & ~(bit)) 180*4882a593Smuzhiyun 181*4882a593Smuzhiyun u8 tw5864_indir_readb(struct tw5864_dev *dev, u16 addr); 182*4882a593Smuzhiyun #define tw_indir_readb(addr) tw5864_indir_readb(dev, addr) 183*4882a593Smuzhiyun void tw5864_indir_writeb(struct tw5864_dev *dev, u16 addr, u8 data); 184*4882a593Smuzhiyun #define tw_indir_writeb(addr, data) tw5864_indir_writeb(dev, addr, data) 185*4882a593Smuzhiyun 186*4882a593Smuzhiyun void tw5864_irqmask_apply(struct tw5864_dev *dev); 187*4882a593Smuzhiyun int tw5864_video_init(struct tw5864_dev *dev, int *video_nr); 188*4882a593Smuzhiyun void tw5864_video_fini(struct tw5864_dev *dev); 189*4882a593Smuzhiyun void tw5864_prepare_frame_headers(struct tw5864_input *input); 190*4882a593Smuzhiyun void tw5864_h264_put_stream_header(u8 **buf, size_t *space_left, int qp, 191*4882a593Smuzhiyun int width, int height); 192*4882a593Smuzhiyun void tw5864_h264_put_slice_header(u8 **buf, size_t *space_left, 193*4882a593Smuzhiyun unsigned int idr_pic_id, 194*4882a593Smuzhiyun unsigned int frame_gop_seqno, 195*4882a593Smuzhiyun int *tail_nb_bits, u8 *tail); 196*4882a593Smuzhiyun void tw5864_request_encoded_frame(struct tw5864_input *input); 197