711 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			711 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0-only
 | |
| #include <linux/module.h>
 | |
| #include <linux/kernel.h>
 | |
| #include <linux/i2c.h>
 | |
| #include <linux/types.h>
 | |
| #include <linux/init.h>
 | |
| #include <linux/errno.h>
 | |
| #include <linux/delay.h>
 | |
| #include <linux/videodev2.h>
 | |
| #include <media/v4l2-common.h>
 | |
| #include <media/tuner.h>
 | |
| #include "tuner-i2c.h"
 | |
| #include "tda9887.h"
 | |
| 
 | |
| 
 | |
| /* Chips:
 | |
|    TDA9885 (PAL, NTSC)
 | |
|    TDA9886 (PAL, SECAM, NTSC)
 | |
|    TDA9887 (PAL, SECAM, NTSC, FM Radio)
 | |
| 
 | |
|    Used as part of several tuners
 | |
| */
 | |
| 
 | |
| static int debug;
 | |
| module_param(debug, int, 0644);
 | |
| MODULE_PARM_DESC(debug, "enable verbose debug messages");
 | |
| 
 | |
| static DEFINE_MUTEX(tda9887_list_mutex);
 | |
| static LIST_HEAD(hybrid_tuner_instance_list);
 | |
| 
 | |
| struct tda9887_priv {
 | |
| 	struct tuner_i2c_props i2c_props;
 | |
| 	struct list_head hybrid_tuner_instance_list;
 | |
| 
 | |
| 	unsigned char	   data[4];
 | |
| 	unsigned int       config;
 | |
| 	unsigned int       mode;
 | |
| 	unsigned int       audmode;
 | |
| 	v4l2_std_id        std;
 | |
| 
 | |
| 	bool               standby;
 | |
| };
 | |
| 
 | |
| /* ---------------------------------------------------------------------- */
 | |
| 
 | |
| #define UNSET       (-1U)
 | |
| 
 | |
| struct tvnorm {
 | |
| 	v4l2_std_id       std;
 | |
| 	char              *name;
 | |
| 	unsigned char     b;
 | |
| 	unsigned char     c;
 | |
| 	unsigned char     e;
 | |
| };
 | |
| 
 | |
| /* ---------------------------------------------------------------------- */
 | |
| 
 | |
| //
 | |
| // TDA defines
 | |
| //
 | |
| 
 | |
| //// first reg (b)
 | |
| #define cVideoTrapBypassOFF     0x00    // bit b0
 | |
| #define cVideoTrapBypassON      0x01    // bit b0
 | |
| 
 | |
| #define cAutoMuteFmInactive     0x00    // bit b1
 | |
| #define cAutoMuteFmActive       0x02    // bit b1
 | |
| 
 | |
| #define cIntercarrier           0x00    // bit b2
 | |
| #define cQSS                    0x04    // bit b2
 | |
| 
 | |
| #define cPositiveAmTV           0x00    // bit b3:4
 | |
| #define cFmRadio                0x08    // bit b3:4
 | |
| #define cNegativeFmTV           0x10    // bit b3:4
 | |
| 
 | |
| 
 | |
| #define cForcedMuteAudioON      0x20    // bit b5
 | |
| #define cForcedMuteAudioOFF     0x00    // bit b5
 | |
| 
 | |
| #define cOutputPort1Active      0x00    // bit b6
 | |
| #define cOutputPort1Inactive    0x40    // bit b6
 | |
| 
 | |
| #define cOutputPort2Active      0x00    // bit b7
 | |
| #define cOutputPort2Inactive    0x80    // bit b7
 | |
| 
 | |
| 
 | |
| //// second reg (c)
 | |
| #define cDeemphasisOFF          0x00    // bit c5
 | |
| #define cDeemphasisON           0x20    // bit c5
 | |
| 
 | |
| #define cDeemphasis75           0x00    // bit c6
 | |
| #define cDeemphasis50           0x40    // bit c6
 | |
| 
 | |
| #define cAudioGain0             0x00    // bit c7
 | |
| #define cAudioGain6             0x80    // bit c7
 | |
| 
 | |
| #define cTopMask                0x1f    // bit c0:4
 | |
| #define cTopDefault		0x10	// bit c0:4
 | |
| 
 | |
| //// third reg (e)
 | |
| #define cAudioIF_4_5             0x00    // bit e0:1
 | |
