162306a36Sopenharmony_ci/* SPDX-License-Identifier: GPL-2.0 */
262306a36Sopenharmony_ci#ifndef _DVB_USB_CXUSB_H_
362306a36Sopenharmony_ci#define _DVB_USB_CXUSB_H_
462306a36Sopenharmony_ci
562306a36Sopenharmony_ci#include <linux/completion.h>
662306a36Sopenharmony_ci#include <linux/i2c.h>
762306a36Sopenharmony_ci#include <linux/list.h>
862306a36Sopenharmony_ci#include <linux/mutex.h>
962306a36Sopenharmony_ci#include <linux/usb.h>
1062306a36Sopenharmony_ci#include <linux/workqueue.h>
1162306a36Sopenharmony_ci#include <media/v4l2-common.h>
1262306a36Sopenharmony_ci#include <media/v4l2-dev.h>
1362306a36Sopenharmony_ci#include <media/v4l2-device.h>
1462306a36Sopenharmony_ci#include <media/videobuf2-core.h>
1562306a36Sopenharmony_ci#include <media/videobuf2-v4l2.h>
1662306a36Sopenharmony_ci
1762306a36Sopenharmony_ci#define DVB_USB_LOG_PREFIX "cxusb"
1862306a36Sopenharmony_ci#include "dvb-usb.h"
1962306a36Sopenharmony_ci
2062306a36Sopenharmony_ci#define CXUSB_VIDEO_URBS (5)
2162306a36Sopenharmony_ci#define CXUSB_VIDEO_URB_MAX_SIZE (512 * 1024)
2262306a36Sopenharmony_ci
2362306a36Sopenharmony_ci#define CXUSB_VIDEO_PKT_SIZE 3030
2462306a36Sopenharmony_ci#define CXUSB_VIDEO_MAX_FRAME_PKTS 346
2562306a36Sopenharmony_ci#define CXUSB_VIDEO_MAX_FRAME_SIZE (CXUSB_VIDEO_MAX_FRAME_PKTS * \
2662306a36Sopenharmony_ci					CXUSB_VIDEO_PKT_SIZE)
2762306a36Sopenharmony_ci
2862306a36Sopenharmony_ci/* usb commands - some of it are guesses, don't have a reference yet */
2962306a36Sopenharmony_ci#define CMD_BLUEBIRD_GPIO_RW 0x05
3062306a36Sopenharmony_ci
3162306a36Sopenharmony_ci#define CMD_I2C_WRITE     0x08
3262306a36Sopenharmony_ci#define CMD_I2C_READ      0x09
3362306a36Sopenharmony_ci
3462306a36Sopenharmony_ci#define CMD_GPIO_READ     0x0d
3562306a36Sopenharmony_ci#define CMD_GPIO_WRITE    0x0e
3662306a36Sopenharmony_ci#define     GPIO_TUNER         0x02
3762306a36Sopenharmony_ci
3862306a36Sopenharmony_ci#define CMD_POWER_OFF     0xdc
3962306a36Sopenharmony_ci#define CMD_POWER_ON      0xde
4062306a36Sopenharmony_ci
4162306a36Sopenharmony_ci#define CMD_STREAMING_ON  0x36
4262306a36Sopenharmony_ci#define CMD_STREAMING_OFF 0x37
4362306a36Sopenharmony_ci
4462306a36Sopenharmony_ci#define CMD_AVER_STREAM_ON  0x18
4562306a36Sopenharmony_ci#define CMD_AVER_STREAM_OFF 0x19
4662306a36Sopenharmony_ci
4762306a36Sopenharmony_ci#define CMD_GET_IR_CODE   0x47
4862306a36Sopenharmony_ci
4962306a36Sopenharmony_ci#define CMD_ANALOG        0x50
5062306a36Sopenharmony_ci#define CMD_DIGITAL       0x51
5162306a36Sopenharmony_ci
5262306a36Sopenharmony_ci#define CXUSB_BT656_PREAMBLE ((const u8 *)"\xff\x00\x00")
5362306a36Sopenharmony_ci
5462306a36Sopenharmony_ci#define CXUSB_BT656_FIELD_MASK BIT(6)
5562306a36Sopenharmony_ci#define CXUSB_BT656_FIELD_1 0
5662306a36Sopenharmony_ci#define CXUSB_BT656_FIELD_2 BIT(6)
5762306a36Sopenharmony_ci
5862306a36Sopenharmony_ci#define CXUSB_BT656_VBI_MASK BIT(5)
5962306a36Sopenharmony_ci#define CXUSB_BT656_VBI_ON BIT(5)
6062306a36Sopenharmony_ci#define CXUSB_BT656_VBI_OFF 0
6162306a36Sopenharmony_ci
6262306a36Sopenharmony_ci#define CXUSB_BT656_SEAV_MASK BIT(4)
6362306a36Sopenharmony_ci#define CXUSB_BT656_SEAV_EAV BIT(4)
6462306a36Sopenharmony_ci#define CXUSB_BT656_SEAV_SAV 0
6562306a36Sopenharmony_ci
6662306a36Sopenharmony_ci/* Max transfer size done by I2C transfer functions */
6762306a36Sopenharmony_ci#define MAX_XFER_SIZE  80
6862306a36Sopenharmony_ci
6962306a36Sopenharmony_cistruct cxusb_state {
7062306a36Sopenharmony_ci	u8 gpio_write_state[3];
7162306a36Sopenharmony_ci	bool gpio_write_refresh[3];
7262306a36Sopenharmony_ci	struct i2c_client *i2c_client_demod;
7362306a36Sopenharmony_ci	struct i2c_client *i2c_client_tuner;
7462306a36Sopenharmony_ci
7562306a36Sopenharmony_ci	unsigned char data[MAX_XFER_SIZE];
7662306a36Sopenharmony_ci
7762306a36Sopenharmony_ci	struct mutex stream_mutex;
7862306a36Sopenharmony_ci	u8 last_lock;
7962306a36Sopenharmony_ci	int (*fe_read_status)(struct dvb_frontend *fe,
8062306a36Sopenharmony_ci			      enum fe_status *status);
8162306a36Sopenharmony_ci};
8262306a36Sopenharmony_ci
8362306a36Sopenharmony_cienum cxusb_open_type {
8462306a36Sopenharmony_ci	CXUSB_OPEN_INIT,
8562306a36Sopenharmony_ci	CXUSB_OPEN_NONE,
8662306a36Sopenharmony_ci	CXUSB_OPEN_ANALOG,
8762306a36Sopenharmony_ci	CXUSB_OPEN_DIGITAL
8862306a36Sopenharmony_ci};
8962306a36Sopenharmony_ci
9062306a36Sopenharmony_cistruct cxusb_medion_auxbuf {
9162306a36Sopenharmony_ci	u8 *buf;
9262306a36Sopenharmony_ci	unsigned int len;
9362306a36Sopenharmony_ci	unsigned int paylen;
9462306a36Sopenharmony_ci};
9562306a36Sopenharmony_ci
9662306a36Sopenharmony_cienum cxusb_bt656_mode {
9762306a36Sopenharmony_ci	NEW_FRAME, FIRST_FIELD, SECOND_FIELD
9862306a36Sopenharmony_ci};
9962306a36Sopenharmony_ci
10062306a36Sopenharmony_cienum cxusb_bt656_fmode {
10162306a36Sopenharmony_ci	START_SEARCH, LINE_SAMPLES, VBI_SAMPLES
10262306a36Sopenharmony_ci};
10362306a36Sopenharmony_ci
10462306a36Sopenharmony_cistruct cxusb_bt656_params {
10562306a36Sopenharmony_ci	enum cxusb_bt656_mode mode;
10662306a36Sopenharmony_ci	enum cxusb_bt656_fmode fmode;
10762306a36Sopenharmony_ci	unsigned int pos;
10862306a36Sopenharmony_ci	unsigned int line;
10962306a36Sopenharmony_ci	unsigned int linesamples;
11062306a36Sopenharmony_ci	u8 *buf;
11162306a36Sopenharmony_ci};
11262306a36Sopenharmony_ci
11362306a36Sopenharmony_cistruct cxusb_medion_dev {
11462306a36Sopenharmony_ci	/* has to be the first one */
11562306a36Sopenharmony_ci	struct cxusb_state state;
11662306a36Sopenharmony_ci
11762306a36Sopenharmony_ci	struct dvb_usb_device *dvbdev;
11862306a36Sopenharmony_ci
11962306a36Sopenharmony_ci	enum cxusb_open_type open_type;
12062306a36Sopenharmony_ci	unsigned int open_ctr;
12162306a36Sopenharmony_ci	struct mutex open_lock;
12262306a36Sopenharmony_ci
12362306a36Sopenharmony_ci#ifdef CONFIG_DVB_USB_CXUSB_ANALOG
12462306a36Sopenharmony_ci	struct v4l2_device v4l2dev;
12562306a36Sopenharmony_ci	struct v4l2_subdev *cx25840;
12662306a36Sopenharmony_ci	struct v4l2_subdev *tuner;
12762306a36Sopenharmony_ci	struct v4l2_subdev *tda9887;
12862306a36Sopenharmony_ci	struct video_device *videodev, *radiodev;
12962306a36Sopenharmony_ci	struct mutex dev_lock;
13062306a36Sopenharmony_ci
13162306a36Sopenharmony_ci	struct vb2_queue videoqueue;
13262306a36Sopenharmony_ci	u32 input;
13362306a36Sopenharmony_ci	bool stop_streaming;
13462306a36Sopenharmony_ci	u32 width, height;
13562306a36Sopenharmony_ci	u32 field_order;
13662306a36Sopenharmony_ci	struct cxusb_medion_auxbuf auxbuf;
13762306a36Sopenharmony_ci	v4l2_std_id norm;
13862306a36Sopenharmony_ci
13962306a36Sopenharmony_ci	struct urb *streamurbs[CXUSB_VIDEO_URBS];
14062306a36Sopenharmony_ci	unsigned long urbcomplete;
14162306a36Sopenharmony_ci	struct work_struct urbwork;
14262306a36Sopenharmony_ci	unsigned int nexturb;
14362306a36Sopenharmony_ci
14462306a36Sopenharmony_ci	struct cxusb_bt656_params bt656;
14562306a36Sopenharmony_ci	struct cxusb_medion_vbuffer *vbuf;
14662306a36Sopenharmony_ci	__u32 vbuf_sequence;
14762306a36Sopenharmony_ci
14862306a36Sopenharmony_ci	struct list_head buflist;
14962306a36Sopenharmony_ci
15062306a36Sopenharmony_ci	struct completion v4l2_release;
15162306a36Sopenharmony_ci#endif
15262306a36Sopenharmony_ci};
15362306a36Sopenharmony_ci
15462306a36Sopenharmony_cistruct cxusb_medion_vbuffer {
15562306a36Sopenharmony_ci	struct vb2_v4l2_buffer vb2;
15662306a36Sopenharmony_ci	struct list_head list;
15762306a36Sopenharmony_ci};
15862306a36Sopenharmony_ci
15962306a36Sopenharmony_ci/* defines for "debug" module parameter */
16062306a36Sopenharmony_ci#define CXUSB_DBG_RC BIT(0)
16162306a36Sopenharmony_ci#define CXUSB_DBG_I2C BIT(1)
16262306a36Sopenharmony_ci#define CXUSB_DBG_MISC BIT(2)
16362306a36Sopenharmony_ci#define CXUSB_DBG_BT656 BIT(3)
16462306a36Sopenharmony_ci#define CXUSB_DBG_URB BIT(4)
16562306a36Sopenharmony_ci#define CXUSB_DBG_OPS BIT(5)
16662306a36Sopenharmony_ci#define CXUSB_DBG_AUXB BIT(6)
16762306a36Sopenharmony_ci
16862306a36Sopenharmony_ciextern int dvb_usb_cxusb_debug;
16962306a36Sopenharmony_ci
17062306a36Sopenharmony_ci#define cxusb_vprintk(dvbdev, lvl, ...) do {				\
17162306a36Sopenharmony_ci		struct cxusb_medion_dev *_cxdev = (dvbdev)->priv;	\
17262306a36Sopenharmony_ci		if (dvb_usb_cxusb_debug & CXUSB_DBG_##lvl)		\
17362306a36Sopenharmony_ci			v4l2_printk(KERN_DEBUG,			\
17462306a36Sopenharmony_ci				    &_cxdev->v4l2dev, __VA_ARGS__);	\
17562306a36Sopenharmony_ci	} while (0)
17662306a36Sopenharmony_ci
17762306a36Sopenharmony_ciint cxusb_ctrl_msg(struct dvb_usb_device *d,
17862306a36Sopenharmony_ci		   u8 cmd, const u8 *wbuf, int wlen, u8 *rbuf, int rlen);
17962306a36Sopenharmony_ci
18062306a36Sopenharmony_ci#ifdef CONFIG_DVB_USB_CXUSB_ANALOG
18162306a36Sopenharmony_ciint cxusb_medion_analog_init(struct dvb_usb_device *dvbdev);
18262306a36Sopenharmony_ciint cxusb_medion_register_analog(struct dvb_usb_device *dvbdev);
18362306a36Sopenharmony_civoid cxusb_medion_unregister_analog(struct dvb_usb_device *dvbdev);
18462306a36Sopenharmony_ci#else
18562306a36Sopenharmony_cistatic inline int cxusb_medion_analog_init(struct dvb_usb_device *dvbdev)
18662306a36Sopenharmony_ci{
18762306a36Sopenharmony_ci	return -EINVAL;
18862306a36Sopenharmony_ci}
18962306a36Sopenharmony_ci
19062306a36Sopenharmony_cistatic inline int cxusb_medion_register_analog(struct dvb_usb_device *dvbdev)
19162306a36Sopenharmony_ci{
19262306a36Sopenharmony_ci	return 0;
19362306a36Sopenharmony_ci}
19462306a36Sopenharmony_ci
19562306a36Sopenharmony_cistatic inline void cxusb_medion_unregister_analog(struct dvb_usb_device *dvbdev)
19662306a36Sopenharmony_ci{
19762306a36Sopenharmony_ci}
19862306a36Sopenharmony_ci#endif
19962306a36Sopenharmony_ci
20062306a36Sopenharmony_ciint cxusb_medion_get(struct dvb_usb_device *dvbdev,
20162306a36Sopenharmony_ci		     enum cxusb_open_type open_type);
20262306a36Sopenharmony_civoid cxusb_medion_put(struct dvb_usb_device *dvbdev);
20362306a36Sopenharmony_ci
20462306a36Sopenharmony_ci#endif
205