diff options
author | Hans Verkuil <hverkuil@xs4all.nl> | 2009-02-07 09:18:05 -0500 |
---|---|---|
committer | Mauro Carvalho Chehab <mchehab@redhat.com> | 2009-03-30 11:42:42 -0400 |
commit | 1b8dac150a01e2312d8e3fedd6462a0ec34c96d0 (patch) | |
tree | ba8fd678a4afb4bd72960e1737f857991fd00a19 /drivers/media/video/saa5249.c | |
parent | d30e21ddcdc948ecedfb46a0ed021d57f310a6f3 (diff) |
V4L/DVB (10499): saa7146: convert saa7146 and mxb in particular to v4l2_subdev.
Modified mxb to load the i2c modules through v4l2_subdev. So no more probing.
Modified tea6415c and tea6420 to use the standard routing ops to do the
routing, rather than using private commands. Dropped the private commands
from tda9840 (they were never used except during initialization of the
module).
Added saa7146 support for VIDIOC_DBG_G_CHIP_IDENT.
Converted saa5246a and saa5249 to v4l2_subdev.
Signed-off-by: Hans Verkuil <hverkuil@xs4all.nl>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
Diffstat (limited to 'drivers/media/video/saa5249.c')
-rw-r--r-- | drivers/media/video/saa5249.c | 70 |
1 files changed, 45 insertions, 25 deletions
diff --git a/drivers/media/video/saa5249.c b/drivers/media/video/saa5249.c index 81e666df0db7..48b27fe48087 100644 --- a/drivers/media/video/saa5249.c +++ b/drivers/media/video/saa5249.c | |||
@@ -50,15 +50,17 @@ | |||
50 | #include <linux/mutex.h> | 50 | #include <linux/mutex.h> |
51 | #include <linux/delay.h> | 51 | #include <linux/delay.h> |
52 | #include <linux/videotext.h> | 52 | #include <linux/videotext.h> |
53 | #include <linux/videodev.h> | 53 | #include <linux/videodev2.h> |
54 | #include <media/v4l2-common.h> | 54 | #include <media/v4l2-device.h> |
55 | #include <media/v4l2-chip-ident.h> | ||
55 | #include <media/v4l2-ioctl.h> | 56 | #include <media/v4l2-ioctl.h> |
56 | #include <media/v4l2-i2c-drv-legacy.h> | 57 | #include <media/v4l2-i2c-drv.h> |
57 | 58 | ||
58 | MODULE_AUTHOR("Michael Geng <linux@MichaelGeng.de>"); | 59 | MODULE_AUTHOR("Michael Geng <linux@MichaelGeng.de>"); |
59 | MODULE_DESCRIPTION("Philips SAA5249 Teletext decoder driver"); | 60 | MODULE_DESCRIPTION("Philips SAA5249 Teletext decoder driver"); |
60 | MODULE_LICENSE("GPL"); | 61 | MODULE_LICENSE("GPL"); |
61 | 62 | ||
63 | |||
62 | #define VTX_VER_MAJ 1 | 64 | #define VTX_VER_MAJ 1 |
63 | #define VTX_VER_MIN 8 | 65 | #define VTX_VER_MIN 8 |
64 | 66 | ||
@@ -95,17 +97,23 @@ typedef struct { | |||
95 | 97 | ||
96 | struct saa5249_device | 98 | struct saa5249_device |
97 | { | 99 | { |
100 | struct v4l2_subdev sd; | ||
101 | struct video_device *vdev; | ||
98 | vdau_t vdau[NUM_DAUS]; /* Data for virtual DAUs (the 5249 only has one */ | 102 | vdau_t vdau[NUM_DAUS]; /* Data for virtual DAUs (the 5249 only has one */ |
99 | /* real DAU, so we have to simulate some more) */ | 103 | /* real DAU, so we have to simulate some more) */ |
100 | int vtx_use_count; | 104 | int vtx_use_count; |
101 | int is_searching[NUM_DAUS]; | 105 | int is_searching[NUM_DAUS]; |
102 | int disp_mode; | 106 | int disp_mode; |
103 | int virtual_mode; | 107 | int virtual_mode; |
104 | struct i2c_client *client; | ||
105 | unsigned long in_use; | 108 | unsigned long in_use; |
106 | struct mutex lock; | 109 | struct mutex lock; |
107 | }; | 110 | }; |
108 | 111 | ||
112 | static inline struct saa5249_device *to_dev(struct v4l2_subdev *sd) | ||
113 | { | ||
114 | return container_of(sd, struct saa5249_device, sd); | ||
115 | } | ||
116 | |||
109 | 117 | ||
110 | #define CCTWR 34 /* I²C write/read-address of vtx-chip */ | 118 | #define CCTWR 34 /* I²C write/read-address of vtx-chip */ |
111 | #define CCTRD 35 | 119 | #define CCTRD 35 |
@@ -147,12 +155,13 @@ static void jdelay(unsigned long delay) | |||
147 | 155 | ||
148 | static int i2c_sendbuf(struct saa5249_device *t, int reg, int count, u8 *data) | 156 | static int i2c_sendbuf(struct saa5249_device *t, int reg, int count, u8 *data) |
149 | { | 157 | { |
158 | struct i2c_client *client = v4l2_get_subdevdata(&t->sd); | ||
150 | char buf[64]; | 159 | char buf[64]; |
151 | 160 | ||
152 | buf[0] = reg; | 161 | buf[0] = reg; |
153 | memcpy(buf+1, data, count); | 162 | memcpy(buf+1, data, count); |
154 | 163 | ||
155 | if (i2c_master_send(t->client, buf, count + 1) == count + 1) | 164 | if (i2c_master_send(client, buf, count + 1) == count + 1) |
156 | return 0; | 165 | return 0; |
157 | return -1; | 166 | return -1; |
158 | } | 167 | } |
@@ -180,7 +189,9 @@ static int i2c_senddata(struct saa5249_device *t, ...) | |||
180 | 189 | ||
181 | static int i2c_getdata(struct saa5249_device *t, int count, u8 *buf) | 190 | static int i2c_getdata(struct saa5249_device *t, int count, u8 *buf) |
182 | { | 191 | { |
183 | if(i2c_master_recv(t->client, buf, count)!=count) | 192 | struct i2c_client *client = v4l2_get_subdevdata(&t->sd); |
193 | |||
194 | if (i2c_master_recv(client, buf, count) != count) | ||
184 | return -1; | 195 | return -1; |
185 | return 0; | 196 | return 0; |
186 | } | 197 | } |
@@ -497,9 +508,6 @@ static int saa5249_open(struct file *file) | |||
497 | struct saa5249_device *t = video_drvdata(file); | 508 | struct saa5249_device *t = video_drvdata(file); |
498 | int pgbuf; | 509 | int pgbuf; |
499 | 510 | ||
500 | if (t->client == NULL) | ||
501 | return -ENODEV; | ||
502 | |||
503 | if (test_and_set_bit(0, &t->in_use)) | 511 | if (test_and_set_bit(0, &t->in_use)) |
504 | return -EBUSY; | 512 | return -EBUSY; |
505 | 513 | ||
@@ -553,18 +561,28 @@ static struct video_device saa_template = | |||
553 | .release = video_device_release, | 561 | .release = video_device_release, |
554 | }; | 562 | }; |
555 | 563 | ||
556 | /* Addresses to scan */ | 564 | static int saa5249_g_chip_ident(struct v4l2_subdev *sd, struct v4l2_dbg_chip_ident *chip) |
557 | static unsigned short normal_i2c[] = { 0x22 >> 1, I2C_CLIENT_END }; | 565 | { |
566 | struct i2c_client *client = v4l2_get_subdevdata(sd); | ||
567 | |||
568 | return v4l2_chip_ident_i2c_client(client, chip, V4L2_IDENT_SAA5249, 0); | ||
569 | } | ||
558 | 570 | ||
559 | I2C_CLIENT_INSMOD; | 571 | static const struct v4l2_subdev_core_ops saa5249_core_ops = { |
572 | .g_chip_ident = saa5249_g_chip_ident, | ||
573 | }; | ||
574 | |||
575 | static const struct v4l2_subdev_ops saa5249_ops = { | ||
576 | .core = &saa5249_core_ops, | ||
577 | }; | ||
560 | 578 | ||
561 | static int saa5249_probe(struct i2c_client *client, | 579 | static int saa5249_probe(struct i2c_client *client, |
562 | const struct i2c_device_id *id) | 580 | const struct i2c_device_id *id) |
563 | { | 581 | { |
564 | int pgbuf; | 582 | int pgbuf; |
565 | int err; | 583 | int err; |
566 | struct video_device *vd; | ||
567 | struct saa5249_device *t; | 584 | struct saa5249_device *t; |
585 | struct v4l2_subdev *sd; | ||
568 | 586 | ||
569 | v4l_info(client, "chip found @ 0x%x (%s)\n", | 587 | v4l_info(client, "chip found @ 0x%x (%s)\n", |
570 | client->addr << 1, client->adapter->name); | 588 | client->addr << 1, client->adapter->name); |
@@ -573,16 +591,17 @@ static int saa5249_probe(struct i2c_client *client, | |||
573 | t = kzalloc(sizeof(*t), GFP_KERNEL); | 591 | t = kzalloc(sizeof(*t), GFP_KERNEL); |
574 | if (t == NULL) | 592 | if (t == NULL) |
575 | return -ENOMEM; | 593 | return -ENOMEM; |
594 | sd = &t->sd; | ||
595 | v4l2_i2c_subdev_init(sd, client, &saa5249_ops); | ||
576 | mutex_init(&t->lock); | 596 | mutex_init(&t->lock); |
577 | 597 | ||
578 | /* Now create a video4linux device */ | 598 | /* Now create a video4linux device */ |
579 | vd = kmalloc(sizeof(struct video_device), GFP_KERNEL); | 599 | t->vdev = video_device_alloc(); |
580 | if (vd == NULL) { | 600 | if (t->vdev == NULL) { |
581 | kfree(client); | 601 | kfree(client); |
582 | return -ENOMEM; | 602 | return -ENOMEM; |
583 | } | 603 | } |
584 | i2c_set_clientdata(client, vd); | 604 | memcpy(t->vdev, &saa_template, sizeof(*t->vdev)); |
585 | memcpy(vd, &saa_template, sizeof(*vd)); | ||
586 | 605 | ||
587 | for (pgbuf = 0; pgbuf < NUM_DAUS; pgbuf++) { | 606 | for (pgbuf = 0; pgbuf < NUM_DAUS; pgbuf++) { |
588 | memset(t->vdau[pgbuf].pgbuf, ' ', sizeof(t->vdau[0].pgbuf)); | 607 | memset(t->vdau[pgbuf].pgbuf, ' ', sizeof(t->vdau[0].pgbuf)); |
@@ -593,26 +612,27 @@ static int saa5249_probe(struct i2c_client *client, | |||
593 | t->vdau[pgbuf].stopped = true; | 612 | t->vdau[pgbuf].stopped = true; |
594 | t->is_searching[pgbuf] = false; | 613 | t->is_searching[pgbuf] = false; |
595 | } | 614 | } |
596 | video_set_drvdata(vd, t); | 615 | video_set_drvdata(t->vdev, t); |
597 | 616 | ||
598 | /* Register it */ | 617 | /* Register it */ |
599 | err = video_register_device(vd, VFL_TYPE_VTX, -1); | 618 | err = video_register_device(t->vdev, VFL_TYPE_VTX, -1); |
600 | if (err < 0) { | 619 | if (err < 0) { |
601 | kfree(t); | 620 | kfree(t); |
602 | kfree(vd); | 621 | video_device_release(t->vdev); |
622 | t->vdev = NULL; | ||
603 | return err; | 623 | return err; |
604 | } | 624 | } |
605 | t->client = client; | ||
606 | return 0; | 625 | return 0; |
607 | } | 626 | } |
608 | 627 | ||
609 | static int saa5249_remove(struct i2c_client *client) | 628 | static int saa5249_remove(struct i2c_client *client) |
610 | { | 629 | { |
611 | struct video_device *vd = i2c_get_clientdata(client); | 630 | struct v4l2_subdev *sd = i2c_get_clientdata(client); |
631 | struct saa5249_device *t = to_dev(sd); | ||
612 | 632 | ||
613 | video_unregister_device(vd); | 633 | video_unregister_device(t->vdev); |
614 | kfree(video_get_drvdata(vd)); | 634 | v4l2_device_unregister_subdev(sd); |
615 | kfree(vd); | 635 | kfree(t); |
616 | return 0; | 636 | return 0; |
617 | } | 637 | } |
618 | 638 | ||