| #define cAudioIF_5_5             0x01    // bit e0:1
 | |
| #define cAudioIF_6_0             0x02    // bit e0:1
 | |
| #define cAudioIF_6_5             0x03    // bit e0:1
 | |
| 
 | |
| 
 | |
| #define cVideoIFMask		0x1c	// bit e2:4
 | |
| /* Video IF selection in TV Mode (bit B3=0) */
 | |
| #define cVideoIF_58_75           0x00    // bit e2:4
 | |
| #define cVideoIF_45_75           0x04    // bit e2:4
 | |
| #define cVideoIF_38_90           0x08    // bit e2:4
 | |
| #define cVideoIF_38_00           0x0C    // bit e2:4
 | |
| #define cVideoIF_33_90           0x10    // bit e2:4
 | |
| #define cVideoIF_33_40           0x14    // bit e2:4
 | |
| #define cRadioIF_45_75           0x18    // bit e2:4
 | |
| #define cRadioIF_38_90           0x1C    // bit e2:4
 | |
| 
 | |
| /* IF1 selection in Radio Mode (bit B3=1) */
 | |
| #define cRadioIF_33_30		0x00	// bit e2,4 (also 0x10,0x14)
 | |
| #define cRadioIF_41_30		0x04	// bit e2,4
 | |
| 
 | |
| /* Output of AFC pin in radio mode when bit E7=1 */
 | |
| #define cRadioAGC_SIF		0x00	// bit e3
 | |
| #define cRadioAGC_FM		0x08	// bit e3
 | |
| 
 | |
| #define cTunerGainNormal         0x00    // bit e5
 | |
| #define cTunerGainLow            0x20    // bit e5
 | |
| 
 | |
| #define cGating_18               0x00    // bit e6
 | |
| #define cGating_36               0x40    // bit e6
 | |
| 
 | |
| #define cAgcOutON                0x80    // bit e7
 | |
| #define cAgcOutOFF               0x00    // bit e7
 | |
| 
 | |
| /* ---------------------------------------------------------------------- */
 | |
| 
 | |
