blob: 72325d774a605ad679335c7104dba33a04afdeca [file] [log] [blame]
Hans Verkuil1c1e45d2008-04-28 20:24:33 -03001/*
2 * cx18 ADEC VBI functions
3 *
4 * Derived from cx25840-vbi.c
5 *
6 * Copyright (C) 2007 Hans Verkuil <hverkuil@xs4all.nl>
7 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License
10 * as published by the Free Software Foundation; either version 2
11 * of the License, or (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
21 * 02110-1301, USA.
22 */
23
24
25#include "cx18-driver.h"
26
Andy Walls302df972009-01-31 00:33:02 -030027/*
28 * For sliced VBI output, we set up to use VIP-1.1, 10-bit mode,
29 * NN counts 4 bytes Dwords, an IDID of 0x00 0x80 or one with the VBI line #.
30 * Thus, according to the VIP-2 Spec, our VBI ancillary data lines
31 * (should!) look like:
32 * 4 byte EAV code: 0xff 0x00 0x00 0xRP
33 * unknown number of possible idle bytes
34 * 3 byte Anc data preamble: 0x00 0xff 0xff
35 * 1 byte data identifier: ne010iii (parity bits, 010, DID bits)
36 * 1 byte secondary data id: nessssss (parity bits, SDID bits)
37 * 1 byte data word count: necccccc (parity bits, NN Dword count)
38 * 2 byte Internal DID: 0x00 0x80 (programmed value)
39 * 4*NN data bytes
40 * 1 byte checksum
41 * Fill bytes needed to fil out to 4*NN bytes of payload
42 *
43 * The RP codes for EAVs when in VIP-1.1 mode, not in raw mode, &
44 * in the vertical blanking interval are:
45 * 0xb0 (Task 0 VerticalBlank HorizontalBlank 0 0 0 0)
46 * 0xf0 (Task EvenField VerticalBlank HorizontalBlank 0 0 0 0)
47 *
48 * Since the V bit is only allowed to toggle in the EAV RP code, just
49 * before the first active region line and for active lines, they are:
50 * 0x90 (Task 0 0 HorizontalBlank 0 0 0 0)
51 * 0xd0 (Task EvenField 0 HorizontalBlank 0 0 0 0)
52 *
53 * The user application DID bytes we care about are:
54 * 0x91 (1 0 010 0 !ActiveLine AncDataPresent)
55 * 0x55 (0 1 010 2ndField !ActiveLine AncDataPresent)
56 *
57 */
58static const u8 sliced_vbi_did[2] = { 0x91, 0x55 };
59
60struct vbi_anc_data {
61 /* u8 eav[4]; */
62 /* u8 idle[]; Variable number of idle bytes */
63 u8 preamble[3];
64 u8 did;
65 u8 sdid;
66 u8 data_count;
67 u8 idid[2];
68 u8 payload[1]; /* 4*data_count of payload */
69 /* u8 checksum; */
70 /* u8 fill[]; Variable number of fill bytes */
71};
72
Hans Verkuil1c1e45d2008-04-28 20:24:33 -030073static int odd_parity(u8 c)
74{
75 c ^= (c >> 4);
76 c ^= (c >> 2);
77 c ^= (c >> 1);
78
79 return c & 1;
80}
81
82static int decode_vps(u8 *dst, u8 *p)
83{
84 static const u8 biphase_tbl[] = {
85 0xf0, 0x78, 0x70, 0xf0, 0xb4, 0x3c, 0x34, 0xb4,
86 0xb0, 0x38, 0x30, 0xb0, 0xf0, 0x78, 0x70, 0xf0,
87 0xd2, 0x5a, 0x52, 0xd2, 0x96, 0x1e, 0x16, 0x96,
88 0x92, 0x1a, 0x12, 0x92, 0xd2, 0x5a, 0x52, 0xd2,
89 0xd0, 0x58, 0x50, 0xd0, 0x94, 0x1c, 0x14, 0x94,
90 0x90, 0x18, 0x10, 0x90, 0xd0, 0x58, 0x50, 0xd0,
91 0xf0, 0x78, 0x70, 0xf0, 0xb4, 0x3c, 0x34, 0xb4,
92 0xb0, 0x38, 0x30, 0xb0, 0xf0, 0x78, 0x70, 0xf0,
93 0xe1, 0x69, 0x61, 0xe1, 0xa5, 0x2d, 0x25, 0xa5,
94 0xa1, 0x29, 0x21, 0xa1, 0xe1, 0x69, 0x61, 0xe1,
95 0xc3, 0x4b, 0x43, 0xc3, 0x87, 0x0f, 0x07, 0x87,
96 0x83, 0x0b, 0x03, 0x83, 0xc3, 0x4b, 0x43, 0xc3,
97 0xc1, 0x49, 0x41, 0xc1, 0x85, 0x0d, 0x05, 0x85,
98 0x81, 0x09, 0x01, 0x81, 0xc1, 0x49, 0x41, 0xc1,
99 0xe1, 0x69, 0x61, 0xe1, 0xa5, 0x2d, 0x25, 0xa5,
100 0xa1, 0x29, 0x21, 0xa1, 0xe1, 0x69, 0x61, 0xe1,
101 0xe0, 0x68, 0x60, 0xe0, 0xa4, 0x2c, 0x24, 0xa4,
102 0xa0, 0x28, 0x20, 0xa0, 0xe0, 0x68, 0x60, 0xe0,
103 0xc2, 0x4a, 0x42, 0xc2, 0x86, 0x0e, 0x06, 0x86,
104 0x82, 0x0a, 0x02, 0x82, 0xc2, 0x4a, 0x42, 0xc2,
105 0xc0, 0x48, 0x40, 0xc0, 0x84, 0x0c, 0x04, 0x84,
106 0x80, 0x08, 0x00, 0x80, 0xc0, 0x48, 0x40, 0xc0,
107 0xe0, 0x68, 0x60, 0xe0, 0xa4, 0x2c, 0x24, 0xa4,
108 0xa0, 0x28, 0x20, 0xa0, 0xe0, 0x68, 0x60, 0xe0,
109 0xf0, 0x78, 0x70, 0xf0, 0xb4, 0x3c, 0x34, 0xb4,
110 0xb0, 0x38, 0x30, 0xb0, 0xf0, 0x78, 0x70, 0xf0,
111 0xd2, 0x5a, 0x52, 0xd2, 0x96, 0x1e, 0x16, 0x96,
112 0x92, 0x1a, 0x12, 0x92, 0xd2, 0x5a, 0x52, 0xd2,
113 0xd0, 0x58, 0x50, 0xd0, 0x94, 0x1c, 0x14, 0x94,
114 0x90, 0x18, 0x10, 0x90, 0xd0, 0x58, 0x50, 0xd0,
115 0xf0, 0x78, 0x70, 0xf0, 0xb4, 0x3c, 0x34, 0xb4,
116 0xb0, 0x38, 0x30, 0xb0, 0xf0, 0x78, 0x70, 0xf0,
117 };
118
119 u8 c, err = 0;
120 int i;
121
122 for (i = 0; i < 2 * 13; i += 2) {
123 err |= biphase_tbl[p[i]] | biphase_tbl[p[i + 1]];
124 c = (biphase_tbl[p[i + 1]] & 0xf) |
125 ((biphase_tbl[p[i]] & 0xf) << 4);
126 dst[i / 2] = c;
127 }
128
129 return err & 0xf0;
130}
131
Hans Verkuil1c1e45d2008-04-28 20:24:33 -0300132int cx18_av_vbi(struct cx18 *cx, unsigned int cmd, void *arg)
133{
134 struct cx18_av_state *state = &cx->av_state;
135 struct v4l2_format *fmt;
136 struct v4l2_sliced_vbi_format *svbi;
137
138 switch (cmd) {
139 case VIDIOC_G_FMT:
140 {
141 static u16 lcr2vbi[] = {
142 0, V4L2_SLICED_TELETEXT_B, 0, /* 1 */
143 0, V4L2_SLICED_WSS_625, 0, /* 4 */
144 V4L2_SLICED_CAPTION_525, /* 6 */
Andy Walls302df972009-01-31 00:33:02 -0300145 V4L2_SLICED_VPS, 0, 0, 0, 0, /* 7 - unlike cx25840 */
Hans Verkuil1c1e45d2008-04-28 20:24:33 -0300146 0, 0, 0, 0
147 };
148 int is_pal = !(state->std & V4L2_STD_525_60);
149 int i;
150
151 fmt = arg;
152 if (fmt->type != V4L2_BUF_TYPE_SLICED_VBI_CAPTURE)
153 return -EINVAL;
154 svbi = &fmt->fmt.sliced;
155 memset(svbi, 0, sizeof(*svbi));
156 /* we're done if raw VBI is active */
157 if ((cx18_av_read(cx, 0x404) & 0x10) == 0)
158 break;
159
160 if (is_pal) {
161 for (i = 7; i <= 23; i++) {
162 u8 v = cx18_av_read(cx, 0x424 + i - 7);
163
164 svbi->service_lines[0][i] = lcr2vbi[v >> 4];
165 svbi->service_lines[1][i] = lcr2vbi[v & 0xf];
166 svbi->service_set |= svbi->service_lines[0][i] |
167 svbi->service_lines[1][i];
168 }
169 } else {
170 for (i = 10; i <= 21; i++) {
171 u8 v = cx18_av_read(cx, 0x424 + i - 10);
172
173 svbi->service_lines[0][i] = lcr2vbi[v >> 4];
174 svbi->service_lines[1][i] = lcr2vbi[v & 0xf];
175 svbi->service_set |= svbi->service_lines[0][i] |
176 svbi->service_lines[1][i];
177 }
178 }
179 break;
180 }
181
182 case VIDIOC_S_FMT:
183 {
184 int is_pal = !(state->std & V4L2_STD_525_60);
185 int vbi_offset = is_pal ? 1 : 0;
186 int i, x;
187 u8 lcr[24];
188
189 fmt = arg;
Andy Wallsdd073432008-12-12 16:24:04 -0300190 if (fmt->type != V4L2_BUF_TYPE_SLICED_VBI_CAPTURE &&
191 fmt->type != V4L2_BUF_TYPE_VBI_CAPTURE)
Hans Verkuil1c1e45d2008-04-28 20:24:33 -0300192 return -EINVAL;
193 svbi = &fmt->fmt.sliced;
Andy Wallsdd073432008-12-12 16:24:04 -0300194 if (fmt->type == V4L2_BUF_TYPE_VBI_CAPTURE) {
Hans Verkuil1c1e45d2008-04-28 20:24:33 -0300195 /* raw VBI */
196 memset(svbi, 0, sizeof(*svbi));
197
Hans Verkuil03b52c32008-07-17 12:33:08 -0300198 /* Setup standard */
199 cx18_av_std_setup(cx);
Hans Verkuil1c1e45d2008-04-28 20:24:33 -0300200
201 /* VBI Offset */
202 cx18_av_write(cx, 0x47f, vbi_offset);
203 cx18_av_write(cx, 0x404, 0x2e);
204 break;
205 }
206
207 for (x = 0; x <= 23; x++)
208 lcr[x] = 0x00;
209
Hans Verkuil03b52c32008-07-17 12:33:08 -0300210 /* Setup standard */
211 cx18_av_std_setup(cx);
Hans Verkuil1c1e45d2008-04-28 20:24:33 -0300212
213 /* Sliced VBI */
214 cx18_av_write(cx, 0x404, 0x32); /* Ancillary data */
215 cx18_av_write(cx, 0x406, 0x13);
216 cx18_av_write(cx, 0x47f, vbi_offset);
217
218 if (is_pal) {
219 for (i = 0; i <= 6; i++)
220 svbi->service_lines[0][i] =
221 svbi->service_lines[1][i] = 0;
222 } else {
223 for (i = 0; i <= 9; i++)
224 svbi->service_lines[0][i] =
225 svbi->service_lines[1][i] = 0;
226
227 for (i = 22; i <= 23; i++)
228 svbi->service_lines[0][i] =
229 svbi->service_lines[1][i] = 0;
230 }
231
232 for (i = 7; i <= 23; i++) {
233 for (x = 0; x <= 1; x++) {
234 switch (svbi->service_lines[1-x][i]) {
235 case V4L2_SLICED_TELETEXT_B:
236 lcr[i] |= 1 << (4 * x);
237 break;
238 case V4L2_SLICED_WSS_625:
239 lcr[i] |= 4 << (4 * x);
240 break;
241 case V4L2_SLICED_CAPTION_525:
242 lcr[i] |= 6 << (4 * x);
243 break;
244 case V4L2_SLICED_VPS:
245 lcr[i] |= 9 << (4 * x);
246 break;
247 }
248 }
249 }
250
251 if (is_pal) {
252 for (x = 1, i = 0x424; i <= 0x434; i++, x++)
253 cx18_av_write(cx, i, lcr[6 + x]);
254 } else {
255 for (x = 1, i = 0x424; i <= 0x430; i++, x++)
256 cx18_av_write(cx, i, lcr[9 + x]);
257 for (i = 0x431; i <= 0x434; i++)
258 cx18_av_write(cx, i, 0);
259 }
260
261 cx18_av_write(cx, 0x43c, 0x16);
262 cx18_av_write(cx, 0x474, is_pal ? 0x2a : 0x22);
263 break;
264 }
265
266 case VIDIOC_INT_DECODE_VBI_LINE:
267 {
268 struct v4l2_decode_vbi_line *vbi = arg;
Andy Walls302df972009-01-31 00:33:02 -0300269 u8 *p;
270 struct vbi_anc_data *anc = (struct vbi_anc_data *) vbi->p;
271 int did, sdid, l, err = 0;
Hans Verkuil1c1e45d2008-04-28 20:24:33 -0300272
Andy Walls302df972009-01-31 00:33:02 -0300273 /*
274 * Check for the ancillary data header for sliced VBI
275 */
276 if (anc->preamble[0] ||
277 anc->preamble[1] != 0xff || anc->preamble[2] != 0xff ||
278 (anc->did != sliced_vbi_did[0] &&
279 anc->did != sliced_vbi_did[1])) {
Hans Verkuil1c1e45d2008-04-28 20:24:33 -0300280 vbi->line = vbi->type = 0;
281 break;
282 }
283
Andy Walls302df972009-01-31 00:33:02 -0300284 did = anc->did;
285 sdid = anc->sdid & 0xf;
286 l = anc->idid[0] & 0x3f;
Hans Verkuil1c1e45d2008-04-28 20:24:33 -0300287 l += state->vbi_line_offset;
Andy Walls302df972009-01-31 00:33:02 -0300288 p = anc->payload;
Hans Verkuil1c1e45d2008-04-28 20:24:33 -0300289
Andy Walls302df972009-01-31 00:33:02 -0300290 /* Decode the SDID set by the slicer */
291 switch (sdid) {
Hans Verkuil1c1e45d2008-04-28 20:24:33 -0300292 case 1:
Andy Walls302df972009-01-31 00:33:02 -0300293 sdid = V4L2_SLICED_TELETEXT_B;
Hans Verkuil1c1e45d2008-04-28 20:24:33 -0300294 break;
295 case 4:
Andy Walls302df972009-01-31 00:33:02 -0300296 sdid = V4L2_SLICED_WSS_625;
Hans Verkuil1c1e45d2008-04-28 20:24:33 -0300297 break;
298 case 6:
Andy Walls302df972009-01-31 00:33:02 -0300299 sdid = V4L2_SLICED_CAPTION_525;
Hans Verkuil1c1e45d2008-04-28 20:24:33 -0300300 err = !odd_parity(p[0]) || !odd_parity(p[1]);
301 break;
Andy Walls302df972009-01-31 00:33:02 -0300302 case 7: /* Differs from cx25840 */
303 sdid = V4L2_SLICED_VPS;
Hans Verkuil1c1e45d2008-04-28 20:24:33 -0300304 if (decode_vps(p, p) != 0)
305 err = 1;
306 break;
307 default:
Andy Walls302df972009-01-31 00:33:02 -0300308 sdid = 0;
Hans Verkuil1c1e45d2008-04-28 20:24:33 -0300309 err = 1;
310 break;
311 }
312
Andy Walls302df972009-01-31 00:33:02 -0300313 vbi->type = err ? 0 : sdid;
Hans Verkuil1c1e45d2008-04-28 20:24:33 -0300314 vbi->line = err ? 0 : l;
Andy Walls302df972009-01-31 00:33:02 -0300315 vbi->is_second_field = err ? 0 : (did == sliced_vbi_did[1]);
Hans Verkuil1c1e45d2008-04-28 20:24:33 -0300316 vbi->p = p;
317 break;
318 }
319 }
320
321 return 0;
322}