summaryrefslogtreecommitdiff
path: root/linux/drivers/media/video/usbvision
diff options
context:
space:
mode:
authorMauro Carvalho Chehab <mchehab@infradead.org>2006-12-04 09:31:30 -0200
committerMauro Carvalho Chehab <mchehab@infradead.org>2006-12-04 09:31:30 -0200
commit6ebb2ed15804a657b118f51ed650e39297420517 (patch)
tree3330a26dfc73cd45e223624daf0eadf01a0659dd /linux/drivers/media/video/usbvision
parent27310b31bc544835fb64b181c5cc18e20896225f (diff)
downloadmediapointer-dvb-s2-6ebb2ed15804a657b118f51ed650e39297420517.tar.gz
mediapointer-dvb-s2-6ebb2ed15804a657b118f51ed650e39297420517.tar.bz2
Usbvision_v4l2: fix norm setting problems
From: Thierry MERLE <thierry.merle@free.fr> Patch contents: - fix i2c command broadcast (caused problems for SECAM norm setting) - default input selection at driver open Signed-off-by: Thierry MERLE <thierry.merle@free.fr> Signed-off-by: Mauro Carvalho Chehab <mchehab@infradead.org>
Diffstat (limited to 'linux/drivers/media/video/usbvision')
-rw-r--r--linux/drivers/media/video/usbvision/usbvision-core.c112
-rw-r--r--linux/drivers/media/video/usbvision/usbvision.h5
2 files changed, 45 insertions, 72 deletions
diff --git a/linux/drivers/media/video/usbvision/usbvision-core.c b/linux/drivers/media/video/usbvision/usbvision-core.c
index 14f884756..be6830808 100644
--- a/linux/drivers/media/video/usbvision/usbvision-core.c
+++ b/linux/drivers/media/video/usbvision/usbvision-core.c
@@ -2699,73 +2699,61 @@ static int usbvision_unrequest_intra (struct usb_usbvision *usbvision)
static void call_i2c_clients(struct usb_usbvision *usbvision, unsigned int cmd,
void *arg)
{
-#if 0 // new version from bttv-if.c
- if (0 != btv->i2c_rc)
- return;
- i2c_clients_command(&btv->i2c_adap, cmd, arg);
-#endif
-
- int i;
-
- for (i = 0; i < USBVISION_I2C_CLIENTS_MAX; i++) {
- if (NULL == usbvision->i2c_clients[i])
- continue;
- if (NULL == usbvision->i2c_clients[i]->driver->command)
- continue;
- usbvision->i2c_clients[i]->driver->command(usbvision->i2c_clients[i], cmd, arg);
- }
+ BUG_ON(NULL == usbvision->i2c_adap.algo_data);
+ i2c_clients_command(&usbvision->i2c_adap, cmd, arg);
}
static int attach_inform(struct i2c_client *client)
{
struct usb_usbvision *usbvision;
-#if LINUX_VERSION_CODE > KERNEL_VERSION(2,6,12)
- struct tuner_setup tun_addr;
-#endif
- int i;
- #if LINUX_VERSION_CODE < KERNEL_VERSION(2,6,0)
- usbvision = (struct usb_usbvision *)client->adapter->data;
- #else
- usbvision = (struct usb_usbvision *)i2c_get_adapdata(client->adapter);
- #endif
+ usbvision = (struct usb_usbvision *)i2c_get_adapdata(client->adapter);
- for (i = 0; i < USBVISION_I2C_CLIENTS_MAX; i++) {
- if (usbvision->i2c_clients[i] == NULL ||
- usbvision->i2c_clients[i]->driver->id ==
- client->driver->id) {
- usbvision->i2c_clients[i] = client;
+ switch (client->addr << 1) {
+ case 0x43:
+ case 0x4b:
+ {
+ struct tuner_setup tun_setup;
+
+ tun_setup.mode_mask = T_ANALOG_TV | T_RADIO;
+ tun_setup.type = TUNER_TDA9887;
+ tun_setup.addr = client->addr;
+
+ call_i2c_clients(usbvision, TUNER_SET_TYPE_ADDR, &tun_setup);
break;
}
- }
-#if LINUX_VERSION_CODE > KERNEL_VERSION(2,6,11)
- if ((usbvision->have_tuner) && (usbvision->tuner_type != -1)) {
- tun_addr.mode_mask = T_ANALOG_TV;
- tun_addr.type = usbvision->tuner_type;
- tun_addr.addr = ADDR_UNSET;
- client->driver->command(client,TUNER_SET_TYPE_ADDR, &tun_addr);
- call_i2c_clients(usbvision, VIDIOC_INT_RESET, NULL);
- call_i2c_clients(usbvision, VIDIOC_S_INPUT, &usbvision->ctl_input);
- call_i2c_clients(usbvision, VIDIOC_STREAMON, NULL);
- }
-#else
- if ((usbvision->have_tuner) && (usbvision->tuner_type != -1)) {
+ case 0x42:
+ PDEBUG(DBG_I2C,"attach_inform: saa7114 detected.\n");
+ break;
+ case 0x4a:
+ PDEBUG(DBG_I2C,"attach_inform: saa7113 detected.\n");
+ break;
+ case 0xa0:
+ PDEBUG(DBG_I2C,"attach_inform: eeprom detected.\n");
+ break;
- call_i2c_clients(usbvision, TUNER_SET_TYPE, &usbvision->tuner_type);
- call_i2c_clients(usbvision, VIDIOC_S_INPUT, &usbvision->ctl_input);
- }
-#endif
- call_i2c_clients(usbvision, VIDIOC_S_STD, &usbvision->tvnorm->id);
+ default:
+ PDEBUG(DBG_I2C,"attach inform: detected I2C address %x\n", client->addr << 1);
+ {
+ struct tuner_setup tun_setup;
- PDEBUG(DBG_I2C, "usbvision[%d] attaches %s", usbvision->nr, client->name);
+ usbvision->tuner_addr = client->addr;
+ if ((usbvision->have_tuner) && (usbvision->tuner_type != -1)) {
+ tun_setup.mode_mask = T_ANALOG_TV | T_RADIO;
+ tun_setup.type = usbvision->tuner_type;
+ tun_setup.addr = usbvision->tuner_addr;
+ call_i2c_clients(usbvision, TUNER_SET_TYPE_ADDR, &tun_setup);
+ }
+ }
+ break;
+ }
return 0;
}
static int detach_inform(struct i2c_client *client)
{
struct usb_usbvision *usbvision;
- int i;
#if LINUX_VERSION_CODE < KERNEL_VERSION(2,6,0)
usbvision = (struct usb_usbvision *)client->adapter->data;
@@ -2774,14 +2762,6 @@ static int detach_inform(struct i2c_client *client)
#endif
PDEBUG(DBG_I2C, "usbvision[%d] detaches %s", usbvision->nr, client->name);
- for (i = 0; i < USBVISION_I2C_CLIENTS_MAX; i++) {
- if (NULL != usbvision->i2c_clients[i] &&
- usbvision->i2c_clients[i]->driver->id ==
- client->driver->id) {
- usbvision->i2c_clients[i] = NULL;
- break;
- }
- }
return 0;
}
@@ -3038,10 +3018,7 @@ static int usbvision_init_i2c(struct usb_usbvision *usbvision)
}
#endif
- usbvision->i2c_ok = usbvision_i2c_usb_add_bus(&usbvision->i2c_adap);
-
- return usbvision->i2c_ok;
-
+ return usbvision_i2c_usb_add_bus(&usbvision->i2c_adap);
}
@@ -4046,6 +4023,8 @@ static int usbvision_v4l2_open(struct inode *inode, struct file *file)
if (!errCode) {
usbvision_begin_streaming(usbvision);
errCode = usbvision_init_isoc(usbvision);
+ /* device needs to be initialized before isoc transfer */
+ usbvision_muxsel(usbvision,0);
usbvision->user++;
}
else {
@@ -4424,7 +4403,7 @@ static int usbvision_v4l2_do_ioctl(struct inode *inode, struct file *file,
memset(ctrl,0,sizeof(*ctrl));
ctrl->id=id;
- i2c_clients_command(&usbvision->i2c_adap, cmd, arg);
+ call_i2c_clients(usbvision, cmd, arg);
if (ctrl->type)
return 0;
@@ -5622,19 +5601,15 @@ static void usbvision_configure_video(struct usb_usbvision *usbvision)
(usbvision->have_tuner ? V4L2_CAP_TUNER : 0);
usbvision->vcap.version = USBVISION_DRIVER_VERSION; /* version */
-
for (i = 0; i < TVNORMS; i++)
if (usbvision_device_data[model].VideoNorm == tvnorms[i].mode)
break;
if (i == TVNORMS)
i = 0;
- usbvision->tvnorm = &tvnorms[i]; /* set default norm */
- call_i2c_clients(usbvision, VIDIOC_S_STD,
- &usbvision->tvnorm->id);
+ usbvision->tvnorm = &tvnorms[i]; /* set default norm */
usbvision->video_inputs = usbvision_device_data[model].VideoChannels;
usbvision->ctl_input = 0;
-/* usbvision_muxsel(usbvision, usbvision->ctl_input); */
/* This should be here to make i2c clients to be able to register */
usbvision_audio_off(usbvision); //first switch off audio
@@ -6106,6 +6081,8 @@ static int __devinit usbvision_probe(struct usb_interface *intf, const struct us
usbvision->tuner_type = usbvision_device_data[model].TunerType;
}
+ usbvision->tuner_addr = ADDR_UNSET;
+
usbvision->DevModel = model;
usbvision->remove_pending = 0;
usbvision->last_error = 0;
@@ -6117,7 +6094,6 @@ static int __devinit usbvision_probe(struct usb_interface *intf, const struct us
usbvision->usb_bandwidth = 0;
usbvision->user = 0;
usbvision->streaming = Stream_Off;
-
usbvision_register_video(usbvision);
usbvision_configure_video(usbvision);
up(&usbvision->lock);
diff --git a/linux/drivers/media/video/usbvision/usbvision.h b/linux/drivers/media/video/usbvision/usbvision.h
index 3ef18a4ed..fdce9be37 100644
--- a/linux/drivers/media/video/usbvision/usbvision.h
+++ b/linux/drivers/media/video/usbvision/usbvision.h
@@ -357,8 +357,6 @@ struct usbvision_frame {
#define BRIDGE_NT1004 1004
#define BRIDGE_NT1005 1005
-#define USBVISION_I2C_CLIENTS_MAX 8
-
struct usbvision_device_data_st {
int idVendor;
int idProduct;
@@ -393,8 +391,6 @@ struct usb_usbvision {
struct i2c_adapter i2c_adap;
struct i2c_algo_usb_data i2c_algo;
struct i2c_client i2c_client;
- int i2c_state, i2c_ok;
- struct i2c_client *i2c_clients[USBVISION_I2C_CLIENTS_MAX];
struct urb *ctrlUrb;
unsigned char ctrlUrbBuffer[8];
@@ -409,6 +405,7 @@ struct usb_usbvision {
int have_tuner;
int tuner_type;
+ int tuner_addr;
int bridgeType; // NT1003, NT1004, NT1005
int channel;
int radio;