| static struct tvnorm tvnorms[] = {
 | |
| 	{
 | |
| 		.std   = V4L2_STD_PAL_BG | V4L2_STD_PAL_H | V4L2_STD_PAL_N,
 | |
| 		.name  = "PAL-BGHN",
 | |
| 		.b     = ( cNegativeFmTV  |
 | |
| 			   cQSS           ),
 | |
| 		.c     = ( cDeemphasisON  |
 | |
| 			   cDeemphasis50  |
 | |
| 			   cTopDefault),
 | |
| 		.e     = ( cGating_36     |
 | |
| 			   cAudioIF_5_5   |
 | |
| 			   cVideoIF_38_90 ),
 | |
| 	},{
 | |
| 		.std   = V4L2_STD_PAL_I,
 | |
| 		.name  = "PAL-I",
 | |
| 		.b     = ( cNegativeFmTV  |
 | |
| 			   cQSS           ),
 | |
| 		.c     = ( cDeemphasisON  |
 | |
| 			   cDeemphasis50  |
 | |
| 			   cTopDefault),
 | |
| 		.e     = ( cGating_36     |
 | |
| 			   cAudioIF_6_0   |
 | |
| 			   cVideoIF_38_90 ),
 | |
| 	},{
 | |
| 		.std   = V4L2_STD_PAL_DK,
 | |
| 		.name  = "PAL-DK",
 | |
| 		.b     = ( cNegativeFmTV  |
 | |
| 			   cQSS           ),
 | |
| 		.c     = ( cDeemphasisON  |
 | |
| 			   cDeemphasis50  |
 | |
| 			   cTopDefault),
 | |
| 		.e     = ( cGating_36     |
 | |
| 			   cAudioIF_6_5   |
 | |
| 			   cVideoIF_38_90 ),
 | |
| 	},{
 | |
| 		.std   = V4L2_STD_PAL_M | V4L2_STD_PAL_Nc,
 | |
| 		.name  = "PAL-M/Nc",
 | |
| 		.b     = ( cNegativeFmTV  |
 | |
| 			   cQSS           ),
 | |
| 		.c     = ( cDeemphasisON  |
 | |
| 			   cDeemphasis75  |
 | |
| 			   cTopDefault),
 | |
| 		.e     = ( cGating_36     |
 | |
| 			   cAudioIF_4_5   |
 | |
| 			   cVideoIF_45_75 ),
 | |
| 	},{
 | |
| 		.std   = V4L2_STD_SECAM_B | V4L2_STD_SECAM_G | V4L2_STD_SECAM_H,
 | |
| 		.name  = "SECAM-BGH",
 | |
| 		.b     = ( cNegativeFmTV  |
 | |
| 			   cQSS           ),
 | |
| 		.c     = ( cTopDefault),
 | |
| 		.e     = ( cAudioIF_5_5   |
 | |
| 			   cVideoIF_38_90 ),
 | |
| 	},{
 | |
| 		.std   = V4L2_STD_SECAM_L,
 | |
| 		.name  = "SECAM-L",
 | |
| 		.b     = ( cPositiveAmTV  |
 | |
| 			   cQSS           ),
 | |
| 		.c     = ( cTopDefault),
 | |
| 		.e     = ( cGating_36	  |
 | |
| 			   cAudioIF_6_5   |
 | |
| 			   cVideoIF_38_90 ),
 | |
| 	},{
 | |
| 		.std   = V4L2_STD_SECAM_LC,
 | |
| 		.name  = "SECAM-L'",
 | |
| 		.b     = ( cOutputPort2Inactive |
 | |
| 			   cPositiveAmTV  |
 | |
| 			   cQSS           ),
 | |
| 		.c     = ( cTopDefault),
 | |
| 		.e     = ( cGating_36	  |
 | |
| 			   cAudioIF_6_5   |
 | |
| 			   cVideoIF_33_90 ),
 | |
| 	},{
 | |
| 		.std   = V4L2_STD_SECAM_DK,
 | |
| 		.name  = "SECAM-DK",
 | |
| 		.b     = ( cNegativeFmTV  |
 | |
| 			   cQSS           ),
 | |
| 		.c     = ( cDeemphasisON  |
 | |
| 			   cDeemphasis50  |
 | |
| 			   cTopDefault),
 | |
| 		.e     = ( cGating_36     |
 | |
| 			   cAudioIF_6_5   |
 | |
| 			   cVideoIF_38_90 ),
 | |
| 	},{
 | |
| 		.std   = V4L2_STD_NTSC_M | V4L2_STD_NTSC_M_KR,
 | |
| 		.name  = "NTSC-M",
 | |
| 		.b     = ( cNegativeFmTV  |
 | |
| 			   cQSS           ),
 | |
| 		.c     = ( cDeemphasisON  |
 | |
| 			   cDeemphasis75  |
 | |
| 			   cTopDefault),
 | |
| 		.e     = ( cGating_36     |
 | |
| 			   cAudioIF_4_5   |
 | |
| 			   cVideoIF_45_75 ),
 | |
| 	},{
 | |
| 		.std   = V4L2_STD_NTSC_M_JP,
 | |
| 		.name  = "NTSC-M-JP",
 | |
| 		.b     = ( cNegativeFmTV  |
 | |
| 			   cQSS           ),
 | |
| 		.c     = ( cDeemphasisON  |
 | |
| 			   cDeemphasis50  |
 | |
| 			   cTopDefault),
 | |
| 		.e     = ( cGating_36     |
 | |
| 			   cAudioIF_4_5   |
 | |
| 			   cVideoIF_58_75 ),
 | |
| 	}
 | |
| };
 | |
| 
 | |
| static struct tvnorm radio_stereo = {
 | |
| 	.name = "Radio Stereo",
 | |
| 	.b    = ( cFmRadio       |
 | |
| 		  cQSS           ),
 | |
| 	.c    = ( cDeemphasisOFF |
 | |
| 		  cAudioGain6    |
 | |
| 		  cTopDefault),
 | |
| 	.e    = ( cTunerGainLow  |
 | |
| 		  cAudioIF_5_5   |
 | |
| 		  cRadioIF_38_90 ),
 | |
| };
 | |
| 
 | |
| static struct tvnorm radio_mono = {
 | |
| 	.name = "Radio Mono",
 | |
| 	.b    = ( cFmRadio       |
 | |
| 		  cQSS           ),
 | |
| 	.c    = ( cDeemphasisON  |
 | |
| 		  cDeemphasis75  |
 | |
| 		  cTopDefault),
 | |
| 	.e    = ( cTunerGainLow  |
 | |
| 		  cAudioIF_5_5   |
 | |
| 		  cRadioIF_38_90 ),
 | |
| };
 | |
