m5602_s5k83a.c 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354
  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 = s5k83a.nctrls;
  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_power_down(struct sd *sd)
  94. {
  95. return 0;
  96. }
  97. void s5k83a_dump_registers(struct sd *sd)
  98. {
  99. int address;
  100. u8 page, old_page;
  101. m5602_read_sensor(sd, S5K83A_PAGE_MAP, &old_page, 1);
  102. for (page = 0; page < 16; page++) {
  103. m5602_write_sensor(sd, S5K83A_PAGE_MAP, &page, 1);
  104. info("Dumping the s5k83a register state for page 0x%x", page);
  105. for (address = 0; address <= 0xff; address++) {
  106. u8 val = 0;
  107. m5602_read_sensor(sd, address, &val, 1);
  108. info("register 0x%x contains 0x%x",
  109. address, val);
  110. }
  111. }
  112. info("s5k83a register state dump complete");
  113. for (page = 0; page < 16; page++) {
  114. m5602_write_sensor(sd, S5K83A_PAGE_MAP, &page, 1);
  115. info("Probing for which registers that are read/write "
  116. "for page 0x%x", page);
  117. for (address = 0; address <= 0xff; address++) {
  118. u8 old_val, ctrl_val, test_val = 0xff;
  119. m5602_read_sensor(sd, address, &old_val, 1);
  120. m5602_write_sensor(sd, address, &test_val, 1);
  121. m5602_read_sensor(sd, address, &ctrl_val, 1);
  122. if (ctrl_val == test_val)
  123. info("register 0x%x is writeable", address);
  124. else
  125. info("register 0x%x is read only", address);
  126. /* Restore original val */
  127. m5602_write_sensor(sd, address, &old_val, 1);
  128. }
  129. }
  130. info("Read/write register probing complete");
  131. m5602_write_sensor(sd, S5K83A_PAGE_MAP, &old_page, 1);
  132. }
  133. int s5k83a_get_brightness(struct gspca_dev *gspca_dev, __s32 *val)
  134. {
  135. int err;
  136. u8 data[2];
  137. struct sd *sd = (struct sd *) gspca_dev;
  138. err = m5602_read_sensor(sd, S5K83A_BRIGHTNESS, data, 2);
  139. if (err < 0)
  140. goto out;
  141. data[1] = data[1] << 1;
  142. *val = data[1];
  143. out:
  144. return err;
  145. }
  146. int s5k83a_set_brightness(struct gspca_dev *gspca_dev, __s32 val)
  147. {
  148. int err;
  149. u8 data[2];
  150. struct sd *sd = (struct sd *) gspca_dev;
  151. data[0] = 0x00;
  152. data[1] = 0x20;
  153. err = m5602_write_sensor(sd, 0x14, data, 2);
  154. if (err < 0)
  155. goto out;
  156. data[0] = 0x01;
  157. data[1] = 0x00;
  158. err = m5602_write_sensor(sd, 0x0d, data, 2);
  159. if (err < 0)
  160. goto out;
  161. /* FIXME: This is not sane, we need to figure out the composition
  162. of these registers */
  163. data[0] = val >> 3; /* brightness, high 5 bits */
  164. data[1] = val >> 1; /* brightness, high 7 bits */
  165. err = m5602_write_sensor(sd, S5K83A_BRIGHTNESS, data, 2);
  166. out:
  167. return err;
  168. }
  169. int s5k83a_get_whiteness(struct gspca_dev *gspca_dev, __s32 *val)
  170. {
  171. int err;
  172. u8 data;
  173. struct sd *sd = (struct sd *) gspca_dev;
  174. err = m5602_read_sensor(sd, S5K83A_WHITENESS, &data, 1);
  175. if (err < 0)
  176. goto out;
  177. *val = data;
  178. out:
  179. return err;
  180. }
  181. int s5k83a_set_whiteness(struct gspca_dev *gspca_dev, __s32 val)
  182. {
  183. int err;
  184. u8 data[1];
  185. struct sd *sd = (struct sd *) gspca_dev;
  186. data[0] = val;
  187. err = m5602_write_sensor(sd, S5K83A_WHITENESS, data, 1);
  188. return err;
  189. }
  190. int s5k83a_get_gain(struct gspca_dev *gspca_dev, __s32 *val)
  191. {
  192. int err;
  193. u8 data[2];
  194. struct sd *sd = (struct sd *) gspca_dev;
  195. err = m5602_read_sensor(sd, S5K83A_GAIN, data, 2);
  196. if (err < 0)
  197. goto out;
  198. data[1] = data[1] & 0x3f;
  199. if (data[1] > S5K83A_MAXIMUM_GAIN)
  200. data[1] = S5K83A_MAXIMUM_GAIN;
  201. *val = data[1];
  202. out:
  203. return err;
  204. }
  205. int s5k83a_set_gain(struct gspca_dev *gspca_dev, __s32 val)
  206. {
  207. int err;
  208. u8 data[2];
  209. struct sd *sd = (struct sd *) gspca_dev;
  210. data[0] = 0;
  211. data[1] = val;
  212. err = m5602_write_sensor(sd, S5K83A_GAIN, data, 2);
  213. return err;
  214. }
  215. int s5k83a_get_vflip(struct gspca_dev *gspca_dev, __s32 *val)
  216. {
  217. int err;
  218. u8 data[1];
  219. struct sd *sd = (struct sd *) gspca_dev;
  220. data[0] = 0x05;
  221. err = m5602_write_sensor(sd, S5K83A_PAGE_MAP, data, 1);
  222. if (err < 0)
  223. goto out;
  224. err = m5602_read_sensor(sd, S5K83A_FLIP, data, 1);
  225. *val = (data[0] | 0x40) ? 1 : 0;
  226. out:
  227. return err;
  228. }
  229. int s5k83a_set_vflip(struct gspca_dev *gspca_dev, __s32 val)
  230. {
  231. int err;
  232. u8 data[1];
  233. struct sd *sd = (struct sd *) gspca_dev;
  234. data[0] = 0x05;
  235. err = m5602_write_sensor(sd, S5K83A_PAGE_MAP, data, 1);
  236. if (err < 0)
  237. goto out;
  238. err = m5602_read_sensor(sd, S5K83A_FLIP, data, 1);
  239. if (err < 0)
  240. goto out;
  241. /* set or zero six bit, seven is hflip */
  242. data[0] = (val) ? (data[0] & 0x80) | 0x40 | S5K83A_FLIP_MASK
  243. : (data[0] & 0x80) | S5K83A_FLIP_MASK;
  244. err = m5602_write_sensor(sd, S5K83A_FLIP, data, 1);
  245. if (err < 0)
  246. goto out;
  247. data[0] = (val) ? 0x0b : 0x0a;
  248. err = m5602_write_sensor(sd, S5K83A_VFLIP_TUNE, data, 1);
  249. out:
  250. return err;
  251. }
  252. int s5k83a_get_hflip(struct gspca_dev *gspca_dev, __s32 *val)
  253. {
  254. int err;
  255. u8 data[1];
  256. struct sd *sd = (struct sd *) gspca_dev;
  257. data[0] = 0x05;
  258. err = m5602_write_sensor(sd, S5K83A_PAGE_MAP, data, 1);
  259. if (err < 0)
  260. goto out;
  261. err = m5602_read_sensor(sd, S5K83A_FLIP, data, 1);
  262. *val = (data[0] | 0x80) ? 1 : 0;
  263. out:
  264. return err;
  265. }
  266. int s5k83a_set_hflip(struct gspca_dev *gspca_dev, __s32 val)
  267. {
  268. int err;
  269. u8 data[1];
  270. struct sd *sd = (struct sd *) gspca_dev;
  271. data[0] = 0x05;
  272. err = m5602_write_sensor(sd, S5K83A_PAGE_MAP, data, 1);
  273. if (err < 0)
  274. goto out;
  275. err = m5602_read_sensor(sd, S5K83A_FLIP, data, 1);
  276. if (err < 0)
  277. goto out;
  278. /* set or zero seven bit, six is vflip */
  279. data[0] = (val) ? (data[0] & 0x40) | 0x80 | S5K83A_FLIP_MASK
  280. : (data[0] & 0x40) | S5K83A_FLIP_MASK;
  281. err = m5602_write_sensor(sd, S5K83A_FLIP, data, 1);
  282. if (err < 0)
  283. goto out;
  284. data[0] = (val) ? 0x0a : 0x0b;
  285. err = m5602_write_sensor(sd, S5K83A_HFLIP_TUNE, data, 1);
  286. out:
  287. return err;
  288. }