aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/media/video/cx25840/cx25840-vbi.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/media/video/cx25840/cx25840-vbi.c')
-rw-r--r--drivers/media/video/cx25840/cx25840-vbi.c314
1 files changed, 148 insertions, 166 deletions
diff --git a/drivers/media/video/cx25840/cx25840-vbi.c b/drivers/media/video/cx25840/cx25840-vbi.c
index 03f09b288eb8..35f6592f6c47 100644
--- a/drivers/media/video/cx25840/cx25840-vbi.c
+++ b/drivers/media/video/cx25840/cx25840-vbi.c
@@ -82,199 +82,181 @@ static int decode_vps(u8 * dst, u8 * p)
82 return err & 0xf0; 82 return err & 0xf0;
83} 83}
84 84
85int cx25840_vbi(struct i2c_client *client, unsigned int cmd, void *arg) 85int cx25840_vbi_g_fmt(struct v4l2_subdev *sd, struct v4l2_format *fmt)
86{ 86{
87 struct cx25840_state *state = to_state(i2c_get_clientdata(client)); 87 struct i2c_client *client = v4l2_get_subdevdata(sd);
88 struct v4l2_format *fmt; 88 struct cx25840_state *state = to_state(sd);
89 struct v4l2_sliced_vbi_format *svbi; 89 struct v4l2_sliced_vbi_format *svbi;
90 static const u16 lcr2vbi[] = {
91 0, V4L2_SLICED_TELETEXT_B, 0, /* 1 */
92 0, V4L2_SLICED_WSS_625, 0, /* 4 */
93 V4L2_SLICED_CAPTION_525, /* 6 */
94 0, 0, V4L2_SLICED_VPS, 0, 0, /* 9 */
95 0, 0, 0, 0
96 };
97 int is_pal = !(state->std & V4L2_STD_525_60);
98 int i;
90 99
91 switch (cmd) { 100 if (fmt->type != V4L2_BUF_TYPE_SLICED_VBI_CAPTURE)
92 case VIDIOC_G_FMT: 101 return -EINVAL;
93 { 102 svbi = &fmt->fmt.sliced;
94 static u16 lcr2vbi[] = { 103 memset(svbi, 0, sizeof(*svbi));
95 0, V4L2_SLICED_TELETEXT_B, 0, /* 1 */ 104 /* we're done if raw VBI is active */
96 0, V4L2_SLICED_WSS_625, 0, /* 4 */ 105 if ((cx25840_read(client, 0x404) & 0x10) == 0)
97 V4L2_SLICED_CAPTION_525, /* 6 */ 106 return 0;
98 0, 0, V4L2_SLICED_VPS, 0, 0, /* 9 */
99 0, 0, 0, 0
100 };
101 int is_pal = !(state->std & V4L2_STD_525_60);
102 int i;
103
104 fmt = arg;
105 if (fmt->type != V4L2_BUF_TYPE_SLICED_VBI_CAPTURE)
106 return -EINVAL;
107 svbi = &fmt->fmt.sliced;
108 memset(svbi, 0, sizeof(*svbi));
109 /* we're done if raw VBI is active */
110 if ((cx25840_read(client, 0x404) & 0x10) == 0)
111 break;
112 107
113 if (is_pal) { 108 if (is_pal) {
114 for (i = 7; i <= 23; i++) { 109 for (i = 7; i <= 23; i++) {
115 u8 v = cx25840_read(client, 0x424 + i - 7); 110 u8 v = cx25840_read(client, 0x424 + i - 7);
116 111
117 svbi->service_lines[0][i] = lcr2vbi[v >> 4]; 112 svbi->service_lines[0][i] = lcr2vbi[v >> 4];
118 svbi->service_lines[1][i] = lcr2vbi[v & 0xf]; 113 svbi->service_lines[1][i] = lcr2vbi[v & 0xf];
119 svbi->service_set |= 114 svbi->service_set |= svbi->service_lines[0][i] |
120 svbi->service_lines[0][i] | svbi->service_lines[1][i]; 115 svbi->service_lines[1][i];
121 }
122 } 116 }
123 else { 117 } else {
124 for (i = 10; i <= 21; i++) { 118 for (i = 10; i <= 21; i++) {
125 u8 v = cx25840_read(client, 0x424 + i - 10); 119 u8 v = cx25840_read(client, 0x424 + i - 10);
126 120
127 svbi->service_lines[0][i] = lcr2vbi[v >> 4]; 121 svbi->service_lines[0][i] = lcr2vbi[v >> 4];
128 svbi->service_lines[1][i] = lcr2vbi[v & 0xf]; 122 svbi->service_lines[1][i] = lcr2vbi[v & 0xf];
129 svbi->service_set |= 123 svbi->service_set |= svbi->service_lines[0][i] |
130 svbi->service_lines[0][i] | svbi->service_lines[1][i]; 124 svbi->service_lines[1][i];
131 }
132 } 125 }
133 break;
134 } 126 }
127 return 0;
128}
135 129
136 case VIDIOC_S_FMT: 130int cx25840_vbi_s_fmt(struct v4l2_subdev *sd, struct v4l2_format *fmt)
137 { 131{
138 int is_pal = !(state->std & V4L2_STD_525_60); 132 struct i2c_client *client = v4l2_get_subdevdata(sd);
139 int vbi_offset = is_pal ? 1 : 0; 133 struct cx25840_state *state = to_state(sd);
140 int i, x; 134 struct v4l2_sliced_vbi_format *svbi;
141 u8 lcr[24]; 135 int is_pal = !(state->std & V4L2_STD_525_60);
142 136 int vbi_offset = is_pal ? 1 : 0;
143 fmt = arg; 137 int i, x;
144 if (fmt->type != V4L2_BUF_TYPE_SLICED_VBI_CAPTURE && 138 u8 lcr[24];
145 fmt->type != V4L2_BUF_TYPE_VBI_CAPTURE) 139
146 return -EINVAL; 140 if (fmt->type != V4L2_BUF_TYPE_SLICED_VBI_CAPTURE &&
147 svbi = &fmt->fmt.sliced; 141 fmt->type != V4L2_BUF_TYPE_VBI_CAPTURE)
148 if (fmt->type == V4L2_BUF_TYPE_VBI_CAPTURE) { 142 return -EINVAL;
149 /* raw VBI */ 143 svbi = &fmt->fmt.sliced;
150 memset(svbi, 0, sizeof(*svbi)); 144 if (fmt->type == V4L2_BUF_TYPE_VBI_CAPTURE) {
151 145 /* raw VBI */
152 /* Setup standard */ 146 memset(svbi, 0, sizeof(*svbi));
153 cx25840_std_setup(client);
154
155 /* VBI Offset */
156 cx25840_write(client, 0x47f, vbi_offset);
157 cx25840_write(client, 0x404, 0x2e);
158 break;
159 }
160
161 for (x = 0; x <= 23; x++)
162 lcr[x] = 0x00;
163 147
164 /* Setup standard */ 148 /* Setup standard */
165 cx25840_std_setup(client); 149 cx25840_std_setup(client);
166 150
167 /* Sliced VBI */ 151 /* VBI Offset */
168 cx25840_write(client, 0x404, 0x32); /* Ancillary data */
169 cx25840_write(client, 0x406, 0x13);
170 cx25840_write(client, 0x47f, vbi_offset); 152 cx25840_write(client, 0x47f, vbi_offset);
153 cx25840_write(client, 0x404, 0x2e);
154 return 0;
155 }
171 156
172 if (is_pal) { 157 for (x = 0; x <= 23; x++)
173 for (i = 0; i <= 6; i++) 158 lcr[x] = 0x00;
174 svbi->service_lines[0][i] = 159
175 svbi->service_lines[1][i] = 0; 160 /* Setup standard */
176 } else { 161 cx25840_std_setup(client);
177 for (i = 0; i <= 9; i++) 162
178 svbi->service_lines[0][i] = 163 /* Sliced VBI */
179 svbi->service_lines[1][i] = 0; 164 cx25840_write(client, 0x404, 0x32); /* Ancillary data */
180 165 cx25840_write(client, 0x406, 0x13);
181 for (i = 22; i <= 23; i++) 166 cx25840_write(client, 0x47f, vbi_offset);
182 svbi->service_lines[0][i] = 167
183 svbi->service_lines[1][i] = 0; 168 if (is_pal) {
184 } 169 for (i = 0; i <= 6; i++)
185 170 svbi->service_lines[0][i] =
186 for (i = 7; i <= 23; i++) { 171 svbi->service_lines[1][i] = 0;
187 for (x = 0; x <= 1; x++) { 172 } else {
188 switch (svbi->service_lines[1-x][i]) { 173 for (i = 0; i <= 9; i++)
189 case V4L2_SLICED_TELETEXT_B: 174 svbi->service_lines[0][i] =
190 lcr[i] |= 1 << (4 * x); 175 svbi->service_lines[1][i] = 0;
191 break; 176
192 case V4L2_SLICED_WSS_625: 177 for (i = 22; i <= 23; i++)
193 lcr[i] |= 4 << (4 * x); 178 svbi->service_lines[0][i] =
194 break; 179 svbi->service_lines[1][i] = 0;
195 case V4L2_SLICED_CAPTION_525: 180 }
196 lcr[i] |= 6 << (4 * x);
197 break;
198 case V4L2_SLICED_VPS:
199 lcr[i] |= 9 << (4 * x);
200 break;
201 }
202 }
203 }
204 181
205 if (is_pal) { 182 for (i = 7; i <= 23; i++) {
206 for (x = 1, i = 0x424; i <= 0x434; i++, x++) { 183 for (x = 0; x <= 1; x++) {
207 cx25840_write(client, i, lcr[6 + x]); 184 switch (svbi->service_lines[1-x][i]) {
208 } 185 case V4L2_SLICED_TELETEXT_B:
209 } 186 lcr[i] |= 1 << (4 * x);
210 else { 187 break;
211 for (x = 1, i = 0x424; i <= 0x430; i++, x++) { 188 case V4L2_SLICED_WSS_625:
212 cx25840_write(client, i, lcr[9 + x]); 189 lcr[i] |= 4 << (4 * x);
213 } 190 break;
214 for (i = 0x431; i <= 0x434; i++) { 191 case V4L2_SLICED_CAPTION_525:
215 cx25840_write(client, i, 0); 192 lcr[i] |= 6 << (4 * x);
193 break;
194 case V4L2_SLICED_VPS:
195 lcr[i] |= 9 << (4 * x);
196 break;
216 } 197 }
217 } 198 }
199 }
218 200
219 cx25840_write(client, 0x43c, 0x16); 201 if (is_pal) {
220 202 for (x = 1, i = 0x424; i <= 0x434; i++, x++)
221 if (is_pal) { 203 cx25840_write(client, i, lcr[6 + x]);
222 cx25840_write(client, 0x474, 0x2a); 204 } else {
223 } else { 205 for (x = 1, i = 0x424; i <= 0x430; i++, x++)
224 cx25840_write(client, 0x474, 0x22); 206 cx25840_write(client, i, lcr[9 + x]);
225 } 207 for (i = 0x431; i <= 0x434; i++)
226 break; 208 cx25840_write(client, i, 0);
227 } 209 }
228 210
229 case VIDIOC_INT_DECODE_VBI_LINE: 211 cx25840_write(client, 0x43c, 0x16);
230 { 212 cx25840_write(client, 0x474, is_pal ? 0x2a : 0x22);
231 struct v4l2_decode_vbi_line *vbi = arg; 213 return 0;
232 u8 *p = vbi->p; 214}
233 int id1, id2, l, err = 0;
234 215
235 if (p[0] || p[1] != 0xff || p[2] != 0xff || 216int cx25840_decode_vbi_line(struct v4l2_subdev *sd, struct v4l2_decode_vbi_line *vbi)
236 (p[3] != 0x55 && p[3] != 0x91)) { 217{
237 vbi->line = vbi->type = 0; 218 struct cx25840_state *state = to_state(sd);
238 break; 219 u8 *p = vbi->p;
239 } 220 int id1, id2, l, err = 0;
221
222 if (p[0] || p[1] != 0xff || p[2] != 0xff ||
223 (p[3] != 0x55 && p[3] != 0x91)) {
224 vbi->line = vbi->type = 0;
225 return 0;
226 }
240 227
241 p += 4; 228 p += 4;
242 id1 = p[-1]; 229 id1 = p[-1];
243 id2 = p[0] & 0xf; 230 id2 = p[0] & 0xf;
244 l = p[2] & 0x3f; 231 l = p[2] & 0x3f;
245 l += state->vbi_line_offset; 232 l += state->vbi_line_offset;
246 p += 4; 233 p += 4;
247 234
248 switch (id2) { 235 switch (id2) {
249 case 1: 236 case 1:
250 id2 = V4L2_SLICED_TELETEXT_B; 237 id2 = V4L2_SLICED_TELETEXT_B;
251 break; 238 break;
252 case 4: 239 case 4:
253 id2 = V4L2_SLICED_WSS_625; 240 id2 = V4L2_SLICED_WSS_625;
254 break; 241 break;
255 case 6: 242 case 6:
256 id2 = V4L2_SLICED_CAPTION_525; 243 id2 = V4L2_SLICED_CAPTION_525;
257 err = !odd_parity(p[0]) || !odd_parity(p[1]); 244 err = !odd_parity(p[0]) || !odd_parity(p[1]);
258 break; 245 break;
259 case 9: 246 case 9:
260 id2 = V4L2_SLICED_VPS; 247 id2 = V4L2_SLICED_VPS;
261 if (decode_vps(p, p) != 0) { 248 if (decode_vps(p, p) != 0)
262 err = 1;
263 }
264 break;
265 default:
266 id2 = 0;
267 err = 1; 249 err = 1;
268 break;
269 }
270
271 vbi->type = err ? 0 : id2;
272 vbi->line = err ? 0 : l;
273 vbi->is_second_field = err ? 0 : (id1 == 0x55);
274 vbi->p = p;
275 break; 250 break;
276 } 251 default:
252 id2 = 0;
253 err = 1;
254 break;
277 } 255 }
278 256
257 vbi->type = err ? 0 : id2;
258 vbi->line = err ? 0 : l;
259 vbi->is_second_field = err ? 0 : (id1 == 0x55);
260 vbi->p = p;
279 return 0; 261 return 0;
280} 262}