| 
 | |
| /* ---------------------------------------------------------------------- */
 | |
| 
 | |
| static void dump_read_message(struct dvb_frontend *fe, unsigned char *buf)
 | |
| {
 | |
| 	struct tda9887_priv *priv = fe->analog_demod_priv;
 | |
| 
 | |
| 	static char *afc[16] = {
 | |
| 		"- 12.5 kHz",
 | |
| 		"- 37.5 kHz",
 | |
| 		"- 62.5 kHz",
 | |
| 		"- 87.5 kHz",
 | |
| 		"-112.5 kHz",
 | |
| 		"-137.5 kHz",
 | |
| 		"-162.5 kHz",
 | |
| 		"-187.5 kHz [min]",
 | |
| 		"+187.5 kHz [max]",
 | |
| 		"+162.5 kHz",
 | |
| 		"+137.5 kHz",
 | |
| 		"+112.5 kHz",
 | |
| 		"+ 87.5 kHz",
 | |
| 		"+ 62.5 kHz",
 | |
| 		"+ 37.5 kHz",
 | |
| 		"+ 12.5 kHz",
 | |
| 	};
 | |
| 	tuner_info("read: 0x%2x\n", buf[0]);
 | |
| 	tuner_info("  after power on : %s\n", (buf[0] & 0x01) ? "yes" : "no");
 | |
| 	tuner_info("  afc            : %s\n", afc[(buf[0] >> 1) & 0x0f]);
 | |
| 	tuner_info("  fmif level     : %s\n", (buf[0] & 0x20) ? "high" : "low");
 | |
| 	tuner_info("  afc window     : %s\n", (buf[0] & 0x40) ? "in" : "out");
 | |
| 	tuner_info("  vfi level      : %s\n", (buf[0] & 0x80) ? "high" : "low");
 | |
| }
 | |
| 
 | |
| static void dump_write_message(struct dvb_frontend *fe, unsigned char *buf)
 | |
