1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-or-later
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun bt866 - BT866 Digital Video Encoder (Rockwell Part)
4*4882a593Smuzhiyun
5*4882a593Smuzhiyun Copyright (C) 1999 Mike Bernson <mike@mlb.org>
6*4882a593Smuzhiyun Copyright (C) 1998 Dave Perks <dperks@ibm.net>
7*4882a593Smuzhiyun
8*4882a593Smuzhiyun Modifications for LML33/DC10plus unified driver
9*4882a593Smuzhiyun Copyright (C) 2000 Serguei Miridonov <mirsev@cicese.mx>
10*4882a593Smuzhiyun
11*4882a593Smuzhiyun This code was modify/ported from the saa7111 driver written
12*4882a593Smuzhiyun by Dave Perks.
13*4882a593Smuzhiyun
14*4882a593Smuzhiyun This code was adapted for the bt866 by Christer Weinigel and ported
15*4882a593Smuzhiyun to 2.6 by Martin Samuelsson.
16*4882a593Smuzhiyun
17*4882a593Smuzhiyun */
18*4882a593Smuzhiyun
19*4882a593Smuzhiyun #include <linux/module.h>
20*4882a593Smuzhiyun #include <linux/types.h>
21*4882a593Smuzhiyun #include <linux/slab.h>
22*4882a593Smuzhiyun #include <linux/ioctl.h>
23*4882a593Smuzhiyun #include <linux/uaccess.h>
24*4882a593Smuzhiyun #include <linux/i2c.h>
25*4882a593Smuzhiyun #include <linux/videodev2.h>
26*4882a593Smuzhiyun #include <media/v4l2-device.h>
27*4882a593Smuzhiyun
28*4882a593Smuzhiyun MODULE_DESCRIPTION("Brooktree-866 video encoder driver");
29*4882a593Smuzhiyun MODULE_AUTHOR("Mike Bernson & Dave Perks");
30*4882a593Smuzhiyun MODULE_LICENSE("GPL");
31*4882a593Smuzhiyun
32*4882a593Smuzhiyun static int debug;
33*4882a593Smuzhiyun module_param(debug, int, 0);
34*4882a593Smuzhiyun MODULE_PARM_DESC(debug, "Debug level (0-1)");
35*4882a593Smuzhiyun
36*4882a593Smuzhiyun
37*4882a593Smuzhiyun /* ----------------------------------------------------------------------- */
38*4882a593Smuzhiyun
39*4882a593Smuzhiyun struct bt866 {
40*4882a593Smuzhiyun struct v4l2_subdev sd;
41*4882a593Smuzhiyun u8 reg[256];
42*4882a593Smuzhiyun };
43*4882a593Smuzhiyun
to_bt866(struct v4l2_subdev * sd)44*4882a593Smuzhiyun static inline struct bt866 *to_bt866(struct v4l2_subdev *sd)
45*4882a593Smuzhiyun {
46*4882a593Smuzhiyun return container_of(sd, struct bt866, sd);
47*4882a593Smuzhiyun }
48*4882a593Smuzhiyun
bt866_write(struct bt866 * encoder,u8 subaddr,u8 data)49*4882a593Smuzhiyun static int bt866_write(struct bt866 *encoder, u8 subaddr, u8 data)
50*4882a593Smuzhiyun {
51*4882a593Smuzhiyun struct i2c_client *client = v4l2_get_subdevdata(&encoder->sd);
52*4882a593Smuzhiyun u8 buffer[2];
53*4882a593Smuzhiyun int err;
54*4882a593Smuzhiyun
55*4882a593Smuzhiyun buffer[0] = subaddr;
56*4882a593Smuzhiyun buffer[1] = data;
57*4882a593Smuzhiyun
58*4882a593Smuzhiyun encoder->reg[subaddr] = data;
59*4882a593Smuzhiyun
60*4882a593Smuzhiyun v4l_dbg(1, debug, client, "write 0x%02x = 0x%02x\n", subaddr, data);
61*4882a593Smuzhiyun
62*4882a593Smuzhiyun for (err = 0; err < 3;) {
63*4882a593Smuzhiyun if (i2c_master_send(client, buffer, 2) == 2)
64*4882a593Smuzhiyun break;
65*4882a593Smuzhiyun err++;
66*4882a593Smuzhiyun v4l_warn(client, "error #%d writing to 0x%02x\n",
67*4882a593Smuzhiyun err, subaddr);
68*4882a593Smuzhiyun schedule_timeout_interruptible(msecs_to_jiffies(100));
69*4882a593Smuzhiyun }
70*4882a593Smuzhiyun if (err == 3) {
71*4882a593Smuzhiyun v4l_warn(client, "giving up\n");
72*4882a593Smuzhiyun return -1;
73*4882a593Smuzhiyun }
74*4882a593Smuzhiyun
75*4882a593Smuzhiyun return 0;
76*4882a593Smuzhiyun }
77*4882a593Smuzhiyun
bt866_s_std_output(struct v4l2_subdev * sd,v4l2_std_id std)78*4882a593Smuzhiyun static int bt866_s_std_output(struct v4l2_subdev *sd, v4l2_std_id std)
79*4882a593Smuzhiyun {
80*4882a593Smuzhiyun v4l2_dbg(1, debug, sd, "set norm %llx\n", (unsigned long long)std);
81*4882a593Smuzhiyun
82*4882a593Smuzhiyun /* Only PAL supported by this driver at the moment! */
83*4882a593Smuzhiyun if (!(std & V4L2_STD_NTSC))
84*4882a593Smuzhiyun return -EINVAL;
85*4882a593Smuzhiyun return 0;
86*4882a593Smuzhiyun }
87*4882a593Smuzhiyun
bt866_s_routing(struct v4l2_subdev * sd,u32 input,u32 output,u32 config)88*4882a593Smuzhiyun static int bt866_s_routing(struct v4l2_subdev *sd,
89*4882a593Smuzhiyun u32 input, u32 output, u32 config)
90*4882a593Smuzhiyun {
91*4882a593Smuzhiyun static const __u8 init[] = {
92*4882a593Smuzhiyun 0xc8, 0xcc, /* CRSCALE */
93*4882a593Smuzhiyun 0xca, 0x91, /* CBSCALE */
94*4882a593Smuzhiyun 0xcc, 0x24, /* YC16 | OSDNUM */
95*4882a593Smuzhiyun 0xda, 0x00, /* */
96*4882a593Smuzhiyun 0xdc, 0x24, /* SETMODE | PAL */
97*4882a593Smuzhiyun 0xde, 0x02, /* EACTIVE */
98*4882a593Smuzhiyun
99*4882a593Smuzhiyun /* overlay colors */
100*4882a593Smuzhiyun 0x70, 0xEB, 0x90, 0x80, 0xB0, 0x80, /* white */
101*4882a593Smuzhiyun 0x72, 0xA2, 0x92, 0x8E, 0xB2, 0x2C, /* yellow */
102*4882a593Smuzhiyun 0x74, 0x83, 0x94, 0x2C, 0xB4, 0x9C, /* cyan */
103*4882a593Smuzhiyun 0x76, 0x70, 0x96, 0x3A, 0xB6, 0x48, /* green */
104*4882a593Smuzhiyun 0x78, 0x54, 0x98, 0xC6, 0xB8, 0xB8, /* magenta */
105*4882a593Smuzhiyun 0x7A, 0x41, 0x9A, 0xD4, 0xBA, 0x64, /* red */
106*4882a593Smuzhiyun 0x7C, 0x23, 0x9C, 0x72, 0xBC, 0xD4, /* blue */
107*4882a593Smuzhiyun 0x7E, 0x10, 0x9E, 0x80, 0xBE, 0x80, /* black */
108*4882a593Smuzhiyun
109*4882a593Smuzhiyun 0x60, 0xEB, 0x80, 0x80, 0xc0, 0x80, /* white */
110*4882a593Smuzhiyun 0x62, 0xA2, 0x82, 0x8E, 0xc2, 0x2C, /* yellow */
111*4882a593Smuzhiyun 0x64, 0x83, 0x84, 0x2C, 0xc4, 0x9C, /* cyan */
112*4882a593Smuzhiyun 0x66, 0x70, 0x86, 0x3A, 0xc6, 0x48, /* green */
113*4882a593Smuzhiyun 0x68, 0x54, 0x88, 0xC6, 0xc8, 0xB8, /* magenta */
114*4882a593Smuzhiyun 0x6A, 0x41, 0x8A, 0xD4, 0xcA, 0x64, /* red */
115*4882a593Smuzhiyun 0x6C, 0x23, 0x8C, 0x72, 0xcC, 0xD4, /* blue */
116*4882a593Smuzhiyun 0x6E, 0x10, 0x8E, 0x80, 0xcE, 0x80, /* black */
117*4882a593Smuzhiyun };
118*4882a593Smuzhiyun struct bt866 *encoder = to_bt866(sd);
119*4882a593Smuzhiyun u8 val;
120*4882a593Smuzhiyun int i;
121*4882a593Smuzhiyun
122*4882a593Smuzhiyun for (i = 0; i < ARRAY_SIZE(init) / 2; i += 2)
123*4882a593Smuzhiyun bt866_write(encoder, init[i], init[i+1]);
124*4882a593Smuzhiyun
125*4882a593Smuzhiyun val = encoder->reg[0xdc];
126*4882a593Smuzhiyun
127*4882a593Smuzhiyun if (input == 0)
128*4882a593Smuzhiyun val |= 0x40; /* CBSWAP */
129*4882a593Smuzhiyun else
130*4882a593Smuzhiyun val &= ~0x40; /* !CBSWAP */
131*4882a593Smuzhiyun
132*4882a593Smuzhiyun bt866_write(encoder, 0xdc, val);
133*4882a593Smuzhiyun
134*4882a593Smuzhiyun val = encoder->reg[0xcc];
135*4882a593Smuzhiyun if (input == 2)
136*4882a593Smuzhiyun val |= 0x01; /* OSDBAR */
137*4882a593Smuzhiyun else
138*4882a593Smuzhiyun val &= ~0x01; /* !OSDBAR */
139*4882a593Smuzhiyun bt866_write(encoder, 0xcc, val);
140*4882a593Smuzhiyun
141*4882a593Smuzhiyun v4l2_dbg(1, debug, sd, "set input %d\n", input);
142*4882a593Smuzhiyun
143*4882a593Smuzhiyun switch (input) {
144*4882a593Smuzhiyun case 0:
145*4882a593Smuzhiyun case 1:
146*4882a593Smuzhiyun case 2:
147*4882a593Smuzhiyun break;
148*4882a593Smuzhiyun default:
149*4882a593Smuzhiyun return -EINVAL;
150*4882a593Smuzhiyun }
151*4882a593Smuzhiyun return 0;
152*4882a593Smuzhiyun }
153*4882a593Smuzhiyun
154*4882a593Smuzhiyun #if 0
155*4882a593Smuzhiyun /* Code to setup square pixels, might be of some use in the future,
156*4882a593Smuzhiyun but is currently unused. */
157*4882a593Smuzhiyun val = encoder->reg[0xdc];
158*4882a593Smuzhiyun if (*iarg)
159*4882a593Smuzhiyun val |= 1; /* SQUARE */
160*4882a593Smuzhiyun else
161*4882a593Smuzhiyun val &= ~1; /* !SQUARE */
162*4882a593Smuzhiyun bt866_write(client, 0xdc, val);
163*4882a593Smuzhiyun #endif
164*4882a593Smuzhiyun
165*4882a593Smuzhiyun /* ----------------------------------------------------------------------- */
166*4882a593Smuzhiyun
167*4882a593Smuzhiyun static const struct v4l2_subdev_video_ops bt866_video_ops = {
168*4882a593Smuzhiyun .s_std_output = bt866_s_std_output,
169*4882a593Smuzhiyun .s_routing = bt866_s_routing,
170*4882a593Smuzhiyun };
171*4882a593Smuzhiyun
172*4882a593Smuzhiyun static const struct v4l2_subdev_ops bt866_ops = {
173*4882a593Smuzhiyun .video = &bt866_video_ops,
174*4882a593Smuzhiyun };
175*4882a593Smuzhiyun
bt866_probe(struct i2c_client * client,const struct i2c_device_id * id)176*4882a593Smuzhiyun static int bt866_probe(struct i2c_client *client,
177*4882a593Smuzhiyun const struct i2c_device_id *id)
178*4882a593Smuzhiyun {
179*4882a593Smuzhiyun struct bt866 *encoder;
180*4882a593Smuzhiyun struct v4l2_subdev *sd;
181*4882a593Smuzhiyun
182*4882a593Smuzhiyun v4l_info(client, "chip found @ 0x%x (%s)\n",
183*4882a593Smuzhiyun client->addr << 1, client->adapter->name);
184*4882a593Smuzhiyun
185*4882a593Smuzhiyun encoder = devm_kzalloc(&client->dev, sizeof(*encoder), GFP_KERNEL);
186*4882a593Smuzhiyun if (encoder == NULL)
187*4882a593Smuzhiyun return -ENOMEM;
188*4882a593Smuzhiyun sd = &encoder->sd;
189*4882a593Smuzhiyun v4l2_i2c_subdev_init(sd, client, &bt866_ops);
190*4882a593Smuzhiyun return 0;
191*4882a593Smuzhiyun }
192*4882a593Smuzhiyun
bt866_remove(struct i2c_client * client)193*4882a593Smuzhiyun static int bt866_remove(struct i2c_client *client)
194*4882a593Smuzhiyun {
195*4882a593Smuzhiyun struct v4l2_subdev *sd = i2c_get_clientdata(client);
196*4882a593Smuzhiyun
197*4882a593Smuzhiyun v4l2_device_unregister_subdev(sd);
198*4882a593Smuzhiyun return 0;
199*4882a593Smuzhiyun }
200*4882a593Smuzhiyun
201*4882a593Smuzhiyun static const struct i2c_device_id bt866_id[] = {
202*4882a593Smuzhiyun { "bt866", 0 },
203*4882a593Smuzhiyun { }
204*4882a593Smuzhiyun };
205*4882a593Smuzhiyun MODULE_DEVICE_TABLE(i2c, bt866_id);
206*4882a593Smuzhiyun
207*4882a593Smuzhiyun static struct i2c_driver bt866_driver = {
208*4882a593Smuzhiyun .driver = {
209*4882a593Smuzhiyun .name = "bt866",
210*4882a593Smuzhiyun },
211*4882a593Smuzhiyun .probe = bt866_probe,
212*4882a593Smuzhiyun .remove = bt866_remove,
213*4882a593Smuzhiyun .id_table = bt866_id,
214*4882a593Smuzhiyun };
215*4882a593Smuzhiyun
216*4882a593Smuzhiyun module_i2c_driver(bt866_driver);
217