📄 radio-mr800.c
字号:
/* vidioc_g_tuner - get tuner attributes */static int vidioc_g_tuner(struct file *file, void *priv, struct v4l2_tuner *v){ struct amradio_device *radio = video_get_drvdata(video_devdata(file)); if (v->index > 0) return -EINVAL;/* TODO: Add function which look is signal stereo or not * amradio_getstat(radio); */ radio->stereo = -1; strcpy(v->name, "FM"); v->type = V4L2_TUNER_RADIO; v->rangelow = FREQ_MIN * FREQ_MUL; v->rangehigh = FREQ_MAX * FREQ_MUL; v->rxsubchans = V4L2_TUNER_SUB_MONO | V4L2_TUNER_SUB_STEREO; v->capability = V4L2_TUNER_CAP_LOW; if (radio->stereo) v->audmode = V4L2_TUNER_MODE_STEREO; else v->audmode = V4L2_TUNER_MODE_MONO; v->signal = 0xffff; /* Can't get the signal strength, sad.. */ v->afc = 0; /* Don't know what is this */ return 0;}/* vidioc_s_tuner - set tuner attributes */static int vidioc_s_tuner(struct file *file, void *priv, struct v4l2_tuner *v){ if (v->index > 0) return -EINVAL; return 0;}/* vidioc_s_frequency - set tuner radio frequency */static int vidioc_s_frequency(struct file *file, void *priv, struct v4l2_frequency *f){ struct amradio_device *radio = video_get_drvdata(video_devdata(file)); radio->curfreq = f->frequency; if (amradio_setfreq(radio, radio->curfreq) < 0) warn("Set frequency failed"); return 0;}/* vidioc_g_frequency - get tuner radio frequency */static int vidioc_g_frequency(struct file *file, void *priv, struct v4l2_frequency *f){ struct amradio_device *radio = video_get_drvdata(video_devdata(file)); f->type = V4L2_TUNER_RADIO; f->frequency = radio->curfreq; return 0;}/* vidioc_queryctrl - enumerate control items */static int vidioc_queryctrl(struct file *file, void *priv, struct v4l2_queryctrl *qc){ int i; for (i = 0; i < ARRAY_SIZE(radio_qctrl); i++) { if (qc->id && qc->id == radio_qctrl[i].id) { memcpy(qc, &(radio_qctrl[i]), sizeof(*qc)); return 0; } } return -EINVAL;}/* vidioc_g_ctrl - get the value of a control */static int vidioc_g_ctrl(struct file *file, void *priv, struct v4l2_control *ctrl){ struct amradio_device *radio = video_get_drvdata(video_devdata(file)); switch (ctrl->id) { case V4L2_CID_AUDIO_MUTE: ctrl->value = radio->muted; return 0; } return -EINVAL;}/* vidioc_s_ctrl - set the value of a control */static int vidioc_s_ctrl(struct file *file, void *priv, struct v4l2_control *ctrl){ struct amradio_device *radio = video_get_drvdata(video_devdata(file)); switch (ctrl->id) { case V4L2_CID_AUDIO_MUTE: if (ctrl->value) { if (amradio_stop(radio) < 0) { warn("amradio_stop() failed"); return -1; } } else { if (amradio_start(radio) < 0) { warn("amradio_start() failed"); return -1; } } return 0; } return -EINVAL;}/* vidioc_g_audio - get audio attributes */static int vidioc_g_audio(struct file *file, void *priv, struct v4l2_audio *a){ if (a->index > 1) return -EINVAL; strcpy(a->name, "Radio"); a->capability = V4L2_AUDCAP_STEREO; return 0;}/* vidioc_s_audio - set audio attributes */static int vidioc_s_audio(struct file *file, void *priv, struct v4l2_audio *a){ if (a->index != 0) return -EINVAL; return 0;}/* vidioc_g_input - get input */static int vidioc_g_input(struct file *filp, void *priv, unsigned int *i){ *i = 0; return 0;}/* vidioc_s_input - set input */static int vidioc_s_input(struct file *filp, void *priv, unsigned int i){ if (i != 0) return -EINVAL; return 0;}/* open device - amradio_start() and amradio_setfreq() */static int usb_amradio_open(struct inode *inode, struct file *file){ struct amradio_device *radio = video_get_drvdata(video_devdata(file)); lock_kernel(); radio->users = 1; radio->muted = 1; if (amradio_start(radio) < 0) { warn("Radio did not start up properly"); radio->users = 0; unlock_kernel(); return -EIO; } if (amradio_setfreq(radio, radio->curfreq) < 0) warn("Set frequency failed"); unlock_kernel(); return 0;}/*close device - free driver structures */static int usb_amradio_close(struct inode *inode, struct file *file){ struct amradio_device *radio = video_get_drvdata(video_devdata(file)); if (!radio) return -ENODEV; radio->users = 0; if (radio->removed) { kfree(radio->buffer); kfree(radio); } return 0;}/* Suspend device - stop device. Need to be checked and fixed */static int usb_amradio_suspend(struct usb_interface *intf, pm_message_t message){ struct amradio_device *radio = usb_get_intfdata(intf); if (amradio_stop(radio) < 0) warn("amradio_stop() failed"); info("radio-mr800: Going into suspend.."); return 0;}/* Resume device - start device. Need to be checked and fixed */static int usb_amradio_resume(struct usb_interface *intf){ struct amradio_device *radio = usb_get_intfdata(intf); if (amradio_start(radio) < 0) warn("amradio_start() failed"); info("radio-mr800: Coming out of suspend.."); return 0;}/* File system interface */static const struct file_operations usb_amradio_fops = { .owner = THIS_MODULE, .open = usb_amradio_open, .release = usb_amradio_close, .ioctl = video_ioctl2,#ifdef CONFIG_COMPAT .compat_ioctl = v4l_compat_ioctl32,#endif .llseek = no_llseek,};static const struct v4l2_ioctl_ops usb_amradio_ioctl_ops = { .vidioc_querycap = vidioc_querycap, .vidioc_g_tuner = vidioc_g_tuner, .vidioc_s_tuner = vidioc_s_tuner, .vidioc_g_frequency = vidioc_g_frequency, .vidioc_s_frequency = vidioc_s_frequency, .vidioc_queryctrl = vidioc_queryctrl, .vidioc_g_ctrl = vidioc_g_ctrl, .vidioc_s_ctrl = vidioc_s_ctrl, .vidioc_g_audio = vidioc_g_audio, .vidioc_s_audio = vidioc_s_audio, .vidioc_g_input = vidioc_g_input, .vidioc_s_input = vidioc_s_input,};/* V4L2 interface */static struct video_device amradio_videodev_template = { .name = "AverMedia MR 800 USB FM Radio", .fops = &usb_amradio_fops, .ioctl_ops = &usb_amradio_ioctl_ops, .release = video_device_release,};/* check if the device is present and register with v4l andusb if it is */static int usb_amradio_probe(struct usb_interface *intf, const struct usb_device_id *id){ struct amradio_device *radio; radio = kmalloc(sizeof(struct amradio_device), GFP_KERNEL); if (!(radio)) return -ENOMEM; radio->buffer = kmalloc(BUFFER_LENGTH, GFP_KERNEL); if (!(radio->buffer)) { kfree(radio); return -ENOMEM; } radio->videodev = video_device_alloc(); if (!(radio->videodev)) { kfree(radio->buffer); kfree(radio); return -ENOMEM; } memcpy(radio->videodev, &amradio_videodev_template, sizeof(amradio_videodev_template)); radio->removed = 0; radio->users = 0; radio->usbdev = interface_to_usbdev(intf); radio->curfreq = 95.16 * FREQ_MUL; mutex_init(&radio->lock); video_set_drvdata(radio->videodev, radio); if (video_register_device(radio->videodev, VFL_TYPE_RADIO, radio_nr)) { warn("Could not register video device"); video_device_release(radio->videodev); kfree(radio->buffer); kfree(radio); return -EIO; } usb_set_intfdata(intf, radio); return 0;}static int __init amradio_init(void){ int retval = usb_register(&usb_amradio_driver); info(DRIVER_VERSION " " DRIVER_DESC); if (retval) err("usb_register failed. Error number %d", retval); return retval;}static void __exit amradio_exit(void){ usb_deregister(&usb_amradio_driver);}module_init(amradio_init);module_exit(amradio_exit);
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -