1be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorin/* Subdriver for the GL860 chip with the OV9655 sensor
2be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorin * Author Olivier LORIN, from logs done by Simon (Sur3) and Almighurt
34f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin * on dsd's weblog
44f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin *
54f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin * This program is free software; you can redistribute it and/or modify
64f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin * it under the terms of the GNU General Public License as published by
74f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin * the Free Software Foundation; either version 2 of the License, or
84f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin * any later version.
94f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin *
104f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin * This program is distributed in the hope that it will be useful,
114f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin * but WITHOUT ANY WARRANTY; without even the implied warranty of
124f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
134f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin * GNU General Public License for more details.
144f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin *
154f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin * You should have received a copy of the GNU General Public License
164f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin * along with this program.  If not, see <http://www.gnu.org/licenses/>.
174f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin */
184f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
194f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin/* Sensor : OV9655 */
204f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
214f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin#include "gl860.h"
224f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
234f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinstatic struct validx tbl_init_at_startup[] = {
244f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x0000, 0x0000}, {0x0010, 0x0010}, {0x0008, 0x00c0}, {0x0001, 0x00c1},
254f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x0001, 0x00c2}, {0x0020, 0x0006}, {0x006a, 0x000d},
264f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
274f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x0040, 0x0000},
284f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin};
294f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
304f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinstatic struct validx tbl_commmon[] = {
314f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x0041, 0x0000}, {0x006a, 0x0007}, {0x0063, 0x0006}, {0x006a, 0x000d},
324f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x0000, 0x00c0}, {0x0010, 0x0010}, {0x0001, 0x00c1}, {0x0041, 0x00c2},
334f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x0004, 0x00d8}, {0x0012, 0x0004}, {0x0000, 0x0058}, {0x0040, 0x0000},
344f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x00f3, 0x0006}, {0x0058, 0x0000}, {0x0048, 0x0000}, {0x0061, 0x0000},
354f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin};
364f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
374f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinstatic s32 tbl_length[] = {12, 56, 52, 54, 56, 42, 32, 12};
384f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
394f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinstatic u8 *tbl_640[] = {
404f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x00\x40\x07\x6a\x06\xf3\x0d\x6a" "\x10\x10\xc1\x01"
414f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	,
424f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x12\x80\x00\x00\x01\x98\x02\x80" "\x03\x12\x04\x03\x0b\x57\x0e\x61"
434f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x0f\x42\x11\x01\x12\x60\x13\x00" "\x14\x3a\x16\x24\x17\x14\x18\x00"
444f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x19\x01\x1a\x3d\x1e\x04\x24\x3c" "\x25\x36\x26\x72\x27\x08\x28\x08"
454f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x29\x15\x2a\x00\x2b\x00\x2c\x08"
464f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	,
474f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x32\xff\x33\x00\x34\x3d\x35\x00" "\x36\xfa\x38\x72\x39\x57\x3a\x00"
484f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x3b\x0c\x3d\x99\x3e\x0c\x3f\xc1" "\x40\xc0\x41\x00\x42\xc0\x43\x0a"
494f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x44\xf0\x45\x46\x46\x62\x47\x2a" "\x48\x3c\x4a\xee\x4b\xe7\x4c\xe7"
504f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x4d\xe7\x4e\xe7"
514f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	,
524f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x4f\x98\x50\x98\x51\x00\x52\x28" "\x53\x70\x54\x98\x58\x1a\x59\x85"
534f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x5a\xa9\x5b\x64\x5c\x84\x5d\x53" "\x5e\x0e\x5f\xf0\x60\xf0\x61\xf0"
544f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x62\x00\x63\x00\x64\x02\x65\x20" "\x66\x00\x69\x0a\x6b\x5a\x6c\x04"
554f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x6d\x55\x6e\x00\x6f\x9d"
564f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	,
574f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x70\x15\x71\x78\x72\x00\x73\x00" "\x74\x3a\x75\x35\x76\x01\x77\x02"
584f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x7a\x24\x7b\x04\x7c\x07\x7d\x10" "\x7e\x28\x7f\x36\x80\x44\x81\x52"
594f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x82\x60\x83\x6c\x84\x78\x85\x8c" "\x86\x9e\x87\xbb\x88\xd2\x89\xe5"
604f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x8a\x23\x8c\x8d\x90\x7c\x91\x7b"
614f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	,
624f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x9d\x02\x9e\x02\x9f\x74\xa0\x73" "\xa1\x40\xa4\x50\xa5\x68\xa6\x70"
634f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\xa8\xc1\xa9\xef\xaa\x92\xab\x04" "\xac\x80\xad\x80\xae\x80\xaf\x80"
644f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\xb2\xf2\xb3\x20\xb4\x20\xb5\x00" "\xb6\xaf"
654f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	,
664f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\xbb\xae\xbc\x4f\xbd\x4e\xbe\x6a" "\xbf\x68\xc0\xaa\xc1\xc0\xc2\x01"
674f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\xc3\x4e\xc6\x85\xc7\x81\xc9\xe0" "\xca\xe8\xcb\xf0\xcc\xd8\xcd\x93"
684f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	,
694f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\xd0\x01\xd1\x08\xd2\xe0\xd3\x01" "\xd4\x10\xd5\x80"
704f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin};
714f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
72d398c0eb62d1788726b501329abcc6fd164d1778Olivier Lorinstatic u8 *tbl_1280[] = {
734f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x00\x40\x07\x6a\x06\xf3\x0d\x6a" "\x10\x10\xc1\x01"
744f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	,
754f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x12\x80\x00\x00\x01\x98\x02\x80" "\x03\x12\x04\x01\x0b\x57\x0e\x61"
764f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x0f\x42\x11\x00\x12\x00\x13\x00" "\x14\x3a\x16\x24\x17\x1b\x18\xbb"
774f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x19\x01\x1a\x81\x1e\x04\x24\x3c" "\x25\x36\x26\x72\x27\x08\x28\x08"
784f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x29\x15\x2a\x00\x2b\x00\x2c\x08"
794f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	,
804f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x32\xa4\x33\x00\x34\x3d\x35\x00" "\x36\xf8\x38\x72\x39\x57\x3a\x00"
814f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x3b\x0c\x3d\x99\x3e\x0c\x3f\xc2" "\x40\xc0\x41\x00\x42\xc0\x43\x0a"
824f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x44\xf0\x45\x46\x46\x62\x47\x2a" "\x48\x3c\x4a\xec\x4b\xe8\x4c\xe8"
834f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x4d\xe8\x4e\xe8"
844f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	,
854f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x4f\x98\x50\x98\x51\x00\x52\x28" "\x53\x70\x54\x98\x58\x1a\x59\x85"
864f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x5a\xa9\x5b\x64\x5c\x84\x5d\x53" "\x5e\x0e\x5f\xf0\x60\xf0\x61\xf0"
874f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x62\x00\x63\x00\x64\x02\x65\x20" "\x66\x00\x69\x02\x6b\x5a\x6c\x04"
884f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x6d\x55\x6e\x00\x6f\x9d"
894f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	,
904f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x70\x08\x71\x78\x72\x00\x73\x01" "\x74\x3a\x75\x35\x76\x01\x77\x02"
914f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x7a\x24\x7b\x04\x7c\x07\x7d\x10" "\x7e\x28\x7f\x36\x80\x44\x81\x52"
924f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x82\x60\x83\x6c\x84\x78\x85\x8c" "\x86\x9e\x87\xbb\x88\xd2\x89\xe5"
934f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x8a\x23\x8c\x0d\x90\x90\x91\x90"
944f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	,
954f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\x9d\x02\x9e\x02\x9f\x94\xa0\x94" "\xa1\x01\xa4\x50\xa5\x68\xa6\x70"
964f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\xa8\xc1\xa9\xef\xaa\x92\xab\x04" "\xac\x80\xad\x80\xae\x80\xaf\x80"
974f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\xb2\xf2\xb3\x20\xb4\x20\xb5\x00" "\xb6\xaf"
984f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	,
994f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\xbb\xae\xbc\x38\xbd\x39\xbe\x01" "\xbf\x01\xc0\xe2\xc1\xc0\xc2\x01"
1004f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\xc3\x4e\xc6\x85\xc7\x81\xc9\xe0" "\xca\xe8\xcb\xf0\xcc\xd8\xcd\x93"
1014f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	,
1024f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	"\xd0\x21\xd1\x18\xd2\xe0\xd3\x01" "\xd4\x28\xd5\x00"
1034f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin};
1044f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
1054f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinstatic u8 c04[] = {0x04};
106be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorinstatic u8 dat_post1[] = "\x04\x00\x10\x20\xa1\x00\x00\x02";
107be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorinstatic u8 dat_post2[] = "\x10\x10\xc1\x02";
108be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorinstatic u8 dat_post3[] = "\x04\x00\x10\x7c\xa1\x00\x00\x04";
109be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorinstatic u8 dat_post4[] = "\x10\x02\xc1\x06";
110be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorinstatic u8 dat_post5[] = "\x04\x00\x10\x7b\xa1\x00\x00\x08";
111be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorinstatic u8 dat_post6[] = "\x10\x10\xc1\x05";
112be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorinstatic u8 dat_post7[] = "\x04\x00\x10\x7c\xa1\x00\x00\x08";
113be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorinstatic u8 dat_post8[] = "\x04\x00\x10\x7c\xa1\x00\x00\x09";
1144f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
1154f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinstatic struct validx tbl_init_post_alt[] = {
1164f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x6032, 0x00ff}, {0x6032, 0x00ff}, {0x6032, 0x00ff}, {0x603c, 0x00ff},
1174f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x6003, 0x00ff}, {0x6032, 0x00ff}, {0x6032, 0x00ff}, {0x6001, 0x00ff},
1184f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x6000, 0x801e},
1194f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0xffff, 0xffff},
1204f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x6004, 0x001e}, {0x6000, 0x801e},
1214f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0xffff, 0xffff},
1224f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x6004, 0x001e}, {0x6012, 0x0003}, {0x6000, 0x801e},
1234f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0xffff, 0xffff},
1244f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x6004, 0x001e}, {0x6000, 0x801e},
1254f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0xffff, 0xffff},
1264f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x6004, 0x001e}, {0x6012, 0x0003},
1274f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0xffff, 0xffff},
1284f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x6000, 0x801e},
1294f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0xffff, 0xffff},
1304f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x6004, 0x001e}, {0x6000, 0x801e},
1314f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0xffff, 0xffff},
1324f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x6004, 0x001e}, {0x6012, 0x0003}, {0x6000, 0x801e},
1334f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0xffff, 0xffff},
1344f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x6004, 0x001e}, {0x6000, 0x801e},
1354f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0xffff, 0xffff},
1364f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x6004, 0x001e}, {0x6012, 0x0003},
1374f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0xffff, 0xffff},
1384f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x6000, 0x801e},
1394f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0xffff, 0xffff},
1404f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x6004, 0x001e}, {0x6000, 0x801e},
1414f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0xffff, 0xffff},
1424f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	{0x6004, 0x001e}, {0x6012, 0x0003},
1434f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin};
1444f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
1454f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinstatic int  ov9655_init_at_startup(struct gspca_dev *gspca_dev);
1464f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinstatic int  ov9655_configure_alt(struct gspca_dev *gspca_dev);
1474f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinstatic int  ov9655_init_pre_alt(struct gspca_dev *gspca_dev);
1484f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinstatic int  ov9655_init_post_alt(struct gspca_dev *gspca_dev);
1494f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinstatic void ov9655_post_unset_alt(struct gspca_dev *gspca_dev);
1504f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinstatic int  ov9655_camera_settings(struct gspca_dev *gspca_dev);
1514f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin/*==========================================================================*/
1524f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
1534f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinvoid ov9655_init_settings(struct gspca_dev *gspca_dev)
1544f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin{
1554f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	struct sd *sd = (struct sd *) gspca_dev;
1564f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
1574f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vcur.backlight  =   0;
1584f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vcur.brightness = 128;
1594f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vcur.sharpness  =   0;
1604f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vcur.contrast   =   0;
1614f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vcur.gamma      =   0;
1624f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vcur.hue        =   0;
1634f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vcur.saturation =   0;
1644f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vcur.whitebal   =   0;
1654f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
1664f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vmax.backlight  =   0;
1674f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vmax.brightness = 255;
1684f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vmax.sharpness  =   0;
1694f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vmax.contrast   =   0;
1704f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vmax.gamma      =   0;
1714f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vmax.hue        =   0 + 1;
1724f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vmax.saturation =   0;
1734f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vmax.whitebal   =   0;
1744f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vmax.mirror     = 0;
1754f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vmax.flip       = 0;
1764f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vmax.AC50Hz     = 0;
1774f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
1784f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->dev_camera_settings = ov9655_camera_settings;
1794f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->dev_init_at_startup = ov9655_init_at_startup;
1804f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->dev_configure_alt   = ov9655_configure_alt;
1814f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->dev_init_pre_alt    = ov9655_init_pre_alt;
1824f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->dev_post_unset_alt  = ov9655_post_unset_alt;
1834f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin}
1844f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
1854f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin/*==========================================================================*/
1864f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
1874f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinstatic int ov9655_init_at_startup(struct gspca_dev *gspca_dev)
1884f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin{
1894f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	fetch_validx(gspca_dev, tbl_init_at_startup,
1904f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin			ARRAY_SIZE(tbl_init_at_startup));
1914f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	fetch_validx(gspca_dev, tbl_commmon, ARRAY_SIZE(tbl_commmon));
1924f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin/*	ctrl_out(gspca_dev, 0x40, 11, 0x0000, 0x0000, 0, NULL);*/
1934f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
1944f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	return 0;
1954f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin}
1964f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
1974f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinstatic int ov9655_init_pre_alt(struct gspca_dev *gspca_dev)
1984f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin{
1994f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	struct sd *sd = (struct sd *) gspca_dev;
2004f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
2014f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vold.brightness = -1;
2024f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	sd->vold.hue = -1;
2034f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
2044f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	fetch_validx(gspca_dev, tbl_commmon, ARRAY_SIZE(tbl_commmon));
2054f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
2064f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	ov9655_init_post_alt(gspca_dev);
2074f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
2084f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	return 0;
2094f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin}
2104f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
2114f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinstatic int ov9655_init_post_alt(struct gspca_dev *gspca_dev)
2124f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin{
2134f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	s32 reso = gspca_dev->cam.cam_mode[(s32) gspca_dev->curr_mode].priv;
214be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorin	s32 n; /* reserved for FETCH functions */
2154f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	s32 i;
2164f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	u8 **tbl;
2174f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
2184f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	ctrl_out(gspca_dev, 0x40, 5, 0x0001, 0x0000, 0, NULL);
2194f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
220d398c0eb62d1788726b501329abcc6fd164d1778Olivier Lorin	tbl = (reso == IMAGE_640) ? tbl_640 : tbl_1280;
2214f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
2224f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	ctrl_out(gspca_dev, 0x40, 3, 0x0000, 0x0200,
2234f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin			tbl_length[0], tbl[0]);
2244f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	for (i = 1; i < 7; i++)
2254f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin		ctrl_out(gspca_dev, 0x40, 3, 0x6000, 0x0200,
2264f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin				tbl_length[i], tbl[i]);
2274f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	ctrl_out(gspca_dev, 0x40, 3, 0x0000, 0x0200,
2284f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin			tbl_length[7], tbl[7]);
2294f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
2304f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	n = fetch_validx(gspca_dev, tbl_init_post_alt,
2314f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin			ARRAY_SIZE(tbl_init_post_alt));
2324f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
2334f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
2344f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
2354f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin					ARRAY_SIZE(tbl_init_post_alt), n);
2364f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
2374f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
2384f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin					ARRAY_SIZE(tbl_init_post_alt), n);
2394f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
2404f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
2414f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin					ARRAY_SIZE(tbl_init_post_alt), n);
2424f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
2434f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
2444f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin					ARRAY_SIZE(tbl_init_post_alt), n);
245be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorin	ctrl_out(gspca_dev, 0x40, 3, 0x6000, 0x0200, 8, dat_post1);
2464f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
2474f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin					ARRAY_SIZE(tbl_init_post_alt), n);
2484f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
2494f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
2504f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
2514f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin					ARRAY_SIZE(tbl_init_post_alt), n);
2524f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
2534f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
2544f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin					ARRAY_SIZE(tbl_init_post_alt), n);
2554f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
2564f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
2574f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin					ARRAY_SIZE(tbl_init_post_alt), n);
2584f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
2594f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
2604f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin					ARRAY_SIZE(tbl_init_post_alt), n);
261be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorin	ctrl_out(gspca_dev, 0x40, 3, 0x6000, 0x0200, 8, dat_post1);
2624f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
2634f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin					ARRAY_SIZE(tbl_init_post_alt), n);
2644f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
2654f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
2664f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
2674f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin					ARRAY_SIZE(tbl_init_post_alt), n);
2684f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	ctrl_in(gspca_dev, 0xc0, 2, 0x6000, 0x801e, 1, c04);
2694f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	keep_on_fetching_validx(gspca_dev, tbl_init_post_alt,
2704f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin					ARRAY_SIZE(tbl_init_post_alt), n);
2714f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
272be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorin	ctrl_out(gspca_dev, 0x40, 3, 0x6000, 0x0200, 8, dat_post1);
2734f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
274be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorin	ctrl_out(gspca_dev, 0x40, 3, 0x0000, 0x0200, 4, dat_post2);
275be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorin	ctrl_out(gspca_dev, 0x40, 3, 0x6000, 0x0200, 8, dat_post3);
2764f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
277be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorin	ctrl_out(gspca_dev, 0x40, 3, 0x0000, 0x0200, 4, dat_post4);
278be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorin	ctrl_out(gspca_dev, 0x40, 3, 0x6000, 0x0200, 8, dat_post5);
2794f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
280be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorin	ctrl_out(gspca_dev, 0x40, 3, 0x0000, 0x0200, 4, dat_post6);
281be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorin	ctrl_out(gspca_dev, 0x40, 3, 0x6000, 0x0200, 8, dat_post7);
2824f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
283be9904bdde05137085af1df98de98a49ddce9ad8Olivier Lorin	ctrl_out(gspca_dev, 0x40, 3, 0x6000, 0x0200, 8, dat_post8);
2844f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
2854f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	ov9655_camera_settings(gspca_dev);
2864f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
2874f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	return 0;
2884f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin}
2894f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
2904f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinstatic int ov9655_configure_alt(struct gspca_dev *gspca_dev)
2914f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin{
2924f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	s32 reso = gspca_dev->cam.cam_mode[(s32) gspca_dev->curr_mode].priv;
2934f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
2944f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	switch (reso) {
2954f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	case IMAGE_640:
2964f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin		gspca_dev->alt = 1 + 1;
2974f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin		break;
2984f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
2994f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	default:
3004f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin		gspca_dev->alt = 1 + 1;
3014f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin		break;
3024f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	}
3034f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	return 0;
3044f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin}
3054f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
3064f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinstatic int ov9655_camera_settings(struct gspca_dev *gspca_dev)
3074f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin{
3084f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	struct sd *sd = (struct sd *) gspca_dev;
3094f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
3104f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	u8 dat_bright[] = "\x04\x00\x10\x7c\xa1\x00\x00\x70";
3114f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
3124f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	s32 bright = sd->vcur.brightness;
3134f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	s32 hue    = sd->vcur.hue;
3144f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
3154f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	if (bright != sd->vold.brightness) {
3164f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin		sd->vold.brightness = bright;
3174f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin		if (bright < 0 || bright > sd->vmax.brightness)
3184f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin			bright = 0;
3194f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
3204f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin		dat_bright[3] = bright;
3214f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin		ctrl_out(gspca_dev, 0x40, 3, 0x6000, 0x0200, 8, dat_bright);
3224f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	}
3234f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
3244f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	if (hue != sd->vold.hue) {
3254f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin		sd->vold.hue = hue;
3264f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin		sd->swapRB = (hue != 0);
3274f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	}
3284f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
3294f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	return 0;
3304f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin}
3314f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin
3324f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorinstatic void ov9655_post_unset_alt(struct gspca_dev *gspca_dev)
3334f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin{
3344f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	ctrl_out(gspca_dev, 0x40, 5, 0x0000, 0x0000, 0, NULL);
3354f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin	ctrl_out(gspca_dev, 0x40, 1, 0x0061, 0x0000, 0, NULL);
3364f7cb8837cec65ade18b0e2655292fd98040234eOlivier Lorin}
337