forked from Minki/linux
34d1324edd
Support for boards based on Techwell TW5864 chip which provides multichannel video & audio grabbing and encoding (H.264, MJPEG, ADPCM G.726). This submission implements only H.264 encoding of all channels at D1 resolution. Thanks to Mark Thompson <sw@jkqxz.net> for help, and for contribution of H.264 startcode emulation prevention code. Signed-off-by: Andrey Utkin <andrey.utkin@corp.bluecherry.net> Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com> Signed-off-by: Mauro Carvalho Chehab <mchehab@s-opensource.com>
206 lines
5.5 KiB
C
206 lines
5.5 KiB
C
/*
|
|
* TW5864 driver - common header file
|
|
*
|
|
* Copyright (C) 2016 Bluecherry, LLC <maintainers@bluecherrydvr.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* 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/pci.h>
|
|
#include <linux/videodev2.h>
|
|
#include <linux/notifier.h>
|
|
#include <linux/delay.h>
|
|
#include <linux/mutex.h>
|
|
#include <linux/io.h>
|
|
#include <linux/interrupt.h>
|
|
|
|
#include <media/v4l2-common.h>
|
|
#include <media/v4l2-ioctl.h>
|
|
#include <media/v4l2-ctrls.h>
|
|
#include <media/v4l2-device.h>
|
|
#include <media/videobuf2-dma-sg.h>
|
|
|
|
#include "tw5864-reg.h"
|
|
|
|
#define PCI_DEVICE_ID_TECHWELL_5864 0x5864
|
|
|
|
#define TW5864_NORMS V4L2_STD_ALL
|
|
|
|
/* ----------------------------------------------------------- */
|
|
/* card configuration */
|
|
|
|
#define TW5864_INPUTS 4
|
|
|
|
/* The TW5864 uses 192 (16x12) detection cells in full screen for motion
|
|
* detection. Each detection cell is composed of 44 pixels and 20 lines for
|
|
* NTSC and 24 lines for PAL.
|
|
*/
|
|
#define MD_CELLS_HOR 16
|
|
#define MD_CELLS_VERT 12
|
|
#define MD_CELLS (MD_CELLS_HOR * MD_CELLS_VERT)
|
|
|
|
#define H264_VLC_BUF_SIZE 0x80000
|
|
#define H264_MV_BUF_SIZE 0x2000 /* device writes 5396 bytes */
|
|
#define QP_VALUE 28
|
|
#define MAX_GOP_SIZE 255
|
|
#define GOP_SIZE MAX_GOP_SIZE
|
|
|
|
enum resolution {
|
|
D1 = 1,
|
|
HD1 = 2, /* half d1 - 360x(240|288) */
|
|
CIF = 3,
|
|
QCIF = 4,
|
|
};
|
|
|
|
/* ----------------------------------------------------------- */
|
|
/* device / file handle status */
|
|
|
|
struct tw5864_dev; /* forward delclaration */
|
|
|
|
/* buffer for one video/vbi/ts frame */
|
|
struct tw5864_buf {
|
|
struct vb2_v4l2_buffer vb;
|
|
struct list_head list;
|
|
|
|
unsigned int size;
|
|
};
|
|
|
|
struct tw5864_dma_buf {
|
|
void *addr;
|
|
dma_addr_t dma_addr;
|
|
};
|
|
|
|
enum tw5864_vid_std {
|
|
STD_NTSC = 0, /* NTSC (M) */
|
|
STD_PAL = 1, /* PAL (B, D, G, H, I) */
|
|
STD_SECAM = 2, /* SECAM */
|
|
STD_NTSC443 = 3, /* NTSC4.43 */
|
|
STD_PAL_M = 4, /* PAL (M) */
|
|
STD_PAL_CN = 5, /* PAL (CN) */
|
|
STD_PAL_60 = 6, /* PAL 60 */
|
|
STD_INVALID = 7,
|
|
STD_AUTO = 7,
|
|
};
|
|
|
|
struct tw5864_input {
|
|
int nr; /* input number */
|
|
struct tw5864_dev *root;
|
|
struct mutex lock; /* used for vidq and vdev */
|
|
spinlock_t slock; /* used for sync between ISR, tasklet & V4L2 API */
|
|
struct video_device vdev;
|
|
struct v4l2_ctrl_handler hdl;
|
|
struct vb2_queue vidq;
|
|
struct list_head active;
|
|
enum resolution resolution;
|
|
unsigned int width, height;
|
|
unsigned int frame_seqno;
|
|
unsigned int frame_gop_seqno;
|
|
unsigned int h264_idr_pic_id;
|
|
int enabled;
|
|
enum tw5864_vid_std std;
|
|
v4l2_std_id v4l2_std;
|
|
int tail_nb_bits;
|
|
u8 tail;
|
|
u8 *buf_cur_ptr;
|
|
int buf_cur_space_left;
|
|
|
|
u32 reg_interlacing;
|
|
u32 reg_vlc;
|
|
u32 reg_dsp_codec;
|
|
u32 reg_dsp;
|
|
u32 reg_emu;
|
|
u32 reg_dsp_qp;
|
|
u32 reg_dsp_ref_mvp_lambda;
|
|
u32 reg_dsp_i4x4_weight;
|
|
u32 buf_id;
|
|
|
|
struct tw5864_buf *vb;
|
|
|
|
struct v4l2_ctrl *md_threshold_grid_ctrl;
|
|
u16 md_threshold_grid_values[12 * 16];
|
|
int qp;
|
|
int gop;
|
|
|
|
/*
|
|
* In (1/MAX_FPS) units.
|
|
* For max FPS (default), set to 1.
|
|
* For 1 FPS, set to e.g. 32.
|
|
*/
|
|
int frame_interval;
|
|
unsigned long new_frame_deadline;
|
|
};
|
|
|
|
struct tw5864_h264_frame {
|
|
struct tw5864_dma_buf vlc;
|
|
struct tw5864_dma_buf mv;
|
|
int vlc_len;
|
|
u32 checksum;
|
|
struct tw5864_input *input;
|
|
u64 timestamp;
|
|
unsigned int seqno;
|
|
unsigned int gop_seqno;
|
|
};
|
|
|
|
/* global device status */
|
|
struct tw5864_dev {
|
|
spinlock_t slock; /* used for sync between ISR, tasklet & V4L2 API */
|
|
struct v4l2_device v4l2_dev;
|
|
struct tw5864_input inputs[TW5864_INPUTS];
|
|
#define H264_BUF_CNT 4
|
|
struct tw5864_h264_frame h264_buf[H264_BUF_CNT];
|
|
int h264_buf_r_index;
|
|
int h264_buf_w_index;
|
|
|
|
struct tasklet_struct tasklet;
|
|
|
|
int encoder_busy;
|
|
/* Input number to check next for ready raw picture (in RR fashion) */
|
|
int next_input;
|
|
|
|
/* pci i/o */
|
|
char name[64];
|
|
struct pci_dev *pci;
|
|
void __iomem *mmio;
|
|
u32 irqmask;
|
|
};
|
|
|
|
#define tw_readl(reg) readl(dev->mmio + reg)
|
|
#define tw_mask_readl(reg, mask) \
|
|
(tw_readl(reg) & (mask))
|
|
#define tw_mask_shift_readl(reg, mask, shift) \
|
|
(tw_mask_readl((reg), ((mask) << (shift))) >> (shift))
|
|
|
|
#define tw_writel(reg, value) writel((value), dev->mmio + reg)
|
|
#define tw_mask_writel(reg, mask, value) \
|
|
tw_writel(reg, (tw_readl(reg) & ~(mask)) | ((value) & (mask)))
|
|
#define tw_mask_shift_writel(reg, mask, shift, value) \
|
|
tw_mask_writel((reg), ((mask) << (shift)), ((value) << (shift)))
|
|
|
|
#define tw_setl(reg, bit) tw_writel((reg), tw_readl(reg) | (bit))
|
|
#define tw_clearl(reg, bit) tw_writel((reg), tw_readl(reg) & ~(bit))
|
|
|
|
u8 tw5864_indir_readb(struct tw5864_dev *dev, u16 addr);
|
|
#define tw_indir_readb(addr) tw5864_indir_readb(dev, addr)
|
|
void tw5864_indir_writeb(struct tw5864_dev *dev, u16 addr, u8 data);
|
|
#define tw_indir_writeb(addr, data) tw5864_indir_writeb(dev, addr, data)
|
|
|
|
void tw5864_irqmask_apply(struct tw5864_dev *dev);
|
|
int tw5864_video_init(struct tw5864_dev *dev, int *video_nr);
|
|
void tw5864_video_fini(struct tw5864_dev *dev);
|
|
void tw5864_prepare_frame_headers(struct tw5864_input *input);
|
|
void tw5864_h264_put_stream_header(u8 **buf, size_t *space_left, int qp,
|
|
int width, int height);
|
|
void tw5864_h264_put_slice_header(u8 **buf, size_t *space_left,
|
|
unsigned int idr_pic_id,
|
|
unsigned int frame_gop_seqno,
|
|
int *tail_nb_bits, u8 *tail);
|
|
void tw5864_request_encoded_frame(struct tw5864_input *input);
|