m5602_s5k83a.c 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390
  1. /*
  2. * Driver for the s5k83a sensor
  3. *
  4. * Copyright (C) 2008 Erik Andrén
  5. * Copyright (C) 2007 Ilyes Gouta. Based on the m5603x Linux Driver Project.
  6. * Copyright (C) 2005 m5603x Linux Driver Project <m5602@x3ng.com.br>
  7. *
  8. * Portions of code to USB interface and ALi driver software,
  9. * Copyright (c) 2006 Willem Duinker
  10. * v4l2 interface modeled after the V4L2 driver
  11. * for SN9C10x PC Camera Controllers
  12. *
  13. * This program is free software; you can redistribute it and/or
  14. * modify it under the terms of the GNU General Public License as
  15. * published by the Free Software Foundation, version 2.
  16. *
  17. */
  18. #include "m5602_s5k83a.h"
  19. static struct v4l2_pix_format s5k83a_modes[] = {
  20. {
  21. 640,
  22. 480,
  23. V4L2_PIX_FMT_SBGGR8,
  24. V4L2_FIELD_NONE,
  25. .sizeimage =
  26. 640 * 480,
  27. .bytesperline = 640,
  28. .colorspace = V4L2_COLORSPACE_SRGB,
  29. .priv = 0
  30. }
  31. };
  32. static void s5k83a_dump_registers(struct sd *sd);
  33. int s5k83a_probe(struct sd *sd)
  34. {
  35. u8 prod_id = 0, ver_id = 0;
  36. int i, err = 0;
  37. if (force_sensor) {
  38. if (force_sensor == S5K83A_SENSOR) {
  39. info("Forcing a %s sensor", s5k83a.name);
  40. goto sensor_found;
  41. }
  42. /* If we want to force another sensor, don't try to probe this
  43. * one */
  44. return -ENODEV;
  45. }
  46. info("Probing for a s5k83a sensor");
  47. /* Preinit the sensor */
  48. for (i = 0; i < ARRAY_SIZE(preinit_s5k83a) && !err; i++) {
  49. u8 data[2] = {preinit_s5k83a[i][2], preinit_s5k83a[i][3]};
  50. if (preinit_s5k83a[i][0] == SENSOR)
  51. err = m5602_write_sensor(sd, preinit_s5k83a[i][1],
  52. data, 2);
  53. else
  54. err = m5602_write_bridge(sd, preinit_s5k83a[i][1],
  55. data[0]);
  56. }
  57. /* We don't know what register (if any) that contain the product id
  58. * Just pick the first addresses that seem to produce the same results
  59. * on multiple machines */
  60. if (m5602_read_sensor(sd, 0x00, &prod_id, 1))
  61. return -ENODEV;
  62. if (m5602_read_sensor(sd, 0x01, &ver_id, 1))
  63. return -ENODEV;
  64. if ((prod_id == 0xff) || (ver_id == 0xff))
  65. return -ENODEV;
  66. else
  67. info("Detected a s5k83a sensor");
  68. sensor_found:
  69. sd->gspca_dev.cam.cam_mode = s5k83a_modes;
  70. sd->gspca_dev.cam.nmodes = ARRAY_SIZE(s5k83a_modes);
  71. sd->desc->ctrls = s5k83a.ctrls;
  72. sd->desc->nctrls = ARRAY_SIZE(s5k83a_ctrls);
  73. return 0;
  74. }
  75. int s5k83a_init(struct sd *sd)
  76. {
  77. int i, err = 0;
  78. for (i = 0; i < ARRAY_SIZE(init_s5k83a) && !err; i++) {
  79. u8 data[2] = {0x00, 0x00};
  80. switch (init_s5k83a[i][0]) {
  81. case BRIDGE:
  82. err = m5602_write_bridge(sd,
  83. init_s5k83a[i][1],
  84. init_s5k83a[i][2]);
  85. break;
  86. case SENSOR:
  87. data[0] = init_s5k83a[i][2];
  88. err = m5602_write_sensor(sd,
  89. init_s5k83a[i][1], data, 1);
  90. break;
  91. case SENSOR_LONG:
  92. data[0] = init_s5k83a[i][2];
  93. data[1] = init_s5k83a[i][3];
  94. err = m5602_write_sensor(sd,
  95. init_s5k83a[i][1], data, 2);
  96. break;
  97. default:
  98. info("Invalid stream command, exiting init");
  99. return -EINVAL;
  100. }
  101. }
  102. if (dump_sensor)
  103. s5k83a_dump_registers(sd);
  104. return (err < 0) ? err : 0;
  105. }
  106. int s5k83a_start(struct sd *sd)
  107. {
  108. return s5k83a_set_led_indication(sd, 1);
  109. }
  110. int s5k83a_stop(struct sd *sd)
  111. {
  112. return s5k83a_set_led_indication(sd, 0);
  113. }
  114. int s5k83a_power_down(struct sd *sd)
  115. {
  116. return 0;
  117. }
  118. static void s5k83a_dump_registers(struct sd *sd)
  119. {
  120. int address;
  121. u8 page, old_page;
  122. m5602_read_sensor(sd, S5K83A_PAGE_MAP, &old_page, 1);
  123. for (page = 0; page < 16; page++) {
  124. m5602_write_sensor(sd, S5K83A_PAGE_MAP, &page, 1);
  125. info("Dumping the s5k83a register state for page 0x%x", page);
  126. for (address = 0; address <= 0xff; address++) {
  127. u8 val = 0;
  128. m5602_read_sensor(sd, address, &val, 1);
  129. info("register 0x%x contains 0x%x",
  130. address, val);
  131. }
  132. }
  133. info("s5k83a register state dump complete");
  134. for (page = 0; page < 16; page++) {
  135. m5602_write_sensor(sd, S5K83A_PAGE_MAP, &page, 1);
  136. info("Probing for which registers that are read/write "
  137. "for page 0x%x", page);
  138. for (address = 0; address <= 0xff; address++) {
  139. u8 old_val, ctrl_val, test_val = 0xff;
  140. m5602_read_sensor(sd, address, &old_val, 1);
  141. m5602_write_sensor(sd, address, &test_val, 1);
  142. m5602_read_sensor(sd, address, &ctrl_val, 1);
  143. if (ctrl_val == test_val)
  144. info("register 0x%x is writeable", address);
  145. else
  146. info("register 0x%x is read only", address);
  147. /* Restore original val */
  148. m5602_write_sensor(sd, address, &old_val, 1);
  149. }
  150. }
  151. info("Read/write register probing complete");
  152. m5602_write_sensor(sd, S5K83A_PAGE_MAP, &old_page, 1);
  153. }
  154. int s5k83a_get_brightness(struct gspca_dev *gspca_dev, __s32 *val)
  155. {
  156. int err;
  157. u8 data[2];
  158. struct sd *sd = (struct sd *) gspca_dev;
  159. err = m5602_read_sensor(sd, S5K83A_BRIGHTNESS, data, 2);
  160. if (err < 0)
  161. return err;
  162. data[1] = data[1] << 1;
  163. *val = data[1];
  164. return err;
  165. }
  166. int s5k83a_set_brightness(struct gspca_dev *gspca_dev, __s32 val)
  167. {
  168. int err;
  169. u8 data[2];
  170. struct sd *sd = (struct sd *) gspca_dev;
  171. data[0] = 0x00;
  172. data[1] = 0x20;
  173. err = m5602_write_sensor(sd, 0x14, data, 2);
  174. if (err < 0)
  175. return err;
  176. data[0] = 0x01;
  177. data[1] = 0x00;
  178. err = m5602_write_sensor(sd, 0x0d, data, 2);
  179. if (err < 0)
  180. return err;
  181. /* FIXME: This is not sane, we need to figure out the composition
  182. of these registers */
  183. data[0] = val >> 3; /* brightness, high 5 bits */
  184. data[1] = val >> 1; /* brightness, high 7 bits */
  185. err = m5602_write_sensor(sd, S5K83A_BRIGHTNESS, data, 2);
  186. return err;
  187. }
  188. int s5k83a_get_whiteness(struct gspca_dev *gspca_dev, __s32 *val)
  189. {
  190. int err;
  191. u8 data;
  192. struct sd *sd = (struct sd *) gspca_dev;
  193. err = m5602_read_sensor(sd, S5K83A_WHITENESS, &data, 1);
  194. if (err < 0)
  195. return err;
  196. *val = data;
  197. return err;
  198. }
  199. int s5k83a_set_whiteness(struct gspca_dev *gspca_dev, __s32 val)
  200. {
  201. int err;
  202. u8 data[1];
  203. struct sd *sd = (struct sd *) gspca_dev;
  204. data[0] = val;
  205. err = m5602_write_sensor(sd, S5K83A_WHITENESS, data, 1);
  206. return err;
  207. }
  208. int s5k83a_get_gain(struct gspca_dev *gspca_dev, __s32 *val)
  209. {
  210. int err;
  211. u8 data[2];
  212. struct sd *sd = (struct sd *) gspca_dev;
  213. err = m5602_read_sensor(sd, S5K83A_GAIN, data, 2);
  214. if (err < 0)
  215. return err;
  216. data[1] = data[1] & 0x3f;
  217. if (data[1] > S5K83A_MAXIMUM_GAIN)
  218. data[1] = S5K83A_MAXIMUM_GAIN;
  219. *val = data[1];
  220. return err;
  221. }
  222. int s5k83a_set_gain(struct gspca_dev *gspca_dev, __s32 val)
  223. {
  224. int err;
  225. u8 data[2];
  226. struct sd *sd = (struct sd *) gspca_dev;
  227. data[0] = 0;
  228. data[1] = val;
  229. err = m5602_write_sensor(sd, S5K83A_GAIN, data, 2);
  230. return err;
  231. }
  232. int s5k83a_get_vflip(struct gspca_dev *gspca_dev, __s32 *val)
  233. {
  234. int err;
  235. u8 data[1];
  236. struct sd *sd = (struct sd *) gspca_dev;
  237. data[0] = 0x05;
  238. err = m5602_write_sensor(sd, S5K83A_PAGE_MAP, data, 1);
  239. if (err < 0)
  240. return err;
  241. err = m5602_read_sensor(sd, S5K83A_FLIP, data, 1);
  242. *val = (data[0] | 0x40) ? 1 : 0;
  243. return err;
  244. }
  245. int s5k83a_set_vflip(struct gspca_dev *gspca_dev, __s32 val)
  246. {
  247. int err;
  248. u8 data[1];
  249. struct sd *sd = (struct sd *) gspca_dev;
  250. data[0] = 0x05;
  251. err = m5602_write_sensor(sd, S5K83A_PAGE_MAP, data, 1);
  252. if (err < 0)
  253. return err;
  254. err = m5602_read_sensor(sd, S5K83A_FLIP, data, 1);
  255. if (err < 0)
  256. return err;
  257. /* set or zero six bit, seven is hflip */
  258. data[0] = (val) ? (data[0] & 0x80) | 0x40 | S5K83A_FLIP_MASK
  259. : (data[0] & 0x80) | S5K83A_FLIP_MASK;
  260. err = m5602_write_sensor(sd, S5K83A_FLIP, data, 1);
  261. if (err < 0)
  262. return err;
  263. data[0] = (val) ? 0x0b : 0x0a;
  264. err = m5602_write_sensor(sd, S5K83A_VFLIP_TUNE, data, 1);
  265. return err;
  266. }
  267. int s5k83a_get_hflip(struct gspca_dev *gspca_dev, __s32 *val)
  268. {
  269. int err;
  270. u8 data[1];
  271. struct sd *sd = (struct sd *) gspca_dev;
  272. data[0] = 0x05;
  273. err = m5602_write_sensor(sd, S5K83A_PAGE_MAP, data, 1);
  274. if (err < 0)
  275. return err;
  276. err = m5602_read_sensor(sd, S5K83A_FLIP, data, 1);
  277. *val = (data[0] | 0x80) ? 1 : 0;
  278. return err;
  279. }
  280. int s5k83a_set_hflip(struct gspca_dev *gspca_dev, __s32 val)
  281. {
  282. int err;
  283. u8 data[1];
  284. struct sd *sd = (struct sd *) gspca_dev;
  285. data[0] = 0x05;
  286. err = m5602_write_sensor(sd, S5K83A_PAGE_MAP, data, 1);
  287. if (err < 0)
  288. return err;
  289. err = m5602_read_sensor(sd, S5K83A_FLIP, data, 1);
  290. if (err < 0)
  291. return err;
  292. /* set or zero seven bit, six is vflip */
  293. data[0] = (val) ? (data[0] & 0x40) | 0x80 | S5K83A_FLIP_MASK
  294. : (data[0] & 0x40) | S5K83A_FLIP_MASK;
  295. err = m5602_write_sensor(sd, S5K83A_FLIP, data, 1);
  296. if (err < 0)
  297. return err;
  298. data[0] = (val) ? 0x0a : 0x0b;
  299. err = m5602_write_sensor(sd, S5K83A_HFLIP_TUNE, data, 1);
  300. return err;
  301. }
  302. int s5k83a_set_led_indication(struct sd *sd, u8 val)
  303. {
  304. int err = 0;
  305. u8 data[1];
  306. err = m5602_read_bridge(sd, M5602_XB_GPIO_DAT, data);
  307. if (err < 0)
  308. return err;
  309. if (val)
  310. data[0] = data[0] | S5K83A_GPIO_LED_MASK;
  311. else
  312. data[0] = data[0] & ~S5K83A_GPIO_LED_MASK;
  313. err = m5602_write_bridge(sd, M5602_XB_GPIO_DAT, data[0]);
  314. return (err < 0) ? err : 0;
  315. }