/* GemTek radio card driver for Linux (C) 1998 Jonas Munsin * * GemTek hasn't released any specs on the card, so the protocol had to * be reverse engineered with dosemu. * * Besides the protocol changes, this is mostly a copy of: * * RadioTrack II driver for Linux radio support (C) 1998 Ben Pfaff * * Based on RadioTrack I/RadioReveal (C) 1997 M. Kirkwood * Converted to new API by Alan Cox * Various bugfixes and enhancements by Russell Kroll * * TODO: Allow for more than one of these foolish entities :-) * * Converted to V4L2 API by Mauro Carvalho Chehab */ #include /* Modules */ #include /* Initdata */ #include /* request_region */ #include /* udelay */ #include /* kernel radio structs */ #include /* for KERNEL_VERSION MACRO */ #include #include /* outb, outb_p */ #include #include #include "compat.h" #define RADIO_VERSION KERNEL_VERSION(0, 0, 3) /* * Module info. */ MODULE_AUTHOR("Jonas Munsin, Pekka Seppänen "); MODULE_DESCRIPTION("A driver for the GemTek Radio card."); MODULE_LICENSE("GPL"); /* * Module params. */ #ifndef CONFIG_RADIO_GEMTEK_PORT #define CONFIG_RADIO_GEMTEK_PORT -1 #endif #ifndef CONFIG_RADIO_GEMTEK_PROBE #define CONFIG_RADIO_GEMTEK_PROBE 1 #endif static int io = CONFIG_RADIO_GEMTEK_PORT; static int probe = CONFIG_RADIO_GEMTEK_PROBE; static int hardmute; static int shutdown = 1; static int keepmuted = 1; static int initmute = 1; static int radio_nr = -1; module_param(io, int, 0444); MODULE_PARM_DESC(io, "Force I/O port for the GemTek Radio card if automatic " "probing is disabled or fails. The most common I/O ports are: 0x20c " "0x30c, 0x24c or 0x34c (0x20c, 0x248 and 0x28c have been reported to " "work for the combined sound/radiocard)."); module_param(probe, bool, 0444); MODULE_PARM_DESC(probe, "Enable automatic device probing. Note: only the most " "common I/O ports used by the card are probed."); module_param(hardmute, bool, 0644); MODULE_PARM_DESC(hardmute, "Enable `hard muting' by shutting down PLL, may " "reduce static noise."); module_param(shutdown, bool, 0644); MODULE_PARM_DESC(shutdown, "Enable shutting down PLL and muting line when " "module is unloaded."); module_param(keepmuted, bool, 0644); MODULE_PARM_DESC(keepmuted, "Keep card muted even when frequency is changed."); module_param(initmute, bool, 0444); MODULE_PARM_DESC(initmute, "Mute card when module is loaded."); module_param(radio_nr, int, 0444); /* * Functions for controlling the card. */ #define GEMTEK_LOWFREQ (87*16000) #define GEMTEK_HIGHFREQ (108*16000) /* * Frequency calculation constants. Intermediate frequency 10.52 MHz (nominal * value 10.7 MHz), reference divisor 6.39 kHz (nominal 6.25 kHz). */ #define FSCALE 8 #define IF_OFFSET ((unsigned int)(10.52 * 16000 * (1<bu2614data = \ ((dev)->bu2614data & ~field##_MASK) | ((data) << field##_SHIFT)) /* * Transmit settings to BU2614FS over GemTek IC. */ static void gemtek_bu2614_transmit(struct gemtek *gt) { int i, bit, q, mute; mutex_lock(>->lock); mute = gt->muted ? GEMTEK_MT : 0x00; outb_p(mute | GEMTEK_DA | GEMTEK_CK, gt->io); udelay(SHORT_DELAY); outb_p(mute | GEMTEK_CE | GEMTEK_DA | GEMTEK_CK, gt->io); udelay(LONG_DELAY); for (i = 0, q = gt->bu2614data; i < 32; i++, q >>= 1) { bit = (q & 1) ? GEMTEK_DA : 0; outb_p(mute | GEMTEK_CE | bit, gt->io); udelay(SHORT_DELAY); outb_p(mute | GEMTEK_CE | bit | GEMTEK_CK, gt->io); udelay(SHORT_DELAY); } outb_p(mute | GEMTEK_DA | GEMTEK_CK, gt->io); udelay(SHORT_DELAY); outb_p(mute | GEMTEK_CE | GEMTEK_DA | GEMTEK_CK, gt->io); udelay(LONG_DELAY); mutex_unlock(>->lock); } /* * Calculate divisor from FM-frequency for BU2614FS (3.125 KHz STDF expected). */ static unsigned long gemtek_convfreq(unsigned long freq) { return ((freq<muted) return; freq = clamp_val(freq, GEMTEK_LOWFREQ, GEMTEK_HIGHFREQ); gt->lastfreq = freq; gt->muted = 0; gemtek_bu2614_set(gt, BU2614_PORT, 0); gemtek_bu2614_set(gt, BU2614_FMES, 0); gemtek_bu2614_set(gt, BU2614_SWIN, 0); /* FM-mode */ gemtek_bu2614_set(gt, BU2614_SWAL, 0); gemtek_bu2614_set(gt, BU2614_FMUN, 1); /* GT bit set */ gemtek_bu2614_set(gt, BU2614_TEST, 0); gemtek_bu2614_set(gt, BU2614_STDF, GEMTEK_STDF_3_125_KHZ); gemtek_bu2614_set(gt, BU2614_FREQ, gemtek_convfreq(freq)); gemtek_bu2614_transmit(gt); } /* * Set mute flag. */ static void gemtek_mute(struct gemtek *gt) { int i; gt->muted = 1; if (hardmute) { /* Turn off PLL, disable data output */ gemtek_bu2614_set(gt, BU2614_PORT, 0); gemtek_bu2614_set(gt, BU2614_FMES, 0); /* CT bit off */ gemtek_bu2614_set(gt, BU2614_SWIN, 0); /* FM-mode */ gemtek_bu2614_set(gt, BU2614_SWAL, 0); gemtek_bu2614_set(gt, BU2614_FMUN, 0); /* GT bit off */ gemtek_bu2614_set(gt, BU2614_TEST, 0); gemtek_bu2614_set(gt, BU2614_STDF, GEMTEK_PLL_OFF); gemtek_bu2614_set(gt, BU2614_FREQ, 0); gemtek_bu2614_transmit(gt); return; } mutex_lock(>->lock); /* Read bus contents (CE, CK and DA). */ i = inb_p(gt->io); /* Write it back with mute flag set. */ outb_p((i >> 5) | GEMTEK_MT, gt->io); udelay(SHORT_DELAY); mutex_unlock(>->lock); } /* * Unset mute flag. */ static void gemtek_unmute(struct gemtek *gt) { int i; gt->muted = 0; if (hardmute) { /* Turn PLL back on. */ gemtek_setfreq(gt, gt->lastfreq); return; } mutex_lock(>->lock); i = inb_p(gt->io); outb_p(i >> 5, gt->io); udelay(SHORT_DELAY); mutex_unlock(>->lock); } /* * Get signal strength (= stereo status). */ static inline int gemtek_getsigstr(struct gemtek *gt) { int sig; mutex_lock(>->lock); sig = inb_p(gt->io) & GEMTEK_NS ? 0 : 1; mutex_unlock(>->lock); return sig; } /* * Check if requested card acts like GemTek Radio card. */ static int gemtek_verify(struct gemtek *gt, int port) { int i, q; if (gt->verified == port) return 1; mutex_lock(>->lock); q = inb_p(port); /* Read bus contents before probing. */ /* Try to turn on CE, CK and DA respectively and check if card responds properly. */ for (i = 0; i < 3; ++i) { outb_p(1 << i, port); udelay(SHORT_DELAY); if ((inb_p(port) & (~GEMTEK_NS)) != (0x17 | (1 << (i + 5)))) { mutex_unlock(>->lock); return 0; } } outb_p(q >> 5, port); /* Write bus contents back. */ udelay(SHORT_DELAY); mutex_unlock(>->lock); gt->verified = port; return 1; } /* * Automatic probing for card. */ static int gemtek_probe(struct gemtek *gt) { struct v4l2_device *v4l2_dev = >->v4l2_dev; int ioports[] = { 0x20c, 0x30c, 0x24c, 0x34c, 0x248, 0x28c }; int i; if (!probe) { v4l2_info(v4l2_dev, "Automatic device probing disabled.\n"); return -1; } v4l2_info(v4l2_dev, "Automatic device probing enabled.\n"); for (i = 0; i < ARRAY_SIZE(ioports); ++i) { v4l2_info(v4l2_dev, "Trying I/O port 0x%x...\n", ioports[i]); if (!request_region(ioports[i], 1, "gemtek-probe")) { v4l2_warn(v4l2_dev, "I/O port 0x%x busy!\n", ioports[i]); continue; } if (gemtek_verify(gt, ioports[i])) { v4l2_info(v4l2_dev, "Card found from I/O port " "0x%x!\n", ioports[i]); release_region(ioports[i], 1); gt->io = ioports[i]; return gt->io; } release_region(ioports[i], 1); } v4l2_err(v4l2_dev, "Automatic probing failed!\n"); return -1; } /* * Video 4 Linux stuff. */ static int gemtek_open(struct file *file) { return 0; } static int gemtek_release(struct file *file) { return 0; } static const struct v4l2_file_operations gemtek_fops = { .owner = THIS_MODULE, .open = gemtek_open, .release = gemtek_release, .ioctl = video_ioctl2, }; static int vidioc_querycap(struct file *file, void *priv, struct v4l2_capability *v) { strlcpy(v->driver, "radio-gemtek", sizeof(v->driver)); strlcpy(v->card, "GemTek", sizeof(v->card)); strlcpy(v->bus_info, "ISA", sizeof(v->bus_info)); v->version = RADIO_VERSION; v->capabilities = V4L2_CAP_TUNER | V4L2_CAP_RADIO; return 0; } static int vidioc_g_tuner(struct file *file, void *priv, struct v4l2_tuner *v) { struct gemtek *gt = video_drvdata(file); if (v->index > 0) return -EINVAL; strlcpy(v->name, "FM", sizeof(v->name)); v->type = V4L2_TUNER_RADIO; v->rangelow = GEMTEK_LOWFREQ; v->rangehigh = GEMTEK_HIGHFREQ; v->capability = V4L2_TUNER_CAP_LOW | V4L2_TUNER_CAP_STEREO; v->signal = 0xffff * gemtek_getsigstr(gt); if (v->signal) { v->audmode = V4L2_TUNER_MODE_STEREO; v->rxsubchans = V4L2_TUNER_SUB_STEREO; } else { v->audmode = V4L2_TUNER_MODE_MONO; v->rxsubchans = V4L2_TUNER_SUB_MONO; } return 0; } static int vidioc_s_tuner(struct file *file, void *priv, struct v4l2_tuner *v) { return (v->index != 0) ? -EINVAL : 0; } static int vidioc_g_frequency(struct file *file, void *priv, struct v4l2_frequency *f) { struct gemtek *gt = video_drvdata(file); if (f->tuner != 0) return -EINVAL; f->type = V4L2_TUNER_RADIO; f->frequency = gt->lastfreq; return 0; } static int vidioc_s_frequency(struct file *file, void *priv, struct v4l2_frequency *f) { struct gemtek *gt = video_drvdata(file); if (f->tuner != 0 || f->type != V4L2_TUNER_RADIO) return -EINVAL; gemtek_setfreq(gt, f->frequency); return 0; } static int vidioc_queryctrl(struct file *file, void *priv, struct v4l2_queryctrl *qc) { switch (qc->id) { case V4L2_CID_AUDIO_MUTE: return v4l2_ctrl_query_fill(qc, 0, 1, 1, 0); default: return -EINVAL; } } static int vidioc_g_ctrl(struct file *file, void *priv, struct v4l2_control *ctrl) { struct gemtek *gt = video_drvdata(file); switch (ctrl->id) { case V4L2_CID_AUDIO_MUTE: ctrl->value = gt->muted; return 0; } return -EINVAL; } static int vidioc_s_ctrl(struct file *file, void *priv, struct v4l2_control *ctrl) { struct gemtek *gt = video_drvdata(file); switch (ctrl->id) { case V4L2_CID_AUDIO_MUTE: if (ctrl->value) gemtek_mute(gt); else gemtek_unmute(gt); return 0; } return -EINVAL; } static int vidioc_g_input(struct file *filp, void *priv, unsigned int *i) { *i = 0; return 0; } static int vidioc_s_input(struct file *filp, void *priv, unsigned int i) { return (i != 0) ? -EINVAL : 0; } static int vidioc_g_audio(struct file *file, void *priv, struct v4l2_audio *a) { a->index = 0; strlcpy(a->name, "Radio", sizeof(a->name)); a->capability = V4L2_AUDCAP_STEREO; return 0; } static int vidioc_s_audio(struct file *file, void *priv, struct v4l2_audio *a) { return (a->index != 0) ? -EINVAL : 0; } static const struct v4l2_ioctl_ops gemtek_ioctl_ops = { .vidioc_querycap = vidioc_querycap, .vidioc_g_tuner = vidioc_g_tuner, .vidioc_s_tuner = vidioc_s_tuner, .vidioc_g_audio = vidioc_g_audio, .vidioc_s_audio = vidioc_s_audio, .vidioc_g_input = vidioc_g_input, .vidioc_s_input = vidioc_s_input, .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 }; /* * Initialization / cleanup related stuff. */ static int __init gemtek_init(void) { struct gemtek *gt = &gemtek_card; struct v4l2_device *v4l2_dev = >->v4l2_dev; int res; strlcpy(v4l2_dev->name, "gemtek", sizeof(v4l2_dev->name)); v4l2_info(v4l2_dev, "GemTek Radio card driver: v0.0.3\n"); mutex_init(>->lock); gt->verified = -1; gt->io = io; gemtek_probe(gt); if (gt->io) { if (!request_region(gt->io, 1, "gemtek")) { v4l2_err(v4l2_dev, "I/O port 0x%x already in use.\n", gt->io); return -EBUSY; } if (!gemtek_verify(gt, gt->io)) v4l2_warn(v4l2_dev, "Card at I/O port 0x%x does not " "respond properly, check your " "configuration.\n", gt->io); else v4l2_info(v4l2_dev, "Using I/O port 0x%x.\n", gt->io); } else if (probe) { v4l2_err(v4l2_dev, "Automatic probing failed and no " "fixed I/O port defined.\n"); return -ENODEV; } else { v4l2_err(v4l2_dev, "Automatic probing disabled but no fixed " "I/O port defined."); return -EINVAL; } res = v4l2_device_register(NULL, v4l2_dev); if (res < 0) { v4l2_err(v4l2_dev, "Could not register v4l2_device\n"); release_region(gt->io, 1); return res; } strlcpy(gt->vdev.name, v4l2_dev->name, sizeof(gt->vdev.name)); gt->vdev.v4l2_dev = v4l2_dev; gt->vdev.fops = &gemtek_fops; gt->vdev.ioctl_ops = &gemtek_ioctl_ops; gt->vdev.release = video_device_release_empty; video_set_drvdata(>->vdev, gt); if (video_register_device(>->vdev, VFL_TYPE_RADIO, radio_nr) < 0) { v4l2_device_unregister(v4l2_dev); release_region(gt->io, 1); return -EBUSY; } /* Set defaults */ gt->lastfreq = GEMTEK_LOWFREQ; gt->bu2614data = 0; if (initmute) gemtek_mute(gt); return 0; } /* * Module cleanup */ static void __exit gemtek_exit(void) { struct gemtek *gt = &gemtek_card; struct v4l2_device *v4l2_dev = >->v4l2_dev; if (shutdown) { hardmute = 1; /* Turn off PLL */ gemtek_mute(gt); } else { v4l2_info(v4l2_dev, "Module unloaded but card not muted!\n"); } video_unregister_device(>->vdev); v4l2_device_unregister(>->v4l2_dev); release_region(gt->io, 1); } module_init(gemtek_init); module_exit(gemtek_exit);