hpm_ov5640.c 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895
  1. /*
  2. * Copyright (c) 2021-2023 HPMicro
  3. *
  4. * SPDX-License-Identifier: BSD-3-Clause
  5. *
  6. */
  7. #include "hpm_ov5640.h"
  8. static const ov5640_reg_val_t ov5640_init_param[] = {
  9. {0x3008, 0x42},
  10. /* System setting. */
  11. {0x3103, 0x03},
  12. {0x3000, 0x00},
  13. {0x3004, 0xff},
  14. {0x3002, 0x1c},
  15. {0x3006, 0xc3},
  16. {0x302e, 0x08},
  17. {0x3037, 0x13},
  18. {0x3108, 0x01},
  19. {0x3618, 0x00},
  20. {0x3612, 0x29},
  21. {0x3708, 0x64},
  22. {0x3709, 0x52},
  23. {0x370c, 0x03},
  24. {0x3820, 0x41},
  25. {0x3821, 0x07},
  26. {0x3630, 0x36},
  27. {0x3631, 0x0e},
  28. {0x3632, 0xe2},
  29. {0x3633, 0x12},
  30. {0x3621, 0xe0},
  31. {0x3704, 0xa0},
  32. {0x3703, 0x5a},
  33. {0x3715, 0x78},
  34. {0x3717, 0x01},
  35. {0x370b, 0x60},
  36. {0x3705, 0x1a},
  37. {0x3905, 0x02},
  38. {0x3906, 0x10},
  39. {0x3901, 0x0a},
  40. {0x3731, 0x12},
  41. {0x3600, 0x08},
  42. {0x3601, 0x33},
  43. {0x302d, 0x60},
  44. {0x3620, 0x52},
  45. {0x371b, 0x20},
  46. {0x471c, 0x50},
  47. {0x3a13, 0x43},
  48. {0x3a18, 0x00},
  49. {0x3a19, 0x7c},
  50. {0x3635, 0x13},
  51. {0x3636, 0x03},
  52. {0x3634, 0x40},
  53. {0x3622, 0x01},
  54. {0x3c01, 0x00},
  55. {0x3a00, 0x58},
  56. {0x4001, 0x02},
  57. {0x4004, 0x02},
  58. {0x4005, 0x1a},
  59. {0x5001, 0xa3},
  60. /* AEC */
  61. {0x3a0f, 0x30},
  62. {0x3a10, 0x28},
  63. {0x3a1b, 0x30},
  64. {0x3a1e, 0x26},
  65. {0x3a11, 0x60},
  66. {0x3a1f, 0x14},
  67. /* AWB */
  68. {0x5180, 0xff},
  69. {0x5181, 0xf2},
  70. {0x5182, 0x00},
  71. {0x5183, 0x14},
  72. {0x5184, 0x25},
  73. {0x5185, 0x24},
  74. {0x5186, 0x09},
  75. {0x5187, 0x09},
  76. {0x5188, 0x09},
  77. {0x5189, 0x88},
  78. {0x518a, 0x54},
  79. {0x518b, 0xee},
  80. {0x518c, 0xb2},
  81. {0x518d, 0x50},
  82. {0x518e, 0x34},
  83. {0x518f, 0x6b},
  84. {0x5190, 0x46},
  85. {0x5191, 0xf8},
  86. {0x5192, 0x04},
  87. {0x5193, 0x70},
  88. {0x5194, 0xf0},
  89. {0x5195, 0xf0},
  90. {0x5196, 0x03},
  91. {0x5197, 0x01},
  92. {0x5198, 0x04},
  93. {0x5199, 0x6c},
  94. {0x519a, 0x04},
  95. {0x519b, 0x00},
  96. {0x519c, 0x09},
  97. {0x519d, 0x2b},
  98. {0x519e, 0x38},
  99. /* Color Matrix */
  100. {0x5381, 0x1e},
  101. {0x5382, 0x5b},
  102. {0x5383, 0x08},
  103. {0x5384, 0x0a},
  104. {0x5385, 0x7e},
  105. {0x5386, 0x88},
  106. {0x5387, 0x7c},
  107. {0x5388, 0x6c},
  108. {0x5389, 0x10},
  109. {0x538a, 0x01},
  110. {0x538b, 0x98},
  111. /* sharp */
  112. {0x5300, 0x08},
  113. {0x5301, 0x30},
  114. {0x5302, 0x10},
  115. {0x5303, 0x00},
  116. {0x5304, 0x08},
  117. {0x5305, 0x30},
  118. {0x5306, 0x08},
  119. {0x5307, 0x16},
  120. {0x5309, 0x08},
  121. {0x530a, 0x30},
  122. {0x530b, 0x04},
  123. {0x530c, 0x06},
  124. /* Gamma */
  125. {0x5480, 0x01},
  126. {0x5481, 0x08},
  127. {0x5482, 0x14},
  128. {0x5483, 0x28},
  129. {0x5484, 0x51},
  130. {0x5485, 0x65},
  131. {0x5486, 0x71},
  132. {0x5487, 0x7d},
  133. {0x5488, 0x87},
  134. {0x5489, 0x91},
  135. {0x548a, 0x9a},
  136. {0x548b, 0xaa},
  137. {0x548c, 0xb8},
  138. {0x548d, 0xcd},
  139. {0x548e, 0xdd},
  140. {0x548f, 0xea},
  141. {0x5490, 0x1d},
  142. /* UV adjust. */
  143. {0x5580, 0x02},
  144. {0x5583, 0x40},
  145. {0x5584, 0x10},
  146. {0x5589, 0x10},
  147. {0x558a, 0x00},
  148. {0x558b, 0xf8},
  149. /* Lens correction. */
  150. {0x5800, 0x23},
  151. {0x5801, 0x14},
  152. {0x5802, 0x0f},
  153. {0x5803, 0x0f},
  154. {0x5804, 0x12},
  155. {0x5805, 0x26},
  156. {0x5806, 0x0c},
  157. {0x5807, 0x08},
  158. {0x5808, 0x05},
  159. {0x5809, 0x05},
  160. {0x580a, 0x08},
  161. {0x580b, 0x0d},
  162. {0x580c, 0x08},
  163. {0x580d, 0x03},
  164. {0x580e, 0x00},
  165. {0x580f, 0x00},
  166. {0x5810, 0x03},
  167. {0x5811, 0x09},
  168. {0x5812, 0x07},
  169. {0x5813, 0x03},
  170. {0x5814, 0x00},
  171. {0x5815, 0x01},
  172. {0x5816, 0x03},
  173. {0x5817, 0x08},
  174. {0x5818, 0x0d},
  175. {0x5819, 0x08},
  176. {0x581a, 0x05},
  177. {0x581b, 0x06},
  178. {0x581c, 0x08},
  179. {0x581d, 0x0e},
  180. {0x581e, 0x29},
  181. {0x581f, 0x17},
  182. {0x5820, 0x11},
  183. {0x5821, 0x11},
  184. {0x5822, 0x15},
  185. {0x5823, 0x28},
  186. {0x5824, 0x46},
  187. {0x5825, 0x26},
  188. {0x5826, 0x08},
  189. {0x5827, 0x26},
  190. {0x5828, 0x64},
  191. {0x5829, 0x26},
  192. {0x582a, 0x24},
  193. {0x582b, 0x22},
  194. {0x582c, 0x24},
  195. {0x582d, 0x24},
  196. {0x582e, 0x06},
  197. {0x582f, 0x22},
  198. {0x5830, 0x40},
  199. {0x5831, 0x42},
  200. {0x5832, 0x24},
  201. {0x5833, 0x26},
  202. {0x5834, 0x24},
  203. {0x5835, 0x22},
  204. {0x5836, 0x22},
  205. {0x5837, 0x26},
  206. {0x5838, 0x44},
  207. {0x5839, 0x24},
  208. {0x583a, 0x26},
  209. {0x583b, 0x28},
  210. {0x583c, 0x42},
  211. {0x583d, 0xce},
  212. {0x481c, 0x00},
  213. {0x481d, 0x1},
  214. /* 50/60Hz detection */
  215. {0x3a02, 0x0b}, /* 60Hz max exposure, night mode 5fps */
  216. {0x3a03, 0x88}, /* 60Hz max exposure */
  217. {0x3a14, 0x0b}, /* 50Hz max exposure, night mode 5fps */
  218. {0x3a15, 0x88}, /* 50Hz max exposure */
  219. {0x3c01, 0x34}, /* Band auto, bit[7] */
  220. {0x3c04, 0x28}, /* threshold low sum */
  221. {0x3c05, 0x98}, /* threshold high sum */
  222. {0x3c06, 0x00}, /* light meter 1 threshold[15:8] */
  223. {0x3c07, 0x08}, /* light meter 1 threshold[7:0] */
  224. {0x3c08, 0x00}, /* light meter 2 threshold[15:8] */
  225. {0x3c09, 0x1c}, /* light meter 2 threshold[7:0] */
  226. {0x3c0a, 0x9c}, /* sample number[15:8] */
  227. {0x3c0b, 0x40}, /* sample number[7:0] */
  228. {0x3708, 0x64},
  229. {0x4001, 0x02}, /* BLC start from line 2 */
  230. {0x4005, 0x1a}, /* BLC always update */
  231. {0x3000, 0x00}, /* enable blocks */
  232. {0x3004, 0xff}, /* enable clocks */
  233. {0x302e, 0x00},
  234. {0x440e, 0x00},
  235. {0x5000, 0xa7}, /* Lenc on, raw gamma on, BPC on, WPC on, CIP on */
  236. };
  237. /* Resolution configuration */
  238. static const ov5640_reg_val_t ov5640_resolution_800_480_param[] = {
  239. {0x3800, 0x00}, {0x3801, 0x00}, {0x3802, 0x00}, {0x3803, 0x04},
  240. {0x3804, 0x0a}, {0x3805, 0x3f}, {0x3806, 0x07}, {0x3807, 0x9b},
  241. {0x3808, 0x03}, {0x3809, 0x20}, {0x380A, 0x01}, {0x380B, 0xe0},
  242. {0x380C, 0x07}, {0x380D, 0x68}, {0x380E, 0x03}, {0x380F, 0xd8},
  243. {0x3810, 0x00}, {0x3811, 0x10}, {0x3812, 0x00}, {0x3813, 0x06}, {0x3814, 0x31}, {0x3815, 0x31},
  244. };
  245. static const ov5640_reg_val_t ov5640_resolution_vga_param[] = {
  246. {0x3800, 0x00}, {0x3801, 0x00}, {0x3802, 0x00}, {0x3803, 0x04},
  247. {0x3804, 0x0a}, {0x3805, 0x3f}, {0x3806, 0x07}, {0x3807, 0x9b},
  248. {0x3808, 0x02}, {0x3809, 0x80}, {0x380A, 0x01}, {0x380B, 0xe0},
  249. {0x380C, 0x07}, {0x380D, 0x68}, {0x380E, 0x03}, {0x380F, 0xd8},
  250. {0x3810, 0x00}, {0x3811, 0x10}, {0x3812, 0x00}, {0x3813, 0x06}, {0x3814, 0x31}, {0x3815, 0x31},
  251. };
  252. static const ov5640_reg_val_t ov5640_resolution_qvga_param[] = {
  253. {0x3800, 0x00}, {0x3801, 0x00}, {0x3802, 0x00}, {0x3803, 0x04},
  254. {0x3804, 0x0a}, {0x3805, 0x3f}, {0x3806, 0x07}, {0x3807, 0x9b},
  255. {0x3808, 0x01}, {0x3809, 0x40}, {0x380A, 0x00}, {0x380B, 0xf0},
  256. {0x380C, 0x07}, {0x380D, 0x68}, {0x380E, 0x03}, {0x380F, 0xd8},
  257. {0x3810, 0x00}, {0x3811, 0x10}, {0x3812, 0x00}, {0x3813, 0x06}, {0x3814, 0x31}, {0x3815, 0x31},
  258. };
  259. static const ov5640_reg_val_t ov5640_resolution_480_272_param[] = {
  260. {0x3800, 0x00}, {0x3801, 0x00}, {0x3802, 0x00}, {0x3803, 0xfa},
  261. {0x3804, 0x0a}, {0x3805, 0x3f}, {0x3806, 0x06}, {0x3807, 0xa9},
  262. {0x3808, 0x01}, {0x3809, 0xE0}, {0x380A, 0x01}, {0x380B, 0x10},
  263. {0x380C, 0x07}, {0x380D, 0x64}, {0x380E, 0x02}, {0x380F, 0xe4},
  264. {0x3810, 0x00}, {0x3811, 0x10}, {0x3812, 0x00}, {0x3813, 0x04}, {0x3814, 0x31}, {0x3815, 0x31},
  265. };
  266. static const ov5640_reg_val_t ov5640_resolution_720p_param[] = {
  267. {0x3800, 0x00}, {0x3801, 0x00}, {0x3802, 0x00}, {0x3803, 0xfa},
  268. {0x3804, 0x0a}, {0x3805, 0x3f}, {0x3806, 0x06}, {0x3807, 0xa9},
  269. {0x3808, 0x05}, {0x3809, 0x00}, {0x380A, 0x02}, {0x380B, 0xd0},
  270. {0x380C, 0x07}, {0x380D, 0x64}, {0x380E, 0x02}, {0x380F, 0xe4},
  271. {0x3810, 0x00}, {0x3811, 0x10}, {0x3812, 0x00}, {0x3813, 0x04}, {0x3814, 0x31}, {0x3815, 0x31},
  272. };
  273. static const ov5640_reg_val_t ov5640_resolution_1080p_param[] = {
  274. {0x3800, 0x01}, {0x3801, 0x50}, {0x3802, 0x01}, {0x3803, 0xb2},
  275. {0x3804, 0x08}, {0x3805, 0xef}, {0x3806, 0x05}, {0x3807, 0xf1},
  276. {0x3808, 0x07}, {0x3809, 0x80}, {0x380A, 0x04}, {0x380B, 0x38},
  277. {0x380C, 0x09}, {0x380D, 0xc4}, {0x380E, 0x04}, {0x380F, 0x60},
  278. {0x3810, 0x00}, {0x3811, 0x10}, {0x3812, 0x00}, {0x3813, 0x04}, {0x3814, 0x11}, {0x3815, 0x11},
  279. };
  280. /* DVP clock */
  281. static const ov5640_clock_config_t ov5640_dvp_clock_configs[] = {
  282. {
  283. .resolution = (uint32_t)video_resolution_800_480,
  284. .pllctrl1 = 0x11,
  285. .pllctrl2 = 0x46,
  286. .vfifoctrl0c = 0x22,
  287. .pclkdiv = 0x02,
  288. .pclkperiod = 0x22,
  289. },
  290. {
  291. .resolution = (uint32_t)video_resolution_vga,
  292. .pllctrl1 = 0x11,
  293. .pllctrl2 = 0x46,
  294. .vfifoctrl0c = 0x22,
  295. .pclkdiv = 0x02,
  296. .pclkperiod = 0x22,
  297. },
  298. {
  299. .resolution = (uint32_t)video_resolution_qvga,
  300. .pllctrl1 = 0x11,
  301. .pllctrl2 = 0x46,
  302. .vfifoctrl0c = 0x22,
  303. .pclkdiv = 0x02,
  304. .pclkperiod = 0x22,
  305. },
  306. {
  307. .resolution = (uint32_t)video_resolution_480_272,
  308. .pllctrl1 = 0x21,
  309. .pllctrl2 = 0x69,
  310. .vfifoctrl0c = 0x20,
  311. .pclkdiv = 0x04,
  312. .pclkperiod = 0x16,
  313. },
  314. {
  315. .resolution = (uint32_t)video_resolution_720p,
  316. .pllctrl1 = 0x21,
  317. .pllctrl2 = 0x69,
  318. .vfifoctrl0c = 0x20,
  319. .pclkdiv = 0x04,
  320. .pclkperiod = 0x16,
  321. },
  322. {
  323. .resolution = (uint32_t)video_resolution_1080p,
  324. .pllctrl1 = 0x21,
  325. .pllctrl2 = 0x69,
  326. .vfifoctrl0c = 0x20,
  327. .pclkdiv = 0x04,
  328. .pclkperiod = 0x16,
  329. },
  330. };
  331. /* MIPI clock */
  332. static const ov5640_clock_config_t ov5640_mipi_clock_configs[] = {
  333. {
  334. .resolution = (uint32_t)video_resolution_800_480,
  335. .pllctrl1 = 0x14,
  336. .pllctrl2 = 0x38,
  337. .vfifoctrl0c = 0x22,
  338. .pclkdiv = 0x02,
  339. .pclkperiod = 0x0a,
  340. },
  341. {
  342. .resolution = (uint32_t)video_resolution_vga,
  343. .pllctrl1 = 0x14,
  344. .pllctrl2 = 0x38,
  345. .vfifoctrl0c = 0x22,
  346. .pclkdiv = 0x02,
  347. .pclkperiod = 0x0a,
  348. },
  349. {
  350. .resolution = (uint32_t)video_resolution_qvga,
  351. .pllctrl1 = 0x14,
  352. .pllctrl2 = 0x38,
  353. .vfifoctrl0c = 0x22,
  354. .pclkdiv = 0x02,
  355. .pclkperiod = 0x0a,
  356. },
  357. {
  358. .resolution = (uint32_t)video_resolution_480_272,
  359. .pllctrl1 = 0x14,
  360. .pllctrl2 = 0x38,
  361. .vfifoctrl0c = 0x22,
  362. .pclkdiv = 0x02,
  363. .pclkperiod = 0x0a,
  364. },
  365. {
  366. .resolution = (uint32_t)video_resolution_720p,
  367. .pllctrl1 = 0x21,
  368. .pllctrl2 = 0x54,
  369. .vfifoctrl0c = 0x20,
  370. .pclkdiv = 0x04,
  371. .pclkperiod = 0x0a,
  372. },
  373. {
  374. .resolution = (uint32_t)video_resolution_1080p,
  375. .pllctrl1 = 0x11,
  376. .pllctrl2 = 0x54,
  377. .vfifoctrl0c = 0x20,
  378. .pclkdiv = 0x04,
  379. .pclkperiod = 0x0a,
  380. },
  381. };
  382. static const ov5640_light_mode_config_t ov5640_light_mode_configs[] = {
  383. /* Auto. */
  384. {
  385. .lightmode = camera_light_mode_auto,
  386. .awbctrl = 0x00,
  387. .awbr_h = 0x04,
  388. .awbr_l = 0x00,
  389. .awbg_h = 0x04,
  390. .awbg_l = 0x00,
  391. .awbb_h = 0x04,
  392. .awbb_l = 0x00,
  393. },
  394. /* Sunny. */
  395. {
  396. .lightmode = camera_light_mode_sunny,
  397. .awbctrl = 0x01,
  398. .awbr_h = 0x06,
  399. .awbr_l = 0x1c,
  400. .awbg_h = 0x04,
  401. .awbg_l = 0x00,
  402. .awbb_h = 0x04,
  403. .awbb_l = 0xf3,
  404. },
  405. /* Office. */
  406. {
  407. .lightmode = camera_light_mode_office,
  408. .awbctrl = 0x01,
  409. .awbr_h = 0x05,
  410. .awbr_l = 0x48,
  411. .awbg_h = 0x04,
  412. .awbg_l = 0x00,
  413. .awbb_h = 0x07,
  414. .awbb_l = 0xcf,
  415. },
  416. /* Cloudy. */
  417. {
  418. .lightmode = camera_light_mode_cloudy,
  419. .awbctrl = 0x01,
  420. .awbr_h = 0x06,
  421. .awbr_l = 0x48,
  422. .awbg_h = 0x04,
  423. .awbg_l = 0x00,
  424. .awbb_h = 0x04,
  425. .awbb_l = 0xd3,
  426. },
  427. /* Home. */
  428. {
  429. .lightmode = camera_light_mode_home,
  430. .awbctrl = 0x01,
  431. .awbr_h = 0x04,
  432. .awbr_l = 0x10,
  433. .awbg_h = 0x04,
  434. .awbg_l = 0x00,
  435. .awbb_h = 0x08,
  436. .awbb_l = 0x40,
  437. },
  438. };
  439. static const ov5640_special_effect_config_t ov5640_special_effect_configs[] = {
  440. /* Normal. */
  441. {
  442. .effect = camera_special_effect_normal,
  443. .sdectrl0 = 0x06,
  444. .sdectrl3 = 0x40,
  445. .sdectrl4 = 0x10,
  446. },
  447. /* Bluish. */
  448. {
  449. .effect = camera_special_effect_bluish,
  450. .sdectrl0 = 0x1e,
  451. .sdectrl3 = 0xa0,
  452. .sdectrl4 = 0x40,
  453. },
  454. /* Redish. */
  455. {
  456. .effect = camera_special_effect_redish,
  457. .sdectrl0 = 0x1e,
  458. .sdectrl3 = 0x80,
  459. .sdectrl4 = 0xc0,
  460. },
  461. /* B & W */
  462. {
  463. .effect = camera_special_effect_bw,
  464. .sdectrl0 = 0x1e,
  465. .sdectrl3 = 0x80,
  466. .sdectrl4 = 0x80,
  467. },
  468. /* Sepia. */
  469. {
  470. .effect = camera_special_effect_sepia,
  471. .sdectrl0 = 0x1e,
  472. .sdectrl3 = 0x40,
  473. .sdectrl4 = 0xa0,
  474. },
  475. /* Negtive. */
  476. {
  477. .effect = camera_special_effect_negtive,
  478. .sdectrl0 = 0x40,
  479. .sdectrl3 = 0x40,
  480. .sdectrl4 = 0x10,
  481. },
  482. /* Greenish. */
  483. {
  484. .effect = camera_special_effect_greenish,
  485. .sdectrl0 = 0x1e,
  486. .sdectrl3 = 0x60,
  487. .sdectrl4 = 0x60,
  488. },
  489. };
  490. static const ov5640_clock_config_t *ov5640_get_clock_config(const camera_config_t *config)
  491. {
  492. uint32_t i;
  493. if (camera_interface_dvp == config->interface) {
  494. for (i = 0; i < ARRAY_SIZE(ov5640_dvp_clock_configs); i++) {
  495. if (HPM_CAMERA_RESOLUTION(config->width, config->height) == ov5640_dvp_clock_configs[i].resolution) {
  496. return &ov5640_dvp_clock_configs[i];
  497. }
  498. }
  499. } else if (camera_interface_mipi == config->interface) {
  500. for (i = 0; i < ARRAY_SIZE(ov5640_mipi_clock_configs); i++) {
  501. if (HPM_CAMERA_RESOLUTION(config->width, config->height) == ov5640_mipi_clock_configs[i].resolution) {
  502. return &ov5640_mipi_clock_configs[i];
  503. }
  504. }
  505. }
  506. return NULL;
  507. }
  508. hpm_stat_t ov5640_read_register(camera_context_t *context, uint16_t reg, uint8_t *buf)
  509. {
  510. uint8_t r[2];
  511. r[0] = reg >> 8;
  512. r[1] = reg & 0xFF;
  513. hpm_stat_t stat = i2c_master_write(context->ptr, context->i2c_device_addr, r, 2);
  514. if (stat != status_success) {
  515. return stat;
  516. }
  517. return i2c_master_read(context->ptr, context->i2c_device_addr, buf, 1);
  518. }
  519. hpm_stat_t ov5640_write_register(camera_context_t *context, uint16_t reg, uint8_t val)
  520. {
  521. uint8_t r[2];
  522. r[0] = reg >> 8;
  523. r[1] = reg & 0xFF;
  524. return i2c_master_address_write(context->ptr, context->i2c_device_addr, r, sizeof(r), &val, 1);
  525. }
  526. hpm_stat_t ov5640_write_multi_registers(camera_context_t *context, const ov5640_reg_val_t regval[], uint32_t len)
  527. {
  528. uint32_t i;
  529. hpm_stat_t stat = status_success;
  530. for (i = 0; i < len; i++) {
  531. HPM_CHECK_RET(ov5640_write_register(context, regval[i].regaddr, regval[i].regval));
  532. }
  533. return stat;
  534. }
  535. hpm_stat_t ov5640_software_reset(camera_context_t *context)
  536. {
  537. hpm_stat_t stat = status_success;
  538. HPM_CHECK_RET(ov5640_write_register(context, 0x3103, 0x11));
  539. HPM_CHECK_RET(ov5640_write_register(context, 0x3008, 0x82));
  540. return stat;
  541. }
  542. hpm_stat_t ov5640_set_pixel_format(camera_context_t *context, display_pixel_format_t pixel_format)
  543. {
  544. hpm_stat_t stat = status_success;
  545. switch (pixel_format) {
  546. case display_pixel_format_y8:
  547. case display_pixel_format_yuv422:
  548. HPM_CHECK_RET(ov5640_write_register(context, 0x4300, 0x30));
  549. HPM_CHECK_RET(ov5640_write_register(context, 0x501f, 0x00));
  550. break;
  551. case display_pixel_format_ycbcr422:
  552. HPM_CHECK_RET(ov5640_write_register(context, 0x4300, 0x32));
  553. HPM_CHECK_RET(ov5640_write_register(context, 0x501f, 0x00));
  554. break;
  555. case display_pixel_format_rgb565:
  556. HPM_CHECK_RET(ov5640_write_register(context, 0x4300, 0x6F));
  557. HPM_CHECK_RET(ov5640_write_register(context, 0x501f, 0x01));
  558. break;
  559. default:
  560. stat = status_invalid_argument;
  561. break;
  562. }
  563. return stat;
  564. }
  565. hpm_stat_t ov5640_check_chip_id(camera_context_t *context)
  566. {
  567. hpm_stat_t stat = status_success;
  568. uint8_t val_h = 0;
  569. uint8_t val_l = 0;
  570. HPM_CHECK_RET(ov5640_read_register(context, OV5640_CHIP_ID_HIGH_BYTE_ADDR, &val_h));
  571. HPM_CHECK_RET(ov5640_read_register(context, OV5640_CHIP_ID_LOW_BYTE_ADDR, &val_l));
  572. if (val_h != OV5640_CHIP_ID_HIGH_BYTE_VALUE) {
  573. return status_fail;
  574. }
  575. if (val_l != OV5640_CHIP_ID_LOW_BYTE_VALUE) {
  576. return status_fail;
  577. }
  578. return stat;
  579. }
  580. hpm_stat_t ov5640_set_image_size(camera_context_t *context, camera_config_t *ov_config)
  581. {
  582. hpm_stat_t stat = status_success;
  583. switch (HPM_CAMERA_RESOLUTION(ov_config->width, ov_config->height)) {
  584. case video_resolution_800_480:
  585. stat = ov5640_write_multi_registers(context, ov5640_resolution_800_480_param, ARRAY_SIZE(ov5640_resolution_800_480_param));
  586. break;
  587. case video_resolution_vga:
  588. stat = ov5640_write_multi_registers(context, ov5640_resolution_vga_param, ARRAY_SIZE(ov5640_resolution_vga_param));
  589. break;
  590. case video_resolution_qvga:
  591. stat = ov5640_write_multi_registers(context, ov5640_resolution_qvga_param, ARRAY_SIZE(ov5640_resolution_qvga_param));
  592. break;
  593. case video_resolution_480_272:
  594. stat = ov5640_write_multi_registers(context, ov5640_resolution_480_272_param, ARRAY_SIZE(ov5640_resolution_480_272_param));
  595. break;
  596. case video_resolution_720p:
  597. stat = ov5640_write_multi_registers(context, ov5640_resolution_720p_param, ARRAY_SIZE(ov5640_resolution_720p_param));
  598. break;
  599. case video_resolution_1080p:
  600. stat = ov5640_write_multi_registers(context, ov5640_resolution_1080p_param, ARRAY_SIZE(ov5640_resolution_1080p_param));
  601. break;
  602. default:
  603. stat = status_invalid_argument;
  604. break;
  605. }
  606. return stat;
  607. }
  608. hpm_stat_t ov5640_set_clock_config(camera_context_t *context, camera_config_t *ov_config)
  609. {
  610. hpm_stat_t stat = status_success;
  611. const ov5640_clock_config_t *clock_config = ov5640_get_clock_config(ov_config);
  612. if (NULL == clock_config) {
  613. return status_invalid_argument;
  614. }
  615. HPM_CHECK_RET(ov5640_write_register(context, 0x3035, clock_config->pllctrl1));
  616. HPM_CHECK_RET(ov5640_write_register(context, 0x3036, clock_config->pllctrl2));
  617. HPM_CHECK_RET(ov5640_write_register(context, 0x460c, clock_config->vfifoctrl0c));
  618. HPM_CHECK_RET(ov5640_write_register(context, 0x3824, clock_config->pclkdiv));
  619. HPM_CHECK_RET(ov5640_write_register(context, 0x4837, clock_config->pclkperiod));
  620. return stat;
  621. }
  622. hpm_stat_t ov5640_set_interface(camera_context_t *context, camera_config_t *ov_config)
  623. {
  624. hpm_stat_t stat = status_success;
  625. if (camera_interface_dvp == ov_config->interface) {
  626. HPM_CHECK_RET(ov5640_write_register(context, 0x3034, 0x1a));
  627. /* Set Frex, Vsync, Href, PCLK, data, GPIO to output. */
  628. HPM_CHECK_RET(ov5640_write_register(context, 0x3017, 0xFF));
  629. HPM_CHECK_RET(ov5640_write_register(context, 0x3018, 0xFF));
  630. /* DVP mode */
  631. HPM_CHECK_RET(ov5640_write_register(context, 0x300e, 0x58));
  632. } else if (camera_interface_mipi == ov_config->interface) {
  633. HPM_CHECK_RET(ov5640_write_register(context, 0x481D, 0x20));
  634. HPM_CHECK_RET(ov5640_write_register(context, 0x481C, 0x0));
  635. HPM_CHECK_RET(ov5640_write_register(context, 0x3034, 0x18));
  636. HPM_CHECK_RET(ov5640_write_register(context, 0x3017, 0x00));
  637. HPM_CHECK_RET(ov5640_write_register(context, 0x3018, 0x00));
  638. /* 2lanes mode */
  639. HPM_CHECK_RET(ov5640_write_register(context, 0x300e, 0x45));
  640. HPM_CHECK_RET(ov5640_write_register(context, 0x302e, 0x08));
  641. HPM_CHECK_RET(ov5640_write_register(context, 0x4800, 0x04));
  642. }
  643. return stat;
  644. }
  645. hpm_stat_t ov5640_start(camera_context_t *context)
  646. {
  647. return ov5640_write_register(context, 0x3008, 0x02);
  648. }
  649. hpm_stat_t ov5640_stop(camera_context_t *context)
  650. {
  651. return ov5640_write_register(context, 0x3008, 0x42);
  652. }
  653. hpm_stat_t ov5640_flip(camera_context_t *context)
  654. {
  655. hpm_stat_t stat = status_success;
  656. HPM_CHECK_RET(ov5640_write_register(context, 0x3821, 1));
  657. return stat;
  658. }
  659. hpm_stat_t ov5640_set_brightness(camera_context_t *context, int32_t brightness)
  660. {
  661. hpm_stat_t stat = status_success;
  662. if ((brightness < -4) || (brightness > 4)) {
  663. return status_invalid_argument;
  664. }
  665. HPM_CHECK_RET(ov5640_write_register(context, 0x3212, 0x03));
  666. if (brightness >= 0) {
  667. HPM_CHECK_RET(ov5640_write_register(context, 0x5588, 0x01));
  668. } else {
  669. brightness = -brightness;
  670. HPM_CHECK_RET(ov5640_write_register(context, 0x5588, 0x09));
  671. }
  672. HPM_CHECK_RET(ov5640_write_register(context, 0x5587, ((uint8_t)brightness) << 4U));
  673. HPM_CHECK_RET(ov5640_write_register(context, 0x3212, 0x13));
  674. HPM_CHECK_RET(ov5640_write_register(context, 0x3212, 0xa3));
  675. return stat;
  676. }
  677. hpm_stat_t ov5640_set_contrast(camera_context_t *context, int32_t contrast)
  678. {
  679. hpm_stat_t stat = status_success;
  680. uint8_t regval;
  681. if ((-4 > contrast) || (4 < contrast)) {
  682. return status_invalid_argument;
  683. }
  684. contrast = 0x20 + contrast * 0x04;
  685. regval = (uint8_t)contrast;
  686. HPM_CHECK_RET(ov5640_write_register(context, 0x3212, 0x03));
  687. HPM_CHECK_RET(ov5640_write_register(context, 0x5580, 0x04));
  688. HPM_CHECK_RET(ov5640_write_register(context, 0x5585, regval));
  689. HPM_CHECK_RET(ov5640_write_register(context, 0x5586, regval));
  690. HPM_CHECK_RET(ov5640_write_register(context, 0x3212, 0x13));
  691. HPM_CHECK_RET(ov5640_write_register(context, 0x3212, 0xa3));
  692. return stat;
  693. }
  694. hpm_stat_t ov5640_set_saturation(camera_context_t *context, int32_t saturation)
  695. {
  696. hpm_stat_t stat = status_success;
  697. uint8_t regval;
  698. if ((-4 > saturation) || (4 < saturation)) {
  699. return status_invalid_argument;
  700. }
  701. saturation = 0x40 + saturation * 0x10;
  702. regval = (uint8_t)saturation;
  703. HPM_CHECK_RET(ov5640_write_register(context, 0x3212, 0x03));
  704. HPM_CHECK_RET(ov5640_write_register(context, 0x5580, 0x02));
  705. HPM_CHECK_RET(ov5640_write_register(context, 0x5583, regval));
  706. HPM_CHECK_RET(ov5640_write_register(context, 0x5584, regval));
  707. HPM_CHECK_RET(ov5640_write_register(context, 0x5588, 0x41));
  708. HPM_CHECK_RET(ov5640_write_register(context, 0x3212, 0x13));
  709. HPM_CHECK_RET(ov5640_write_register(context, 0x3212, 0xa3));
  710. return stat;
  711. }
  712. hpm_stat_t ov5640_set_light_mode(camera_context_t *context, int32_t lightmode)
  713. {
  714. hpm_stat_t stat = status_success;
  715. uint8_t i;
  716. for (i = 0; i < ARRAY_SIZE(ov5640_light_mode_configs); i++) {
  717. if (lightmode == (int32_t)ov5640_light_mode_configs[i].lightmode) {
  718. HPM_CHECK_RET(ov5640_write_register(context, 0x3212, 0x03));
  719. HPM_CHECK_RET(ov5640_write_register(context, 0x3406, ov5640_light_mode_configs[i].awbctrl));
  720. HPM_CHECK_RET(ov5640_write_register(context, 0x3400, ov5640_light_mode_configs[i].awbr_h));
  721. HPM_CHECK_RET(ov5640_write_register(context, 0x3401, ov5640_light_mode_configs[i].awbr_l));
  722. HPM_CHECK_RET(ov5640_write_register(context, 0x3402, ov5640_light_mode_configs[i].awbg_h));
  723. HPM_CHECK_RET(ov5640_write_register(context, 0x3403, ov5640_light_mode_configs[i].awbg_l));
  724. HPM_CHECK_RET(ov5640_write_register(context, 0x3404, ov5640_light_mode_configs[i].awbb_h));
  725. HPM_CHECK_RET(ov5640_write_register(context, 0x3405, ov5640_light_mode_configs[i].awbb_l));
  726. HPM_CHECK_RET(ov5640_write_register(context, 0x3212, 0x13));
  727. HPM_CHECK_RET(ov5640_write_register(context, 0x3212, 0xa3));
  728. return stat;
  729. }
  730. }
  731. /* No configuration found. */
  732. return status_invalid_argument;
  733. }
  734. hpm_stat_t ov5640_set_special_effect(camera_context_t *context, int32_t effect)
  735. {
  736. hpm_stat_t stat = status_success;
  737. uint8_t i;
  738. for (i = 0; i < ARRAY_SIZE(ov5640_special_effect_configs); i++) {
  739. if (effect == (int32_t)ov5640_special_effect_configs[i].effect) {
  740. HPM_CHECK_RET(ov5640_write_register(context, 0x3212, 0x03));
  741. HPM_CHECK_RET(ov5640_write_register(context, 0x5580, ov5640_special_effect_configs[i].sdectrl0));
  742. HPM_CHECK_RET(ov5640_write_register(context, 0x5583, ov5640_special_effect_configs[i].sdectrl3));
  743. HPM_CHECK_RET(ov5640_write_register(context, 0x5584, ov5640_special_effect_configs[i].sdectrl4));
  744. HPM_CHECK_RET(ov5640_write_register(context, 0x5003, 0x08));
  745. HPM_CHECK_RET(ov5640_write_register(context, 0x3212, 0x13));
  746. HPM_CHECK_RET(ov5640_write_register(context, 0x3212, 0xa3));
  747. return stat;
  748. }
  749. }
  750. /* No configuration found. */
  751. return status_invalid_argument;
  752. }
  753. hpm_stat_t ov5640_init(camera_context_t *context, camera_config_t *ov_config)
  754. {
  755. hpm_stat_t stat = status_success;
  756. /* check the chip id */
  757. HPM_CHECK_RET(ov5640_check_chip_id(context));
  758. /* Initialize: load registers value */
  759. HPM_CHECK_RET(ov5640_write_multi_registers(context, ov5640_init_param, ARRAY_SIZE(ov5640_init_param)));
  760. /* configure image windowing */
  761. HPM_CHECK_RET(ov5640_set_image_size(context, ov_config));
  762. HPM_CHECK_RET(ov5640_flip(context));
  763. /* configure Pixel format */
  764. HPM_CHECK_RET(ov5640_set_pixel_format(context, ov_config->pixel_format));
  765. /* configure PCLK clock */
  766. HPM_CHECK_RET(ov5640_set_clock_config(context, ov_config));
  767. /* configure interface */
  768. HPM_CHECK_RET(ov5640_set_interface(context, ov_config));
  769. /* configure contrast */
  770. HPM_CHECK_RET(ov5640_set_contrast(context, 2));
  771. /* camera start */
  772. HPM_CHECK_RET(ov5640_write_register(context, 0x3008, 0x02));
  773. return stat;
  774. }
  775. void ov5640_power_up(camera_context_t *context)
  776. {
  777. assert(context->delay_ms != NULL);
  778. if (context->write_rst) {
  779. context->write_rst(OV5640_RST_ACTIVE);
  780. }
  781. if (context->write_pwdn) {
  782. context->write_pwdn(OV5640_PWDN_ACTIVE);
  783. }
  784. context->delay_ms(5);
  785. if (context->write_pwdn) {
  786. context->write_pwdn(OV5640_PWDN_INACTIVE);
  787. }
  788. context->delay_ms(2);
  789. if (context->write_rst) {
  790. context->write_rst(OV5640_RST_INACTIVE);
  791. }
  792. context->delay_ms(20);
  793. }