| {
 | |
| 	struct tda9887_priv *priv = fe->analog_demod_priv;
 | |
| 
 | |
| 	static char *sound[4] = {
 | |
| 		"AM/TV",
 | |
| 		"FM/radio",
 | |
| 		"FM/TV",
 | |
| 		"FM/radio"
 | |
| 	};
 | |
| 	static char *adjust[32] = {
 | |
| 		"-16", "-15", "-14", "-13", "-12", "-11", "-10", "-9",
 | |
| 		"-8",  "-7",  "-6",  "-5",  "-4",  "-3",  "-2",  "-1",
 | |
| 		"0",   "+1",  "+2",  "+3",  "+4",  "+5",  "+6",  "+7",
 | |
| 		"+8",  "+9",  "+10", "+11", "+12", "+13", "+14", "+15"
 | |
| 	};
 | |
| 	static char *deemph[4] = {
 | |
| 		"no", "no", "75", "50"
 | |
| 	};
 | |
| 	static char *carrier[4] = {
 | |
| 		"4.5 MHz",
 | |
| 		"5.5 MHz",
 | |
| 		"6.0 MHz",
 | |
| 		"6.5 MHz / AM"
 | |
| 	};
 | |
| 	static char *vif[8] = {
 | |
| 		"58.75 MHz",
 | |
| 		"45.75 MHz",
 | |
| 		"38.9 MHz",
 | |
| 		"38.0 MHz",
 | |
| 		"33.9 MHz",
 | |
| 		"33.4 MHz",
 | |
| 		"45.75 MHz + pin13",
 | |
| 		"38.9 MHz + pin13",
 | |
| 	};
 | |
| 	static char *rif[4] = {
 | |
| 		"44 MHz",
 | |
| 		"52 MHz",
 | |
| 		"52 MHz",
 | |
| 		"44 MHz",
 | |
| 	};
 | |
| 
 | |
| 	tuner_info("write: byte B 0x%02x\n", buf[1]);
 | |
| 	tuner_info("  B0   video mode      : %s\n",
 | |
| 		   (buf[1] & 0x01) ? "video trap" : "sound trap");
 | |
| 	tuner_info("  B1   auto mute fm    : %s\n",
 | |
| 		   (buf[1] & 0x02) ? "yes" : "no");
 | |
| 	tuner_info("  B2   carrier mode    : %s\n",
 | |
| 		   (buf[1] & 0x04) ? "QSS" : "Intercarrier");
 | |
| 	tuner_info("  B3-4 tv sound/radio  : %s\n",
 | |
| 		   sound[(buf[1] & 0x18) >> 3]);
 | |
| 	tuner_info("  B5   force mute audio: %s\n",
 | |
| 		   (buf[1] & 0x20) ? "yes" : "no");
 | |
| 	tuner_info("  B6   output port 1   : %s\n",
 | |
| 		   (buf[1] & 0x40) ? "high (inactive)" : "low (active)");
 | |
| 	tuner_info("  B7   output port 2   : %s\n",
 | |
| 		   (buf[1] & 0x80) ? "high (inactive)" : "low (active)");
 | |
| 
 | |
| 	tuner_info("write: byte C 0x%02x\n", buf[2]);
 | |
| 	tuner_info("  C0-4 top adjustment  : %s dB\n",
 | |
| 		   adjust[buf[2] & 0x1f]);
 | |
| 	tuner_info("  C5-6 de-emphasis     : %s\n",
 | |
| 		   deemph[(buf[2] & 0x60) >> 5]);
 | |
| 	tuner_info("  C7   audio gain      : %s\n",
 | |
| 		   (buf[2] & 0x80) ? "-6" : "0");
 | |
| 
 | |
| 	tuner_info("write: byte E 0x%02x\n", buf[3]);
 | |
| 	tuner_info("  E0-1 sound carrier   : %s\n",
 | |
| 		   carrier[(buf[3] & 0x03)]);
 | |
| 	tuner_info("  E6   l pll gating   : %s\n",
 | |
| 		   (buf[3] & 0x40) ? "36" : "13");
 | |
| 
 | |
| 	if (buf[1] & 0x08) {
 | |
| 		/* radio */
 | |
| 		tuner_info("  E2-4 video if        : %s\n",
 | |
| 			   rif[(buf[3] & 0x0c) >> 2]);
 | |
| 		tuner_info("  E7   vif agc output  : %s\n",
 | |
| 			   (buf[3] & 0x80)
 | |
| 			   ? ((buf[3] & 0x10) ? "fm-agc radio" :
 | |
| 						"sif-agc radio")
 | |
| 			   : "fm radio carrier afc");
 | |
| 	} else {
 | |
| 		/* video */
 | |
| 		tuner_info("  E2-4 video if        : %s\n",
 | |
| 			   vif[(buf[3] & 0x1c) >> 2]);
 | |
| 		tuner_info("  E5   tuner gain      : %s\n",
 | |
| 			   (buf[3] & 0x80)
 | |
| 			   ? ((buf[3] & 0x20) ? "external" : "normal")
 | |
| 			   : ((buf[3] & 0x20) ? "minimum"  : "normal"));
 | |
| 		tuner_info("  E7   vif agc output  : %s\n",
 | |
| 			   (buf[3] & 0x80) ? ((buf[3] & 0x20)
 | |
| 				? "pin3 port, pin22 vif agc out"
 | |
| 				: "pin22 port, pin3 vif acg ext in")
 | |
| 				: "pin3+pin22 port");
 | |
| 	}
 | |
| 	tuner_info("--\n");
 | |
| }
 | |
| 
 | |
| /* ---------------------------------------------------------------------- */
 | |
| 
 | |
| static int tda9887_set_tvnorm(struct dvb_frontend *fe)
 | |
