diff options
author | Hans Verkuil <hverkuil@xs4all.nl> | 2007-11-04 15:03:36 +0100 |
---|---|---|
committer | Hans Verkuil <hverkuil@xs4all.nl> | 2007-11-04 15:03:36 +0100 |
commit | c5f2e2eb39ba373d8e9560854ffac028d1f275b0 (patch) | |
tree | 394e67f95a5eb9ff526e876ab6af7f37d4fae1a6 /linux/drivers/media/video/tuner-core.c | |
parent | ba20dface9c85c8aee7d5fb076c4c3cf65cef6ba (diff) | |
download | mediapointer-dvb-s2-c5f2e2eb39ba373d8e9560854ffac028d1f275b0.tar.gz mediapointer-dvb-s2-c5f2e2eb39ba373d8e9560854ffac028d1f275b0.tar.bz2 |
tuner: convert to bus-based I2C API
From: Hans Verkuil <hverkuil@xs4all.nl>
Signed-off-by: Hans Verkuil <hverkuil@xs4all.nl>
Diffstat (limited to 'linux/drivers/media/video/tuner-core.c')
-rw-r--r-- | linux/drivers/media/video/tuner-core.c | 172 |
1 files changed, 42 insertions, 130 deletions
diff --git a/linux/drivers/media/video/tuner-core.c b/linux/drivers/media/video/tuner-core.c index 100eec64f..b1cf04f33 100644 --- a/linux/drivers/media/video/tuner-core.c +++ b/linux/drivers/media/video/tuner-core.c @@ -20,6 +20,7 @@ #include <media/tuner.h> #include <media/tuner-types.h> #include <media/v4l2-common.h> +#include <media/v4l2-i2c-drv-legacy.h> #include "tuner-driver.h" #if LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0) #include "i2c-compat.h" @@ -34,7 +35,11 @@ #define UNSET (-1U) -#define PREFIX "tuner " +#if LINUX_VERSION_CODE <= KERNEL_VERSION(2, 6, 15) +#define PREFIX t->i2c->driver->name +#else +#define PREFIX t->i2c->driver->driver.name +#endif /* standard i2c insmod options */ static unsigned short normal_i2c[] = { @@ -100,9 +105,6 @@ MODULE_DESCRIPTION("device driver for various TV and TV+FM radio tuners"); MODULE_AUTHOR("Ralph Metzler, Gerd Knorr, Gunther Mayer"); MODULE_LICENSE("GPL"); -static struct i2c_driver driver; -static struct i2c_client client_template; - /* ---------------------------------------------------------------------- */ static void fe_set_freq(struct dvb_frontend *fe, unsigned int freq) @@ -957,47 +959,20 @@ static int tuner_command(struct i2c_client *client, unsigned int cmd, void *arg) return 0; } -#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 5, 0) -#if LINUX_VERSION_CODE > KERNEL_VERSION(2, 6, 14) -#if LINUX_VERSION_CODE > KERNEL_VERSION(2, 6, 20) static int tuner_suspend(struct i2c_client *c, pm_message_t state) -#else -static int tuner_suspend(struct device *dev, pm_message_t state) -#endif -#else -#if LINUX_VERSION_CODE > KERNEL_VERSION(2, 6, 13) -static int tuner_suspend(struct device *dev, pm_message_t state, u32 level) -#else -static int tuner_suspend(struct device *dev, u32 state, u32 level) -#endif -#endif { -#if LINUX_VERSION_CODE <= KERNEL_VERSION(2, 6, 20) - struct i2c_client *c = container_of (dev, struct i2c_client, dev); -#endif - struct tuner *t = i2c_get_clientdata (c); + struct tuner *t = i2c_get_clientdata(c); - tuner_dbg ("suspend\n"); + tuner_dbg("suspend\n"); /* FIXME: power down ??? */ return 0; } -#if LINUX_VERSION_CODE > KERNEL_VERSION(2, 6, 14) -#if LINUX_VERSION_CODE > KERNEL_VERSION(2, 6, 20) static int tuner_resume(struct i2c_client *c) -#else -static int tuner_resume(struct device *dev) -#endif -#else -static int tuner_resume(struct device *dev, u32 level) -#endif { -#if LINUX_VERSION_CODE <= KERNEL_VERSION(2, 6, 20) - struct i2c_client *c = container_of (dev, struct i2c_client, dev); -#endif - struct tuner *t = i2c_get_clientdata (c); + struct tuner *t = i2c_get_clientdata(c); - tuner_dbg ("resume\n"); + tuner_dbg("resume\n"); if (V4L2_TUNER_RADIO == t->mode) { if (t->radio_freq) set_freq(c, t->radio_freq); @@ -1007,14 +982,13 @@ static int tuner_resume(struct device *dev, u32 level) } return 0; } -#endif /* ---------------------------------------------------------------------- */ LIST_HEAD(tuner_list); /* Search for existing radio and/or TV tuners on the given I2C adapter. - Note that when this function is called from tuner_attach you can be + Note that when this function is called from tuner_probe you can be certain no other devices will be added/deleted at the same time, I2C core protects against that. */ static void tuner_lookup(struct i2c_adapter *adap, @@ -1045,33 +1019,19 @@ static void tuner_lookup(struct i2c_adapter *adap, } /* During client attach, set_type is called by adapter's attach_inform callback. - set_type must then be completed by tuner_attach. + set_type must then be completed by tuner_probe. */ -#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 5, 0) -static int tuner_attach(struct i2c_adapter *adap, int addr, int kind) -#else -static int tuner_attach(struct i2c_adapter *adap, int addr, - unsigned short flags, int kind) -#endif +static int tuner_probe(struct i2c_client *client) { - struct i2c_client *client; struct tuner *t; struct tuner *radio; struct tuner *tv; - client = kmalloc(sizeof(struct i2c_client), GFP_KERNEL); - if (NULL == client) - return -ENOMEM; - t = kzalloc(sizeof(struct tuner), GFP_KERNEL); - if (NULL == t) { - kfree(client); + if (NULL == t) return -ENOMEM; - } t->i2c = client; - client_template.adapter = adap; - client_template.addr = addr; - memcpy(client, &client_template, sizeof(struct i2c_client)); + strlcpy(client->name, "(tuner unset)", sizeof(client->name)); i2c_set_clientdata(client, t); t->type = UNSET; t->audmode = V4L2_TUNER_MODE_STEREO; @@ -1088,14 +1048,16 @@ static int tuner_attach(struct i2c_adapter *adap, int addr, printk(KERN_CONT "%02x ", buffer[i]); printk("\n"); } - /* HACK: This test were added to avoid tuner to probe tda9840 and + /* HACK: This test was added to avoid tuner to probe tda9840 and tea6415c on the MXB card */ - if (adap->id == I2C_HW_SAA7146 && addr < 0x4a) + if (client->adapter->id == I2C_HW_SAA7146 && client->addr < 0x4a) { + kfree(t); return -ENODEV; + } /* autodetection code based on the i2c addr */ if (!no_autodetect) { - switch (addr) { + switch (client->addr) { case 0x10: if (tea5761_autodetection(t->i2c->adapter, t->i2c->addr) != EINVAL) { @@ -1165,7 +1127,8 @@ static int tuner_attach(struct i2c_adapter *adap, int addr, /* Should be just before return */ register_client: - tuner_info("chip found @ 0x%x (%s)\n", addr << 1, adap->name); + tuner_info("chip found @ 0x%x (%s)\n", client->addr << 1, + client->adapter->name); /* Sets a default mode */ if (t->mode_mask & T_ANALOG_TV) { @@ -1175,22 +1138,25 @@ register_client: } else { t->mode = T_DIGITAL_TV; } - - i2c_attach_client(client); set_type(client, t->type, t->mode_mask, t->config, t->tuner_callback); -#if LINUX_VERSION_CODE < KERNEL_VERSION(2, 5, 0) -MOD_INC_USE_COUNT; -#endif + list_add_tail(&t->list, &tuner_list); return 0; } -static int tuner_probe(struct i2c_adapter *adap) +static int tuner_legacy_probe(struct i2c_adapter *adap) { if (0 != addr) { normal_i2c[0] = addr; normal_i2c[1] = I2C_CLIENT_END; } +#ifdef I2C_CLASS_TV_ANALOG + if ((adap->class & I2C_CLASS_TV_ANALOG) == 0) +#else + if (adap->id != I2C_HW_B_BT848) +#endif + return 0; + /* HACK: Ignore 0x6b and 0x6f on cx88 boards. * FusionHDTV5 RT Gold has an ir receiver at 0x6b * and an RTC at 0x6f which can get corrupted if probed. @@ -1212,92 +1178,38 @@ static int tuner_probe(struct i2c_adapter *adap) "too many options specified " "in i2c probe ignore list!\n"); } - -#ifdef I2C_CLASS_TV_ANALOG - if (adap->class & I2C_CLASS_TV_ANALOG) - return i2c_probe(adap, &addr_data, tuner_attach); -#else - if (adap->id == I2C_HW_B_BT848) - return i2c_probe(adap, &addr_data, tuner_attach); -#endif - return 0; + return 1; } -static int tuner_detach(struct i2c_client *client) +static int tuner_remove(struct i2c_client *client) { struct tuner *t = i2c_get_clientdata(client); struct analog_tuner_ops *ops = t->fe.ops.analog_demod_ops; - int err; - err = i2c_detach_client(t->i2c); - if (err) { - tuner_warn - ("Client deregistration failed, client not detached.\n"); - return err; - } - -#if LINUX_VERSION_CODE < KERNEL_VERSION(2, 5, 0) -MOD_DEC_USE_COUNT; -#endif if (ops && ops->release) ops->release(&t->fe); list_del(&t->list); kfree(t); - kfree(client); return 0; } /* ----------------------------------------------------------------------- */ -static struct i2c_driver driver = { -#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2,6,0))&&(LINUX_VERSION_CODE <= KERNEL_VERSION(2,6,15)) - .owner = THIS_MODULE, -#endif -#if LINUX_VERSION_CODE <= KERNEL_VERSION(2,6,15) +static struct v4l2_i2c_driver_data v4l2_i2c_data = { .name = "tuner", - .flags = I2C_DF_NOTIFY, -#endif - .id = I2C_DRIVERID_TUNER, - .attach_adapter = tuner_probe, - .detach_client = tuner_detach, + .driverid = I2C_DRIVERID_TUNER, .command = tuner_command, -#if LINUX_VERSION_CODE > KERNEL_VERSION(2,6,20) + .probe = tuner_probe, + .remove = tuner_remove, .suspend = tuner_suspend, - .resume = tuner_resume, -#endif -#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0) - .driver = { -#if LINUX_VERSION_CODE > KERNEL_VERSION(2,6,15) - .name = "tuner", -#endif -#if LINUX_VERSION_CODE <= KERNEL_VERSION(2,6,20) - .suspend = tuner_suspend, - .resume = tuner_resume, -#endif - }, -#endif + .resume = tuner_resume, + .legacy_probe = tuner_legacy_probe, }; -static struct i2c_client client_template = { - .name = "(tuner unset)", -#if LINUX_VERSION_CODE <= KERNEL_VERSION(2,6,15) - .flags = I2C_CLIENT_ALLOW_USE, -#endif - .driver = &driver, -}; - -static int __init tuner_init_module(void) -{ - return i2c_add_driver(&driver); -} -static void __exit tuner_cleanup_module(void) -{ - i2c_del_driver(&driver); -} - -module_init(tuner_init_module); -module_exit(tuner_cleanup_module); +#if LINUX_VERSION_CODE < KERNEL_VERSION(2, 5, 0) +EXPORT_NO_SYMBOLS; +#endif /* * Overrides for Emacs so that we follow Linus's tabbing style. |