xref: /OK3568_Linux_fs/kernel/drivers/media/pci/tw5864/tw5864.h (revision 4882a59341e53eb6f0b4789bf948001014eff981)
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