aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/media/radio/radio-terratec.c247
1 files changed, 138 insertions, 109 deletions
diff --git a/drivers/media/radio/radio-terratec.c b/drivers/media/radio/radio-terratec.c
index d59a27accb84..e43acfd7e533 100644
--- a/drivers/media/radio/radio-terratec.c
+++ b/drivers/media/radio/radio-terratec.c
@@ -205,135 +205,152 @@ static int tt_getsigstr(struct tt_device *dev) /* TODO */
205 return 1; /* signal present */ 205 return 1; /* signal present */
206} 206}
207 207
208static int vidioc_querycap(struct file *file, void *priv,
209 struct v4l2_capability *v)
210{
211 strlcpy(v->driver, "radio-terratec", sizeof(v->driver));
212 strlcpy(v->card, "ActiveRadio", sizeof(v->card));
213 sprintf(v->bus_info, "ISA");
214 v->version = RADIO_VERSION;
215 v->capabilities = V4L2_CAP_TUNER;
216 return 0;
217}
208 218
209/* implement the video4linux api */ 219static int vidioc_g_tuner(struct file *file, void *priv,
210 220 struct v4l2_tuner *v)
211static int tt_do_ioctl(struct inode *inode, struct file *file,
212 unsigned int cmd, void *arg)
213{ 221{
214 struct video_device *dev = video_devdata(file); 222 struct video_device *dev = video_devdata(file);
215 struct tt_device *tt=dev->priv; 223 struct tt_device *tt = dev->priv;
216 224
217 switch(cmd) 225 if (v->index > 0)
218 { 226 return -EINVAL;
219 case VIDIOC_QUERYCAP:
220 {
221 struct v4l2_capability *v = arg;
222 memset(v,0,sizeof(*v));
223 strlcpy(v->driver, "radio-terratec", sizeof (v->driver));
224 strlcpy(v->card, "ActiveRadio", sizeof (v->card));
225 sprintf(v->bus_info,"ISA");
226 v->version = RADIO_VERSION;
227 v->capabilities = V4L2_CAP_TUNER;
228 227
229 return 0; 228 strcpy(v->name, "FM");
230 } 229 v->type = V4L2_TUNER_RADIO;
231 case VIDIOC_G_TUNER: 230 v->rangelow = (87*16000);
232 { 231 v->rangehigh = (108*16000);
233 struct v4l2_tuner *v = arg; 232 v->rxsubchans = V4L2_TUNER_SUB_MONO;
233 v->capability = V4L2_TUNER_CAP_LOW;
234 v->audmode = V4L2_TUNER_MODE_MONO;
235 v->signal = 0xFFFF*tt_getsigstr(tt);
236 return 0;
237}
234 238
235 if (v->index > 0) 239static int vidioc_s_tuner(struct file *file, void *priv,
236 return -EINVAL; 240 struct v4l2_tuner *v)
241{
242 if (v->index > 0)
243 return -EINVAL;
244 return 0;
245}
237 246
238 memset(v,0,sizeof(*v)); 247static int vidioc_s_frequency(struct file *file, void *priv,
239 strcpy(v->name, "FM"); 248 struct v4l2_frequency *f)
240 v->type = V4L2_TUNER_RADIO; 249{
250 struct video_device *dev = video_devdata(file);
251 struct tt_device *tt = dev->priv;
241 252
242 v->rangelow=(87*16000); 253 tt->curfreq = f->frequency;
243 v->rangehigh=(108*16000); 254 tt_setfreq(tt, tt->curfreq);
244 v->rxsubchans =V4L2_TUNER_SUB_MONO; 255 return 0;
245 v->capability=V4L2_TUNER_CAP_LOW; 256}
246 v->audmode = V4L2_TUNER_MODE_MONO;
247 v->signal=0xFFFF*tt_getsigstr(tt);
248 257
249 return 0; 258static int vidioc_g_frequency(struct file *file, void *priv,
250 } 259 struct v4l2_frequency *f)
251 case VIDIOC_S_TUNER: 260{
252 { 261 struct video_device *dev = video_devdata(file);
253 struct v4l2_tuner *v = arg; 262 struct tt_device *tt = dev->priv;
254 263
255 if (v->index > 0) 264 f->type = V4L2_TUNER_RADIO;
256 return -EINVAL; 265 f->frequency = tt->curfreq;
266 return 0;
267}
257 268
258 return 0; 269static int vidioc_queryctrl(struct file *file, void *priv,
259 } 270 struct v4l2_queryctrl *qc)
260 case VIDIOC_S_FREQUENCY: 271{
261 { 272 int i;
262 struct v4l2_frequency *f = arg;
263 273
264 tt->curfreq = f->frequency; 274 for (i = 0; i < ARRAY_SIZE(radio_qctrl); i++) {
265 tt_setfreq(tt, tt->curfreq); 275 if (qc->id && qc->id == radio_qctrl[i].id) {
276 memcpy(qc, &(radio_qctrl[i]),
277 sizeof(*qc));
266 return 0; 278 return 0;
267 } 279 }
268 case VIDIOC_G_FREQUENCY: 280 }
269 { 281 return -EINVAL;
270 struct v4l2_frequency *f = arg; 282}
271 283
272 f->type = V4L2_TUNER_RADIO; 284static int vidioc_g_ctrl(struct file *file, void *priv,
273 f->frequency = tt->curfreq; 285 struct v4l2_control *ctrl)
286{
287 struct video_device *dev = video_devdata(file);
288 struct tt_device *tt = dev->priv;
274 289
275 return 0; 290 switch (ctrl->id) {
276 } 291 case V4L2_CID_AUDIO_MUTE:
277 case VIDIOC_QUERYCTRL: 292 if (tt->muted)
278 { 293 ctrl->value = 1;
279 struct v4l2_queryctrl *qc = arg; 294 else
280 int i; 295 ctrl->value = 0;
281 296 return 0;
282 for (i = 0; i < ARRAY_SIZE(radio_qctrl); i++) { 297 case V4L2_CID_AUDIO_VOLUME:
283 if (qc->id && qc->id == radio_qctrl[i].id) { 298 ctrl->value = tt->curvol * 6554;
284 memcpy(qc, &(radio_qctrl[i]), 299 return 0;
285 sizeof(*qc)); 300 }
286 return (0); 301 return -EINVAL;
287 } 302}
288 }
289 return -EINVAL;
290 }
291 case VIDIOC_G_CTRL:
292 {
293 struct v4l2_control *ctrl= arg;
294
295 switch (ctrl->id) {
296 case V4L2_CID_AUDIO_MUTE:
297 if (tt->muted)
298 ctrl->value=1;
299 else
300 ctrl->value=0;
301 return (0);
302 case V4L2_CID_AUDIO_VOLUME:
303 ctrl->value=tt->curvol * 6554;
304 return (0);
305 }
306 return -EINVAL;
307 }
308 case VIDIOC_S_CTRL:
309 {
310 struct v4l2_control *ctrl= arg;
311
312 switch (ctrl->id) {
313 case V4L2_CID_AUDIO_MUTE:
314 if (ctrl->value) {
315 tt_mute(tt);
316 } else {
317 tt_setvol(tt,tt->curvol);
318 }
319 return (0);
320 case V4L2_CID_AUDIO_VOLUME:
321 tt_setvol(tt,ctrl->value);
322 return (0);
323 }
324 return -EINVAL;
325 }
326 303
327 default: 304static int vidioc_s_ctrl(struct file *file, void *priv,
328 return v4l_compat_translate_ioctl(inode,file,cmd,arg, 305 struct v4l2_control *ctrl)
329 tt_do_ioctl); 306{
307 struct video_device *dev = video_devdata(file);
308 struct tt_device *tt = dev->priv;
309
310 switch (ctrl->id) {
311 case V4L2_CID_AUDIO_MUTE:
312 if (ctrl->value)
313 tt_mute(tt);
314 else
315 tt_setvol(tt,tt->curvol);
316 return 0;
317 case V4L2_CID_AUDIO_VOLUME:
318 tt_setvol(tt,ctrl->value);
319 return 0;
330 } 320 }
321 return -EINVAL;
322}
323
324static int vidioc_g_audio(struct file *file, void *priv,
325 struct v4l2_audio *a)
326{
327 if (a->index > 1)
328 return -EINVAL;
329
330 strcpy(a->name, "Radio");
331 a->capability = V4L2_AUDCAP_STEREO;
332 return 0;
333}
334
335static int vidioc_g_input(struct file *filp, void *priv, unsigned int *i)
336{
337 *i = 0;
338 return 0;
331} 339}
332 340
333static int tt_ioctl(struct inode *inode, struct file *file, 341static int vidioc_s_input(struct file *filp, void *priv, unsigned int i)
334 unsigned int cmd, unsigned long arg)
335{ 342{
336 return video_usercopy(inode, file, cmd, arg, tt_do_ioctl); 343 if (i != 0)
344 return -EINVAL;
345 return 0;
346}
347
348static int vidioc_s_audio(struct file *file, void *priv,
349 struct v4l2_audio *a)
350{
351 if (a->index != 0)
352 return -EINVAL;
353 return 0;
337} 354}
338 355
339static struct tt_device terratec_unit; 356static struct tt_device terratec_unit;
@@ -342,7 +359,7 @@ static const struct file_operations terratec_fops = {
342 .owner = THIS_MODULE, 359 .owner = THIS_MODULE,
343 .open = video_exclusive_open, 360 .open = video_exclusive_open,
344 .release = video_exclusive_release, 361 .release = video_exclusive_release,
345 .ioctl = tt_ioctl, 362 .ioctl = video_ioctl2,
346 .compat_ioctl = v4l_compat_ioctl32, 363 .compat_ioctl = v4l_compat_ioctl32,
347 .llseek = no_llseek, 364 .llseek = no_llseek,
348}; 365};
@@ -354,6 +371,18 @@ static struct video_device terratec_radio=
354 .type = VID_TYPE_TUNER, 371 .type = VID_TYPE_TUNER,
355 .hardware = 0, 372 .hardware = 0,
356 .fops = &terratec_fops, 373 .fops = &terratec_fops,
374 .vidioc_querycap = vidioc_querycap,
375 .vidioc_g_tuner = vidioc_g_tuner,
376 .vidioc_s_tuner = vidioc_s_tuner,
377 .vidioc_g_frequency = vidioc_g_frequency,
378 .vidioc_s_frequency = vidioc_s_frequency,
379 .vidioc_queryctrl = vidioc_queryctrl,
380 .vidioc_g_ctrl = vidioc_g_ctrl,
381 .vidioc_s_ctrl = vidioc_s_ctrl,
382 .vidioc_g_audio = vidioc_g_audio,
383 .vidioc_s_audio = vidioc_s_audio,
384 .vidioc_g_input = vidioc_g_input,
385 .vidioc_s_input = vidioc_s_input,
357}; 386};
358 387
359static int __init terratec_init(void) 388static int __init terratec_init(void)