/* SPDX-License-Identifier: GPL-2.0 */ #ifndef _DVB_USB_CXUSB_H_ #define _DVB_USB_CXUSB_H_ #include <linux/completion.h> #include <linux/i2c.h> #include <linux/list.h> #include <linux/mutex.h> #include <linux/usb.h> #include <linux/workqueue.h> #include <media/v4l2-common.h> #include <media/v4l2-dev.h> #include <media/v4l2-device.h> #include <media/videobuf2-core.h> #include <media/videobuf2-v4l2.h> #define DVB_USB_LOG_PREFIX "cxusb" #include "dvb-usb.h" #define CXUSB_VIDEO_URBS (5) #define CXUSB_VIDEO_URB_MAX_SIZE (512 * 1024) #define CXUSB_VIDEO_PKT_SIZE 3030 #define CXUSB_VIDEO_MAX_FRAME_PKTS 346 #define CXUSB_VIDEO_MAX_FRAME_SIZE (CXUSB_VIDEO_MAX_FRAME_PKTS * \ CXUSB_VIDEO_PKT_SIZE) /* usb commands - some of it are guesses, don't have a reference yet */ #define CMD_BLUEBIRD_GPIO_RW 0x05 #define CMD_I2C_WRITE 0x08 #define CMD_I2C_READ 0x09 #define CMD_GPIO_READ 0x0d #define CMD_GPIO_WRITE 0x0e #define GPIO_TUNER 0x02 #define CMD_POWER_OFF 0xdc #define CMD_POWER_ON 0xde #define CMD_STREAMING_ON 0x36 #define CMD_STREAMING_OFF 0x37 #define CMD_AVER_STREAM_ON 0x18 #define CMD_AVER_STREAM_OFF 0x19 #define CMD_GET_IR_CODE 0x47 #define CMD_ANALOG 0x50 #define CMD_DIGITAL 0x51 #define CXUSB_BT656_PREAMBLE ((const u8 *)"\xff\x00\x00") #define CXUSB_BT656_FIELD_MASK BIT(6) #define CXUSB_BT656_FIELD_1 0 #define CXUSB_BT656_FIELD_2 BIT(6) #define CXUSB_BT656_VBI_MASK BIT(5) #define CXUSB_BT656_VBI_ON BIT(5) #define CXUSB_BT656_VBI_OFF 0 #define CXUSB_BT656_SEAV_MASK BIT(4) #define CXUSB_BT656_SEAV_EAV BIT(4) #define CXUSB_BT656_SEAV_SAV 0 /* Max transfer size done by I2C transfer functions */ #define MAX_XFER_SIZE 80 struct cxusb_state { u8 gpio_write_state[3]; bool gpio_write_refresh[3]; struct i2c_client *i2c_client_demod; struct i2c_client *i2c_client_tuner; unsigned char data[MAX_XFER_SIZE]; struct mutex stream_mutex; u8 last_lock; int (*fe_read_status)(struct dvb_frontend *fe, enum fe_status *status); }; enum cxusb_open_type { CXUSB_OPEN_INIT, CXUSB_OPEN_NONE, CXUSB_OPEN_ANALOG, CXUSB_OPEN_DIGITAL }; struct cxusb_medion_auxbuf { u8 *buf; unsigned int len; unsigned int paylen; }; enum cxusb_bt656_mode { NEW_FRAME, FIRST_FIELD, SECOND_FIELD }; enum cxusb_bt656_fmode { START_SEARCH, LINE_SAMPLES, VBI_SAMPLES }; struct cxusb_bt656_params { enum cxusb_bt656_mode mode; enum cxusb_bt656_fmode fmode; unsigned int pos; unsigned int line; unsigned int linesamples; u8 *buf; }; struct cxusb_medion_dev { /* has to be the first one */ struct cxusb_state state; struct dvb_usb_device *dvbdev; enum cxusb_open_type open_type; unsigned int open_ctr; struct mutex open_lock; #ifdef CONFIG_DVB_USB_CXUSB_ANALOG struct v4l2_device v4l2dev; struct v4l2_subdev *cx25840; struct v4l2_subdev *tuner; struct v4l2_subdev *tda9887; struct video_device *videodev, *radiodev; struct mutex dev_lock; struct vb2_queue videoqueue; u32 input; bool stop_streaming; u32 width, height; u32 field_order; struct cxusb_medion_auxbuf auxbuf; v4l2_std_id norm; struct urb *streamurbs[CXUSB_VIDEO_URBS]; unsigned long urbcomplete; struct work_struct urbwork; unsigned int nexturb; struct cxusb_bt656_params bt656; struct cxusb_medion_vbuffer *vbuf; __u32 vbuf_sequence; struct list_head buflist; struct completion v4l2_release; #endif }; struct cxusb_medion_vbuffer { struct vb2_v4l2_buffer vb2; struct list_head list; }; /* defines for "debug" module parameter */ #define CXUSB_DBG_RC BIT(0) #define CXUSB_DBG_I2C BIT(1) #define CXUSB_DBG_MISC BIT(2) #define CXUSB_DBG_BT656 BIT(3) #define CXUSB_DBG_URB BIT(4) #define CXUSB_DBG_OPS BIT(5) #define CXUSB_DBG_AUXB BIT(6) extern int dvb_usb_cxusb_debug; #define cxusb_vprintk(dvbdev, lvl, ...) do { \ struct cxusb_medion_dev *_cxdev = (dvbdev)->priv; \ if (dvb_usb_cxusb_debug & CXUSB_DBG_##lvl) \ v4l2_printk(KERN_DEBUG, \ &_cxdev->v4l2dev, __VA_ARGS__); \ } while (0) int cxusb_ctrl_msg(struct dvb_usb_device *d, u8 cmd, const u8 *wbuf, int wlen, u8 *rbuf, int rlen); #ifdef CONFIG_DVB_USB_CXUSB_ANALOG int cxusb_medion_analog_init(struct dvb_usb_device *dvbdev); int cxusb_medion_register_analog(struct dvb_usb_device *dvbdev); void cxusb_medion_unregister_analog(struct dvb_usb_device *dvbdev); #else static inline int cxusb_medion_analog_init(struct dvb_usb_device *dvbdev) { return -EINVAL; } static inline int cxusb_medion_register_analog(struct dvb_usb_device *dvbdev) { return 0; } static inline void cxusb_medion_unregister_analog(struct dvb_usb_device *dvbdev) { } #endif int cxusb_medion_get(struct dvb_usb_device *dvbdev, enum cxusb_open_type open_type); void cxusb_medion_put(struct dvb_usb_device *dvbdev); #endif