| {
 | |
| 	struct tda9887_priv *priv = fe->analog_demod_priv;
 | |
| 	struct tvnorm *norm = NULL;
 | |
| 	char *buf = priv->data;
 | |
| 	int i;
 | |
| 
 | |
| 	if (priv->mode == V4L2_TUNER_RADIO) {
 | |
| 		if (priv->audmode == V4L2_TUNER_MODE_MONO)
 | |
| 			norm = &radio_mono;
 | |
| 		else
 | |
| 			norm = &radio_stereo;
 | |
| 	} else {
 | |
| 		for (i = 0; i < ARRAY_SIZE(tvnorms); i++) {
 | |
| 			if (tvnorms[i].std & priv->std) {
 | |
| 				norm = tvnorms+i;
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	if (NULL == norm) {
 | |
| 		tuner_dbg("Unsupported tvnorm entry - audio muted\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	tuner_dbg("configure for: %s\n", norm->name);
 | |
| 	buf[1] = norm->b;
 | |
| 	buf[2] = norm->c;
 | |
| 	buf[3] = norm->e;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static unsigned int port1  = UNSET;
 | |
| static unsigned int port2  = UNSET;
 | |
| static unsigned int qss    = UNSET;
 | |
| static unsigned int adjust = UNSET;
 | |
| 
 | |
| module_param(port1, int, 0644);
 | |
| module_param(port2, int, 0644);
 | |
| module_param(qss, int, 0644);
 | |
| module_param(adjust, int, 0644);
 | |
| 
 | |
| static int tda9887_set_insmod(struct dvb_frontend *fe)
 | |
| {
 | |
| 	struct tda9887_priv *priv = fe->analog_demod_priv;
 | |
| 	char *buf = priv->data;
 | |
| 
 | |
| 	if (UNSET != port1) {
 | |
| 		if (port1)
 | |
| 			buf[1] |= cOutputPort1Inactive;
 | |
| 		else
 | |
| 			buf[1] &= ~cOutputPort1Inactive;
 | |
| 	}
 | |
| 	if (UNSET != port2) {
 | |
| 		if (port2)
 | |
| 			buf[1] |= cOutputPort2Inactive;
 | |
| 		else
 | |
| 			buf[1] &= ~cOutputPort2Inactive;
 | |
| 	}
 | |
| 
 | |
| 	if (UNSET != qss) {
 | |
| 		if (qss)
 | |
| 			buf[1] |= cQSS;
 | |
| 		else
 | |
| 			buf[1] &= ~cQSS;
 | |
| 	}
 | |
| 
 | |
| 	if (adjust < 0x20) {
 | |
| 		buf[2] &= ~cTopMask;
 | |
| 		buf[2] |= adjust;
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int tda9887_do_config(struct dvb_frontend *fe)
 | |
| {
 | |
| 	struct tda9887_priv *priv = fe->analog_demod_priv;
 | |
| 	char *buf = priv->data;
 | |
| 
 | |
| 	if (priv->config & TDA9887_PORT1_ACTIVE)
 | |
| 		buf[1] &= ~cOutputPort1Inactive;
 | |
| 	if (priv->config & TDA9887_PORT1_INACTIVE)
 | |
| 		buf[1] |= cOutputPort1Inactive;
 | |
| 	if (priv->config & TDA9887_PORT2_ACTIVE)
 | |
| 		buf[1] &= ~cOutputPort2Inactive;
 | |
| 	if (priv->config & TDA9887_PORT2_INACTIVE)
 | |
| 		buf[1] |= cOutputPort2Inactive;
 | |
| 
 | |
| 	if (priv->config & TDA9887_QSS)
 | |
| 		buf[1] |= cQSS;
 | |
| 	if (priv->config & TDA9887_INTERCARRIER)
 | |
| 		buf[1] &= ~cQSS;
 | |
| 
 | |
| 	if (priv->config & TDA9887_AUTOMUTE)
 | |
| 		buf[1] |= cAutoMuteFmActive;
 | |
| 	if (priv->config & TDA9887_DEEMPHASIS_MASK) {
 | |
| 		buf[2] &= ~0x60;
 | |
| 		switch (priv->config & TDA9887_DEEMPHASIS_MASK) {
 | |
| 		case TDA9887_DEEMPHASIS_NONE:
 | |
| 			buf[2] |= cDeemphasisOFF;
 | |
| 			break;
 | |
| 		case TDA9887_DEEMPHASIS_50:
 | |
| 			buf[2] |= cDeemphasisON | cDeemphasis50;
 | |
| 			break;
 | |
| 		case TDA9887_DEEMPHASIS_75:
 | |
| 			buf[2] |= cDeemphasisON | cDeemphasis75;
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 	if (priv->config & TDA9887_TOP_SET) {
 | |
| 		buf[2] &= ~cTopMask;
 | |
| 		buf[2] |= (priv->config >> 8) & cTopMask;
 | |
| 	}
 | |
| 	if ((priv->config & TDA9887_INTERCARRIER_NTSC) &&
 | |
| 	    (priv->std & V4L2_STD_NTSC))
 | |
| 		buf[1] &= ~cQSS;
 | |
| 	if (priv->config & TDA9887_GATING_18)
 | |
| 		buf[3] &= ~cGating_36;
 | |
| 
 | |
| 	if (priv->mode == V4L2_TUNER_RADIO) {
 | |
| 		if (priv->config & TDA9887_RIF_41_3) {
 | |
| 			buf[3] &= ~cVideoIFMask;
 | |
| 			buf[3] |= cRadioIF_41_30;
 | |
| 		}
 | |
| 		if (priv->config & TDA9887_GAIN_NORMAL)
 | |
| 			buf[3] &= ~cTunerGainLow;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /* ---------------------------------------------------------------------- */
 | |
| 
 | |
| static int tda9887_status(struct dvb_frontend *fe)
 | |
| {
 | |
| 	struct tda9887_priv *priv = fe->analog_demod_priv;
 | |
| 	unsigned char buf[1];
 | |
| 	int rc;
 | |
| 
 | |
| 	rc = tuner_i2c_xfer_recv(&priv->i2c_props, buf, 1);
 | |
| 	if (rc != 1)
 | |
| 		tuner_info("i2c i/o error: rc == %d (should be 1)\n", rc);
 | |
| 	dump_read_message(fe, buf);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static void tda9887_configure(struct dvb_frontend *fe)
 | |
| {
 | |
| 	struct tda9887_priv *priv = fe->analog_demod_priv;
 | |
| 	int rc;
 | |
| 
 | |
| 	memset(priv->data,0,sizeof(priv->data));
 | |
| 	tda9887_set_tvnorm(fe);
 | |
| 
 | |
| 	/* A note on the port settings:
 | |
| 	   These settings tend to depend on the specifics of the board.
 | |
| 	   By default they are set to inactive (bit value 1) by this driver,
 | |
| 	   overwriting any changes made by the tvnorm. This means that it
 | |
| 	   is the responsibility of the module using the tda9887 to set
 | |
| 	   these values in case of changes in the tvnorm.
 | |
| 	   In many cases port 2 should be made active (0) when selecting
 | |
| 	   SECAM-L, and port 2 should remain inactive (1) for SECAM-L'.
 | |
| 
 | |
| 	   For the other standards the tda9887 application note says that
 | |
| 	   the ports should be set to active (0), but, again, that may
 | |
| 	   differ depending on the precise hardware configuration.
 | |
| 	 */
 | |
| 	priv->data[1] |= cOutputPort1Inactive;
 | |
| 	priv->data[1] |= cOutputPort2Inactive;
 | |
| 
 | |
| 	tda9887_do_config(fe);
 | |
| 	tda9887_set_insmod(fe);
 | |
| 
 | |
| 	if (priv->standby)
 | |
| 		priv->data[1] |= cForcedMuteAudioON;
 | |
| 
 | |
| 	tuner_dbg("writing: b=0x%02x c=0x%02x e=0x%02x\n",
 | |
| 		  priv->data[1], priv->data[2], priv->data[3]);
 | |
| 	if (debug > 1)
 | |
| 		dump_write_message(fe, priv->data);
 | |
| 
 | |
| 	if (4 != (rc = tuner_i2c_xfer_send(&priv->i2c_props,priv->data,4)))
 | |
| 		tuner_info("i2c i/o error: rc == %d (should be 4)\n", rc);
 | |
| 
 | |
| 	if (debug > 2) {
 | |
| 		msleep_interruptible(1000);
 | |
| 		tda9887_status(fe);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /* ---------------------------------------------------------------------- */
 | |
| 
 | |
| static void tda9887_tuner_status(struct dvb_frontend *fe)
 | |
| {
 | |
| 	struct tda9887_priv *priv = fe->analog_demod_priv;
 | |
| 	tuner_info("Data bytes: b=0x%02x c=0x%02x e=0x%02x\n",
 | |
| 		   priv->data[1], priv->data[2], priv->data[3]);
 | |
| }
 | |
| 
 | |
| static int tda9887_get_afc(struct dvb_frontend *fe, s32 *afc)
 | |
| {
 | |
| 	struct tda9887_priv *priv = fe->analog_demod_priv;
 | |
| 	static const int AFC_BITS_2_kHz[] = {
 | |
| 		-12500,  -37500,  -62500,  -97500,
 | |
| 		-112500, -137500, -162500, -187500,
 | |
| 		187500,  162500,  137500,  112500,
 | |
| 		97500 ,  62500,   37500 ,  12500
 | |
| 	};
 | |
| 	__u8 reg = 0;
 | |
| 
 | |
| 	if (priv->mode != V4L2_TUNER_RADIO)
 | |
| 		return 0;
 | |
| 	if (1 == tuner_i2c_xfer_recv(&priv->i2c_props, ®, 1))
 | |
| 		*afc = AFC_BITS_2_kHz[(reg >> 1) & 0x0f];
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static void tda9887_standby(struct dvb_frontend *fe)
 | |
| {
 | |
| 	struct tda9887_priv *priv = fe->analog_demod_priv;
 | |
| 
 | |
| 	priv->standby = true;
 | |
| 
 | |
| 	tda9887_configure(fe);
 | |
| }
 | |
| 
 | |
| static void tda9887_set_params(struct dvb_frontend *fe,
 | |
| 			       struct analog_parameters *params)
 | |
| {
 | |
| 	struct tda9887_priv *priv = fe->analog_demod_priv;
 | |
| 
 | |
| 	priv->standby = false;
 | |
| 	priv->mode    = params->mode;
 | |
| 	priv->audmode = params->audmode;
 | |
| 	priv->std     = params->std;
 | |
| 	tda9887_configure(fe);
 | |
| }
 | |
| 
 | |
| static int tda9887_set_config(struct dvb_frontend *fe, void *priv_cfg)
 | |
| {
 | |
| 	struct tda9887_priv *priv = fe->analog_demod_priv;
 | |
| 
 | |
| 	priv->config = *(unsigned int *)priv_cfg;
 | |
| 	tda9887_configure(fe);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static void tda9887_release(struct dvb_frontend *fe)
 | |
| {
 | |
| 	struct tda9887_priv *priv = fe->analog_demod_priv;
 | |
| 
 | |
| 	mutex_lock(&tda9887_list_mutex);
 | |
| 
 | |
| 	if (priv)
 | |
| 		hybrid_tuner_release_state(priv);
 | |
| 
 | |
| 	mutex_unlock(&tda9887_list_mutex);
 | |
| 
 | |
| 	fe->analog_demod_priv = NULL;
 | |
| }
 | |
| 
 | |
| static const struct analog_demod_ops tda9887_ops = {
 | |
| 	.info		= {
 | |
| 		.name	= "tda9887",
 | |
| 	},
 | |
| 	.set_params     = tda9887_set_params,
 | |
| 	.standby        = tda9887_standby,
 | |
| 	.tuner_status   = tda9887_tuner_status,
 | |
| 	.get_afc        = tda9887_get_afc,
 | |
| 	.release        = tda9887_release,
 | |
| 	.set_config     = tda9887_set_config,
 | |
| };
 | |
| 
 | |
| struct dvb_frontend *tda9887_attach(struct dvb_frontend *fe,
 | |
| 				    struct i2c_adapter *i2c_adap,
 | |
| 				    u8 i2c_addr)
 | |
| {
 | |
| 	struct tda9887_priv *priv = NULL;
 | |
| 	int instance;
 | |
| 
 | |
| 	mutex_lock(&tda9887_list_mutex);
 | |
| 
 | |
| 	instance = hybrid_tuner_request_state(struct tda9887_priv, priv,
 | |
| 					      hybrid_tuner_instance_list,
 | |
| 					      i2c_adap, i2c_addr, "tda9887");
 | |
| 	switch (instance) {
 | |
| 	case 0:
 | |
| 		mutex_unlock(&tda9887_list_mutex);
 | |
| 		return NULL;
 | |
| 	case 1:
 | |
| 		fe->analog_demod_priv = priv;
 | |
| 		priv->standby = true;
 | |
| 		tuner_info("tda988[5/6/7] found\n");
 | |
| 		break;
 | |
| 	default:
 | |
| 		fe->analog_demod_priv = priv;
 | |
| 		break;
 | |
| 	}
 | |
| 
 | |
| 	mutex_unlock(&tda9887_list_mutex);
 | |
| 
 | |
| 	memcpy(&fe->ops.analog_ops, &tda9887_ops,
 | |
| 	       sizeof(struct analog_demod_ops));
 | |
| 
 | |
| 	return fe;
 | |
| }
 | |
| EXPORT_SYMBOL_GPL(tda9887_attach);
 | |
| 
 | |
| MODULE_LICENSE("GPL");
 |