m5602_s5k83a.c 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376
  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 void s5k83a_dump_registers(struct sd *sd);
  20. int s5k83a_probe(struct sd *sd)
  21. {
  22. u8 prod_id = 0, ver_id = 0;
  23. int i, err = 0;
  24. if (force_sensor) {
  25. if (force_sensor == S5K83A_SENSOR) {
  26. info("Forcing a %s sensor", s5k83a.name);
  27. goto sensor_found;
  28. }
  29. /* If we want to force another sensor, don't try to probe this
  30. * one */
  31. return -ENODEV;
  32. }
  33. info("Probing for a s5k83a sensor");
  34. /* Preinit the sensor */
  35. for (i = 0; i < ARRAY_SIZE(preinit_s5k83a) && !err; i++) {
  36. u8 data[2] = {preinit_s5k83a[i][2], preinit_s5k83a[i][3]};
  37. if (preinit_s5k83a[i][0] == SENSOR)
  38. err = m5602_write_sensor(sd, preinit_s5k83a[i][1],
  39. data, 2);
  40. else
  41. err = m5602_write_bridge(sd, preinit_s5k83a[i][1],
  42. data[0]);
  43. }
  44. /* We don't know what register (if any) that contain the product id
  45. * Just pick the first addresses that seem to produce the same results
  46. * on multiple machines */
  47. if (m5602_read_sensor(sd, 0x00, &prod_id, 1))
  48. return -ENODEV;
  49. if (m5602_read_sensor(sd, 0x01, &ver_id, 1))
  50. return -ENODEV;
  51. if ((prod_id == 0xff) || (ver_id == 0xff))
  52. return -ENODEV;
  53. else
  54. info("Detected a s5k83a sensor");
  55. sensor_found:
  56. sd->gspca_dev.cam.cam_mode = s5k83a.modes;
  57. sd->gspca_dev.cam.nmodes = s5k83a.nmodes;
  58. sd->desc->ctrls = s5k83a.ctrls;
  59. sd->desc->nctrls = ARRAY_SIZE(s5k83a_ctrls);
  60. return 0;
  61. }
  62. int s5k83a_init(struct sd *sd)
  63. {
  64. int i, err = 0;
  65. for (i = 0; i < ARRAY_SIZE(init_s5k83a) && !err; i++) {
  66. u8 data[2] = {0x00, 0x00};
  67. switch (init_s5k83a[i][0]) {
  68. case BRIDGE:
  69. err = m5602_write_bridge(sd,
  70. init_s5k83a[i][1],
  71. init_s5k83a[i][2]);
  72. break;
  73. case SENSOR:
  74. data[0] = init_s5k83a[i][2];
  75. err = m5602_write_sensor(sd,
  76. init_s5k83a[i][1], data, 1);
  77. break;
  78. case SENSOR_LONG:
  79. data[0] = init_s5k83a[i][2];
  80. data[1] = init_s5k83a[i][3];
  81. err = m5602_write_sensor(sd,
  82. init_s5k83a[i][1], data, 2);
  83. break;
  84. default:
  85. info("Invalid stream command, exiting init");
  86. return -EINVAL;
  87. }
  88. }
  89. if (dump_sensor)
  90. s5k83a_dump_registers(sd);
  91. return (err < 0) ? err : 0;
  92. }
  93. int s5k83a_start(struct sd *sd)
  94. {
  95. return s5k83a_set_led_indication(sd, 1);
  96. }
  97. int s5k83a_stop(struct sd *sd)
  98. {
  99. return s5k83a_set_led_indication(sd, 0);
  100. }
  101. int s5k83a_power_down(struct sd *sd)
  102. {
  103. return 0;
  104. }
  105. static void s5k83a_dump_registers(struct sd *sd)
  106. {
  107. int address;
  108. u8 page, old_page;
  109. m5602_read_sensor(sd, S5K83A_PAGE_MAP, &old_page, 1);
  110. for (page = 0; page < 16; page++) {
  111. m5602_write_sensor(sd, S5K83A_PAGE_MAP, &page, 1);
  112. info("Dumping the s5k83a register state for page 0x%x", page);
  113. for (address = 0; address <= 0xff; address++) {
  114. u8 val = 0;
  115. m5602_read_sensor(sd, address, &val, 1);
  116. info("register 0x%x contains 0x%x",
  117. address, val);
  118. }
  119. }
  120. info("s5k83a register state dump complete");
  121. for (page = 0; page < 16; page++) {
  122. m5602_write_sensor(sd, S5K83A_PAGE_MAP, &page, 1);
  123. info("Probing for which registers that are read/write "
  124. "for page 0x%x", page);
  125. for (address = 0; address <= 0xff; address++) {
  126. u8 old_val, ctrl_val, test_val = 0xff;
  127. m5602_read_sensor(sd, address, &old_val, 1);
  128. m5602_write_sensor(sd, address, &test_val, 1);
  129. m5602_read_sensor(sd, address, &ctrl_val, 1);
  130. if (ctrl_val == test_val)
  131. info("register 0x%x is writeable", address);
  132. else
  133. info("register 0x%x is read only", address);
  134. /* Restore original val */
  135. m5602_write_sensor(sd, address, &old_val, 1);
  136. }
  137. }
  138. info("Read/write register probing complete");
  139. m5602_write_sensor(sd, S5K83A_PAGE_MAP, &old_page, 1);
  140. }
  141. int s5k83a_get_brightness(struct gspca_dev *gspca_dev, __s32 *val)
  142. {
  143. int err;
  144. u8 data[2];
  145. struct sd *sd = (struct sd *) gspca_dev;
  146. err = m5602_read_sensor(sd, S5K83A_BRIGHTNESS, data, 2);
  147. if (err < 0)
  148. return err;
  149. data[1] = data[1] << 1;
  150. *val = data[1];
  151. return err;
  152. }
  153. int s5k83a_set_brightness(struct gspca_dev *gspca_dev, __s32 val)
  154. {
  155. int err;
  156. u8 data[2];
  157. struct sd *sd = (struct sd *) gspca_dev;
  158. data[0] = 0x00;
  159. data[1] = 0x20;
  160. err = m5602_write_sensor(sd, 0x14, data, 2);
  161. if (err < 0)
  162. return err;
  163. data[0] = 0x01;
  164. data[1] = 0x00;
  165. err = m5602_write_sensor(sd, 0x0d, data, 2);
  166. if (err < 0)
  167. return err;
  168. /* FIXME: This is not sane, we need to figure out the composition
  169. of these registers */
  170. data[0] = val >> 3; /* brightness, high 5 bits */
  171. data[1] = val >> 1; /* brightness, high 7 bits */
  172. err = m5602_write_sensor(sd, S5K83A_BRIGHTNESS, data, 2);
  173. return err;
  174. }
  175. int s5k83a_get_whiteness(struct gspca_dev *gspca_dev, __s32 *val)
  176. {
  177. int err;
  178. u8 data;
  179. struct sd *sd = (struct sd *) gspca_dev;
  180. err = m5602_read_sensor(sd, S5K83A_WHITENESS, &data, 1);
  181. if (err < 0)
  182. return err;
  183. *val = data;
  184. return err;
  185. }
  186. int s5k83a_set_whiteness(struct gspca_dev *gspca_dev, __s32 val)
  187. {
  188. int err;
  189. u8 data[1];
  190. struct sd *sd = (struct sd *) gspca_dev;
  191. data[0] = val;
  192. err = m5602_write_sensor(sd, S5K83A_WHITENESS, data, 1);
  193. return err;
  194. }
  195. int s5k83a_get_gain(struct gspca_dev *gspca_dev, __s32 *val)
  196. {
  197. int err;
  198. u8 data[2];
  199. struct sd *sd = (struct sd *) gspca_dev;
  200. err = m5602_read_sensor(sd, S5K83A_GAIN, data, 2);
  201. if (err < 0)
  202. return err;
  203. data[1] = data[1] & 0x3f;
  204. if (data[1] > S5K83A_MAXIMUM_GAIN)
  205. data[1] = S5K83A_MAXIMUM_GAIN;
  206. *val = data[1];
  207. return err;
  208. }
  209. int s5k83a_set_gain(struct gspca_dev *gspca_dev, __s32 val)
  210. {
  211. int err;
  212. u8 data[2];
  213. struct sd *sd = (struct sd *) gspca_dev;
  214. data[0] = 0;
  215. data[1] = val;
  216. err = m5602_write_sensor(sd, S5K83A_GAIN, data, 2);
  217. return err;
  218. }
  219. int s5k83a_get_vflip(struct gspca_dev *gspca_dev, __s32 *val)
  220. {
  221. int err;
  222. u8 data[1];
  223. struct sd *sd = (struct sd *) gspca_dev;
  224. data[0] = 0x05;
  225. err = m5602_write_sensor(sd, S5K83A_PAGE_MAP, data, 1);
  226. if (err < 0)
  227. return err;
  228. err = m5602_read_sensor(sd, S5K83A_FLIP, data, 1);
  229. *val = (data[0] | 0x40) ? 1 : 0;
  230. return err;
  231. }
  232. int s5k83a_set_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. if (err < 0)
  243. return err;
  244. /* set or zero six bit, seven is hflip */
  245. data[0] = (val) ? (data[0] & 0x80) | 0x40 | S5K83A_FLIP_MASK
  246. : (data[0] & 0x80) | S5K83A_FLIP_MASK;
  247. err = m5602_write_sensor(sd, S5K83A_FLIP, data, 1);
  248. if (err < 0)
  249. return err;
  250. data[0] = (val) ? 0x0b : 0x0a;
  251. err = m5602_write_sensor(sd, S5K83A_VFLIP_TUNE, data, 1);
  252. return err;
  253. }
  254. int s5k83a_get_hflip(struct gspca_dev *gspca_dev, __s32 *val)
  255. {
  256. int err;
  257. u8 data[1];
  258. struct sd *sd = (struct sd *) gspca_dev;
  259. data[0] = 0x05;
  260. err = m5602_write_sensor(sd, S5K83A_PAGE_MAP, data, 1);
  261. if (err < 0)
  262. return err;
  263. err = m5602_read_sensor(sd, S5K83A_FLIP, data, 1);
  264. *val = (data[0] | 0x80) ? 1 : 0;
  265. return err;
  266. }
  267. int s5k83a_set_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. if (err < 0)
  278. return err;
  279. /* set or zero seven bit, six is vflip */
  280. data[0] = (val) ? (data[0] & 0x40) | 0x80 | S5K83A_FLIP_MASK
  281. : (data[0] & 0x40) | S5K83A_FLIP_MASK;
  282. err = m5602_write_sensor(sd, S5K83A_FLIP, data, 1);
  283. if (err < 0)
  284. return err;
  285. data[0] = (val) ? 0x0a : 0x0b;
  286. err = m5602_write_sensor(sd, S5K83A_HFLIP_TUNE, data, 1);
  287. return err;
  288. }
  289. int s5k83a_set_led_indication(struct sd *sd, u8 val)
  290. {
  291. int err = 0;
  292. u8 data[1];
  293. err = m5602_read_bridge(sd, M5602_XB_GPIO_DAT, data);
  294. if (err < 0)
  295. return err;
  296. if (val)
  297. data[0] = data[0] | S5K83A_GPIO_LED_MASK;
  298. else
  299. data[0] = data[0] & ~S5K83A_GPIO_LED_MASK;
  300. err = m5602_write_bridge(sd, M5602_XB_GPIO_DAT, data[0]);
  301. return (err < 0) ? err : 0;
  302. }