summaryrefslogtreecommitdiff
path: root/linux/drivers/media
diff options
context:
space:
mode:
Diffstat (limited to 'linux/drivers/media')
-rw-r--r--linux/drivers/media/video/msp3400-driver.c619
-rw-r--r--linux/drivers/media/video/msp3400.c619
-rw-r--r--linux/drivers/media/video/msp3400.h22
3 files changed, 1066 insertions, 194 deletions
diff --git a/linux/drivers/media/video/msp3400-driver.c b/linux/drivers/media/video/msp3400-driver.c
index 74e57e172..6f930254c 100644
--- a/linux/drivers/media/video/msp3400-driver.c
+++ b/linux/drivers/media/video/msp3400-driver.c
@@ -61,14 +61,23 @@
#include "msp3400.h"
#include "compat.h"
+#define OPMODE_AUTO -1
+#define OPMODE_MANUAL 0
+#define OPMODE_SIMPLE 1 /* use short programming (>= msp3410 only) */
+#define OPMODE_SIMPLER 2 /* use shorter programming (>= msp34xxG) */
+
/* insmod parameters */
+static int opmode = OPMODE_AUTO;
static int debug = 0; /* debug output */
static int once = 0; /* no continous stereo monitoring */
static int amsound = 0; /* hard-wire AM sound at 6.5 Hz (france),
- the autoscan seems work well only with FM... */
-static int simple = -1; /* use short programming (>= msp3410 only) */
+ the autoscan seems work well only with FM... */
+static int standard = 1; /* Override auto detect of audio standard, if needed. */
static int dolby = 0;
+static int stereo_threshold = 0x190; /* a2 threshold for stereo/bilingual
+ (msp34xxg only) 0x00a0-0x03c0 */
+
#define DFP_COUNT 0x41
static const int bl_dfp[] = {
0x00, 0x01, 0x02, 0x03, 0x06, 0x08, 0x09, 0x0a,
@@ -78,7 +87,7 @@ static const int bl_dfp[] = {
struct msp3400c {
int rev1,rev2;
- int simple;
+ int opmode;
int mode;
int norm;
int stereo;
@@ -86,6 +95,7 @@ struct msp3400c {
int acb;
int main, second; /* sound carrier */
int input;
+ int source; /* see msp34xxg_set_source */
int muted;
int volume, balance;
@@ -102,6 +112,9 @@ struct msp3400c {
int active:1;
int restart:1;
int rmmod:1;
+#if 0
+ struct semaphore *notify;
+#endif
int watch_stereo;
struct timer_list wake_stereo;
@@ -109,6 +122,7 @@ struct msp3400c {
#define HAVE_NICAM(msp) (((msp->rev2>>8) & 0xff) != 00)
#define HAVE_SIMPLE(msp) ((msp->rev1 & 0xff) >= 'D'-'@')
+#define HAVE_SIMPLER(msp) ((msp->rev1 & 0xff) >= 'G'-'@')
#define HAVE_RADIO(msp) ((msp->rev1 & 0xff) >= 'G'-'@')
#define VIDEO_MODE_RADIO 16 /* norm magic for radio mode */
@@ -119,9 +133,15 @@ struct msp3400c {
#define d2printk if (debug >= 2) printk
MODULE_PARM(once,"i");
+MODULE_PARM_DESC(once, "No continuous stereo monitoring");
MODULE_PARM(debug,"i");
-MODULE_PARM(simple,"i");
+MODULE_PARM_DESC(debug, "Enable debug messages");
+MODULE_PARM(opmode,"i");
+MODULE_PARM(stereo_threshold,"i");
+MODULE_PARM(standard,"i");
+MODULE_PARM_DESC(standard, "Specify audio standard: 32 = NTSC, 64 = radio, Default: Autodetect");
MODULE_PARM(amsound,"i");
+MODULE_PARM_DESC(amsound, "Hardwire AM sound at 6.5Hz (France), FM can autoscan");
MODULE_PARM(dolby,"i");
MODULE_DESCRIPTION("device driver for msp34xx TV sound processor");
@@ -205,7 +225,7 @@ msp3400c_read(struct i2c_client *client, int dev, int addr)
msp3400c_reset(client);
return -1;
}
- return read[0] << 8 | read[1];
+ return read[0] << 8 | read[1];
}
static int
@@ -336,16 +356,6 @@ static struct CARRIER_DETECT carrier_detect_65[] = {
/* ----------------------------------------------------------------------- */
-#define SCART_MASK 0
-#define SCART_IN1 1
-#define SCART_IN2 2
-#define SCART_IN1_DA 3
-#define SCART_IN2_DA 4
-#define SCART_IN3 5
-#define SCART_IN4 6
-#define SCART_MONO 7
-#define SCART_MUTE 8
-
static int scarts[3][9] = {
/* MASK IN1 IN2 IN1_DA IN2_DA IN3 IN4 MONO MUTE */
{ 0x0320, 0x0000, 0x0200, -1, -1, 0x0300, 0x0020, 0x0100, 0x0320 },
@@ -399,8 +409,8 @@ static void msp3400c_setvolume(struct i2c_client *client,
muted ? "on" : "off", volume, balance, val>>8, bal);
msp3400c_write(client,I2C_MSP3400C_DFP, 0x0000, val); /* loudspeaker */
msp3400c_write(client,I2C_MSP3400C_DFP, 0x0006, val); /* headphones */
- /* scart - on/off only */
- msp3400c_write(client,I2C_MSP3400C_DFP, 0x0007, val ? 0x4000 : 0);
+ msp3400c_write(client,I2C_MSP3400C_DFP, 0x0007,
+ muted ? 0x01 : (val | 0x01));
msp3400c_write(client,I2C_MSP3400C_DFP, 0x0001, bal << 8);
}
@@ -477,6 +487,18 @@ static void msp3400c_setmode(struct i2c_client *client, int type)
}
}
+// given a bitmask of VIDEO_SOUND_XXX returns the "best" in the bitmask
+static int best_audio_mode(int mode)
+{
+ if (mode & VIDEO_SOUND_STEREO)
+ return VIDEO_SOUND_STEREO;
+ if (mode & VIDEO_SOUND_LANG1)
+ return VIDEO_SOUND_LANG1;
+ if (mode & VIDEO_SOUND_LANG2)
+ return VIDEO_SOUND_LANG2;
+ return VIDEO_SOUND_MONO;
+}
+
/* turn on/off nicam + stereo */
static void msp3400c_setstereo(struct i2c_client *client, int mode)
{
@@ -493,6 +515,8 @@ static void msp3400c_setstereo(struct i2c_client *client, int mode)
int nicam=0; /* channel source: FM/AM or nicam */
int src=0;
+ BUG_ON(msp->opmode == OPMODE_SIMPLER);
+
/* switch demodulator */
switch (msp->mode) {
case MSP_MODE_FM_TERRA:
@@ -628,6 +652,18 @@ msp3400c_restore_dfp(struct i2c_client *client)
}
}
+/* if the dfp_regs is set, set what's in there.
+ * Otherwise, set the default value */
+static int msp3400c_write_dfp_with_default(struct i2c_client *client, int addr,
+ int default_value)
+{
+ struct msp3400c *msp = i2c_get_clientdata(client);
+ int value=default_value;
+ if ( addr< DFP_COUNT && -1 != msp->dfp_regs[addr] )
+ value=msp->dfp_regs[addr];
+ return msp3400c_write(client, I2C_MSP3400C_DFP, addr, value);
+}
+
/* ----------------------------------------------------------------------- */
struct REGISTER_DUMP {
@@ -767,16 +803,8 @@ static void watch_stereo(struct i2c_client *client)
{
struct msp3400c *msp = i2c_get_clientdata(client);
- if (autodetect_stereo(client)) {
- if (msp->stereo & VIDEO_SOUND_STEREO)
- msp3400c_setstereo(client,VIDEO_SOUND_STEREO);
- else if (msp->stereo & VIDEO_SOUND_LANG1)
- msp3400c_setstereo(client,VIDEO_SOUND_LANG1);
- else if (msp->stereo & VIDEO_SOUND_LANG2)
- msp3400c_setstereo(client,VIDEO_SOUND_LANG2);
- else
- msp3400c_setstereo(client,VIDEO_SOUND_MONO);
- }
+ if (autodetect_stereo(client))
+ msp3400c_setstereo(client,best_audio_mode(msp->stereo));
if (once)
msp->watch_stereo = 0;
if (msp->watch_stereo)
@@ -867,10 +895,12 @@ static int msp3400c_thread(void *data)
/* carrier detect pass #2 -- second (stereo) carrier */
switch (max1) {
case 1: /* 5.5 */
- cd = carrier_detect_55; count = CARRIER_COUNT(carrier_detect_55);
+ cd = carrier_detect_55;
+ count = CARRIER_COUNT(carrier_detect_55);
break;
case 3: /* 6.5 */
- cd = carrier_detect_65; count = CARRIER_COUNT(carrier_detect_65);
+ cd = carrier_detect_65;
+ count = CARRIER_COUNT(carrier_detect_65);
break;
case 0: /* 4.5 */
case 2: /* 6.0 */
@@ -988,7 +1018,7 @@ static int msp3400c_thread(void *data)
done:
msp->active = 0;
dprintk(KERN_DEBUG "msp3400: thread: exit\n");
- complete_and_exit(&msp->texit, 0);
+ complete_and_exit(&msp->texit, 0);
}
/* ----------------------------------------------------------------------- */
@@ -1021,6 +1051,49 @@ static struct MODES {
{ 0x0060, MSP_CARRIER(7.2), MSP_CARRIER(7.2), "7.2 SAT ADR" },
{ -1, 0, 0, NULL }, /* EOF */
};
+
+static inline const char *msp34xx_standard_mode_name(int mode)
+{
+ int i;
+ for (i = 0; modelist[i].name != NULL; i++)
+ if (modelist[i].retval == mode)
+ return modelist[i].name;
+ return "unknown";
+}
+
+static int msp34xx_modus(int norm)
+{
+ switch (norm) {
+ case VIDEO_MODE_PAL:
+ return 0x1003;
+ case VIDEO_MODE_NTSC: /* BTSC */
+ return 0x2003;
+ case VIDEO_MODE_SECAM:
+ return 0x0003;
+ case VIDEO_MODE_RADIO:
+ return 0x0003;
+ case VIDEO_MODE_AUTO:
+ return 0x2003;
+ default:
+ return 0x0003;
+ }
+}
+
+static int msp34xx_standard(int norm)
+{
+ switch (norm) {
+ case VIDEO_MODE_PAL:
+ return 1;
+ case VIDEO_MODE_NTSC: /* BTSC */
+ return 0x0020;
+ case VIDEO_MODE_SECAM:
+ return 1;
+ case VIDEO_MODE_RADIO:
+ return 0x0040;
+ default:
+ return 1;
+ }
+}
static int msp3410d_thread(void *data)
{
@@ -1074,39 +1147,14 @@ static int msp3410d_thread(void *data)
msp3400c_reset(client);
/* start autodetect */
- switch (msp->norm) {
- case VIDEO_MODE_PAL:
- mode = 0x1003;
- std = 1;
- break;
- case VIDEO_MODE_NTSC: /* BTSC */
- mode = 0x2003;
- std = 0x0020;
- break;
- case VIDEO_MODE_SECAM:
- mode = 0x0003;
- std = 1;
- break;
- case VIDEO_MODE_RADIO:
- mode = 0x0003;
- std = 0x0040;
- break;
- default:
- mode = 0x0003;
- std = 1;
- break;
- }
+ mode = msp34xx_modus(msp->norm);
+ std = msp34xx_standard(msp->norm);
msp3400c_write(client, I2C_MSP3400C_DEM, 0x30, mode);
msp3400c_write(client, I2C_MSP3400C_DEM, 0x20, std);
- if (debug) {
- int i;
- for (i = 0; modelist[i].name != NULL; i++)
- if (modelist[i].retval == std)
- break;
+ if (debug)
printk(KERN_DEBUG "msp3410: setting mode: %s (0x%04x)\n",
- modelist[i].name ? modelist[i].name : "unknown",std);
- }
+ msp34xx_standard_mode_name(std) ,std);
if (std != 1) {
/* programmed some specific mode */
@@ -1211,12 +1259,14 @@ static int msp3410d_thread(void *data)
msp->watch_stereo = 1;
break;
}
-
- /* unmute + restore dfp registers */
+
+ /* unmute, restore misc registers */
msp3400c_setbass(client, msp->bass);
msp3400c_settreble(client, msp->treble);
msp3400c_setvolume(client, msp->muted,
msp->volume, msp->balance);
+
+ msp3400c_write(client, I2C_MSP3400C_DFP, 0x0013, msp->acb);
msp3400c_restore_dfp(client);
if (msp->watch_stereo)
@@ -1233,6 +1283,296 @@ done:
}
/* ----------------------------------------------------------------------- */
+/* msp34xxG + (simpler no-thread) */
+/* this one uses both automatic standard detection and automatic sound */
+/* select which are available in the newer G versions */
+/* struct msp: only norm, acb and source are really used in this mode */
+
+static void msp34xxg_set_source(struct i2c_client *client, int source);
+
+/* (re-)initialize the msp34xxg, according to the current norm in msp->norm
+ * return 0 if it worked, -1 if it failed
+ */
+static int msp34xxg_reset(struct i2c_client *client)
+{
+ struct msp3400c *msp = i2c_get_clientdata(client);
+ int i;
+ int modus;
+ int std;
+
+ if (msp3400c_reset(client))
+ return -1;
+
+ /* make sure that input/output is muted (paranoid mode) */
+ if (msp3400c_write(client,
+ I2C_MSP3400C_DFP,
+ 0x13, /* ACB */
+ 0x0f20 /* mute DSP input, mute SCART 1 */))
+ return -1;
+
+ /* step-by-step initialisation, as described in the manual */
+ modus = msp34xx_modus(msp->norm);
+ modus &= ~0x03; /* STATUS_CHANGE=0 */
+ modus |= 0x01; /* AUTOMATIC_SOUND_DETECTION=1 */
+ if (msp3400c_write(client,
+ I2C_MSP3400C_DEM,
+ 0x30/*MODUS*/,
+ modus))
+ return -1;
+
+ /* write the dfps that may have an influence on standard/audio autodetection right now */
+ msp34xxg_set_source(client, msp->source);
+
+ if (msp3400c_write_dfp_with_default(client,
+ 0x0e, /* AM/FM Prescale */
+ 0x3000 /* default: [15:8] 75khz deviation */))
+ return -1;
+
+ if (msp3400c_write_dfp_with_default(client,
+ 0x10, /* NICAM Prescale */
+ 0x5a00 /* default: 9db gain (as recommended) */))
+ return -1;
+
+ std = standard;
+ if (msp3400c_write(client,
+ I2C_MSP3400C_DEM,
+ 0x20, /* STANDARD SELECT */
+ std /* default: 0x01 for automatic standard select*/))
+ return -1;
+
+
+ if (std == 0x01) {
+ dprintk("msp34xxg: triggered autodetect, waiting for result\n");
+
+ /* triggered autodetect */
+ for (i = 0; i < 10; i++) {
+ int val;
+ current->state = TASK_INTERRUPTIBLE;
+ schedule_timeout(HZ/10);
+ if (signal_pending(current))
+ return -1; /* failed */
+
+ /* check results */
+ val = msp3400c_read(client, I2C_MSP3400C_DEM, 0x7e);
+ if (val < 0x07ff) {
+ std=val;
+ break;
+ }
+ dprintk("msp34xxg: detection still in progress\n");
+ }
+ if (i == 10) {
+ dprintk("msp34xxg: detection still in progress after 10 tries. giving up.\n");
+ return -1;
+ }
+ }
+ dprintk("msp34xxg: current mode: %s (0x%04x)\n",
+ msp34xx_standard_mode_name(std), std);
+
+ /* unmute: dispatch sound to scart output, set scart volume */
+ dprintk("msp34xxg: unmute\n");
+
+ msp3400c_setbass(client, msp->bass);
+ msp3400c_settreble(client, msp->treble);
+ msp3400c_setvolume(client, msp->muted, msp->volume, msp->balance);
+
+ /* restore other dfp's */
+ msp3400c_restore_dfp(client);
+
+ /* restore ACB */
+ if (msp3400c_write(client,
+ I2C_MSP3400C_DFP,
+ 0x13, /* ACB */
+ msp->acb))
+ return -1;
+
+ return 0;
+}
+
+/* set the same 'source' for the loudspeaker, scart and quasi-peak detector
+ * the value for source is the same as bit 15:8 of DFP registers 0x08,
+ * 0x0a and 0x0c: 0=mono, 1=stereo or A|B, 2=SCART, 3=stereo or A, 4=stereo or B
+ *
+ * this function replaces msp3400c_setstereo
+ */
+static void msp34xxg_set_source(struct i2c_client *client, int source)
+{
+ struct msp3400c *msp = i2c_get_clientdata(client);
+
+ /* fix matrix mode to stereo and let the msp choose what
+ * to output according to 'source', as recommended
+ */
+ int value = (source&0x07)<<8|(source==0 ? 0x00:0x20);
+ dprintk("msp34xxg: set source to %d (0x%x)\n", source, value);
+ msp3400c_write(client,
+ I2C_MSP3400C_DFP,
+ 0x08, /* Loudspeaker Output */
+ value);
+ msp3400c_write(client,
+ I2C_MSP3400C_DFP,
+ 0x0a, /* SCART1 DA Output */
+ value);
+ msp3400c_write(client,
+ I2C_MSP3400C_DFP,
+ 0x0c, /* Quasi-peak detector */
+ value);
+ /*
+ * set identification threshold. Personally, I
+ * I set it to a higher value that the default
+ * of 0x190 to ignore noisy stereo signals.
+ * this needs tuning. (recommended range 0x00a0-0x03c0)
+ * 0x7f0 = forced mono mode
+ */
+ msp3400c_write(client,
+ I2C_MSP3400C_DEM,
+ 0x22, /* a2 threshold for stereo/bilingual */
+ source==0 ? 0x7f0:stereo_threshold);
+ msp->source=source;
+}
+
+/* get the current stereo mode and return it as a V4L1 stereo value (Use V4L2 calls whenever possible) */
+static int msp34xxg_get_v4l1_stereo(struct i2c_client *client)
+{
+ /* This is not really clear: the API says that the mode should
+ * be the current mode, but the old driver returned what the
+ * mode *could* be (like rxsubchans in v4l2). I'll just
+ * follow what the API says...
+ *
+ * The most important point is, I think, that if someone does a get
+ * and then a set with this value, nothing should have changed. It
+ * used not to be the case and that was extremely confusing: one would
+ * do a GET, change mute to 1, then a SET with the same pointer and the
+ * stereo mode would have changed, the watcher thread killed... (see
+ * handler for VIDIOCSAUDIO)
+ */
+ struct msp3400c *msp = i2c_get_clientdata(client);
+ switch (msp->source) {
+ case 1: /* stereo or A|B */
+ return VIDEO_SOUND_LANG1|VIDEO_SOUND_LANG2|VIDEO_SOUND_STEREO;
+ case 3: /* stereo or A */
+ return VIDEO_SOUND_LANG1|VIDEO_SOUND_STEREO;
+ case 4: /* stereo or B */
+ return VIDEO_SOUND_LANG1|VIDEO_SOUND_STEREO;
+ case 0: /* mono */
+ return VIDEO_SOUND_MONO;
+ default: /* scart input */
+ return VIDEO_SOUND_STEREO;
+ }
+}
+
+/* set the current stereo mode given a v4l1 mode (Use V4L2 calls whenever possible) */
+static void msp34xxg_set_v4l1_stereo(struct i2c_client *client, int stereo)
+{
+ int source;
+
+ if ((stereo & VIDEO_SOUND_LANG1) && (stereo & VIDEO_SOUND_LANG2))
+ source = 1; /* stereo or A|B */
+ else if (stereo & VIDEO_SOUND_LANG1)
+ source = 3; /* stereo or A */
+ else if (stereo & VIDEO_SOUND_LANG2)
+ source = 4; /* stereo or B */
+ else if (stereo & VIDEO_SOUND_STEREO)
+ source = 3; /* stereo or A */
+ else
+ source = 0; /* mono only */
+
+ msp34xxg_set_source(client, source);
+}
+
+#if 0
+static void msp34xxg_get_v4l2_stereo(struct i2c_client *client, int *rxsubchans,
+ int *audmode)
+{
+ struct msp3400c *msp = i2c_get_clientdata(client);
+
+ if (rxsubchans) {
+ int status = msp3400c_read(client,
+ I2C_MSP3400C_DEM,
+ 0x0200 /* STATUS */);
+ int is_bilingual = status&0x100;
+ int is_stereo = status&0x40;
+ int val=0;
+ if (is_stereo)
+ val |= V4L2_TUNER_SUB_STEREO;
+ else
+ val |= V4L2_TUNER_SUB_MONO;
+ if (is_bilingual) {
+ val |= V4L2_TUNER_SUB_LANG1|V4L2_TUNER_SUB_LANG2;
+ /* I'm supposed to check whether it's SAP or not
+ * and set only LANG2/SAP in this case. Yet, the MSP
+ * does a lot of work to hide this and handle everything
+ * the same way. I don't want to work around it so unless
+ * this is a problem, I'll handle SAP just like lang1/lang2.
+ */
+ }
+ dprintk("msp34xxg: status=0x%x, stereo=%d, bilingual=%d -> rxsubchans=%d\n",
+ status, is_stereo, is_bilingual, val);
+ *rxsubchans=val;
+ }
+ if (audmode) {
+ int val;
+ switch (msp->source) {
+ case 0: /* mono only */
+ val = V4L2_TUNER_MODE_MONO;
+ break;
+
+ case 1: /* stereo or A|B */
+ case 2: /* scart input (stereo) */
+ /* I'm surprised, but according to v4l2 spec, that
+ * is what V4L2_TUNER_MODE_STEREO *may* mean:
+ * > When the tuner receives bilingual audio it may play
+ * > different languages on the left and right channel or
+ * > the primary language on both channels.
+ * I chose the 2nd interpretation since there's no
+ * BILINGUAL mode.
+ */
+ val = V4L2_TUNER_MODE_STEREO;
+ break;
+
+ default:
+ case 3: /* stereo or A */
+ val = V4L2_TUNER_MODE_LANG1;
+ break;
+ case 4: /* stereo or B */
+ val = V4L2_TUNER_MODE_LANG2;
+ break;
+ }
+ dprintk("msp34xxg: source=%d -> audmode=%d\n",
+ msp->source, val);
+ *audmode=val;
+ }
+}
+
+static void msp34xxg_set_v4l2_stereo(struct i2c_client *client, int audmode)
+{
+ int source = 0;
+
+ switch (audmode) {
+ case V4L2_TUNER_MODE_MONO:
+ source=0; /* mono only */
+ break;
+
+ case V4L2_TUNER_MODE_STEREO:
+ source=1; /* stereo or A|B, see comment in msp34xxg_get_v4l2_stereo() */
+ /* problem: that could also mean 2 (scart input) */
+ break;
+
+ case V4L2_TUNER_MODE_LANG1:
+ source=3; /* stereo or A */
+ break;
+
+ case V4L2_TUNER_MODE_LANG2:
+ source=4; /* stereo or B */
+ break;
+
+ default: /* doing nothing: a safe, sane default */
+ return;
+ }
+
+ msp34xxg_set_source(client, source);
+}
+#endif
+
+/* ----------------------------------------------------------------------- */
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0)
static int msp_attach(struct i2c_adapter *adap, int addr, int kind);
@@ -1272,6 +1612,7 @@ static int msp_attach(struct i2c_adapter *adap, int addr,
{
struct msp3400c *msp;
struct i2c_client *c;
+ int (*thread_func)(void *data) = NULL;
int i;
client_template.adapter = adap;
@@ -1291,12 +1632,12 @@ static int msp_attach(struct i2c_adapter *adap, int addr,
}
memset(msp,0,sizeof(struct msp3400c));
- msp->volume = 65535;
+ msp->volume = 58880; /* 0db gain */
msp->balance = 32768;
- msp->bass = 32768;
- msp->treble = 32768;
- msp->input = -1;
- msp->muted = 1;
+ msp->bass = 32768;
+ msp->treble = 32768;
+ msp->input = -1;
+ msp->muted = 1;
for (i = 0; i < DFP_COUNT; i++)
msp->dfp_regs[i] = -1;
@@ -1330,12 +1671,14 @@ static int msp_attach(struct i2c_adapter *adap, int addr,
(msp->rev2>>8)&0xff, (msp->rev1&0xff)+'@',
((msp->rev1>>8)&0xff)+'@', msp->rev2&0x1f);
- if (simple == -1) {
- /* default mode */
- msp->simple = HAVE_SIMPLE(msp);
- } else {
- /* use insmod option */
- msp->simple = simple;
+ msp->opmode = opmode;
+ if (OPMODE_AUTO == msp->opmode) {
+ if (HAVE_SIMPLER(msp))
+ msp->opmode = OPMODE_SIMPLER;
+ else if (HAVE_SIMPLE(msp))
+ msp->opmode = OPMODE_SIMPLE;
+ else
+ msp->opmode = OPMODE_MANUAL;
}
/* timer for stereo checking */
@@ -1349,22 +1692,40 @@ static int msp_attach(struct i2c_adapter *adap, int addr,
printk(" +nicam");
if (HAVE_SIMPLE(msp))
printk(" +simple");
+ if (HAVE_SIMPLER(msp))
+ printk(" +simpler");
if (HAVE_RADIO(msp))
printk(" +radio");
printk("\n");
- /* startup control thread */
+ /* version-specific initialization */
+ switch (msp->opmode) {
+ case OPMODE_MANUAL:
+ thread_func = msp3400c_thread;
+ break;
+ case OPMODE_SIMPLE:
+ thread_func = msp3410d_thread;
+ break;
+ case OPMODE_SIMPLER:
+ msp34xxg_reset(c);
+ break;
+ }
+
+ /* startup control thread if needed */
+ if (thread_func) {
+ init_completion(&msp->texit);
+ msp->tpid = kernel_thread(thread_func, (void *)c, 0);
+ if (msp->tpid < 0)
+ printk(KERN_WARNING "msp34xx: kernel_thread() failed\n");
+ wake_up_interruptible(&msp->wq);
+ } else {
+ msp->tpid = -1;
+ }
+
+ /* done */
#if LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0)
MOD_INC_USE_COUNT;
#endif
- init_completion(&msp->texit);
- msp->tpid = kernel_thread(msp->simple ? msp3410d_thread : msp3400c_thread,
- (void *)c, 0);
- if (msp->tpid < 0)
- printk(KERN_WARNING "msp34xx: kernel_thread() failed\n");
- wake_up_interruptible(&msp->wq);
-
- /* done */
i2c_attach_client(c);
return 0;
}
@@ -1372,7 +1733,7 @@ static int msp_attach(struct i2c_adapter *adap, int addr,
static int msp_detach(struct i2c_client *client)
{
struct msp3400c *msp = i2c_get_clientdata(client);
-
+
/* shutdown control thread */
del_timer_sync(&msp->wake_stereo);
if (msp->tpid >= 0) {
@@ -1412,6 +1773,8 @@ static void msp_wake_thread(struct i2c_client *client)
{
struct msp3400c *msp = i2c_get_clientdata(client);
+ if (-1 == msp->tpid)
+ return;
msp3400c_setvolume(client,msp->muted,0,0);
msp->watch_stereo=0;
del_timer(&msp->wake_stereo);
@@ -1462,7 +1825,8 @@ static int msp_command(struct i2c_client *client, unsigned int cmd, void *arg)
msp->stereo = VIDEO_SOUND_STEREO;
msp3400c_set_scart(client,scart,0);
msp3400c_write(client,I2C_MSP3400C_DFP,0x000d,0x1900);
- msp3400c_setstereo(client,msp->stereo);
+ if (msp->opmode != OPMODE_SIMPLER)
+ msp3400c_setstereo(client, msp->stereo);
}
if (msp->active)
msp->restart = 1;
@@ -1471,22 +1835,26 @@ static int msp_command(struct i2c_client *client, unsigned int cmd, void *arg)
case AUDC_SET_RADIO:
dprintk(KERN_DEBUG "msp34xx: AUDC_SET_RADIO\n");
msp->norm = VIDEO_MODE_RADIO;
+ dprintk(KERN_DEBUG "msp34xx: switching to radio mode\n");
msp->watch_stereo=0;
del_timer(&msp->wake_stereo);
- dprintk(KERN_DEBUG "msp34xx: switching to radio mode\n");
- if (msp->simple) {
- /* the thread will do for us */
- msp_wake_thread(client);
- } else {
+ switch (msp->opmode) {
+ case OPMODE_MANUAL:
/* set msp3400 to FM radio mode */
msp3400c_setmode(client,MSP_MODE_FM_RADIO);
msp3400c_setcarrier(client, MSP_CARRIER(10.7),
MSP_CARRIER(10.7));
msp3400c_setvolume(client, msp->muted,
- msp->volume, msp->balance);
+ msp->volume, msp->balance);
+ break;
+ case OPMODE_SIMPLE:
+ /* the thread will do for us */
+ msp_wake_thread(client);
+ break;
+ case OPMODE_SIMPLER:
+ msp34xxg_reset(client);
+ break;
}
- if (msp->active)
- msp->restart = 1;
break;
#if 1
@@ -1536,7 +1904,9 @@ static int msp_command(struct i2c_client *client, unsigned int cmd, void *arg)
va->bass = msp->bass;
va->treble = msp->treble;
- if (msp->norm != VIDEO_MODE_RADIO) {
+ if (msp->opmode == OPMODE_SIMPLER) {
+ va->mode = msp34xxg_get_v4l1_stereo(client);
+ } else if (msp->norm != VIDEO_MODE_RADIO) {
autodetect_stereo(client);
va->mode = msp->stereo;
}
@@ -1559,10 +1929,14 @@ static int msp_command(struct i2c_client *client, unsigned int cmd, void *arg)
msp3400c_settreble(client,msp->treble);
if (va->mode != 0 && msp->norm != VIDEO_MODE_RADIO) {
- msp->watch_stereo=0;
- del_timer(&msp->wake_stereo);
- msp->stereo = va->mode & 0x0f;
- msp3400c_setstereo(client,va->mode & 0x0f);
+ if (msp->opmode == OPMODE_SIMPLER) {
+ msp34xxg_set_v4l1_stereo(client, va->mode);
+ } else {
+ msp->watch_stereo=0;
+ del_timer(&msp->wake_stereo);
+ msp->stereo = va->mode & 0x0f;
+ msp3400c_setstereo(client,va->mode & 0x0f);
+ }
}
break;
}
@@ -1572,13 +1946,64 @@ static int msp_command(struct i2c_client *client, unsigned int cmd, void *arg)
dprintk(KERN_DEBUG "msp34xx: VIDIOCSCHAN (norm=%d)\n",vc->norm);
msp->norm = vc->norm;
+ switch (msp->opmode) {
+ case OPMODE_MANUAL:
+ case OPMODE_SIMPLE:
+ msp_wake_thread(client);
+ break;
+ case OPMODE_SIMPLER:
+ msp34xxg_reset(client);
+ break;
+ }
break;
}
case VIDIOCSFREQ:
{
/* new channel -- kick audio carrier scan */
dprintk(KERN_DEBUG "msp34xx: VIDIOCSFREQ\n");
- msp_wake_thread(client);
+ switch (msp->opmode) {
+ case OPMODE_MANUAL:
+ case OPMODE_SIMPLE:
+ msp_wake_thread(client);
+ break;
+ case OPMODE_SIMPLER:
+ msp34xxg_reset(client);
+ break;
+ }
+ break;
+ }
+
+#if 0
+ case VIDIOC_G_TUNER:
+ {
+ struct v4l2_tuner *vt=(struct v4l2_tuner *)arg;
+ vt->capability|=V4L2_TUNER_CAP_STEREO|V4L2_TUNER_CAP_LANG1|V4L2_TUNER_CAP_LANG2;
+
+ /* get rxsubchans and audmode */
+ if (IS_MSP34XX_G(msp)) {
+ msp34xxg_get_v4l2_stereo(client, &vt->rxsubchans, &vt->audmode);
+ }
+ break;
+ }
+ case VIDIOC_S_TUNER:
+ {
+ struct v4l2_tuner *vt=(struct v4l2_tuner *)arg;
+ /* only set audmode */
+ if (vt->audmode!=-1) {
+ if (IS_MSP34XX_G(msp)) {
+ msp34xxg_set_v4l2_stereo(client, vt->audmode);
+ }
+ }
+ break;
+ }
+#endif
+
+ case MSP_SET_MATRIX:
+ {
+ struct msp_matrix *mspm = arg;
+
+ dprintk(KERN_DEBUG "msp34xx: MSP_SET_MATRIX\n");
+ msp3400c_set_scart(client, mspm->input, mspm->output);
break;
}
diff --git a/linux/drivers/media/video/msp3400.c b/linux/drivers/media/video/msp3400.c
index 74e57e172..6f930254c 100644
--- a/linux/drivers/media/video/msp3400.c
+++ b/linux/drivers/media/video/msp3400.c
@@ -61,14 +61,23 @@
#include "msp3400.h"
#include "compat.h"
+#define OPMODE_AUTO -1
+#define OPMODE_MANUAL 0
+#define OPMODE_SIMPLE 1 /* use short programming (>= msp3410 only) */
+#define OPMODE_SIMPLER 2 /* use shorter programming (>= msp34xxG) */
+
/* insmod parameters */
+static int opmode = OPMODE_AUTO;
static int debug = 0; /* debug output */
static int once = 0; /* no continous stereo monitoring */
static int amsound = 0; /* hard-wire AM sound at 6.5 Hz (france),
- the autoscan seems work well only with FM... */
-static int simple = -1; /* use short programming (>= msp3410 only) */
+ the autoscan seems work well only with FM... */
+static int standard = 1; /* Override auto detect of audio standard, if needed. */
static int dolby = 0;
+static int stereo_threshold = 0x190; /* a2 threshold for stereo/bilingual
+ (msp34xxg only) 0x00a0-0x03c0 */
+
#define DFP_COUNT 0x41
static const int bl_dfp[] = {
0x00, 0x01, 0x02, 0x03, 0x06, 0x08, 0x09, 0x0a,
@@ -78,7 +87,7 @@ static const int bl_dfp[] = {
struct msp3400c {
int rev1,rev2;
- int simple;
+ int opmode;
int mode;
int norm;
int stereo;
@@ -86,6 +95,7 @@ struct msp3400c {
int acb;
int main, second; /* sound carrier */
int input;
+ int source; /* see msp34xxg_set_source */
int muted;
int volume, balance;
@@ -102,6 +112,9 @@ struct msp3400c {
int active:1;
int restart:1;
int rmmod:1;
+#if 0
+ struct semaphore *notify;
+#endif
int watch_stereo;
struct timer_list wake_stereo;
@@ -109,6 +122,7 @@ struct msp3400c {
#define HAVE_NICAM(msp) (((msp->rev2>>8) & 0xff) != 00)
#define HAVE_SIMPLE(msp) ((msp->rev1 & 0xff) >= 'D'-'@')
+#define HAVE_SIMPLER(msp) ((msp->rev1 & 0xff) >= 'G'-'@')
#define HAVE_RADIO(msp) ((msp->rev1 & 0xff) >= 'G'-'@')
#define VIDEO_MODE_RADIO 16 /* norm magic for radio mode */
@@ -119,9 +133,15 @@ struct msp3400c {
#define d2printk if (debug >= 2) printk
MODULE_PARM(once,"i");
+MODULE_PARM_DESC(once, "No continuous stereo monitoring");
MODULE_PARM(debug,"i");
-MODULE_PARM(simple,"i");
+MODULE_PARM_DESC(debug, "Enable debug messages");
+MODULE_PARM(opmode,"i");
+MODULE_PARM(stereo_threshold,"i");
+MODULE_PARM(standard,"i");
+MODULE_PARM_DESC(standard, "Specify audio standard: 32 = NTSC, 64 = radio, Default: Autodetect");
MODULE_PARM(amsound,"i");
+MODULE_PARM_DESC(amsound, "Hardwire AM sound at 6.5Hz (France), FM can autoscan");
MODULE_PARM(dolby,"i");
MODULE_DESCRIPTION("device driver for msp34xx TV sound processor");
@@ -205,7 +225,7 @@ msp3400c_read(struct i2c_client *client, int dev, int addr)
msp3400c_reset(client);
return -1;
}
- return read[0] << 8 | read[1];
+ return read[0] << 8 | read[1];
}
static int
@@ -336,16 +356,6 @@ static struct CARRIER_DETECT carrier_detect_65[] = {
/* ----------------------------------------------------------------------- */
-#define SCART_MASK 0
-#define SCART_IN1 1
-#define SCART_IN2 2
-#define SCART_IN1_DA 3
-#define SCART_IN2_DA 4
-#define SCART_IN3 5
-#define SCART_IN4 6
-#define SCART_MONO 7
-#define SCART_MUTE 8
-
static int scarts[3][9] = {
/* MASK IN1 IN2 IN1_DA IN2_DA IN3 IN4 MONO MUTE */
{ 0x0320, 0x0000, 0x0200, -1, -1, 0x0300, 0x0020, 0x0100, 0x0320 },
@@ -399,8 +409,8 @@ static void msp3400c_setvolume(struct i2c_client *client,
muted ? "on" : "off", volume, balance, val>>8, bal);
msp3400c_write(client,I2C_MSP3400C_DFP, 0x0000, val); /* loudspeaker */
msp3400c_write(client,I2C_MSP3400C_DFP, 0x0006, val); /* headphones */
- /* scart - on/off only */
- msp3400c_write(client,I2C_MSP3400C_DFP, 0x0007, val ? 0x4000 : 0);
+ msp3400c_write(client,I2C_MSP3400C_DFP, 0x0007,
+ muted ? 0x01 : (val | 0x01));
msp3400c_write(client,I2C_MSP3400C_DFP, 0x0001, bal << 8);
}
@@ -477,6 +487,18 @@ static void msp3400c_setmode(struct i2c_client *client, int type)
}
}
+// given a bitmask of VIDEO_SOUND_XXX returns the "best" in the bitmask
+static int best_audio_mode(int mode)
+{
+ if (mode & VIDEO_SOUND_STEREO)
+ return VIDEO_SOUND_STEREO;
+ if (mode & VIDEO_SOUND_LANG1)
+ return VIDEO_SOUND_LANG1;
+ if (mode & VIDEO_SOUND_LANG2)
+ return VIDEO_SOUND_LANG2;
+ return VIDEO_SOUND_MONO;
+}
+
/* turn on/off nicam + stereo */
static void msp3400c_setstereo(struct i2c_client *client, int mode)
{
@@ -493,6 +515,8 @@ static void msp3400c_setstereo(struct i2c_client *client, int mode)
int nicam=0; /* channel source: FM/AM or nicam */
int src=0;
+ BUG_ON(msp->opmode == OPMODE_SIMPLER);
+
/* switch demodulator */
switch (msp->mode) {
case MSP_MODE_FM_TERRA:
@@ -628,6 +652,18 @@ msp3400c_restore_dfp(struct i2c_client *client)
}
}
+/* if the dfp_regs is set, set what's in there.
+ * Otherwise, set the default value */
+static int msp3400c_write_dfp_with_default(struct i2c_client *client, int addr,
+ int default_value)
+{
+ struct msp3400c *msp = i2c_get_clientdata(client);
+ int value=default_value;
+ if ( addr< DFP_COUNT && -1 != msp->dfp_regs[addr] )
+ value=msp->dfp_regs[addr];
+ return msp3400c_write(client, I2C_MSP3400C_DFP, addr, value);
+}
+
/* ----------------------------------------------------------------------- */
struct REGISTER_DUMP {
@@ -767,16 +803,8 @@ static void watch_stereo(struct i2c_client *client)
{
struct msp3400c *msp = i2c_get_clientdata(client);
- if (autodetect_stereo(client)) {
- if (msp->stereo & VIDEO_SOUND_STEREO)
- msp3400c_setstereo(client,VIDEO_SOUND_STEREO);
- else if (msp->stereo & VIDEO_SOUND_LANG1)
- msp3400c_setstereo(client,VIDEO_SOUND_LANG1);
- else if (msp->stereo & VIDEO_SOUND_LANG2)
- msp3400c_setstereo(client,VIDEO_SOUND_LANG2);
- else
- msp3400c_setstereo(client,VIDEO_SOUND_MONO);
- }
+ if (autodetect_stereo(client))
+ msp3400c_setstereo(client,best_audio_mode(msp->stereo));
if (once)
msp->watch_stereo = 0;
if (msp->watch_stereo)
@@ -867,10 +895,12 @@ static int msp3400c_thread(void *data)
/* carrier detect pass #2 -- second (stereo) carrier */
switch (max1) {
case 1: /* 5.5 */
- cd = carrier_detect_55; count = CARRIER_COUNT(carrier_detect_55);
+ cd = carrier_detect_55;
+ count = CARRIER_COUNT(carrier_detect_55);
break;
case 3: /* 6.5 */
- cd = carrier_detect_65; count = CARRIER_COUNT(carrier_detect_65);
+ cd = carrier_detect_65;
+ count = CARRIER_COUNT(carrier_detect_65);
break;
case 0: /* 4.5 */
case 2: /* 6.0 */
@@ -988,7 +1018,7 @@ static int msp3400c_thread(void *data)
done:
msp->active = 0;
dprintk(KERN_DEBUG "msp3400: thread: exit\n");
- complete_and_exit(&msp->texit, 0);
+ complete_and_exit(&msp->texit, 0);
}
/* ----------------------------------------------------------------------- */
@@ -1021,6 +1051,49 @@ static struct MODES {
{ 0x0060, MSP_CARRIER(7.2), MSP_CARRIER(7.2), "7.2 SAT ADR" },
{ -1, 0, 0, NULL }, /* EOF */
};
+
+static inline const char *msp34xx_standard_mode_name(int mode)
+{
+ int i;
+ for (i = 0; modelist[i].name != NULL; i++)
+ if (modelist[i].retval == mode)
+ return modelist[i].name;
+ return "unknown";
+}
+
+static int msp34xx_modus(int norm)
+{
+ switch (norm) {
+ case VIDEO_MODE_PAL:
+ return 0x1003;
+ case VIDEO_MODE_NTSC: /* BTSC */
+ return 0x2003;
+ case VIDEO_MODE_SECAM:
+ return 0x0003;
+ case VIDEO_MODE_RADIO:
+ return 0x0003;
+ case VIDEO_MODE_AUTO:
+ return 0x2003;
+ default:
+ return 0x0003;
+ }
+}
+
+static int msp34xx_standard(int norm)
+{
+ switch (norm) {
+ case VIDEO_MODE_PAL:
+ return 1;
+ case VIDEO_MODE_NTSC: /* BTSC */
+ return 0x0020;
+ case VIDEO_MODE_SECAM:
+ return 1;
+ case VIDEO_MODE_RADIO:
+ return 0x0040;
+ default:
+ return 1;
+ }
+}
static int msp3410d_thread(void *data)
{
@@ -1074,39 +1147,14 @@ static int msp3410d_thread(void *data)
msp3400c_reset(client);
/* start autodetect */
- switch (msp->norm) {
- case VIDEO_MODE_PAL:
- mode = 0x1003;
- std = 1;
- break;
- case VIDEO_MODE_NTSC: /* BTSC */
- mode = 0x2003;
- std = 0x0020;
- break;
- case VIDEO_MODE_SECAM:
- mode = 0x0003;
- std = 1;
- break;
- case VIDEO_MODE_RADIO:
- mode = 0x0003;
- std = 0x0040;
- break;
- default:
- mode = 0x0003;
- std = 1;
- break;
- }
+ mode = msp34xx_modus(msp->norm);
+ std = msp34xx_standard(msp->norm);
msp3400c_write(client, I2C_MSP3400C_DEM, 0x30, mode);
msp3400c_write(client, I2C_MSP3400C_DEM, 0x20, std);
- if (debug) {
- int i;
- for (i = 0; modelist[i].name != NULL; i++)
- if (modelist[i].retval == std)
- break;
+ if (debug)
printk(KERN_DEBUG "msp3410: setting mode: %s (0x%04x)\n",
- modelist[i].name ? modelist[i].name : "unknown",std);
- }
+ msp34xx_standard_mode_name(std) ,std);
if (std != 1) {
/* programmed some specific mode */
@@ -1211,12 +1259,14 @@ static int msp3410d_thread(void *data)
msp->watch_stereo = 1;
break;
}
-
- /* unmute + restore dfp registers */
+
+ /* unmute, restore misc registers */
msp3400c_setbass(client, msp->bass);
msp3400c_settreble(client, msp->treble);
msp3400c_setvolume(client, msp->muted,
msp->volume, msp->balance);
+
+ msp3400c_write(client, I2C_MSP3400C_DFP, 0x0013, msp->acb);
msp3400c_restore_dfp(client);
if (msp->watch_stereo)
@@ -1233,6 +1283,296 @@ done:
}
/* ----------------------------------------------------------------------- */
+/* msp34xxG + (simpler no-thread) */
+/* this one uses both automatic standard detection and automatic sound */
+/* select which are available in the newer G versions */
+/* struct msp: only norm, acb and source are really used in this mode */
+
+static void msp34xxg_set_source(struct i2c_client *client, int source);
+
+/* (re-)initialize the msp34xxg, according to the current norm in msp->norm
+ * return 0 if it worked, -1 if it failed
+ */
+static int msp34xxg_reset(struct i2c_client *client)
+{
+ struct msp3400c *msp = i2c_get_clientdata(client);
+ int i;
+ int modus;
+ int std;
+
+ if (msp3400c_reset(client))
+ return -1;
+
+ /* make sure that input/output is muted (paranoid mode) */
+ if (msp3400c_write(client,
+ I2C_MSP3400C_DFP,
+ 0x13, /* ACB */
+ 0x0f20 /* mute DSP input, mute SCART 1 */))
+ return -1;
+
+ /* step-by-step initialisation, as described in the manual */
+ modus = msp34xx_modus(msp->norm);
+ modus &= ~0x03; /* STATUS_CHANGE=0 */
+ modus |= 0x01; /* AUTOMATIC_SOUND_DETECTION=1 */
+ if (msp3400c_write(client,
+ I2C_MSP3400C_DEM,
+ 0x30/*MODUS*/,
+ modus))
+ return -1;
+
+ /* write the dfps that may have an influence on standard/audio autodetection right now */
+ msp34xxg_set_source(client, msp->source);
+
+ if (msp3400c_write_dfp_with_default(client,
+ 0x0e, /* AM/FM Prescale */
+ 0x3000 /* default: [15:8] 75khz deviation */))
+ return -1;
+
+ if (msp3400c_write_dfp_with_default(client,
+ 0x10, /* NICAM Prescale */
+ 0x5a00 /* default: 9db gain (as recommended) */))
+ return -1;
+
+ std = standard;
+ if (msp3400c_write(client,
+ I2C_MSP3400C_DEM,
+ 0x20, /* STANDARD SELECT */
+ std /* default: 0x01 for automatic standard select*/))
+ return -1;
+
+
+ if (std == 0x01) {
+ dprintk("msp34xxg: triggered autodetect, waiting for result\n");
+
+ /* triggered autodetect */
+ for (i = 0; i < 10; i++) {
+ int val;
+ current->state = TASK_INTERRUPTIBLE;
+ schedule_timeout(HZ/10);
+ if (signal_pending(current))
+ return -1; /* failed */
+
+ /* check results */
+ val = msp3400c_read(client, I2C_MSP3400C_DEM, 0x7e);
+ if (val < 0x07ff) {
+ std=val;
+ break;
+ }
+ dprintk("msp34xxg: detection still in progress\n");
+ }
+ if (i == 10) {
+ dprintk("msp34xxg: detection still in progress after 10 tries. giving up.\n");
+ return -1;
+ }
+ }
+ dprintk("msp34xxg: current mode: %s (0x%04x)\n",
+ msp34xx_standard_mode_name(std), std);
+
+ /* unmute: dispatch sound to scart output, set scart volume */
+ dprintk("msp34xxg: unmute\n");
+
+ msp3400c_setbass(client, msp->bass);
+ msp3400c_settreble(client, msp->treble);
+ msp3400c_setvolume(client, msp->muted, msp->volume, msp->balance);
+
+ /* restore other dfp's */
+ msp3400c_restore_dfp(client);
+
+ /* restore ACB */
+ if (msp3400c_write(client,
+ I2C_MSP3400C_DFP,
+ 0x13, /* ACB */
+ msp->acb))
+ return -1;
+
+ return 0;
+}
+
+/* set the same 'source' for the loudspeaker, scart and quasi-peak detector
+ * the value for source is the same as bit 15:8 of DFP registers 0x08,
+ * 0x0a and 0x0c: 0=mono, 1=stereo or A|B, 2=SCART, 3=stereo or A, 4=stereo or B
+ *
+ * this function replaces msp3400c_setstereo
+ */
+static void msp34xxg_set_source(struct i2c_client *client, int source)
+{
+ struct msp3400c *msp = i2c_get_clientdata(client);
+
+ /* fix matrix mode to stereo and let the msp choose what
+ * to output according to 'source', as recommended
+ */
+ int value = (source&0x07)<<8|(source==0 ? 0x00:0x20);
+ dprintk("msp34xxg: set source to %d (0x%x)\n", source, value);
+ msp3400c_write(client,
+ I2C_MSP3400C_DFP,
+ 0x08, /* Loudspeaker Output */
+ value);
+ msp3400c_write(client,
+ I2C_MSP3400C_DFP,
+ 0x0a, /* SCART1 DA Output */
+ value);
+ msp3400c_write(client,
+ I2C_MSP3400C_DFP,
+ 0x0c, /* Quasi-peak detector */
+ value);
+ /*
+ * set identification threshold. Personally, I
+ * I set it to a higher value that the default
+ * of 0x190 to ignore noisy stereo signals.
+ * this needs tuning. (recommended range 0x00a0-0x03c0)
+ * 0x7f0 = forced mono mode
+ */
+ msp3400c_write(client,
+ I2C_MSP3400C_DEM,
+ 0x22, /* a2 threshold for stereo/bilingual */
+ source==0 ? 0x7f0:stereo_threshold);
+ msp->source=source;
+}
+
+/* get the current stereo mode and return it as a V4L1 stereo value (Use V4L2 calls whenever possible) */
+static int msp34xxg_get_v4l1_stereo(struct i2c_client *client)
+{
+ /* This is not really clear: the API says that the mode should
+ * be the current mode, but the old driver returned what the
+ * mode *could* be (like rxsubchans in v4l2). I'll just
+ * follow what the API says...
+ *
+ * The most important point is, I think, that if someone does a get
+ * and then a set with this value, nothing should have changed. It
+ * used not to be the case and that was extremely confusing: one would
+ * do a GET, change mute to 1, then a SET with the same pointer and the
+ * stereo mode would have changed, the watcher thread killed... (see
+ * handler for VIDIOCSAUDIO)
+ */
+ struct msp3400c *msp = i2c_get_clientdata(client);
+ switch (msp->source) {
+ case 1: /* stereo or A|B */
+ return VIDEO_SOUND_LANG1|VIDEO_SOUND_LANG2|VIDEO_SOUND_STEREO;
+ case 3: /* stereo or A */
+ return VIDEO_SOUND_LANG1|VIDEO_SOUND_STEREO;
+ case 4: /* stereo or B */
+ return VIDEO_SOUND_LANG1|VIDEO_SOUND_STEREO;
+ case 0: /* mono */
+ return VIDEO_SOUND_MONO;
+ default: /* scart input */
+ return VIDEO_SOUND_STEREO;
+ }
+}
+
+/* set the current stereo mode given a v4l1 mode (Use V4L2 calls whenever possible) */
+static void msp34xxg_set_v4l1_stereo(struct i2c_client *client, int stereo)
+{
+ int source;
+
+ if ((stereo & VIDEO_SOUND_LANG1) && (stereo & VIDEO_SOUND_LANG2))
+ source = 1; /* stereo or A|B */
+ else if (stereo & VIDEO_SOUND_LANG1)
+ source = 3; /* stereo or A */
+ else if (stereo & VIDEO_SOUND_LANG2)
+ source = 4; /* stereo or B */
+ else if (stereo & VIDEO_SOUND_STEREO)
+ source = 3; /* stereo or A */
+ else
+ source = 0; /* mono only */
+
+ msp34xxg_set_source(client, source);
+}
+
+#if 0
+static void msp34xxg_get_v4l2_stereo(struct i2c_client *client, int *rxsubchans,
+ int *audmode)
+{
+ struct msp3400c *msp = i2c_get_clientdata(client);
+
+ if (rxsubchans) {
+ int status = msp3400c_read(client,
+ I2C_MSP3400C_DEM,
+ 0x0200 /* STATUS */);
+ int is_bilingual = status&0x100;
+ int is_stereo = status&0x40;
+ int val=0;
+ if (is_stereo)
+ val |= V4L2_TUNER_SUB_STEREO;
+ else
+ val |= V4L2_TUNER_SUB_MONO;
+ if (is_bilingual) {
+ val |= V4L2_TUNER_SUB_LANG1|V4L2_TUNER_SUB_LANG2;
+ /* I'm supposed to check whether it's SAP or not
+ * and set only LANG2/SAP in this case. Yet, the MSP
+ * does a lot of work to hide this and handle everything
+ * the same way. I don't want to work around it so unless
+ * this is a problem, I'll handle SAP just like lang1/lang2.
+ */
+ }
+ dprintk("msp34xxg: status=0x%x, stereo=%d, bilingual=%d -> rxsubchans=%d\n",
+ status, is_stereo, is_bilingual, val);
+ *rxsubchans=val;
+ }
+ if (audmode) {
+ int val;
+ switch (msp->source) {
+ case 0: /* mono only */
+ val = V4L2_TUNER_MODE_MONO;
+ break;
+
+ case 1: /* stereo or A|B */
+ case 2: /* scart input (stereo) */
+ /* I'm surprised, but according to v4l2 spec, that
+ * is what V4L2_TUNER_MODE_STEREO *may* mean:
+ * > When the tuner receives bilingual audio it may play
+ * > different languages on the left and right channel or
+ * > the primary language on both channels.
+ * I chose the 2nd interpretation since there's no
+ * BILINGUAL mode.
+ */
+ val = V4L2_TUNER_MODE_STEREO;
+ break;
+
+ default:
+ case 3: /* stereo or A */
+ val = V4L2_TUNER_MODE_LANG1;
+ break;
+ case 4: /* stereo or B */
+ val = V4L2_TUNER_MODE_LANG2;
+ break;
+ }
+ dprintk("msp34xxg: source=%d -> audmode=%d\n",
+ msp->source, val);
+ *audmode=val;
+ }
+}
+
+static void msp34xxg_set_v4l2_stereo(struct i2c_client *client, int audmode)
+{
+ int source = 0;
+
+ switch (audmode) {
+ case V4L2_TUNER_MODE_MONO:
+ source=0; /* mono only */
+ break;
+
+ case V4L2_TUNER_MODE_STEREO:
+ source=1; /* stereo or A|B, see comment in msp34xxg_get_v4l2_stereo() */
+ /* problem: that could also mean 2 (scart input) */
+ break;
+
+ case V4L2_TUNER_MODE_LANG1:
+ source=3; /* stereo or A */
+ break;
+
+ case V4L2_TUNER_MODE_LANG2:
+ source=4; /* stereo or B */
+ break;
+
+ default: /* doing nothing: a safe, sane default */
+ return;
+ }
+
+ msp34xxg_set_source(client, source);
+}
+#endif
+
+/* ----------------------------------------------------------------------- */
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0)
static int msp_attach(struct i2c_adapter *adap, int addr, int kind);
@@ -1272,6 +1612,7 @@ static int msp_attach(struct i2c_adapter *adap, int addr,
{
struct msp3400c *msp;
struct i2c_client *c;
+ int (*thread_func)(void *data) = NULL;
int i;
client_template.adapter = adap;
@@ -1291,12 +1632,12 @@ static int msp_attach(struct i2c_adapter *adap, int addr,
}
memset(msp,0,sizeof(struct msp3400c));
- msp->volume = 65535;
+ msp->volume = 58880; /* 0db gain */
msp->balance = 32768;
- msp->bass = 32768;
- msp->treble = 32768;
- msp->input = -1;
- msp->muted = 1;
+ msp->bass = 32768;
+ msp->treble = 32768;
+ msp->input = -1;
+ msp->muted = 1;
for (i = 0; i < DFP_COUNT; i++)
msp->dfp_regs[i] = -1;
@@ -1330,12 +1671,14 @@ static int msp_attach(struct i2c_adapter *adap, int addr,
(msp->rev2>>8)&0xff, (msp->rev1&0xff)+'@',
((msp->rev1>>8)&0xff)+'@', msp->rev2&0x1f);
- if (simple == -1) {
- /* default mode */
- msp->simple = HAVE_SIMPLE(msp);
- } else {
- /* use insmod option */
- msp->simple = simple;
+ msp->opmode = opmode;
+ if (OPMODE_AUTO == msp->opmode) {
+ if (HAVE_SIMPLER(msp))
+ msp->opmode = OPMODE_SIMPLER;
+ else if (HAVE_SIMPLE(msp))
+ msp->opmode = OPMODE_SIMPLE;
+ else
+ msp->opmode = OPMODE_MANUAL;
}
/* timer for stereo checking */
@@ -1349,22 +1692,40 @@ static int msp_attach(struct i2c_adapter *adap, int addr,
printk(" +nicam");
if (HAVE_SIMPLE(msp))
printk(" +simple");
+ if (HAVE_SIMPLER(msp))
+ printk(" +simpler");
if (HAVE_RADIO(msp))
printk(" +radio");
printk("\n");
- /* startup control thread */
+ /* version-specific initialization */
+ switch (msp->opmode) {
+ case OPMODE_MANUAL:
+ thread_func = msp3400c_thread;
+ break;
+ case OPMODE_SIMPLE:
+ thread_func = msp3410d_thread;
+ break;
+ case OPMODE_SIMPLER:
+ msp34xxg_reset(c);
+ break;
+ }
+
+ /* startup control thread if needed */
+ if (thread_func) {
+ init_completion(&msp->texit);
+ msp->tpid = kernel_thread(thread_func, (void *)c, 0);
+ if (msp->tpid < 0)
+ printk(KERN_WARNING "msp34xx: kernel_thread() failed\n");
+ wake_up_interruptible(&msp->wq);
+ } else {
+ msp->tpid = -1;
+ }
+
+ /* done */
#if LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0)
MOD_INC_USE_COUNT;
#endif
- init_completion(&msp->texit);
- msp->tpid = kernel_thread(msp->simple ? msp3410d_thread : msp3400c_thread,
- (void *)c, 0);
- if (msp->tpid < 0)
- printk(KERN_WARNING "msp34xx: kernel_thread() failed\n");
- wake_up_interruptible(&msp->wq);
-
- /* done */
i2c_attach_client(c);
return 0;
}
@@ -1372,7 +1733,7 @@ static int msp_attach(struct i2c_adapter *adap, int addr,
static int msp_detach(struct i2c_client *client)
{
struct msp3400c *msp = i2c_get_clientdata(client);
-
+
/* shutdown control thread */
del_timer_sync(&msp->wake_stereo);
if (msp->tpid >= 0) {
@@ -1412,6 +1773,8 @@ static void msp_wake_thread(struct i2c_client *client)
{
struct msp3400c *msp = i2c_get_clientdata(client);
+ if (-1 == msp->tpid)
+ return;
msp3400c_setvolume(client,msp->muted,0,0);
msp->watch_stereo=0;
del_timer(&msp->wake_stereo);
@@ -1462,7 +1825,8 @@ static int msp_command(struct i2c_client *client, unsigned int cmd, void *arg)
msp->stereo = VIDEO_SOUND_STEREO;
msp3400c_set_scart(client,scart,0);
msp3400c_write(client,I2C_MSP3400C_DFP,0x000d,0x1900);
- msp3400c_setstereo(client,msp->stereo);
+ if (msp->opmode != OPMODE_SIMPLER)
+ msp3400c_setstereo(client, msp->stereo);
}
if (msp->active)
msp->restart = 1;
@@ -1471,22 +1835,26 @@ static int msp_command(struct i2c_client *client, unsigned int cmd, void *arg)
case AUDC_SET_RADIO:
dprintk(KERN_DEBUG "msp34xx: AUDC_SET_RADIO\n");
msp->norm = VIDEO_MODE_RADIO;
+ dprintk(KERN_DEBUG "msp34xx: switching to radio mode\n");
msp->watch_stereo=0;
del_timer(&msp->wake_stereo);
- dprintk(KERN_DEBUG "msp34xx: switching to radio mode\n");
- if (msp->simple) {
- /* the thread will do for us */
- msp_wake_thread(client);
- } else {
+ switch (msp->opmode) {
+ case OPMODE_MANUAL:
/* set msp3400 to FM radio mode */
msp3400c_setmode(client,MSP_MODE_FM_RADIO);
msp3400c_setcarrier(client, MSP_CARRIER(10.7),
MSP_CARRIER(10.7));
msp3400c_setvolume(client, msp->muted,
- msp->volume, msp->balance);
+ msp->volume, msp->balance);
+ break;
+ case OPMODE_SIMPLE:
+ /* the thread will do for us */
+ msp_wake_thread(client);
+ break;
+ case OPMODE_SIMPLER:
+ msp34xxg_reset(client);
+ break;
}
- if (msp->active)
- msp->restart = 1;
break;
#if 1
@@ -1536,7 +1904,9 @@ static int msp_command(struct i2c_client *client, unsigned int cmd, void *arg)
va->bass = msp->bass;
va->treble = msp->treble;
- if (msp->norm != VIDEO_MODE_RADIO) {
+ if (msp->opmode == OPMODE_SIMPLER) {
+ va->mode = msp34xxg_get_v4l1_stereo(client);
+ } else if (msp->norm != VIDEO_MODE_RADIO) {
autodetect_stereo(client);
va->mode = msp->stereo;
}
@@ -1559,10 +1929,14 @@ static int msp_command(struct i2c_client *client, unsigned int cmd, void *arg)
msp3400c_settreble(client,msp->treble);
if (va->mode != 0 && msp->norm != VIDEO_MODE_RADIO) {
- msp->watch_stereo=0;
- del_timer(&msp->wake_stereo);
- msp->stereo = va->mode & 0x0f;
- msp3400c_setstereo(client,va->mode & 0x0f);
+ if (msp->opmode == OPMODE_SIMPLER) {
+ msp34xxg_set_v4l1_stereo(client, va->mode);
+ } else {
+ msp->watch_stereo=0;
+ del_timer(&msp->wake_stereo);
+ msp->stereo = va->mode & 0x0f;
+ msp3400c_setstereo(client,va->mode & 0x0f);
+ }
}
break;
}
@@ -1572,13 +1946,64 @@ static int msp_command(struct i2c_client *client, unsigned int cmd, void *arg)
dprintk(KERN_DEBUG "msp34xx: VIDIOCSCHAN (norm=%d)\n",vc->norm);
msp->norm = vc->norm;
+ switch (msp->opmode) {
+ case OPMODE_MANUAL:
+ case OPMODE_SIMPLE:
+ msp_wake_thread(client);
+ break;
+ case OPMODE_SIMPLER:
+ msp34xxg_reset(client);
+ break;
+ }
break;
}
case VIDIOCSFREQ:
{
/* new channel -- kick audio carrier scan */
dprintk(KERN_DEBUG "msp34xx: VIDIOCSFREQ\n");
- msp_wake_thread(client);
+ switch (msp->opmode) {
+ case OPMODE_MANUAL:
+ case OPMODE_SIMPLE:
+ msp_wake_thread(client);
+ break;
+ case OPMODE_SIMPLER:
+ msp34xxg_reset(client);
+ break;
+ }
+ break;
+ }
+
+#if 0
+ case VIDIOC_G_TUNER:
+ {
+ struct v4l2_tuner *vt=(struct v4l2_tuner *)arg;
+ vt->capability|=V4L2_TUNER_CAP_STEREO|V4L2_TUNER_CAP_LANG1|V4L2_TUNER_CAP_LANG2;
+
+ /* get rxsubchans and audmode */
+ if (IS_MSP34XX_G(msp)) {
+ msp34xxg_get_v4l2_stereo(client, &vt->rxsubchans, &vt->audmode);
+ }
+ break;
+ }
+ case VIDIOC_S_TUNER:
+ {
+ struct v4l2_tuner *vt=(struct v4l2_tuner *)arg;
+ /* only set audmode */
+ if (vt->audmode!=-1) {
+ if (IS_MSP34XX_G(msp)) {
+ msp34xxg_set_v4l2_stereo(client, vt->audmode);
+ }
+ }
+ break;
+ }
+#endif
+
+ case MSP_SET_MATRIX:
+ {
+ struct msp_matrix *mspm = arg;
+
+ dprintk(KERN_DEBUG "msp34xx: MSP_SET_MATRIX\n");
+ msp3400c_set_scart(client, mspm->input, mspm->output);
break;
}
diff --git a/linux/drivers/media/video/msp3400.h b/linux/drivers/media/video/msp3400.h
index 9673133f2..d70a954e1 100644
--- a/linux/drivers/media/video/msp3400.h
+++ b/linux/drivers/media/video/msp3400.h
@@ -8,7 +8,29 @@ struct msp_dfpreg {
int value;
};
+struct msp_matrix {
+ int input;
+ int output;
+};
+
#define MSP_SET_DFPREG _IOW('m',15,struct msp_dfpreg)
#define MSP_GET_DFPREG _IOW('m',16,struct msp_dfpreg)
+/* ioctl for MSP_SET_MATRIX will have to be registered */
+#define MSP_SET_MATRIX _IOW('m',17,struct msp_matrix)
+
+#define SCART_MASK 0
+#define SCART_IN1 1
+#define SCART_IN2 2
+#define SCART_IN1_DA 3
+#define SCART_IN2_DA 4
+#define SCART_IN3 5
+#define SCART_IN4 6
+#define SCART_MONO 7
+#define SCART_MUTE 8
+
+#define SCART_DSP_IN 0
+#define SCART1_OUT 1
+#define SCART2_OUT 2
+
#endif /* MSP3400_H */