2019-05-19 13:08:20 +01:00
// SPDX-License-Identifier: GPL-2.0-only
2006-04-08 16:06:16 -03:00
/* radio-trust.c - Trust FM Radio card driver for Linux 2.2
2005-04-16 15:20:36 -07:00
* by Eric Lammerts < eric @ scintilla . utwente . nl >
*
* Based on radio - aztech . c . Original notes :
*
2006-04-08 16:06:16 -03:00
* Adapted to support the Video for Linux API by
2005-04-16 15:20:36 -07:00
* Russell Kroll < rkroll @ exploits . org > . Based on original tuner code by :
*
* Quay Ly
* Donald Song
2006-04-08 16:06:16 -03:00
* Jason Lewis ( jlewis @ twilight . vtc . vsc . edu )
2005-04-16 15:20:36 -07:00
* Scott McGrath ( smcgrath @ twilight . vtc . vsc . edu )
* William McGrath ( wmcgrath @ twilight . vtc . vsc . edu )
*
MAINTAINERS & files: Canonize the e-mails I use at files
From now on, I'll start using my @kernel.org as my development e-mail.
As such, let's remove the entries that point to the old
mchehab@s-opensource.com at MAINTAINERS file.
For the files written with a copyright with mchehab@s-opensource,
let's keep Samsung on their names, using mchehab+samsung@kernel.org,
in order to keep pointing to my employer, with sponsors the work.
For the files written before I join Samsung (on July, 4 2013),
let's just use mchehab@kernel.org.
For bug reports, we can simply point to just kernel.org, as
this will reach my mchehab+samsung inbox anyway.
Signed-off-by: Mauro Carvalho Chehab <mchehab@s-opensource.com>
Signed-off-by: Brian Warner <brian.warner@samsung.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab+samsung@kernel.org>
2018-04-25 05:34:48 -04:00
* Converted to V4L2 API by Mauro Carvalho Chehab < mchehab @ kernel . org >
2005-04-16 15:20:36 -07:00
*/
# include <linux/module.h>
# include <linux/init.h>
# include <linux/ioport.h>
2006-08-08 09:10:05 -03:00
# include <linux/videodev2.h>
2009-03-06 13:54:23 -03:00
# include <linux/io.h>
2012-02-29 05:50:27 -03:00
# include <linux/slab.h>
2009-03-06 13:54:23 -03:00
# include <media/v4l2-device.h>
2008-07-20 08:12:02 -03:00
# include <media/v4l2-ioctl.h>
2012-01-16 05:16:29 -03:00
# include "radio-isa.h"
2005-04-16 15:20:36 -07:00
2009-03-06 13:54:23 -03:00
MODULE_AUTHOR ( " Eric Lammerts, Russell Kroll, Quay Lu, Donald Song, Jason Lewis, Scott McGrath, William McGrath " ) ;
MODULE_DESCRIPTION ( " A driver for the Trust FM Radio card. " ) ;
MODULE_LICENSE ( " GPL " ) ;
2012-01-16 05:16:29 -03:00
MODULE_VERSION ( " 0.1.99 " ) ;
2006-08-08 09:10:05 -03:00
2005-04-16 15:20:36 -07:00
/* acceptable ports: 0x350 (JP3 shorted), 0x358 (JP3 open) */
# ifndef CONFIG_RADIO_TRUST_PORT
# define CONFIG_RADIO_TRUST_PORT -1
# endif
2012-01-16 05:16:29 -03:00
# define TRUST_MAX 2
2009-03-06 13:54:23 -03:00
2012-01-16 05:16:29 -03:00
static int io [ TRUST_MAX ] = { [ 0 ] = CONFIG_RADIO_TRUST_PORT ,
[ 1 . . . ( TRUST_MAX - 1 ) ] = - 1 } ;
static int radio_nr [ TRUST_MAX ] = { [ 0 . . . ( TRUST_MAX - 1 ) ] = - 1 } ;
module_param_array ( io , int , NULL , 0444 ) ;
MODULE_PARM_DESC ( io , " I/O addresses of the Trust FM Radio card (0x350 or 0x358) " ) ;
module_param_array ( radio_nr , int , NULL , 0444 ) ;
MODULE_PARM_DESC ( radio_nr , " Radio device numbers " ) ;
2009-03-06 13:54:23 -03:00
struct trust {
2012-01-16 05:16:29 -03:00
struct radio_isa_card isa ;
2009-03-06 13:54:23 -03:00
int ioval ;
} ;
2012-01-16 05:16:29 -03:00
static struct radio_isa_card * trust_alloc ( void )
{
struct trust * tr = kzalloc ( sizeof ( * tr ) , GFP_KERNEL ) ;
return tr ? & tr - > isa : NULL ;
}
2005-04-16 15:20:36 -07:00
/* i2c addresses */
# define TDA7318_ADDR 0x88
# define TSA6060T_ADDR 0xc4
2012-01-16 05:16:29 -03:00
# define TR_DELAY do { inb(tr->isa.io); inb(tr->isa.io); inb(tr->isa.io); } while (0)
# define TR_SET_SCL outb(tr->ioval |= 2, tr->isa.io)
# define TR_CLR_SCL outb(tr->ioval &= 0xfd, tr->isa.io)
# define TR_SET_SDA outb(tr->ioval |= 1, tr->isa.io)
# define TR_CLR_SDA outb(tr->ioval &= 0xfe, tr->isa.io)
2005-04-16 15:20:36 -07:00
2009-03-06 13:54:23 -03:00
static void write_i2c ( struct trust * tr , int n , . . . )
2005-04-16 15:20:36 -07:00
{
unsigned char val , mask ;
va_list args ;
va_start ( args , n ) ;
/* start condition */
TR_SET_SDA ;
TR_SET_SCL ;
TR_DELAY ;
TR_CLR_SDA ;
TR_CLR_SCL ;
TR_DELAY ;
2012-01-16 05:16:29 -03:00
for ( ; n ; n - - ) {
2005-04-16 15:20:36 -07:00
val = va_arg ( args , unsigned ) ;
2012-01-16 05:16:29 -03:00
for ( mask = 0x80 ; mask ; mask > > = 1 ) {
if ( val & mask )
2005-04-16 15:20:36 -07:00
TR_SET_SDA ;
else
TR_CLR_SDA ;
TR_SET_SCL ;
TR_DELAY ;
TR_CLR_SCL ;
TR_DELAY ;
}
/* acknowledge bit */
TR_SET_SDA ;
TR_SET_SCL ;
TR_DELAY ;
TR_CLR_SCL ;
TR_DELAY ;
}
/* stop condition */
TR_CLR_SDA ;
TR_DELAY ;
TR_SET_SCL ;
TR_DELAY ;
TR_SET_SDA ;
TR_DELAY ;
va_end ( args ) ;
}
2012-01-16 05:16:29 -03:00
static int trust_s_mute_volume ( struct radio_isa_card * isa , bool mute , int vol )
2005-04-16 15:20:36 -07:00
{
2012-01-16 05:16:29 -03:00
struct trust * tr = container_of ( isa , struct trust , isa ) ;
2005-04-16 15:20:36 -07:00
2012-01-16 05:16:29 -03:00
tr - > ioval = ( tr - > ioval & 0xf7 ) | ( mute < < 3 ) ;
outb ( tr - > ioval , isa - > io ) ;
write_i2c ( tr , 2 , TDA7318_ADDR , vol ^ 0x1f ) ;
return 0 ;
2005-04-16 15:20:36 -07:00
}
2012-01-16 05:16:29 -03:00
static int trust_s_stereo ( struct radio_isa_card * isa , bool stereo )
2005-04-16 15:20:36 -07:00
{
2012-01-16 05:16:29 -03:00
struct trust * tr = container_of ( isa , struct trust , isa ) ;
2005-04-16 15:20:36 -07:00
2012-01-16 05:16:29 -03:00
tr - > ioval = ( tr - > ioval & 0xfb ) | ( ! stereo < < 2 ) ;
outb ( tr - > ioval , isa - > io ) ;
return 0 ;
2005-04-16 15:20:36 -07:00
}
2012-01-16 05:16:29 -03:00
static u32 trust_g_signal ( struct radio_isa_card * isa )
2005-04-16 15:20:36 -07:00
{
int i , v ;
2006-04-08 16:06:16 -03:00
2009-03-06 13:54:23 -03:00
for ( i = 0 , v = 0 ; i < 100 ; i + + )
2012-01-16 05:16:29 -03:00
v | = inb ( isa - > io ) ;
2009-03-06 13:54:23 -03:00
return ( v & 1 ) ? 0 : 0xffff ;
2005-04-16 15:20:36 -07:00
}
2012-01-16 05:16:29 -03:00
static int trust_s_frequency ( struct radio_isa_card * isa , u32 freq )
2005-04-16 15:20:36 -07:00
{
2012-01-16 05:16:29 -03:00
struct trust * tr = container_of ( isa , struct trust , isa ) ;
2005-04-16 15:20:36 -07:00
2012-01-16 05:16:29 -03:00
freq / = 160 ; /* Convert to 10 kHz units */
freq + = 1070 ; /* Add 10.7 MHz IF */
write_i2c ( tr , 5 , TSA6060T_ADDR , ( freq < < 1 ) | 1 ,
freq > > 7 , 0x60 | ( ( freq > > 15 ) & 1 ) , 0 ) ;
2007-04-20 18:22:19 -03:00
return 0 ;
}
2005-04-16 15:20:36 -07:00
2012-01-16 05:16:29 -03:00
static int basstreble2chip [ 15 ] = {
0 , 1 , 2 , 3 , 4 , 5 , 6 , 7 , 14 , 13 , 12 , 11 , 10 , 9 , 8
} ;
2006-08-08 09:10:05 -03:00
2012-01-16 05:16:29 -03:00
static int trust_s_ctrl ( struct v4l2_ctrl * ctrl )
2007-04-20 18:22:19 -03:00
{
2012-01-16 05:16:29 -03:00
struct radio_isa_card * isa =
container_of ( ctrl - > handler , struct radio_isa_card , hdl ) ;
struct trust * tr = container_of ( isa , struct trust , isa ) ;
2009-03-06 13:54:23 -03:00
2007-04-20 18:22:19 -03:00
switch ( ctrl - > id ) {
case V4L2_CID_AUDIO_BASS :
2012-01-16 05:16:29 -03:00
write_i2c ( tr , 2 , TDA7318_ADDR , 0x60 | basstreble2chip [ ctrl - > val ] ) ;
2007-04-20 18:22:19 -03:00
return 0 ;
case V4L2_CID_AUDIO_TREBLE :
2012-01-16 05:16:29 -03:00
write_i2c ( tr , 2 , TDA7318_ADDR , 0x70 | basstreble2chip [ ctrl - > val ] ) ;
2007-04-20 18:22:19 -03:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2007-04-20 18:22:19 -03:00
return - EINVAL ;
2005-04-16 15:20:36 -07:00
}
2012-01-16 05:16:29 -03:00
static const struct v4l2_ctrl_ops trust_ctrl_ops = {
. s_ctrl = trust_s_ctrl ,
2005-04-16 15:20:36 -07:00
} ;
2012-01-16 05:16:29 -03:00
static int trust_initialize ( struct radio_isa_card * isa )
2005-04-16 15:20:36 -07:00
{
2012-01-16 05:16:29 -03:00
struct trust * tr = container_of ( isa , struct trust , isa ) ;
2009-03-06 13:54:23 -03:00
tr - > ioval = 0xf ;
write_i2c ( tr , 2 , TDA7318_ADDR , 0x80 ) ; /* speaker att. LF = 0 dB */
write_i2c ( tr , 2 , TDA7318_ADDR , 0xa0 ) ; /* speaker att. RF = 0 dB */
write_i2c ( tr , 2 , TDA7318_ADDR , 0xc0 ) ; /* speaker att. LR = 0 dB */
write_i2c ( tr , 2 , TDA7318_ADDR , 0xe0 ) ; /* speaker att. RR = 0 dB */
write_i2c ( tr , 2 , TDA7318_ADDR , 0x40 ) ; /* stereo 1 input, gain = 18.75 dB */
2005-04-16 15:20:36 -07:00
2012-01-16 05:16:29 -03:00
v4l2_ctrl_new_std ( & isa - > hdl , & trust_ctrl_ops ,
V4L2_CID_AUDIO_BASS , 0 , 15 , 1 , 8 ) ;
v4l2_ctrl_new_std ( & isa - > hdl , & trust_ctrl_ops ,
V4L2_CID_AUDIO_TREBLE , 0 , 15 , 1 , 8 ) ;
return isa - > hdl . error ;
}
2005-04-16 15:20:36 -07:00
2012-01-16 05:16:29 -03:00
static const struct radio_isa_ops trust_ops = {
. init = trust_initialize ,
. alloc = trust_alloc ,
. s_mute_volume = trust_s_mute_volume ,
. s_frequency = trust_s_frequency ,
. s_stereo = trust_s_stereo ,
. g_signal = trust_g_signal ,
} ;
2010-11-14 09:36:23 -03:00
2012-01-16 05:16:29 -03:00
static const int trust_ioports [ ] = { 0x350 , 0x358 } ;
static struct radio_isa_driver trust_driver = {
. driver = {
. match = radio_isa_match ,
. probe = radio_isa_probe ,
. remove = radio_isa_remove ,
. driver = {
. name = " radio-trust " ,
} ,
} ,
. io_params = io ,
. radio_nr_params = radio_nr ,
. io_ports = trust_ioports ,
. num_of_io_ports = ARRAY_SIZE ( trust_ioports ) ,
. region_size = 2 ,
. card = " Trust FM Radio " ,
. ops = & trust_ops ,
. has_stereo = true ,
. max_volume = 31 ,
} ;
2010-11-14 09:36:23 -03:00
2012-01-16 05:16:29 -03:00
static int __init trust_init ( void )
{
return isa_register_driver ( & trust_driver . driver , TRUST_MAX ) ;
2005-04-16 15:20:36 -07:00
}
2012-01-16 05:16:29 -03:00
static void __exit trust_exit ( void )
2005-04-16 15:20:36 -07:00
{
2012-01-16 05:16:29 -03:00
isa_unregister_driver ( & trust_driver . driver ) ;
2005-04-16 15:20:36 -07:00
}
module_init ( trust_init ) ;
2012-01-16 05:16:29 -03:00
module_exit ( trust_exit ) ;