#include <sys/param.h>
#include <sys/systm.h>
#include <sys/kernel.h>
#include <sys/vnode.h>
#include <dev/ic/bt8xx.h>
#include <dev/pci/bktr/bktr_reg.h>
#include <dev/pci/bktr/bktr_tuner.h>
#include <dev/pci/bktr/bktr_core.h>
#if defined( TUNER_AFC )
#define AFC_DELAY 10000
#define AFC_BITS 0x07
#define AFC_FREQ_MINUS_125 0x00
#define AFC_FREQ_MINUS_62 0x01
#define AFC_FREQ_CENTERED 0x02
#define AFC_FREQ_PLUS_62 0x03
#define AFC_FREQ_PLUS_125 0x04
#define AFC_MAX_STEP (5 * FREQFACTOR)
#endif
#define TTYPE_XXX 0
#define TTYPE_NTSC 1
#define TTYPE_NTSC_J 2
#define TTYPE_PAL 3
#define TTYPE_PAL_M 4
#define TTYPE_PAL_N 5
#define TTYPE_SECAM 6
#define TSA552x_CB_MSB (0x80)
#define TSA552x_CB_CP (1<<6)
#define TSA552x_CB_T2 (1<<5)
#define TSA552x_CB_T1 (1<<4)
#define TSA552x_CB_T0 (1<<3)
#define TSA552x_CB_RSA (1<<2)
#define TSA552x_CB_RSB (1<<1)
#define TSA552x_CB_OS (1<<0)
#define TSA552x_RADIO (TSA552x_CB_MSB | \
TSA552x_CB_T0)
#define TSA552x_FCONTROL (TSA552x_CB_MSB | \
TSA552x_CB_CP | \
TSA552x_CB_T0 | \
TSA552x_CB_RSA | \
TSA552x_CB_RSB)
#define TSA552x_SCONTROL (TSA552x_CB_MSB | \
TSA552x_CB_T0 | \
TSA552x_CB_RSA | \
TSA552x_CB_RSB)
#define TSCH5_FCONTROL 0x82
#define TSCH5_RADIO 0x86
#define TSBH1_FCONTROL 0xce
static const struct TUNER tuners[] = {
{ "<no>",
TTYPE_XXX,
{ 0x00,
0x00,
0x00,
0x00 },
{ 0x00, 0x00 },
{ 0x00, 0x00, 0x00,0x00} },
{ "Temic NTSC",
TTYPE_NTSC,
{ TSA552x_SCONTROL,
TSA552x_SCONTROL,
TSA552x_SCONTROL,
0x00 },
{ 0x00, 0x00},
{ 0x02, 0x04, 0x01, 0x00 } },
{ "Temic PAL",
TTYPE_PAL,
{ TSA552x_SCONTROL,
TSA552x_SCONTROL,
TSA552x_SCONTROL,
0x00 },
{ 0x00, 0x00 },
{ 0x02, 0x04, 0x01, 0x00 } },
{ "Temic SECAM",
TTYPE_SECAM,
{ TSA552x_SCONTROL,
TSA552x_SCONTROL,
TSA552x_SCONTROL,
0x00 },
{ 0x00, 0x00 },
{ 0x02, 0x04, 0x01,0x00 } },
{ "Philips NTSC",
TTYPE_NTSC,
{ TSA552x_SCONTROL,
TSA552x_SCONTROL,
TSA552x_SCONTROL,
0x00 },
{ 0x00, 0x00 },
{ 0xa0, 0x90, 0x30, 0x00 } },
{ "Philips PAL",
TTYPE_PAL,
{ TSA552x_SCONTROL,
TSA552x_SCONTROL,
TSA552x_SCONTROL,
0x00 },
{ 0x00, 0x00 },
{ 0xa0, 0x90, 0x30, 0x00 } },
{ "Philips SECAM",
TTYPE_SECAM,
{ TSA552x_SCONTROL,
TSA552x_SCONTROL,
TSA552x_SCONTROL,
0x00 },
{ 0x00, 0x00 },
{ 0xa7, 0x97, 0x37, 0x00 } },
{ "Temic PAL I",
TTYPE_PAL,
{ TSA552x_SCONTROL,
TSA552x_SCONTROL,
TSA552x_SCONTROL,
0x00 },
{ 0x00, 0x00 },
{ 0x02, 0x04, 0x01,0x00 } },
{ "Philips PAL I",
TTYPE_PAL,
{ TSA552x_SCONTROL,
TSA552x_SCONTROL,
TSA552x_SCONTROL,
0x00 },
{ 0x00, 0x00 },
{ 0xa0, 0x90, 0x30,0x00 } },
{ "Philips FR1236 NTSC FM",
TTYPE_NTSC,
{ TSA552x_FCONTROL,
TSA552x_FCONTROL,
TSA552x_FCONTROL,
TSA552x_RADIO },
{ 0x00, 0x00 },
{ 0xa0, 0x90, 0x30,0xa4 } },
{ "Philips FR1216 PAL FM" ,
TTYPE_PAL,
{ TSA552x_FCONTROL,
TSA552x_FCONTROL,
TSA552x_FCONTROL,
TSA552x_RADIO },
{ 0x00, 0x00 },
{ 0xa0, 0x90, 0x30, 0xa4 } },
{ "Philips FR1236 SECAM FM",
TTYPE_SECAM,
{ TSA552x_FCONTROL,
TSA552x_FCONTROL,
TSA552x_FCONTROL,
TSA552x_RADIO },
{ 0x00, 0x00 },
{ 0xa7, 0x97, 0x37, 0xa4 } },
{ "ALPS TSCH5 NTSC FM",
TTYPE_NTSC,
{ TSCH5_FCONTROL,
TSCH5_FCONTROL,
TSCH5_FCONTROL,
TSCH5_RADIO },
{ 0x00, 0x00 },
{ 0x14, 0x12, 0x11, 0x04 } },
{ "ALPS TSBH1 NTSC",
TTYPE_NTSC,
{ TSBH1_FCONTROL,
TSBH1_FCONTROL,
TSBH1_FCONTROL,
0x00 },
{ 0x00, 0x00 },
{ 0x01, 0x02, 0x08, 0x00 } },
{ "Tivision TVF5533-MF NTSC",
TTYPE_NTSC,
{ TSBH1_FCONTROL,
TSBH1_FCONTROL,
TSBH1_FCONTROL,
0x00 },
{ 0x00, 0x00 },
{ 0x01, 0x02, 0x04, 0x00 } },
};
#define FREQFACTOR 16
#define OFFSET 6.00
static const int nabcst[] = {
83, (int)( 45.75 * FREQFACTOR), 0,
14, (int)(471.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
7, (int)(175.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
5, (int)( 77.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
2, (int)( 55.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
0
};
#undef OFFSET
#define OFFSET 6.00
static const int irccable[] = {
116, (int)( 45.75 * FREQFACTOR), 0,
100, (int)(649.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
95, (int)( 91.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
23, (int)(217.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
14, (int)(121.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
7, (int)(175.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
5, (int)( 77.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
2, (int)( 55.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
0
};
#undef OFFSET
#define OFFSET 6.00
static const int hrccable[] = {
116, (int)( 45.75 * FREQFACTOR), 0,
100, (int)(648.00 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
95, (int)( 90.00 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
23, (int)(216.00 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
14, (int)(120.00 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
7, (int)(174.00 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
5, (int)( 78.00 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
2, (int)( 54.00 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
0
};
#undef OFFSET
static const int weurope[] = {
121, (int)( 38.90 * FREQFACTOR), 0,
100, (int)(303.25 * FREQFACTOR), (int)(8.00 * FREQFACTOR),
90, (int)(231.25 * FREQFACTOR), (int)(7.00 * FREQFACTOR),
80, (int)(105.25 * FREQFACTOR), (int)(7.00 * FREQFACTOR),
74, (int)( 69.25 * FREQFACTOR), (int)(7.00 * FREQFACTOR),
21, (int)(471.25 * FREQFACTOR), (int)(8.00 * FREQFACTOR),
17, (int)(183.25 * FREQFACTOR), (int)(9.00 * FREQFACTOR),
16, (int)(175.25 * FREQFACTOR), (int)(9.00 * FREQFACTOR),
15, (int)(82.25 * FREQFACTOR), (int)(8.50 * FREQFACTOR),
13, (int)(53.75 * FREQFACTOR), (int)(8.50 * FREQFACTOR),
5, (int)(175.25 * FREQFACTOR), (int)(7.00 * FREQFACTOR),
2, (int)(48.25 * FREQFACTOR), (int)(7.00 * FREQFACTOR),
0
};
#define OFFSET 6.00
#define IF_FREQ 45.75
static const int jpnbcst[] = {
62, (int)(IF_FREQ * FREQFACTOR), 0,
13, (int)(471.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
8, (int)(193.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
4, (int)(171.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
1, (int)( 91.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
0
};
#undef IF_FREQ
#undef OFFSET
#define OFFSET 6.00
#define IF_FREQ 45.75
static const int jpncable[] = {
63, (int)(IF_FREQ * FREQFACTOR), 0,
23, (int)(223.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
22, (int)(165.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
13, (int)(109.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
8, (int)(193.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
4, (int)(171.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
1, (int)( 91.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
0
};
#undef IF_FREQ
#undef OFFSET
#define IF_FREQ 38.90
static const int xussr[] = {
107, (int)(IF_FREQ * FREQFACTOR), 0,
78, (int)(231.25 * FREQFACTOR), (int)(8.00 * FREQFACTOR),
70, (int)(111.25 * FREQFACTOR), (int)(8.00 * FREQFACTOR),
35, (int)(583.25 * FREQFACTOR), (int)(8.00 * FREQFACTOR),
21, (int)(471.25 * FREQFACTOR), (int)(8.00 * FREQFACTOR),
6, (int)(175.25 * FREQFACTOR), (int)(8.00 * FREQFACTOR),
3, (int)( 77.25 * FREQFACTOR), (int)(8.00 * FREQFACTOR),
1, (int)( 49.75 * FREQFACTOR), (int)(9.50 * FREQFACTOR),
0
};
#undef IF_FREQ
#define OFFSET 7.00
#define IF_FREQ 38.90
static const int australia[] = {
83, (int)(IF_FREQ * FREQFACTOR), 0,
28, (int)(527.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
10, (int)(209.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
6, (int)(175.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
4, (int)( 95.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
3, (int)( 86.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
1, (int)( 57.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
0
};
#undef OFFSET
#undef IF_FREQ
#define OFFSET 8.00
#define IF_FREQ 38.90
static const int france[] = {
69, (int)(IF_FREQ * FREQFACTOR), 0,
21, (int)(471.25 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
5, (int)(176.00 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
4, (int)( 63.75 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
3, (int)( 60.50 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
1, (int)( 47.75 * FREQFACTOR), (int)(OFFSET * FREQFACTOR),
0
};
#undef OFFSET
#undef IF_FREQ
static const struct {
const int *ptr;
char name[BT848_MAX_CHNLSET_NAME_LEN];
} freqTable[] = {
{NULL, ""},
{nabcst, "nabcst"},
{irccable, "cableirc"},
{hrccable, "cablehrc"},
{weurope, "weurope"},
{jpnbcst, "jpnbcst"},
{jpncable, "jpncable"},
{xussr, "xussr"},
{australia, "australia"},
{france, "france"},
};
#define TBL_CHNL freqTable[ bktr->tuner.chnlset ].ptr[ x ]
#define TBL_BASE_FREQ freqTable[ bktr->tuner.chnlset ].ptr[ x + 1 ]
#define TBL_OFFSET freqTable[ bktr->tuner.chnlset ].ptr[ x + 2 ]
static int
frequency_lookup( bktr_ptr_t bktr, int channel )
{
int x;
x = 0;
if ( channel > TBL_CHNL )
return( -1 );
for ( x = 3; TBL_CHNL; x += 3 ) {
if ( channel >= TBL_CHNL ) {
return( TBL_BASE_FREQ +
((channel - TBL_CHNL) * TBL_OFFSET) );
}
}
return( -1 );
}
#undef TBL_OFFSET
#undef TBL_BASE_FREQ
#undef TBL_CHNL
#define TBL_IF freqTable[ bktr->tuner.chnlset ].ptr[ 1 ]
void select_tuner( bktr_ptr_t bktr, int tuner_type ) {
if (tuner_type < Bt848_MAX_TUNER) {
bktr->card.tuner = &tuners[ tuner_type ];
} else {
bktr->card.tuner = NULL;
}
}
#define LOW_BAND 0
#define MID_BAND 1
#define HIGH_BAND 2
#define FM_RADIO_BAND 3
#define STATUSBIT_COLD 0x80
#define STATUSBIT_LOCK 0x40
#define STATUSBIT_TV 0x20
#define STATUSBIT_STEREO 0x10
#define STATUSBIT_ADC 0x07
int
tv_freq( bktr_ptr_t bktr, int frequency, int type )
{
const struct TUNER* tuner;
u_char addr;
u_char control;
u_char band;
int N;
int band_select = 0;
#if defined( TEST_TUNER_AFC )
int oldFrequency, afcDelta;
#endif
tuner = bktr->card.tuner;
if ( tuner == NULL )
return( -1 );
if (type == TV_FREQUENCY) {
if ( frequency < (160 * FREQFACTOR ) )
band_select = LOW_BAND;
else if ( frequency < (454 * FREQFACTOR ) )
band_select = MID_BAND;
else
band_select = HIGH_BAND;
bktr->tuner.tuner_mode = BT848_TUNER_MODE_TV;
#if defined( TEST_TUNER_AFC )
if ( bktr->tuner.afc )
frequency -= 4;
#endif
N = frequency + TBL_IF;
addr = bktr->card.tuner_pllAddr;
control = tuner->pllControl[ band_select ];
band = tuner->bandAddrs[ band_select ];
if(!(band && control))
return(-1);
if ( frequency > bktr->tuner.frequency ) {
i2cWrite( bktr, addr, (N>>8) & 0x7f, N & 0xff );
i2cWrite( bktr, addr, control, band );
}
else {
i2cWrite( bktr, addr, control, band );
i2cWrite( bktr, addr, (N>>8) & 0x7f, N & 0xff );
}
#if defined( TUNER_AFC )
if ( bktr->tuner.afc == TRUE ) {
#if defined( TEST_TUNER_AFC )
oldFrequency = frequency;
#endif
if ( (N = do_afc( bktr, addr, N )) < 0 ) {
N = frequency + TBL_IF;
#if defined( TEST_TUNER_AFC )
printf("%s: do_afc: failed to lock\n",
bktr_name(bktr));
#endif
i2cWrite( bktr, addr, (N>>8) & 0x7f, N & 0xff );
}
else
frequency = N - TBL_IF;
#if defined( TEST_TUNER_AFC )
printf("%s: do_afc: returned freq %d (%d %% %d)\n", bktr_name(bktr), frequency, frequency / 16, frequency % 16);
afcDelta = frequency - oldFrequency;
printf("%s: changed by: %d clicks (%d mod %d)\n", bktr_name(bktr), afcDelta, afcDelta / 16, afcDelta % 16);
#endif
}
#endif
bktr->tuner.frequency = frequency;
}
if ( type == FM_RADIO_FREQUENCY ) {
band_select = FM_RADIO_BAND;
bktr->tuner.tuner_mode = BT848_TUNER_MODE_RADIO;
N = (frequency + 1070)/5;
addr = bktr->card.tuner_pllAddr;
control = tuner->pllControl[ band_select ];
band = tuner->bandAddrs[ band_select ];
if(!(band && control))
return(-1);
band |= bktr->tuner.radio_mode;
i2cWrite( bktr, addr, control, band );
i2cWrite( bktr, addr, (N>>8) & 0x7f, N & 0xff );
bktr->tuner.frequency = (N * 5) - 1070;
}
return( 0 );
}
#if defined( TUNER_AFC )
int
do_afc( bktr_ptr_t bktr, int addr, int frequency )
{
int step;
int status;
int origFrequency;
origFrequency = frequency;
tsleep_nsec( BKTR_SLEEP, PZERO, "tuning", MSEC_TO_NSEC(1000 / 8) );
if ( (status = i2cRead( bktr, addr + 1 )) < 0 )
return( -1 );
#if defined( TEST_TUNER_AFC )
printf( "%s: Original freq: %d, status: 0x%02x\n", bktr_name(bktr), frequency, status );
#endif
for ( step = 0; step < AFC_MAX_STEP; ++step ) {
if ( (status = i2cRead( bktr, addr + 1 )) < 0 )
goto fubar;
if ( !(status & 0x40) ) {
#if defined( TEST_TUNER_AFC )
printf( "%s: no lock!\n", bktr_name(bktr) );
#endif
goto fubar;
}
switch( status & AFC_BITS ) {
case AFC_FREQ_CENTERED:
#if defined( TEST_TUNER_AFC )
printf( "%s: Centered, freq: %d, status: 0x%02x\n", bktr_name(bktr), frequency, status );
#endif
return( frequency );
case AFC_FREQ_MINUS_125:
case AFC_FREQ_MINUS_62:
#if defined( TEST_TUNER_AFC )
printf( "%s: Low, freq: %d, status: 0x%02x\n", bktr_name(bktr), frequency, status );
#endif
--frequency;
break;
case AFC_FREQ_PLUS_62:
case AFC_FREQ_PLUS_125:
#if defined( TEST_TUNER_AFC )
printf( "%s: Hi, freq: %d, status: 0x%02x\n", bktr_name(bktr), frequency, status );
#endif
++frequency;
break;
}
i2cWrite( bktr, addr,
(frequency>>8) & 0x7f, frequency & 0xff );
DELAY( AFC_DELAY );
}
fubar:
i2cWrite( bktr, addr,
(origFrequency>>8) & 0x7f, origFrequency & 0xff );
return( -1 );
}
#endif
#undef TBL_IF
int get_tuner_status( bktr_ptr_t bktr ) {
return i2cRead( bktr, bktr->card.tuner_pllAddr + 1 );
}
int
tv_channel( bktr_ptr_t bktr, int channel )
{
int frequency;
if ( (frequency = frequency_lookup( bktr, channel )) < 0 )
return( -1 );
if ( tv_freq( bktr, frequency, TV_FREQUENCY ) < 0 )
return( -1 );
return( (bktr->tuner.channel = channel) );
}
int
tuner_getchnlset(struct bktr_chnlset *chnlset)
{
if (( chnlset->index < CHNLSET_MIN ) ||
( chnlset->index > CHNLSET_MAX ))
return( EINVAL );
memcpy(&chnlset->name, &freqTable[chnlset->index].name,
BT848_MAX_CHNLSET_NAME_LEN);
chnlset->max_channel=freqTable[chnlset->index].ptr[0];
return( 0 );
}