2005-04-16 15:20:36 -07:00
/* GemTek radio card driver for Linux (C) 1998 Jonas Munsin <jmunsin@iki.fi>
*
* 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
2006-04-08 16:06:16 -03:00
*
2005-04-16 15:20:36 -07:00
* Based on RadioTrack I / RadioReveal ( C ) 1997 M . Kirkwood
2008-10-27 15:13:47 -03:00
* Converted to new API by Alan Cox < alan @ lxorguk . ukuu . org . uk >
2005-04-16 15:20:36 -07:00
* Various bugfixes and enhancements by Russell Kroll < rkroll @ exploits . org >
*
* TODO : Allow for more than one of these foolish entities : - )
*
2006-08-08 09:10:01 -03:00
* Converted to V4L2 API by Mauro Carvalho Chehab < mchehab @ infradead . org >
2005-04-16 15:20:36 -07:00
*/
# include <linux/module.h> /* Modules */
# include <linux/init.h> /* Initdata */
2005-09-13 01:25:15 -07:00
# include <linux/ioport.h> /* request_region */
2005-04-16 15:20:36 -07:00
# include <linux/delay.h> /* udelay */
2006-08-08 09:10:01 -03:00
# include <linux/videodev2.h> /* kernel radio structs */
2009-03-06 13:50:42 -03:00
# include <linux/version.h> /* for KERNEL_VERSION MACRO */
# include <linux/mutex.h>
# include <linux/io.h> /* outb, outb_p */
2008-07-20 08:12:02 -03:00
# include <media/v4l2-ioctl.h>
2009-03-06 13:50:42 -03:00
# include <media/v4l2-device.h>
2005-04-16 15:20:36 -07:00
2009-03-06 13:50:42 -03:00
# define RADIO_VERSION KERNEL_VERSION(0, 0, 3)
2006-08-08 09:10:01 -03:00
2007-10-01 00:27:55 -03:00
/*
* Module info .
*/
MODULE_AUTHOR ( " Jonas Munsin, Pekka Sepp<70> nen <pexu@kapsi.fi> " ) ;
MODULE_DESCRIPTION ( " A driver for the GemTek Radio card. " ) ;
MODULE_LICENSE ( " GPL " ) ;
/*
* Module params .
*/
2006-08-08 09:10:01 -03:00
2005-04-16 15:20:36 -07:00
# ifndef CONFIG_RADIO_GEMTEK_PORT
# define CONFIG_RADIO_GEMTEK_PORT -1
# endif
2007-10-01 00:27:55 -03:00
# ifndef CONFIG_RADIO_GEMTEK_PROBE
# define CONFIG_RADIO_GEMTEK_PROBE 1
# endif
2005-04-16 15:20:36 -07:00
2007-10-01 00:27:55 -03:00
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 ;
2005-04-16 15:20:36 -07:00
2007-10-01 00:27:55 -03:00
module_param ( io , int , 0444 ) ;
2007-11-19 22:48:16 -03:00
MODULE_PARM_DESC ( io , " Force I/O port for the GemTek Radio card if automatic "
2007-10-01 00:27:55 -03:00
" 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 "
2007-11-19 22:48:16 -03:00
" work for the combined sound/radiocard). " ) ;
2007-10-01 00:27:55 -03:00
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)
2007-10-01 00:32:25 -03:00
/*
* 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<<FSCALE)))
# define REF_FREQ ((unsigned int)(6.39 * 16 * (1<<FSCALE)))
2007-10-01 00:27:55 -03:00
# define GEMTEK_CK 0x01 /* Clock signal */
# define GEMTEK_DA 0x02 /* Serial data */
# define GEMTEK_CE 0x04 /* Chip enable */
# define GEMTEK_NS 0x08 /* No signal */
# define GEMTEK_MT 0x10 /* Line mute */
# define GEMTEK_STDF_3_125_KHZ 0x01 /* Standard frequency 3.125 kHz */
# define GEMTEK_PLL_OFF 0x07 /* PLL off */
# define BU2614_BUS_SIZE 32 /* BU2614 / BU2614FS bus size */
# define SHORT_DELAY 5 /* usec */
# define LONG_DELAY 75 /* usec */
2009-03-06 13:50:42 -03:00
struct gemtek {
struct v4l2_device v4l2_dev ;
struct video_device vdev ;
struct mutex lock ;
2007-10-01 00:27:55 -03:00
unsigned long lastfreq ;
2005-04-16 15:20:36 -07:00
int muted ;
2009-03-06 13:50:42 -03:00
int verified ;
int io ;
2007-10-01 00:38:30 -03:00
u32 bu2614data ;
2005-04-16 15:20:36 -07:00
} ;
2009-03-06 13:50:42 -03:00
static struct gemtek gemtek_card ;
2007-10-01 00:38:30 -03:00
# define BU2614_FREQ_BITS 16 /* D0..D15, Frequency data */
# define BU2614_PORT_BITS 3 /* P0..P2, Output port control data */
# define BU2614_VOID_BITS 4 /* unused */
# define BU2614_FMES_BITS 1 /* CT, Frequency measurement beginning data */
# define BU2614_STDF_BITS 3 /* R0..R2, Standard frequency data */
# define BU2614_SWIN_BITS 1 /* S, Switch between FMIN / AMIN */
# define BU2614_SWAL_BITS 1 /* PS, Swallow counter division (AMIN only)*/
# define BU2614_VOID2_BITS 1 /* unused */
# define BU2614_FMUN_BITS 1 /* GT, Frequency measurement time & unlock */
# define BU2614_TEST_BITS 1 /* TS, Test data is input */
# define BU2614_FREQ_SHIFT 0
# define BU2614_PORT_SHIFT (BU2614_FREQ_BITS + BU2614_FREQ_SHIFT)
# define BU2614_VOID_SHIFT (BU2614_PORT_BITS + BU2614_PORT_SHIFT)
# define BU2614_FMES_SHIFT (BU2614_VOID_BITS + BU2614_VOID_SHIFT)
# define BU2614_STDF_SHIFT (BU2614_FMES_BITS + BU2614_FMES_SHIFT)
# define BU2614_SWIN_SHIFT (BU2614_STDF_BITS + BU2614_STDF_SHIFT)
# define BU2614_SWAL_SHIFT (BU2614_SWIN_BITS + BU2614_SWIN_SHIFT)
# define BU2614_VOID2_SHIFT (BU2614_SWAL_BITS + BU2614_SWAL_SHIFT)
# define BU2614_FMUN_SHIFT (BU2614_VOID2_BITS + BU2614_VOID2_SHIFT)
# define BU2614_TEST_SHIFT (BU2614_FMUN_BITS + BU2614_FMUN_SHIFT)
# define MKMASK(field) (((1<<BU2614_##field##_BITS) - 1) << \
BU2614_ # # field # # _SHIFT )
# define BU2614_PORT_MASK MKMASK(PORT)
# define BU2614_FREQ_MASK MKMASK(FREQ)
# define BU2614_VOID_MASK MKMASK(VOID)
# define BU2614_FMES_MASK MKMASK(FMES)
# define BU2614_STDF_MASK MKMASK(STDF)
# define BU2614_SWIN_MASK MKMASK(SWIN)
# define BU2614_SWAL_MASK MKMASK(SWAL)
# define BU2614_VOID2_MASK MKMASK(VOID2)
# define BU2614_FMUN_MASK MKMASK(FMUN)
# define BU2614_TEST_MASK MKMASK(TEST)
2007-10-01 00:27:55 -03:00
/*
* Set data which will be sent to BU2614FS .
2005-04-16 15:20:36 -07:00
*/
2007-10-01 00:38:30 -03:00
# define gemtek_bu2614_set(dev, field, data) ((dev)->bu2614data = \
( ( dev ) - > bu2614data & ~ field # # _MASK ) | ( ( data ) < < field # # _SHIFT ) )
2007-10-01 00:27:55 -03:00
/*
* Transmit settings to BU2614FS over GemTek IC .
*/
2009-03-06 13:50:42 -03:00
static void gemtek_bu2614_transmit ( struct gemtek * gt )
2007-10-01 00:27:55 -03:00
{
int i , bit , q , mute ;
2009-03-06 13:50:42 -03:00
mutex_lock ( & gt - > lock ) ;
2007-10-01 00:27:55 -03:00
2009-03-06 13:50:42 -03:00
mute = gt - > muted ? GEMTEK_MT : 0x00 ;
2007-10-01 00:27:55 -03:00
2009-03-06 13:50:42 -03:00
outb_p ( mute | GEMTEK_DA | GEMTEK_CK , gt - > io ) ;
2007-10-01 00:27:55 -03:00
udelay ( SHORT_DELAY ) ;
2009-03-06 13:50:42 -03:00
outb_p ( mute | GEMTEK_CE | GEMTEK_DA | GEMTEK_CK , gt - > io ) ;
2007-10-01 00:27:55 -03:00
udelay ( LONG_DELAY ) ;
2009-03-06 13:50:42 -03:00
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 ) ;
2007-10-01 00:27:55 -03:00
}
2009-03-06 13:50:42 -03:00
outb_p ( mute | GEMTEK_DA | GEMTEK_CK , gt - > io ) ;
2007-10-01 00:27:55 -03:00
udelay ( SHORT_DELAY ) ;
2009-03-06 13:50:42 -03:00
outb_p ( mute | GEMTEK_CE | GEMTEK_DA | GEMTEK_CK , gt - > io ) ;
2007-10-01 00:27:55 -03:00
udelay ( LONG_DELAY ) ;
2009-03-06 13:50:42 -03:00
mutex_unlock ( & gt - > lock ) ;
2005-04-16 15:20:36 -07:00
}
2007-10-01 00:27:55 -03:00
/*
2007-10-01 00:32:25 -03:00
* Calculate divisor from FM - frequency for BU2614FS ( 3.125 KHz STDF expected ) .
2007-10-01 00:27:55 -03:00
*/
2007-10-01 00:32:25 -03:00
static unsigned long gemtek_convfreq ( unsigned long freq )
2005-04-16 15:20:36 -07:00
{
2007-10-01 00:32:25 -03:00
return ( ( freq < < FSCALE ) + IF_OFFSET + REF_FREQ / 2 ) / REF_FREQ ;
2007-10-01 00:27:55 -03:00
}
/*
* Set FM - frequency .
*/
2009-03-06 13:50:42 -03:00
static void gemtek_setfreq ( struct gemtek * gt , unsigned long freq )
2007-10-01 00:27:55 -03:00
{
2009-03-06 13:50:42 -03:00
if ( keepmuted & & hardmute & & gt - > muted )
2005-04-16 15:20:36 -07:00
return ;
2007-10-01 00:27:55 -03:00
2009-03-06 13:50:42 -03:00
freq = clamp_val ( freq , GEMTEK_LOWFREQ , GEMTEK_HIGHFREQ ) ;
2007-10-01 00:27:55 -03:00
2009-03-06 13:50:42 -03:00
gt - > lastfreq = freq ;
gt - > muted = 0 ;
2007-10-01 00:27:55 -03:00
2009-03-06 13:50:42 -03:00
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 ) ;
2007-10-01 00:27:55 -03:00
2009-03-06 13:50:42 -03:00
gemtek_bu2614_set ( gt , BU2614_STDF , GEMTEK_STDF_3_125_KHZ ) ;
gemtek_bu2614_set ( gt , BU2614_FREQ , gemtek_convfreq ( freq ) ) ;
2007-10-01 00:27:55 -03:00
2009-03-06 13:50:42 -03:00
gemtek_bu2614_transmit ( gt ) ;
2005-04-16 15:20:36 -07:00
}
2007-10-01 00:27:55 -03:00
/*
* Set mute flag .
*/
2009-03-06 13:50:42 -03:00
static void gemtek_mute ( struct gemtek * gt )
2005-04-16 15:20:36 -07:00
{
2007-10-01 00:27:55 -03:00
int i ;
2009-03-06 13:50:42 -03:00
gt - > muted = 1 ;
2007-10-01 00:27:55 -03:00
if ( hardmute ) {
/* Turn off PLL, disable data output */
2009-03-06 13:50:42 -03:00
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 ;
}
2007-10-01 00:27:55 -03:00
2009-03-06 13:50:42 -03:00
mutex_lock ( & gt - > lock ) ;
2007-10-01 00:27:55 -03:00
2009-03-06 13:50:42 -03:00
/* 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 ( & gt - > lock ) ;
2005-04-16 15:20:36 -07:00
}
2007-10-01 00:27:55 -03:00
/*
* Unset mute flag .
*/
2009-03-06 13:50:42 -03:00
static void gemtek_unmute ( struct gemtek * gt )
2005-04-16 15:20:36 -07:00
{
2007-10-01 00:27:55 -03:00
int i ;
2009-03-06 13:50:42 -03:00
gt - > muted = 0 ;
2007-10-01 00:27:55 -03:00
if ( hardmute ) {
/* Turn PLL back on. */
2009-03-06 13:50:42 -03:00
gemtek_setfreq ( gt , gt - > lastfreq ) ;
return ;
}
mutex_lock ( & gt - > lock ) ;
2007-10-01 00:27:55 -03:00
2009-03-06 13:50:42 -03:00
i = inb_p ( gt - > io ) ;
outb_p ( i > > 5 , gt - > io ) ;
udelay ( SHORT_DELAY ) ;
2007-10-01 00:27:55 -03:00
2009-03-06 13:50:42 -03:00
mutex_unlock ( & gt - > lock ) ;
2005-04-16 15:20:36 -07:00
}
2007-10-01 00:27:55 -03:00
/*
* Get signal strength ( = stereo status ) .
*/
2009-03-06 13:50:42 -03:00
static inline int gemtek_getsigstr ( struct gemtek * gt )
2005-04-16 15:20:36 -07:00
{
2009-03-06 13:50:42 -03:00
int sig ;
mutex_lock ( & gt - > lock ) ;
sig = inb_p ( gt - > io ) & GEMTEK_NS ? 0 : 1 ;
mutex_unlock ( & gt - > lock ) ;
return sig ;
2007-10-01 00:27:55 -03:00
}
2005-04-16 15:20:36 -07:00
2007-10-01 00:27:55 -03:00
/*
* Check if requested card acts like GemTek Radio card .
*/
2009-03-06 13:50:42 -03:00
static int gemtek_verify ( struct gemtek * gt , int port )
2007-10-01 00:27:55 -03:00
{
int i , q ;
2005-04-16 15:20:36 -07:00
2009-03-06 13:50:42 -03:00
if ( gt - > verified = = port )
2007-10-01 00:27:55 -03:00
return 1 ;
2005-04-16 15:20:36 -07:00
2009-03-06 13:50:42 -03:00
mutex_lock ( & gt - > lock ) ;
2006-04-08 16:06:16 -03:00
2007-10-01 00:27:55 -03:00
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 ) ;
2005-04-16 15:20:36 -07:00
2007-10-01 00:27:55 -03:00
if ( ( inb_p ( port ) & ( ~ GEMTEK_NS ) ) ! = ( 0x17 | ( 1 < < ( i + 5 ) ) ) ) {
2009-03-06 13:50:42 -03:00
mutex_unlock ( & gt - > lock ) ;
2007-10-01 00:27:55 -03:00
return 0 ;
}
}
outb_p ( q > > 5 , port ) ; /* Write bus contents back. */
udelay ( SHORT_DELAY ) ;
2005-04-16 15:20:36 -07:00
2009-03-06 13:50:42 -03:00
mutex_unlock ( & gt - > lock ) ;
gt - > verified = port ;
2006-04-08 16:06:16 -03:00
2007-10-01 00:27:55 -03:00
return 1 ;
2005-04-16 15:20:36 -07:00
}
2007-10-01 00:27:55 -03:00
/*
* Automatic probing for card .
*/
2009-03-06 13:50:42 -03:00
static int gemtek_probe ( struct gemtek * gt )
2005-04-16 15:20:36 -07:00
{
2009-03-06 13:50:42 -03:00
struct v4l2_device * v4l2_dev = & gt - > v4l2_dev ;
2007-10-01 00:27:55 -03:00
int ioports [ ] = { 0x20c , 0x30c , 0x24c , 0x34c , 0x248 , 0x28c } ;
int i ;
if ( ! probe ) {
2009-03-06 13:50:42 -03:00
v4l2_info ( v4l2_dev , " Automatic device probing disabled. \n " ) ;
2007-10-01 00:27:55 -03:00
return - 1 ;
}
2009-03-06 13:50:42 -03:00
v4l2_info ( v4l2_dev , " Automatic device probing enabled. \n " ) ;
2007-10-01 00:27:55 -03:00
for ( i = 0 ; i < ARRAY_SIZE ( ioports ) ; + + i ) {
2009-03-06 13:50:42 -03:00
v4l2_info ( v4l2_dev , " Trying I/O port 0x%x... \n " , ioports [ i ] ) ;
2007-10-01 00:27:55 -03:00
if ( ! request_region ( ioports [ i ] , 1 , " gemtek-probe " ) ) {
2009-03-06 13:50:42 -03:00
v4l2_warn ( v4l2_dev , " I/O port 0x%x busy! \n " ,
2007-10-01 00:27:55 -03:00
ioports [ i ] ) ;
continue ;
}
2009-03-06 13:50:42 -03:00
if ( gemtek_verify ( gt , ioports [ i ] ) ) {
v4l2_info ( v4l2_dev , " Card found from I/O port "
2007-10-01 00:27:55 -03:00
" 0x%x! \n " , ioports [ i ] ) ;
release_region ( ioports [ i ] , 1 ) ;
2009-03-06 13:50:42 -03:00
gt - > io = ioports [ i ] ;
return gt - > io ;
2007-10-01 00:27:55 -03:00
}
release_region ( ioports [ i ] , 1 ) ;
}
2009-03-06 13:50:42 -03:00
v4l2_err ( v4l2_dev , " Automatic probing failed! \n " ) ;
2007-10-01 00:27:55 -03:00
return - 1 ;
2005-04-16 15:20:36 -07:00
}
2007-10-01 00:27:55 -03:00
/*
* Video 4 Linux stuff .
*/
2008-08-23 04:49:13 -03:00
2008-12-30 06:58:20 -03:00
static const struct v4l2_file_operations gemtek_fops = {
2007-10-01 00:27:55 -03:00
. owner = THIS_MODULE ,
2010-11-14 09:36:23 -03:00
. unlocked_ioctl = video_ioctl2 ,
2007-10-01 00:27:55 -03:00
} ;
static int vidioc_querycap ( struct file * file , void * priv ,
struct v4l2_capability * v )
2007-04-25 00:15:46 -03:00
{
strlcpy ( v - > driver , " radio-gemtek " , sizeof ( v - > driver ) ) ;
strlcpy ( v - > card , " GemTek " , sizeof ( v - > card ) ) ;
2009-03-06 13:50:42 -03:00
strlcpy ( v - > bus_info , " ISA " , sizeof ( v - > bus_info ) ) ;
2007-04-25 00:15:46 -03:00
v - > version = RADIO_VERSION ;
2009-03-06 13:50:42 -03:00
v - > capabilities = V4L2_CAP_TUNER | V4L2_CAP_RADIO ;
2007-04-25 00:15:46 -03:00
return 0 ;
}
2007-10-01 00:27:55 -03:00
static int vidioc_g_tuner ( struct file * file , void * priv , struct v4l2_tuner * v )
2005-04-16 15:20:36 -07:00
{
2009-03-06 13:50:42 -03:00
struct gemtek * gt = video_drvdata ( file ) ;
2007-04-25 00:15:46 -03:00
if ( v - > index > 0 )
return - EINVAL ;
2006-08-08 09:10:01 -03:00
2009-03-06 13:50:42 -03:00
strlcpy ( v - > name , " FM " , sizeof ( v - > name ) ) ;
2007-04-25 00:15:46 -03:00
v - > type = V4L2_TUNER_RADIO ;
2007-10-01 00:27:55 -03:00
v - > rangelow = GEMTEK_LOWFREQ ;
v - > rangehigh = GEMTEK_HIGHFREQ ;
v - > capability = V4L2_TUNER_CAP_LOW | V4L2_TUNER_CAP_STEREO ;
2009-03-06 13:50:42 -03:00
v - > signal = 0xffff * gemtek_getsigstr ( gt ) ;
2007-10-01 00:27:55 -03:00
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 ;
}
2007-04-25 00:15:46 -03:00
return 0 ;
}
2007-10-01 00:27:55 -03:00
static int vidioc_s_tuner ( struct file * file , void * priv , struct v4l2_tuner * v )
2007-04-25 00:15:46 -03:00
{
2009-03-06 13:50:42 -03:00
return ( v - > index ! = 0 ) ? - EINVAL : 0 ;
2007-04-25 00:15:46 -03:00
}
2006-08-08 09:10:01 -03:00
2009-03-06 13:50:42 -03:00
static int vidioc_g_frequency ( struct file * file , void * priv ,
2007-10-01 00:27:55 -03:00
struct v4l2_frequency * f )
2007-04-25 00:15:46 -03:00
{
2009-03-06 13:50:42 -03:00
struct gemtek * gt = video_drvdata ( file ) ;
2007-10-01 00:27:55 -03:00
2009-03-06 13:50:42 -03:00
if ( f - > tuner ! = 0 )
return - EINVAL ;
f - > type = V4L2_TUNER_RADIO ;
f - > frequency = gt - > lastfreq ;
2007-04-25 00:15:46 -03:00
return 0 ;
}
2006-08-08 09:10:01 -03:00
2009-03-06 13:50:42 -03:00
static int vidioc_s_frequency ( struct file * file , void * priv ,
2007-10-01 00:27:55 -03:00
struct v4l2_frequency * f )
2007-04-25 00:15:46 -03:00
{
2009-03-06 13:50:42 -03:00
struct gemtek * gt = video_drvdata ( file ) ;
2006-08-08 09:10:01 -03:00
2009-03-06 13:50:42 -03:00
if ( f - > tuner ! = 0 | | f - > type ! = V4L2_TUNER_RADIO )
return - EINVAL ;
gemtek_setfreq ( gt , f - > frequency ) ;
2007-04-25 00:15:46 -03:00
return 0 ;
}
2006-08-08 09:10:01 -03:00
2007-04-25 00:15:46 -03:00
static int vidioc_queryctrl ( struct file * file , void * priv ,
2007-10-01 00:27:55 -03:00
struct v4l2_queryctrl * qc )
2007-04-25 00:15:46 -03:00
{
2009-03-06 13:50:42 -03:00
switch ( qc - > id ) {
case V4L2_CID_AUDIO_MUTE :
return v4l2_ctrl_query_fill ( qc , 0 , 1 , 1 , 0 ) ;
default :
return - EINVAL ;
2007-04-25 00:15:46 -03:00
}
}
2005-04-16 15:20:36 -07:00
2007-04-25 00:15:46 -03:00
static int vidioc_g_ctrl ( struct file * file , void * priv ,
2007-10-01 00:27:55 -03:00
struct v4l2_control * ctrl )
2007-04-25 00:15:46 -03:00
{
2009-03-06 13:50:42 -03:00
struct gemtek * gt = video_drvdata ( file ) ;
2005-04-16 15:20:36 -07:00
2007-04-25 00:15:46 -03:00
switch ( ctrl - > id ) {
case V4L2_CID_AUDIO_MUTE :
2009-03-06 13:50:42 -03:00
ctrl - > value = gt - > muted ;
2007-04-25 00:15:46 -03:00
return 0 ;
}
return - EINVAL ;
}
static int vidioc_s_ctrl ( struct file * file , void * priv ,
2007-10-01 00:27:55 -03:00
struct v4l2_control * ctrl )
2007-04-25 00:15:46 -03:00
{
2009-03-06 13:50:42 -03:00
struct gemtek * gt = video_drvdata ( file ) ;
2007-04-25 00:15:46 -03:00
switch ( ctrl - > id ) {
case V4L2_CID_AUDIO_MUTE :
if ( ctrl - > value )
2009-03-06 13:50:42 -03:00
gemtek_mute ( gt ) ;
2007-04-25 00:15:46 -03:00
else
2009-03-06 13:50:42 -03:00
gemtek_unmute ( gt ) ;
2007-04-25 00:15:46 -03:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2007-04-25 00:15:46 -03:00
return - EINVAL ;
2005-04-16 15:20:36 -07:00
}
2007-04-25 00:15:46 -03:00
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 )
{
2009-03-06 13:50:42 -03:00
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 ;
2007-04-25 00:15:46 -03:00
return 0 ;
}
2007-10-01 00:27:55 -03:00
static int vidioc_s_audio ( struct file * file , void * priv , struct v4l2_audio * a )
2007-04-25 00:15:46 -03:00
{
2009-03-06 13:50:42 -03:00
return ( a - > index ! = 0 ) ? - EINVAL : 0 ;
2005-04-16 15:20:36 -07:00
}
2008-07-21 02:57:38 -03:00
static const struct v4l2_ioctl_ops gemtek_ioctl_ops = {
2007-10-01 00:27:55 -03:00
. 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
2005-04-16 15:20:36 -07:00
} ;
2007-10-01 00:27:55 -03:00
/*
* Initialization / cleanup related stuff .
*/
2005-04-16 15:20:36 -07:00
static int __init gemtek_init ( void )
{
2009-03-06 13:50:42 -03:00
struct gemtek * gt = & gemtek_card ;
struct v4l2_device * v4l2_dev = & gt - > v4l2_dev ;
int res ;
2005-04-16 15:20:36 -07:00
2009-03-06 13:50:42 -03:00
strlcpy ( v4l2_dev - > name , " gemtek " , sizeof ( v4l2_dev - > name ) ) ;
2006-04-08 16:06:16 -03:00
2009-03-06 13:50:42 -03:00
v4l2_info ( v4l2_dev , " GemTek Radio card driver: v0.0.3 \n " ) ;
mutex_init ( & gt - > 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 ) ;
2007-10-01 00:27:55 -03:00
return - EBUSY ;
}
2009-03-06 13:50:42 -03:00
if ( ! gemtek_verify ( gt , gt - > io ) )
v4l2_warn ( v4l2_dev , " Card at I/O port 0x%x does not "
2007-10-01 00:27:55 -03:00
" respond properly, check your "
2009-03-06 13:50:42 -03:00
" configuration. \n " , gt - > io ) ;
2007-10-01 00:27:55 -03:00
else
2009-03-06 13:50:42 -03:00
v4l2_info ( v4l2_dev , " Using I/O port 0x%x. \n " , gt - > io ) ;
2007-10-01 00:27:55 -03:00
} else if ( probe ) {
2009-03-06 13:50:42 -03:00
v4l2_err ( v4l2_dev , " Automatic probing failed and no "
2007-10-01 00:27:55 -03:00
" fixed I/O port defined. \n " ) ;
return - ENODEV ;
} else {
2009-03-06 13:50:42 -03:00
v4l2_err ( v4l2_dev , " Automatic probing disabled but no fixed "
2007-10-01 00:27:55 -03:00
" I/O port defined. " ) ;
2005-04-16 15:20:36 -07:00
return - EINVAL ;
}
2009-03-06 13:50:42 -03:00
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 ;
}
2007-10-01 00:27:55 -03:00
2009-03-06 13:50:42 -03:00
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 ( & gt - > vdev , gt ) ;
2007-10-01 00:27:55 -03:00
/* Set defaults */
2009-03-06 13:50:42 -03:00
gt - > lastfreq = GEMTEK_LOWFREQ ;
gt - > bu2614data = 0 ;
2005-04-16 15:20:36 -07:00
2007-10-01 00:27:55 -03:00
if ( initmute )
2009-03-06 13:50:42 -03:00
gemtek_mute ( gt ) ;
2005-04-16 15:20:36 -07:00
2010-11-14 09:36:23 -03:00
if ( video_register_device ( & gt - > vdev , VFL_TYPE_RADIO , radio_nr ) < 0 ) {
v4l2_device_unregister ( v4l2_dev ) ;
release_region ( gt - > io , 1 ) ;
return - EBUSY ;
}
2005-04-16 15:20:36 -07:00
return 0 ;
}
2007-10-01 00:27:55 -03:00
/*
* Module cleanup
*/
static void __exit gemtek_exit ( void )
2005-04-16 15:20:36 -07:00
{
2009-03-06 13:50:42 -03:00
struct gemtek * gt = & gemtek_card ;
struct v4l2_device * v4l2_dev = & gt - > v4l2_dev ;
2007-10-01 00:27:55 -03:00
if ( shutdown ) {
hardmute = 1 ; /* Turn off PLL */
2009-03-06 13:50:42 -03:00
gemtek_mute ( gt ) ;
2007-10-01 00:27:55 -03:00
} else {
2009-03-06 13:50:42 -03:00
v4l2_info ( v4l2_dev , " Module unloaded but card not muted! \n " ) ;
2007-10-01 00:27:55 -03:00
}
2009-03-06 13:50:42 -03:00
video_unregister_device ( & gt - > vdev ) ;
v4l2_device_unregister ( & gt - > v4l2_dev ) ;
release_region ( gt - > io , 1 ) ;
2005-04-16 15:20:36 -07:00
}
module_init ( gemtek_init ) ;
2007-10-01 00:27:55 -03:00
module_exit ( gemtek_exit ) ;