mirror of https://gitee.com/openkylin/linux.git
V4L/DVB (4932): Usbvision_v4l2: fix norm setting problems
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>
This commit is contained in:
parent
ee5407c501
commit
957883d0b5
|
@ -2570,58 +2570,61 @@ static int usbvision_unrequest_intra (struct usb_usbvision *usbvision)
|
||||||
static void call_i2c_clients(struct usb_usbvision *usbvision, unsigned int cmd,
|
static void call_i2c_clients(struct usb_usbvision *usbvision, unsigned int cmd,
|
||||||
void *arg)
|
void *arg)
|
||||||
{
|
{
|
||||||
|
BUG_ON(NULL == usbvision->i2c_adap.algo_data);
|
||||||
int i;
|
i2c_clients_command(&usbvision->i2c_adap, cmd, arg);
|
||||||
|
|
||||||
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);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int attach_inform(struct i2c_client *client)
|
static int attach_inform(struct i2c_client *client)
|
||||||
{
|
{
|
||||||
struct usb_usbvision *usbvision;
|
struct usb_usbvision *usbvision;
|
||||||
struct tuner_setup tun_addr;
|
|
||||||
int i;
|
|
||||||
|
|
||||||
#if LINUX_VERSION_CODE < KERNEL_VERSION(2,6,0)
|
usbvision = (struct usb_usbvision *)i2c_get_adapdata(client->adapter);
|
||||||
usbvision = (struct usb_usbvision *)client->adapter->data;
|
|
||||||
#else
|
|
||||||
usbvision = (struct usb_usbvision *)i2c_get_adapdata(client->adapter);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
for (i = 0; i < USBVISION_I2C_CLIENTS_MAX; i++) {
|
switch (client->addr << 1) {
|
||||||
if (usbvision->i2c_clients[i] == NULL ||
|
case 0x43:
|
||||||
usbvision->i2c_clients[i]->driver->id ==
|
case 0x4b:
|
||||||
client->driver->id) {
|
{
|
||||||
usbvision->i2c_clients[i] = client;
|
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;
|
break;
|
||||||
}
|
}
|
||||||
}
|
case 0x42:
|
||||||
if ((usbvision->have_tuner) && (usbvision->tuner_type != -1)) {
|
PDEBUG(DBG_I2C,"attach_inform: saa7114 detected.\n");
|
||||||
tun_addr.mode_mask = T_ANALOG_TV;
|
break;
|
||||||
tun_addr.type = usbvision->tuner_type;
|
case 0x4a:
|
||||||
tun_addr.addr = ADDR_UNSET;
|
PDEBUG(DBG_I2C,"attach_inform: saa7113 detected.\n");
|
||||||
client->driver->command(client,TUNER_SET_TYPE_ADDR, &tun_addr);
|
break;
|
||||||
call_i2c_clients(usbvision, VIDIOC_INT_RESET, NULL);
|
case 0xa0:
|
||||||
call_i2c_clients(usbvision, VIDIOC_S_INPUT, &usbvision->ctl_input);
|
PDEBUG(DBG_I2C,"attach_inform: eeprom detected.\n");
|
||||||
call_i2c_clients(usbvision, VIDIOC_STREAMON, NULL);
|
break;
|
||||||
}
|
|
||||||
call_i2c_clients(usbvision, VIDIOC_S_STD, &usbvision->tvnorm->id);
|
|
||||||
|
|
||||||
PDEBUG(DBG_I2C, "usbvision[%d] attaches %s", usbvision->nr, client->name);
|
default:
|
||||||
|
PDEBUG(DBG_I2C,"attach inform: detected I2C address %x\n", client->addr << 1);
|
||||||
|
{
|
||||||
|
struct tuner_setup tun_setup;
|
||||||
|
|
||||||
|
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;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int detach_inform(struct i2c_client *client)
|
static int detach_inform(struct i2c_client *client)
|
||||||
{
|
{
|
||||||
struct usb_usbvision *usbvision;
|
struct usb_usbvision *usbvision;
|
||||||
int i;
|
|
||||||
|
|
||||||
#if LINUX_VERSION_CODE < KERNEL_VERSION(2,6,0)
|
#if LINUX_VERSION_CODE < KERNEL_VERSION(2,6,0)
|
||||||
usbvision = (struct usb_usbvision *)client->adapter->data;
|
usbvision = (struct usb_usbvision *)client->adapter->data;
|
||||||
|
@ -2630,14 +2633,6 @@ static int detach_inform(struct i2c_client *client)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
PDEBUG(DBG_I2C, "usbvision[%d] detaches %s", usbvision->nr, client->name);
|
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;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2886,10 +2881,7 @@ static int usbvision_init_i2c(struct usb_usbvision *usbvision)
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
usbvision->i2c_ok = usbvision_i2c_usb_add_bus(&usbvision->i2c_adap);
|
return usbvision_i2c_usb_add_bus(&usbvision->i2c_adap);
|
||||||
|
|
||||||
return usbvision->i2c_ok;
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -3811,6 +3803,8 @@ static int usbvision_v4l2_open(struct inode *inode, struct file *file)
|
||||||
if (!errCode) {
|
if (!errCode) {
|
||||||
usbvision_begin_streaming(usbvision);
|
usbvision_begin_streaming(usbvision);
|
||||||
errCode = usbvision_init_isoc(usbvision);
|
errCode = usbvision_init_isoc(usbvision);
|
||||||
|
/* device needs to be initialized before isoc transfer */
|
||||||
|
usbvision_muxsel(usbvision,0);
|
||||||
usbvision->user++;
|
usbvision->user++;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
@ -4169,7 +4163,7 @@ static int usbvision_v4l2_do_ioctl(struct inode *inode, struct file *file,
|
||||||
memset(ctrl,0,sizeof(*ctrl));
|
memset(ctrl,0,sizeof(*ctrl));
|
||||||
ctrl->id=id;
|
ctrl->id=id;
|
||||||
|
|
||||||
i2c_clients_command(&usbvision->i2c_adap, cmd, arg);
|
call_i2c_clients(usbvision, cmd, arg);
|
||||||
|
|
||||||
if (ctrl->type)
|
if (ctrl->type)
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -5300,19 +5294,15 @@ static void usbvision_configure_video(struct usb_usbvision *usbvision)
|
||||||
(usbvision->have_tuner ? V4L2_CAP_TUNER : 0);
|
(usbvision->have_tuner ? V4L2_CAP_TUNER : 0);
|
||||||
usbvision->vcap.version = USBVISION_DRIVER_VERSION; /* version */
|
usbvision->vcap.version = USBVISION_DRIVER_VERSION; /* version */
|
||||||
|
|
||||||
|
|
||||||
for (i = 0; i < TVNORMS; i++)
|
for (i = 0; i < TVNORMS; i++)
|
||||||
if (usbvision_device_data[model].VideoNorm == tvnorms[i].mode)
|
if (usbvision_device_data[model].VideoNorm == tvnorms[i].mode)
|
||||||
break;
|
break;
|
||||||
if (i == TVNORMS)
|
if (i == TVNORMS)
|
||||||
i = 0;
|
i = 0;
|
||||||
usbvision->tvnorm = &tvnorms[i]; /* set default norm */
|
usbvision->tvnorm = &tvnorms[i]; /* set default norm */
|
||||||
call_i2c_clients(usbvision, VIDIOC_S_STD,
|
|
||||||
&usbvision->tvnorm->id);
|
|
||||||
|
|
||||||
usbvision->video_inputs = usbvision_device_data[model].VideoChannels;
|
usbvision->video_inputs = usbvision_device_data[model].VideoChannels;
|
||||||
usbvision->ctl_input = 0;
|
usbvision->ctl_input = 0;
|
||||||
/* usbvision_muxsel(usbvision, usbvision->ctl_input); */
|
|
||||||
|
|
||||||
/* This should be here to make i2c clients to be able to register */
|
/* This should be here to make i2c clients to be able to register */
|
||||||
usbvision_audio_off(usbvision); //first switch off audio
|
usbvision_audio_off(usbvision); //first switch off audio
|
||||||
|
@ -5678,6 +5668,8 @@ static int __devinit usbvision_probe(struct usb_interface *intf, const struct us
|
||||||
usbvision->tuner_type = usbvision_device_data[model].TunerType;
|
usbvision->tuner_type = usbvision_device_data[model].TunerType;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
usbvision->tuner_addr = ADDR_UNSET;
|
||||||
|
|
||||||
usbvision->DevModel = model;
|
usbvision->DevModel = model;
|
||||||
usbvision->remove_pending = 0;
|
usbvision->remove_pending = 0;
|
||||||
usbvision->last_error = 0;
|
usbvision->last_error = 0;
|
||||||
|
@ -5689,7 +5681,6 @@ static int __devinit usbvision_probe(struct usb_interface *intf, const struct us
|
||||||
usbvision->usb_bandwidth = 0;
|
usbvision->usb_bandwidth = 0;
|
||||||
usbvision->user = 0;
|
usbvision->user = 0;
|
||||||
usbvision->streaming = Stream_Off;
|
usbvision->streaming = Stream_Off;
|
||||||
|
|
||||||
usbvision_register_video(usbvision);
|
usbvision_register_video(usbvision);
|
||||||
usbvision_configure_video(usbvision);
|
usbvision_configure_video(usbvision);
|
||||||
up(&usbvision->lock);
|
up(&usbvision->lock);
|
||||||
|
|
|
@ -319,8 +319,6 @@ struct usbvision_frame {
|
||||||
#define BRIDGE_NT1004 1004
|
#define BRIDGE_NT1004 1004
|
||||||
#define BRIDGE_NT1005 1005
|
#define BRIDGE_NT1005 1005
|
||||||
|
|
||||||
#define USBVISION_I2C_CLIENTS_MAX 8
|
|
||||||
|
|
||||||
struct usbvision_device_data_st {
|
struct usbvision_device_data_st {
|
||||||
int idVendor;
|
int idVendor;
|
||||||
int idProduct;
|
int idProduct;
|
||||||
|
@ -355,8 +353,6 @@ struct usb_usbvision {
|
||||||
struct i2c_adapter i2c_adap;
|
struct i2c_adapter i2c_adap;
|
||||||
struct i2c_algo_usb_data i2c_algo;
|
struct i2c_algo_usb_data i2c_algo;
|
||||||
struct i2c_client i2c_client;
|
struct i2c_client i2c_client;
|
||||||
int i2c_state, i2c_ok;
|
|
||||||
struct i2c_client *i2c_clients[USBVISION_I2C_CLIENTS_MAX];
|
|
||||||
|
|
||||||
struct urb *ctrlUrb;
|
struct urb *ctrlUrb;
|
||||||
unsigned char ctrlUrbBuffer[8];
|
unsigned char ctrlUrbBuffer[8];
|
||||||
|
@ -367,6 +363,7 @@ struct usb_usbvision {
|
||||||
|
|
||||||
int have_tuner;
|
int have_tuner;
|
||||||
int tuner_type;
|
int tuner_type;
|
||||||
|
int tuner_addr;
|
||||||
int bridgeType; // NT1003, NT1004, NT1005
|
int bridgeType; // NT1003, NT1004, NT1005
|
||||||
int channel;
|
int channel;
|
||||||
int radio;
|
int radio;
|
||||||
|
|
Loading…
Reference in New Issue