diff options
-rw-r--r-- | drivers/media/video/gspca/pac7311.c | 107 |
1 files changed, 105 insertions, 2 deletions
diff --git a/drivers/media/video/gspca/pac7311.c b/drivers/media/video/gspca/pac7311.c index 3abc018f9690..a89cc9fd1531 100644 --- a/drivers/media/video/gspca/pac7311.c +++ b/drivers/media/video/gspca/pac7311.c | |||
@@ -40,6 +40,8 @@ struct sd { | |||
40 | unsigned char contrast; | 40 | unsigned char contrast; |
41 | unsigned char colors; | 41 | unsigned char colors; |
42 | unsigned char autogain; | 42 | unsigned char autogain; |
43 | __u8 hflip; | ||
44 | __u8 vflip; | ||
43 | 45 | ||
44 | char tosof; /* number of bytes before next start of frame */ | 46 | char tosof; /* number of bytes before next start of frame */ |
45 | signed char ag_cnt; | 47 | signed char ag_cnt; |
@@ -59,6 +61,10 @@ static int sd_setcolors(struct gspca_dev *gspca_dev, __s32 val); | |||
59 | static int sd_getcolors(struct gspca_dev *gspca_dev, __s32 *val); | 61 | static int sd_getcolors(struct gspca_dev *gspca_dev, __s32 *val); |
60 | static int sd_setautogain(struct gspca_dev *gspca_dev, __s32 val); | 62 | static int sd_setautogain(struct gspca_dev *gspca_dev, __s32 val); |
61 | static int sd_getautogain(struct gspca_dev *gspca_dev, __s32 *val); | 63 | static int sd_getautogain(struct gspca_dev *gspca_dev, __s32 *val); |
64 | static int sd_sethflip(struct gspca_dev *gspca_dev, __s32 val); | ||
65 | static int sd_gethflip(struct gspca_dev *gspca_dev, __s32 *val); | ||
66 | static int sd_setvflip(struct gspca_dev *gspca_dev, __s32 val); | ||
67 | static int sd_getvflip(struct gspca_dev *gspca_dev, __s32 *val); | ||
62 | 68 | ||
63 | static struct ctrl sd_ctrls[] = { | 69 | static struct ctrl sd_ctrls[] = { |
64 | { | 70 | { |
@@ -97,7 +103,8 @@ static struct ctrl sd_ctrls[] = { | |||
97 | .type = V4L2_CTRL_TYPE_INTEGER, | 103 | .type = V4L2_CTRL_TYPE_INTEGER, |
98 | .name = "Saturation", | 104 | .name = "Saturation", |
99 | .minimum = 0, | 105 | .minimum = 0, |
100 | .maximum = 255, | 106 | #define COLOR_MAX 255 |
107 | .maximum = COLOR_MAX, | ||
101 | .step = 1, | 108 | .step = 1, |
102 | #define COLOR_DEF 127 | 109 | #define COLOR_DEF 127 |
103 | .default_value = COLOR_DEF, | 110 | .default_value = COLOR_DEF, |
@@ -119,6 +126,35 @@ static struct ctrl sd_ctrls[] = { | |||
119 | .set = sd_setautogain, | 126 | .set = sd_setautogain, |
120 | .get = sd_getautogain, | 127 | .get = sd_getautogain, |
121 | }, | 128 | }, |
129 | /* next controls work with pac7302 only */ | ||
130 | { | ||
131 | { | ||
132 | .id = V4L2_CID_HFLIP, | ||
133 | .type = V4L2_CTRL_TYPE_BOOLEAN, | ||
134 | .name = "Mirror", | ||
135 | .minimum = 0, | ||
136 | .maximum = 1, | ||
137 | .step = 1, | ||
138 | #define HFLIP_DEF 0 | ||
139 | .default_value = HFLIP_DEF, | ||
140 | }, | ||
141 | .set = sd_sethflip, | ||
142 | .get = sd_gethflip, | ||
143 | }, | ||
144 | { | ||
145 | { | ||
146 | .id = V4L2_CID_VFLIP, | ||
147 | .type = V4L2_CTRL_TYPE_BOOLEAN, | ||
148 | .name = "Vflip", | ||
149 | .minimum = 0, | ||
150 | .maximum = 1, | ||
151 | .step = 1, | ||
152 | #define VFLIP_DEF 0 | ||
153 | .default_value = VFLIP_DEF, | ||
154 | }, | ||
155 | .set = sd_setvflip, | ||
156 | .get = sd_getvflip, | ||
157 | }, | ||
122 | }; | 158 | }; |
123 | 159 | ||
124 | static struct v4l2_pix_format vga_mode[] = { | 160 | static struct v4l2_pix_format vga_mode[] = { |
@@ -493,8 +529,25 @@ static void setcolors(struct gspca_dev *gspca_dev) | |||
493 | { | 529 | { |
494 | struct sd *sd = (struct sd *) gspca_dev; | 530 | struct sd *sd = (struct sd *) gspca_dev; |
495 | 531 | ||
496 | if (sd->sensor == SENSOR_PAC7302) | 532 | if (sd->sensor == SENSOR_PAC7302) { |
533 | int i, v; | ||
534 | static const int a[9] = | ||
535 | {217, -212, 0, -101, 170, -67, -38, -315, 355}; | ||
536 | static const int b[9] = | ||
537 | {19, 106, 0, 19, 106, 1, 19, 106, 1}; | ||
538 | |||
539 | reg_w(gspca_dev, 0xff, 0x03); /* page 3 */ | ||
540 | reg_w(gspca_dev, 0x11, 0x01); | ||
541 | reg_w(gspca_dev, 0xff, 0x00); /* page 0 */ | ||
542 | reg_w(gspca_dev, 0xff, 0x00); /* page 0 */ | ||
543 | for (i = 0; i < 9; i++) { | ||
544 | v = a[i] * sd->colors / COLOR_MAX + b[i]; | ||
545 | reg_w(gspca_dev, 0x0f + 2 * i, (v >> 8) & 0x07); | ||
546 | reg_w(gspca_dev, 0x0f + 2 * i + 1, v); | ||
547 | } | ||
548 | reg_w(gspca_dev, 0xdc, 0x01); | ||
497 | return; | 549 | return; |
550 | } | ||
498 | reg_w(gspca_dev, 0xff, 0x01); | 551 | reg_w(gspca_dev, 0xff, 0x01); |
499 | reg_w(gspca_dev, 0x10, sd->colors); | 552 | reg_w(gspca_dev, 0x10, sd->colors); |
500 | /* load registers to sensor (Bit 0, auto clear) */ | 553 | /* load registers to sensor (Bit 0, auto clear) */ |
@@ -514,6 +567,19 @@ static void setautogain(struct gspca_dev *gspca_dev) | |||
514 | } | 567 | } |
515 | } | 568 | } |
516 | 569 | ||
570 | /* this function is used by pac7302 only */ | ||
571 | static void sethvflip(struct gspca_dev *gspca_dev) | ||
572 | { | ||
573 | struct sd *sd = (struct sd *) gspca_dev; | ||
574 | __u8 data; | ||
575 | |||
576 | reg_w(gspca_dev, 0xff, 0x03); /* page 3 */ | ||
577 | data = (sd->hflip ? 0x00 : 0x08) | ||
578 | | (sd->vflip ? 0x04 : 0x00); | ||
579 | reg_w(gspca_dev, 0x21, data); | ||
580 | reg_w(gspca_dev, 0x11, 0x01); | ||
581 | } | ||
582 | |||
517 | /* this function is called at open time */ | 583 | /* this function is called at open time */ |
518 | static int sd_open(struct gspca_dev *gspca_dev) | 584 | static int sd_open(struct gspca_dev *gspca_dev) |
519 | { | 585 | { |
@@ -574,6 +640,7 @@ static void sd_start(struct gspca_dev *gspca_dev) | |||
574 | /* start stream */ | 640 | /* start stream */ |
575 | reg_w(gspca_dev, 0xff, 0x01); | 641 | reg_w(gspca_dev, 0xff, 0x01); |
576 | if (sd->sensor == SENSOR_PAC7302) { | 642 | if (sd->sensor == SENSOR_PAC7302) { |
643 | sethvflip(gspca_dev); | ||
577 | reg_w(gspca_dev, 0x78, 0x01); | 644 | reg_w(gspca_dev, 0x78, 0x01); |
578 | reg_w(gspca_dev, 0xff, 0x01); | 645 | reg_w(gspca_dev, 0xff, 0x01); |
579 | reg_w(gspca_dev, 0x78, 0x01); | 646 | reg_w(gspca_dev, 0x78, 0x01); |
@@ -790,6 +857,42 @@ static int sd_getautogain(struct gspca_dev *gspca_dev, __s32 *val) | |||
790 | return 0; | 857 | return 0; |
791 | } | 858 | } |
792 | 859 | ||
860 | static int sd_sethflip(struct gspca_dev *gspca_dev, __s32 val) | ||
861 | { | ||
862 | struct sd *sd = (struct sd *) gspca_dev; | ||
863 | |||
864 | sd->hflip = val; | ||
865 | if (gspca_dev->streaming) | ||
866 | sethvflip(gspca_dev); | ||
867 | return 0; | ||
868 | } | ||
869 | |||
870 | static int sd_gethflip(struct gspca_dev *gspca_dev, __s32 *val) | ||
871 | { | ||
872 | struct sd *sd = (struct sd *) gspca_dev; | ||
873 | |||
874 | *val = sd->hflip; | ||
875 | return 0; | ||
876 | } | ||
877 | |||
878 | static int sd_setvflip(struct gspca_dev *gspca_dev, __s32 val) | ||
879 | { | ||
880 | struct sd *sd = (struct sd *) gspca_dev; | ||
881 | |||
882 | sd->vflip = val; | ||
883 | if (gspca_dev->streaming) | ||
884 | sethvflip(gspca_dev); | ||
885 | return 0; | ||
886 | } | ||
887 | |||
888 | static int sd_getvflip(struct gspca_dev *gspca_dev, __s32 *val) | ||
889 | { | ||
890 | struct sd *sd = (struct sd *) gspca_dev; | ||
891 | |||
892 | *val = sd->vflip; | ||
893 | return 0; | ||
894 | } | ||
895 | |||
793 | /* sub-driver description */ | 896 | /* sub-driver description */ |
794 | static struct sd_desc sd_desc = { | 897 | static struct sd_desc sd_desc = { |
795 | .name = MODULE_NAME, | 898 | .name = MODULE_NAME, |