cam_cdm_util.c 35 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2017-2021, The Linux Foundation. All rights reserved.
  4. * Copyright (c) 2022-2024, Qualcomm Innovation Center, Inc. All rights reserved.
  5. */
  6. #include <linux/types.h>
  7. #include <linux/kernel.h>
  8. #include <linux/errno.h>
  9. #include <linux/bug.h>
  10. #include "cam_cdm_intf_api.h"
  11. #include "cam_cdm_util.h"
  12. #include "cam_cdm.h"
  13. #include "cam_io_util.h"
  14. #define CAM_CDM_DWORD 4
  15. #define CAM_CDM_SW_CMD_COUNT 2
  16. #define CAM_CMD_LENGTH_MASK 0xFFFF
  17. #define CAM_CDM_REG_OFFSET_MASK 0x00FFFFFF
  18. #define CAM_CDM_DMI_DATA_HI_OFFSET 8
  19. #define CAM_CDM_DMI_DATA_OFFSET 8
  20. #define CAM_CDM_DMI_DATA_LO_OFFSET 12
  21. static unsigned int CDMCmdHeaderSizes[
  22. CAM_CDM_CMD_PRIVATE_BASE + CAM_CDM_SW_CMD_COUNT] = {
  23. 0, /* UNUSED*/
  24. 3, /* DMI*/
  25. 0, /* UNUSED*/
  26. 2, /* RegContinuous*/
  27. 1, /* RegRandom*/
  28. 2, /* BUFFER_INDIREC*/
  29. 2, /* GenerateIRQ*/
  30. 3, /* WaitForEvent*/
  31. 1, /* ChangeBase*/
  32. 1, /* PERF_CONTROL*/
  33. 3, /* DMI32*/
  34. 3, /* DMI64*/
  35. 3, /* WaitCompEvent*/
  36. 3, /* ClearCompEvent*/
  37. 3, /* WaitPrefetchDisable*/
  38. };
  39. /**
  40. * struct cdm_regrandom_cmd - Definition for CDM random register command.
  41. * @count: Number of register writes
  42. * @reserved: reserved bits
  43. * @cmd: Command ID (CDMCmd)
  44. */
  45. struct cdm_regrandom_cmd {
  46. unsigned int count : 16;
  47. unsigned int reserved : 8;
  48. unsigned int cmd : 8;
  49. } __attribute__((__packed__));
  50. /**
  51. * struct cdm_regcontinuous_cmd - Definition for a CDM register range command.
  52. * @count: Number of register writes
  53. * @reserved0: reserved bits
  54. * @cmd: Command ID (CDMCmd)
  55. * @offset: Start address of the range of registers
  56. * @reserved1: reserved bits
  57. */
  58. struct cdm_regcontinuous_cmd {
  59. unsigned int count : 16;
  60. unsigned int reserved0 : 8;
  61. unsigned int cmd : 8;
  62. unsigned int offset : 24;
  63. unsigned int reserved1 : 8;
  64. } __attribute__((__packed__));
  65. /**
  66. * struct cdm_dmi_cmd - Definition for a CDM DMI command.
  67. * @length: Number of bytes in LUT - 1
  68. * @reserved: reserved bits
  69. * @cmd: Command ID (CDMCmd)
  70. * @addr: Address of the LUT in memory
  71. * @DMIAddr: Address of the target DMI config register
  72. * @DMISel: DMI identifier
  73. */
  74. struct cdm_dmi_cmd {
  75. unsigned int length : 16;
  76. unsigned int reserved : 8;
  77. unsigned int cmd : 8;
  78. unsigned int addr;
  79. unsigned int DMIAddr : 24;
  80. unsigned int DMISel : 8;
  81. } __attribute__((__packed__));
  82. /**
  83. * struct cdm_indirect_cmd - Definition for a CDM indirect buffer command.
  84. * @length: Number of bytes in buffer - 1
  85. * @reserved: reserved bits
  86. * @cmd: Command ID (CDMCmd)
  87. * @addr: Device address of the indirect buffer
  88. */
  89. struct cdm_indirect_cmd {
  90. unsigned int length : 16;
  91. unsigned int reserved : 8;
  92. unsigned int cmd : 8;
  93. unsigned int addr;
  94. } __attribute__((__packed__));
  95. /**
  96. * struct cdm_changebase_cmd - Definition for CDM base address change command.
  97. * @base: Base address to be changed to
  98. * @cmd:Command ID (CDMCmd)
  99. */
  100. struct cdm_changebase_cmd {
  101. unsigned int base : 24;
  102. unsigned int cmd : 8;
  103. } __attribute__((__packed__));
  104. /**
  105. * struct cdm_wait_event_cmd - Definition for a CDM Gen IRQ command.
  106. * @mask: Mask for the events
  107. * @id: ID to read back for debug
  108. * @iw_reserved: reserved bits
  109. * @iw: iw AHB write bit
  110. * @cmd:Command ID (CDMCmd)
  111. * @offset: Offset to where data is written
  112. * @offset_reserved: reserved bits
  113. * @data: data returned in IRQ_USR_DATA
  114. */
  115. struct cdm_wait_event_cmd {
  116. unsigned int mask : 8;
  117. unsigned int id : 8;
  118. unsigned int iw_reserved : 7;
  119. unsigned int iw : 1;
  120. unsigned int cmd : 8;
  121. unsigned int offset : 24;
  122. unsigned int offset_reserved : 8;
  123. unsigned int data;
  124. } __attribute__((__packed__));
  125. /**
  126. * struct cdm_genirq_cmd - Definition for a CDM Wait event command.
  127. * @reserved: reserved bits
  128. * @cmd:Command ID (CDMCmd)
  129. * @userdata: userdata returned in IRQ_USR_DATA
  130. */
  131. struct cdm_genirq_cmd {
  132. unsigned int reserved : 24;
  133. unsigned int cmd : 8;
  134. unsigned int userdata;
  135. } __attribute__((__packed__));
  136. /**
  137. * struct cdm_perf_ctrl_cmd_t - Definition for CDM perf control command.
  138. * @perf: perf command
  139. * @reserved: reserved bits
  140. * @cmd:Command ID (CDMCmd)
  141. */
  142. struct cdm_perf_ctrl_cmd {
  143. unsigned int perf : 2;
  144. unsigned int reserved : 22;
  145. unsigned int cmd : 8;
  146. } __attribute__((__packed__));
  147. struct cdm_wait_comp_event_cmd {
  148. unsigned int reserved : 8;
  149. unsigned int id : 8;
  150. unsigned int id_reserved: 8;
  151. unsigned int cmd : 8;
  152. unsigned int mask1;
  153. unsigned int mask2;
  154. } __attribute__((__packed__));
  155. struct cdm_clear_comp_event_cmd {
  156. unsigned int reserved : 8;
  157. unsigned int id : 8;
  158. unsigned int id_reserved: 8;
  159. unsigned int cmd : 8;
  160. unsigned int mask1;
  161. unsigned int mask2;
  162. } __attribute__((__packed__));
  163. struct cdm_prefetch_disable_event_cmd {
  164. unsigned int reserved : 8;
  165. unsigned int id : 8;
  166. unsigned int id_reserved: 8;
  167. unsigned int cmd : 8;
  168. unsigned int mask1;
  169. unsigned int mask2;
  170. } __attribute__((__packed__));
  171. uint32_t cam_cdm_get_cmd_header_size(unsigned int command)
  172. {
  173. return CDMCmdHeaderSizes[command];
  174. }
  175. uint32_t cam_cdm_required_size_dmi(void)
  176. {
  177. return cam_cdm_get_cmd_header_size(CAM_CDM_CMD_DMI);
  178. }
  179. uint32_t cam_cdm_required_size_reg_continuous(uint32_t numVals)
  180. {
  181. if (!numVals) {
  182. CAM_WARN(CAM_CDM, "numVals cant be 0");
  183. return 0;
  184. }
  185. return cam_cdm_get_cmd_header_size(CAM_CDM_CMD_REG_CONT) + numVals;
  186. }
  187. uint32_t cam_cdm_required_size_reg_random(uint32_t numRegVals)
  188. {
  189. return cam_cdm_get_cmd_header_size(CAM_CDM_CMD_REG_RANDOM) +
  190. (2 * numRegVals);
  191. }
  192. uint32_t cam_cdm_required_size_indirect(void)
  193. {
  194. return cam_cdm_get_cmd_header_size(CAM_CDM_CMD_BUFF_INDIRECT);
  195. }
  196. uint32_t cam_cdm_required_size_genirq(void)
  197. {
  198. return cam_cdm_get_cmd_header_size(CAM_CDM_CMD_GEN_IRQ);
  199. }
  200. uint32_t cam_cdm_required_size_wait_event(void)
  201. {
  202. return cam_cdm_get_cmd_header_size(CAM_CDM_CMD_WAIT_EVENT);
  203. }
  204. uint32_t cam_cdm_required_size_changebase(void)
  205. {
  206. return cam_cdm_get_cmd_header_size(CAM_CDM_CMD_CHANGE_BASE);
  207. }
  208. uint32_t cam_cdm_required_size_comp_wait(void)
  209. {
  210. return cam_cdm_get_cmd_header_size(CAM_CDM_CMD_COMP_WAIT);
  211. }
  212. uint32_t cam_cdm_required_size_clear_comp_event(void)
  213. {
  214. return cam_cdm_get_cmd_header_size(CAM_CDM_CLEAR_COMP_WAIT);
  215. }
  216. uint32_t cam_cdm_required_size_prefetch_disable(void)
  217. {
  218. return cam_cdm_get_cmd_header_size(CAM_CDM_WAIT_PREFETCH_DISABLE);
  219. }
  220. uint32_t cam_cdm_offsetof_dmi_addr(void)
  221. {
  222. return offsetof(struct cdm_dmi_cmd, addr);
  223. }
  224. uint32_t cam_cdm_offsetof_indirect_addr(void)
  225. {
  226. return offsetof(struct cdm_indirect_cmd, addr);
  227. }
  228. uint32_t *cam_cdm_write_dmi(uint32_t *pCmdBuffer, uint8_t dmiCmd,
  229. uint32_t DMIAddr, uint8_t DMISel, uint32_t dmiBufferAddr,
  230. uint32_t length)
  231. {
  232. struct cdm_dmi_cmd *pHeader = (struct cdm_dmi_cmd *)pCmdBuffer;
  233. pHeader->cmd = CAM_CDM_CMD_DMI;
  234. pHeader->addr = dmiBufferAddr;
  235. pHeader->length = length;
  236. pHeader->DMIAddr = DMIAddr;
  237. pHeader->DMISel = DMISel;
  238. pCmdBuffer += cam_cdm_get_cmd_header_size(CAM_CDM_CMD_DMI);
  239. return pCmdBuffer;
  240. }
  241. uint32_t *cam_cdm_write_regcontinuous(uint32_t *pCmdBuffer, uint32_t reg,
  242. uint32_t numVals, uint32_t *pVals)
  243. {
  244. uint32_t i;
  245. struct cdm_regcontinuous_cmd *pHeader =
  246. (struct cdm_regcontinuous_cmd *)pCmdBuffer;
  247. pHeader->count = numVals;
  248. pHeader->cmd = CAM_CDM_CMD_REG_CONT;
  249. pHeader->reserved0 = 0;
  250. pHeader->reserved1 = 0;
  251. pHeader->offset = reg;
  252. pCmdBuffer += cam_cdm_get_cmd_header_size(CAM_CDM_CMD_REG_CONT);
  253. for (i = 0; i < numVals; i++)
  254. (((uint32_t *)pCmdBuffer)[i]) = (((uint32_t *)pVals)[i]);
  255. pCmdBuffer += numVals;
  256. return pCmdBuffer;
  257. }
  258. uint32_t *cam_cdm_write_regrandom(uint32_t *pCmdBuffer, uint32_t numRegVals,
  259. uint32_t *pRegVals)
  260. {
  261. uint32_t i;
  262. uint32_t *dst, *src;
  263. struct cdm_regrandom_cmd *pHeader =
  264. (struct cdm_regrandom_cmd *)pCmdBuffer;
  265. if (!numRegVals) {
  266. CAM_ERR(CAM_CDM, "Number of reg-val pairs can not be 0");
  267. return pCmdBuffer;
  268. }
  269. pHeader->count = numRegVals;
  270. pHeader->cmd = CAM_CDM_CMD_REG_RANDOM;
  271. pHeader->reserved = 0;
  272. pCmdBuffer += cam_cdm_get_cmd_header_size(CAM_CDM_CMD_REG_RANDOM);
  273. dst = pCmdBuffer;
  274. src = pRegVals;
  275. for (i = 0; i < numRegVals; i++) {
  276. *dst++ = *src++;
  277. *dst++ = *src++;
  278. }
  279. return dst;
  280. }
  281. uint32_t *cam_cdm_write_indirect(uint32_t *pCmdBuffer, uint32_t indirectBufAddr,
  282. uint32_t length)
  283. {
  284. struct cdm_indirect_cmd *pHeader =
  285. (struct cdm_indirect_cmd *)pCmdBuffer;
  286. pHeader->cmd = CAM_CDM_CMD_BUFF_INDIRECT;
  287. pHeader->addr = indirectBufAddr;
  288. pHeader->length = length - 1;
  289. pCmdBuffer += cam_cdm_get_cmd_header_size(CAM_CDM_CMD_BUFF_INDIRECT);
  290. return pCmdBuffer;
  291. }
  292. void cam_cdm_write_genirq(uint32_t *pCmdBuffer, uint32_t userdata,
  293. bool bit_wr_enable, uint32_t fifo_idx)
  294. {
  295. struct cdm_genirq_cmd *pHeader = (struct cdm_genirq_cmd *)pCmdBuffer;
  296. CAM_DBG(CAM_CDM, "userdata 0x%x, fifo_idx %d",
  297. userdata, fifo_idx);
  298. if (bit_wr_enable)
  299. pHeader->reserved = (unsigned int)((fifo_idx << 1)
  300. | (unsigned int)(bit_wr_enable));
  301. pHeader->cmd = CAM_CDM_CMD_GEN_IRQ;
  302. pHeader->userdata = (userdata << (8 * fifo_idx));
  303. }
  304. uint32_t *cam_cdm_write_wait_event(uint32_t *pcmdbuffer, uint32_t iw,
  305. uint32_t id, uint32_t mask,
  306. uint32_t offset, uint32_t data)
  307. {
  308. struct cdm_wait_event_cmd *pheader =
  309. (struct cdm_wait_event_cmd *)pcmdbuffer;
  310. pheader->cmd = CAM_CDM_CMD_WAIT_EVENT;
  311. pheader->mask = mask;
  312. pheader->data = data;
  313. pheader->id = id;
  314. pheader->iw = iw;
  315. pheader->offset = offset;
  316. pheader->iw_reserved = 0;
  317. pheader->offset_reserved = 0;
  318. pcmdbuffer += cam_cdm_get_cmd_header_size(CAM_CDM_CMD_WAIT_EVENT);
  319. return pcmdbuffer;
  320. }
  321. uint32_t *cam_cdm_write_changebase(uint32_t *pCmdBuffer, uint32_t base)
  322. {
  323. struct cdm_changebase_cmd *pHeader =
  324. (struct cdm_changebase_cmd *)pCmdBuffer;
  325. CAM_DBG(CAM_CDM, "Change to base 0x%x", base);
  326. pHeader->cmd = CAM_CDM_CMD_CHANGE_BASE;
  327. pHeader->base = base;
  328. pCmdBuffer += cam_cdm_get_cmd_header_size(CAM_CDM_CMD_CHANGE_BASE);
  329. return pCmdBuffer;
  330. }
  331. uint32_t *cam_cdm_write_wait_comp_event(
  332. uint32_t *pCmdBuffer, uint32_t mask1, uint32_t mask2)
  333. {
  334. struct cdm_wait_comp_event_cmd *pHeader =
  335. (struct cdm_wait_comp_event_cmd *)pCmdBuffer;
  336. pHeader->cmd = CAM_CDM_CMD_COMP_WAIT;
  337. pHeader->mask1 = mask1;
  338. pHeader->mask2 = mask2;
  339. pCmdBuffer += cam_cdm_get_cmd_header_size(CAM_CDM_CMD_COMP_WAIT);
  340. return pCmdBuffer;
  341. }
  342. uint32_t *cam_cdm_write_clear_comp_event(
  343. uint32_t *pCmdBuffer, uint32_t mask1, uint32_t mask2)
  344. {
  345. struct cdm_clear_comp_event_cmd *pHeader =
  346. (struct cdm_clear_comp_event_cmd *)pCmdBuffer;
  347. pHeader->cmd = CAM_CDM_CLEAR_COMP_WAIT;
  348. pHeader->mask1 = mask1;
  349. pHeader->mask2 = mask2;
  350. pCmdBuffer += cam_cdm_get_cmd_header_size(CAM_CDM_CLEAR_COMP_WAIT);
  351. return pCmdBuffer;
  352. }
  353. uint32_t *cam_cdm_write_wait_prefetch_disable(
  354. uint32_t *pCmdBuffer,
  355. uint32_t id,
  356. uint32_t mask1,
  357. uint32_t mask2)
  358. {
  359. struct cdm_prefetch_disable_event_cmd *pHeader =
  360. (struct cdm_prefetch_disable_event_cmd *)pCmdBuffer;
  361. pHeader->cmd = CAM_CDM_WAIT_PREFETCH_DISABLE;
  362. pHeader->id = id;
  363. pHeader->mask1 = mask1;
  364. pHeader->mask2 = mask2;
  365. pCmdBuffer += cam_cdm_get_cmd_header_size(CAM_CDM_WAIT_PREFETCH_DISABLE);
  366. return pCmdBuffer;
  367. }
  368. struct cam_cdm_utils_ops CDM170_ops = {
  369. .cdm_get_cmd_header_size = cam_cdm_get_cmd_header_size,
  370. .cdm_required_size_dmi = cam_cdm_required_size_dmi,
  371. .cdm_required_size_reg_continuous = cam_cdm_required_size_reg_continuous,
  372. .cdm_required_size_reg_random = cam_cdm_required_size_reg_random,
  373. .cdm_required_size_indirect = cam_cdm_required_size_indirect,
  374. .cdm_required_size_genirq = cam_cdm_required_size_genirq,
  375. .cdm_required_size_wait_event = cam_cdm_required_size_wait_event,
  376. .cdm_required_size_changebase = cam_cdm_required_size_changebase,
  377. .cdm_required_size_comp_wait = cam_cdm_required_size_comp_wait,
  378. .cdm_required_size_clear_comp_event = cam_cdm_required_size_clear_comp_event,
  379. .cdm_required_size_prefetch_disable = cam_cdm_required_size_prefetch_disable,
  380. .cdm_offsetof_dmi_addr = cam_cdm_offsetof_dmi_addr,
  381. .cdm_offsetof_indirect_addr = cam_cdm_offsetof_indirect_addr,
  382. .cdm_write_dmi = cam_cdm_write_dmi,
  383. .cdm_write_regcontinuous = cam_cdm_write_regcontinuous,
  384. .cdm_write_regrandom = cam_cdm_write_regrandom,
  385. .cdm_write_indirect = cam_cdm_write_indirect,
  386. .cdm_write_genirq = cam_cdm_write_genirq,
  387. .cdm_write_wait_event = cam_cdm_write_wait_event,
  388. .cdm_write_changebase = cam_cdm_write_changebase,
  389. .cdm_write_wait_comp_event = cam_cdm_write_wait_comp_event,
  390. .cdm_write_clear_comp_event = cam_cdm_write_clear_comp_event,
  391. .cdm_write_wait_prefetch_disable = cam_cdm_write_wait_prefetch_disable,
  392. };
  393. int cam_cdm_get_ioremap_from_base(uint32_t hw_base,
  394. uint32_t base_array_size,
  395. struct cam_soc_reg_map *base_table[CAM_SOC_MAX_BLOCK],
  396. void __iomem **device_base)
  397. {
  398. int ret = -EINVAL, i;
  399. for (i = 0; i < base_array_size; i++) {
  400. if (base_table[i])
  401. CAM_DBG(CAM_CDM, "In loop %d ioremap for %x addr=%x",
  402. i, (base_table[i])->mem_cam_base, hw_base);
  403. if ((base_table[i]) &&
  404. ((base_table[i])->mem_cam_base == hw_base)) {
  405. *device_base = (base_table[i])->mem_base;
  406. ret = 0;
  407. break;
  408. }
  409. }
  410. return ret;
  411. }
  412. static int cam_cdm_util_cmd_buf_validation(void __iomem *base_addr,
  413. uint32_t base_array_size,
  414. struct cam_soc_reg_map *base_table[CAM_SOC_MAX_BLOCK],
  415. uint32_t cmd_buf_size, uint32_t *cmd_buf, void *buf,
  416. resource_size_t *size,
  417. enum cam_cdm_command cmd_type)
  418. {
  419. int i, ret = 0;
  420. if (!base_addr) {
  421. CAM_ERR(CAM_CDM, "invalid base address");
  422. return -EINVAL;
  423. }
  424. for (i = 0; i < base_array_size; i++) {
  425. if ((base_table[i]) &&
  426. ((base_table[i])->mem_base == base_addr)) {
  427. *size = (base_table[i])->size;
  428. break;
  429. }
  430. }
  431. if (*size == 0) {
  432. CAM_ERR(CAM_CDM, "Could not retrieve ioremap size, address not mapped!");
  433. return -EINVAL;
  434. }
  435. switch (cmd_type) {
  436. case CAM_CDM_CMD_REG_RANDOM: {
  437. struct cdm_regrandom_cmd *reg_random = (struct cdm_regrandom_cmd *)buf;
  438. uint32_t *data, offset;
  439. if ((!reg_random->count) || (((reg_random->count * (sizeof(uint32_t) * 2)) +
  440. cam_cdm_get_cmd_header_size(CAM_CDM_CMD_REG_RANDOM)) >
  441. cmd_buf_size)) {
  442. CAM_ERR(CAM_CDM, "invalid reg_count %d cmd_buf_size %d",
  443. reg_random->count, cmd_buf_size);
  444. ret = -EINVAL;
  445. }
  446. data = cmd_buf + cam_cdm_get_cmd_header_size(CAM_CDM_CMD_REG_RANDOM);
  447. for (i = 0; i < reg_random->count; i++) {
  448. offset = data[0];
  449. if (offset > *size) {
  450. CAM_ERR(CAM_CDM, "Offset out of mapped range! size:%llu offset:%u",
  451. *size, offset);
  452. return -EINVAL;
  453. }
  454. data += 2;
  455. }
  456. }
  457. break;
  458. case CAM_CDM_CMD_REG_CONT: {
  459. struct cdm_regcontinuous_cmd *reg_cont = (struct cdm_regcontinuous_cmd *) buf;
  460. if ((!reg_cont->count) || (((reg_cont->count * sizeof(uint32_t)) +
  461. cam_cdm_get_cmd_header_size(CAM_CDM_CMD_REG_CONT)) >
  462. cmd_buf_size)) {
  463. CAM_ERR(CAM_CDM, "buffer size %d is not sufficient for count%d",
  464. cmd_buf_size, reg_cont->count);
  465. ret = -EINVAL;
  466. }
  467. if ((reg_cont->offset > *size) && ((reg_cont->offset +
  468. (reg_cont->count * sizeof(uint32_t))) > *size)) {
  469. CAM_ERR(CAM_CDM, "Offset out of mapped range! size: %lu, offset: %u",
  470. *size, reg_cont->offset);
  471. return -EINVAL;
  472. }
  473. }
  474. break;
  475. case CAM_CDM_CMD_SWD_DMI_64: {
  476. struct cdm_dmi_cmd *swd_dmi = (struct cdm_dmi_cmd *) buf;
  477. if (cmd_buf_size < (cam_cdm_required_size_dmi() + swd_dmi->length + 1)) {
  478. CAM_ERR(CAM_CDM, "invalid CDM_SWD_DMI length %d",
  479. swd_dmi->length + 1);
  480. ret = -EINVAL;
  481. }
  482. if ((swd_dmi->DMIAddr + CAM_CDM_DMI_DATA_LO_OFFSET > *size) ||
  483. (swd_dmi->DMIAddr + CAM_CDM_DMI_DATA_HI_OFFSET > *size)) {
  484. CAM_ERR(CAM_CDM,
  485. "Offset out of mapped range! size:%llu lo_offset:%u hi_offset:%u",
  486. *size, swd_dmi->DMIAddr + CAM_CDM_DMI_DATA_LO_OFFSET,
  487. swd_dmi->DMIAddr + CAM_CDM_DMI_DATA_LO_OFFSET);
  488. return -EINVAL;
  489. }
  490. }
  491. break;
  492. case CAM_CDM_CMD_SWD_DMI_32: {
  493. struct cdm_dmi_cmd *swd_dmi = (struct cdm_dmi_cmd *) buf;
  494. if (cmd_buf_size < (cam_cdm_required_size_dmi() + swd_dmi->length + 1)) {
  495. CAM_ERR(CAM_CDM, "invalid CDM_SWD_DMI length %d",
  496. swd_dmi->length + 1);
  497. ret = -EINVAL;
  498. }
  499. if (swd_dmi->DMIAddr + CAM_CDM_DMI_DATA_LO_OFFSET > *size) {
  500. CAM_ERR(CAM_CDM,
  501. "Offset out of mapped range! size:%llu lo_offset:%u",
  502. *size, swd_dmi->DMIAddr + CAM_CDM_DMI_DATA_LO_OFFSET);
  503. return -EINVAL;
  504. }
  505. }
  506. break;
  507. case CAM_CDM_CMD_DMI: {
  508. struct cdm_dmi_cmd *swd_dmi = (struct cdm_dmi_cmd *) buf;
  509. if (cmd_buf_size < (cam_cdm_required_size_dmi() + swd_dmi->length + 1)) {
  510. CAM_ERR(CAM_CDM, "invalid CDM_SWD_DMI length %d",
  511. swd_dmi->length + 1);
  512. ret = -EINVAL;
  513. }
  514. if (swd_dmi->DMIAddr + CAM_CDM_DMI_DATA_OFFSET > *size) {
  515. CAM_ERR(CAM_CDM,
  516. "Offset out of mapped range! size:%llu offset:%u",
  517. *size, swd_dmi->DMIAddr + CAM_CDM_DMI_DATA_OFFSET);
  518. return -EINVAL;
  519. }
  520. }
  521. break;
  522. default:
  523. CAM_ERR(CAM_CDM, "unsupported cdm_cmd_type type 0%x",
  524. cmd_type);
  525. ret = -EINVAL;
  526. break;
  527. }
  528. return ret;
  529. }
  530. static int cam_cdm_util_reg_cont_write(void __iomem *base_addr,
  531. uint32_t *cmd_buf, uint32_t cmd_buf_size, uint32_t *used_bytes,
  532. uint32_t base_array_size,
  533. struct cam_soc_reg_map *base_table[CAM_SOC_MAX_BLOCK])
  534. {
  535. int rc;
  536. uint32_t *data;
  537. struct cdm_regcontinuous_cmd reg_cont;
  538. resource_size_t size = 0;
  539. memcpy(&reg_cont, cmd_buf, sizeof(struct cdm_regcontinuous_cmd));
  540. rc = cam_cdm_util_cmd_buf_validation(base_addr, base_array_size, base_table,
  541. cmd_buf_size, cmd_buf, (void *)&reg_cont,
  542. &size, CAM_CDM_CMD_REG_CONT);
  543. if (rc) {
  544. CAM_ERR(CAM_CDM, "Validation failed! rc=%d", rc);
  545. return rc;
  546. }
  547. data = cmd_buf + cam_cdm_get_cmd_header_size(CAM_CDM_CMD_REG_CONT);
  548. cam_io_memcpy(base_addr + reg_cont.offset, data,
  549. reg_cont.count * sizeof(uint32_t));
  550. *used_bytes = (reg_cont.count * sizeof(uint32_t)) +
  551. (4 * cam_cdm_get_cmd_header_size(CAM_CDM_CMD_REG_CONT));
  552. return rc;
  553. }
  554. static int cam_cdm_util_reg_random_write(void __iomem *base_addr,
  555. uint32_t *cmd_buf, uint32_t cmd_buf_size, uint32_t *used_bytes,
  556. uint32_t base_array_size,
  557. struct cam_soc_reg_map *base_table[CAM_SOC_MAX_BLOCK])
  558. {
  559. int i, rc;
  560. struct cdm_regrandom_cmd reg_random;
  561. uint32_t *data, offset;
  562. resource_size_t size = 0;
  563. memcpy(&reg_random, cmd_buf, sizeof(struct cdm_regrandom_cmd));
  564. rc = cam_cdm_util_cmd_buf_validation(base_addr, base_array_size, base_table,
  565. cmd_buf_size, cmd_buf, (void *)&reg_random,
  566. &size, CAM_CDM_CMD_REG_RANDOM);
  567. if (rc) {
  568. CAM_ERR(CAM_CDM, "Validation failed! rc=%d", rc);
  569. return rc;
  570. }
  571. data = cmd_buf + cam_cdm_get_cmd_header_size(CAM_CDM_CMD_REG_RANDOM);
  572. for (i = 0; i < reg_random.count; i++) {
  573. offset = data[0];
  574. CAM_DBG(CAM_CDM, "reg random: offset %pK, value 0x%x",
  575. ((void __iomem *)(base_addr + offset)),
  576. data[1]);
  577. cam_io_w(data[1], base_addr + offset);
  578. data += 2;
  579. }
  580. *used_bytes = ((reg_random.count * (sizeof(uint32_t) * 2)) +
  581. (4 * cam_cdm_get_cmd_header_size(CAM_CDM_CMD_REG_RANDOM)));
  582. return rc;
  583. }
  584. static int cam_cdm_util_swd_dmi_write(uint32_t cdm_cmd_type,
  585. void __iomem *base_addr, uint32_t *cmd_buf, uint32_t cmd_buf_size,
  586. uint32_t *used_bytes, uint32_t base_array_size,
  587. struct cam_soc_reg_map *base_table[CAM_SOC_MAX_BLOCK])
  588. {
  589. int i, rc;
  590. struct cdm_dmi_cmd swd_dmi;
  591. uint32_t *data;
  592. resource_size_t size = 0;
  593. memcpy(&swd_dmi, cmd_buf, sizeof(struct cdm_dmi_cmd));
  594. rc = cam_cdm_util_cmd_buf_validation(base_addr, base_array_size, base_table,
  595. cmd_buf_size, cmd_buf, (void *)&swd_dmi,
  596. &size, cdm_cmd_type);
  597. if (rc) {
  598. CAM_ERR(CAM_CDM, "Validation failed! rc=%d", rc);
  599. return rc;
  600. }
  601. data = cmd_buf + cam_cdm_required_size_dmi();
  602. if (cdm_cmd_type == CAM_CDM_CMD_SWD_DMI_64) {
  603. for (i = 0; i < (swd_dmi.length + 1)/8; i++) {
  604. cam_io_w_mb(data[0], base_addr +
  605. swd_dmi.DMIAddr + CAM_CDM_DMI_DATA_LO_OFFSET);
  606. cam_io_w_mb(data[1], base_addr +
  607. swd_dmi.DMIAddr + CAM_CDM_DMI_DATA_HI_OFFSET);
  608. data += 2;
  609. }
  610. } else if (cdm_cmd_type == CAM_CDM_CMD_DMI) {
  611. for (i = 0; i < (swd_dmi.length + 1)/4; i++) {
  612. cam_io_w_mb(data[0], base_addr +
  613. swd_dmi.DMIAddr + CAM_CDM_DMI_DATA_OFFSET);
  614. data += 1;
  615. }
  616. } else {
  617. for (i = 0; i < (swd_dmi.length + 1)/4; i++) {
  618. cam_io_w_mb(data[0], base_addr +
  619. swd_dmi.DMIAddr + CAM_CDM_DMI_DATA_LO_OFFSET);
  620. data += 1;
  621. }
  622. }
  623. *used_bytes = (4 * cam_cdm_required_size_dmi()) + swd_dmi.length + 1;
  624. return rc;
  625. }
  626. int cam_cdm_util_cmd_buf_write(void __iomem **current_device_base,
  627. uint32_t *cmd_buf, uint32_t cmd_buf_size,
  628. struct cam_soc_reg_map *base_table[CAM_SOC_MAX_BLOCK],
  629. uint32_t base_array_size, uint8_t bl_tag)
  630. {
  631. int ret = 0;
  632. uint32_t cdm_cmd_type = 0;
  633. uint32_t used_bytes = 0;
  634. while (cmd_buf_size > 0) {
  635. CAM_DBG(CAM_CDM, "cmd data=%x", *cmd_buf);
  636. cdm_cmd_type = (*cmd_buf >> CAM_CDM_COMMAND_OFFSET);
  637. switch (cdm_cmd_type) {
  638. case CAM_CDM_CMD_REG_CONT: {
  639. ret = cam_cdm_util_reg_cont_write(*current_device_base,
  640. cmd_buf, cmd_buf_size, &used_bytes,
  641. base_array_size, base_table);
  642. if (ret)
  643. break;
  644. if (used_bytes > 0) {
  645. cmd_buf_size -= used_bytes;
  646. cmd_buf += used_bytes/4;
  647. }
  648. }
  649. break;
  650. case CAM_CDM_CMD_REG_RANDOM: {
  651. ret = cam_cdm_util_reg_random_write(
  652. *current_device_base, cmd_buf, cmd_buf_size,
  653. &used_bytes, base_array_size, base_table);
  654. if (ret)
  655. break;
  656. if (used_bytes > 0) {
  657. cmd_buf_size -= used_bytes;
  658. cmd_buf += used_bytes / 4;
  659. }
  660. }
  661. break;
  662. case CAM_CDM_CMD_DMI:
  663. case CAM_CDM_CMD_SWD_DMI_32:
  664. case CAM_CDM_CMD_SWD_DMI_64: {
  665. if (*current_device_base == 0) {
  666. CAM_ERR(CAM_CDM,
  667. "Got SWI DMI cmd =%d for invalid hw",
  668. cdm_cmd_type);
  669. ret = -EINVAL;
  670. break;
  671. }
  672. ret = cam_cdm_util_swd_dmi_write(cdm_cmd_type,
  673. *current_device_base, cmd_buf, cmd_buf_size,
  674. &used_bytes, base_array_size, base_table);
  675. if (ret)
  676. break;
  677. if (used_bytes > 0) {
  678. cmd_buf_size -= used_bytes;
  679. cmd_buf += used_bytes / 4;
  680. }
  681. }
  682. break;
  683. case CAM_CDM_CMD_CHANGE_BASE: {
  684. struct cdm_changebase_cmd *change_base_cmd =
  685. (struct cdm_changebase_cmd *)cmd_buf;
  686. ret = cam_cdm_get_ioremap_from_base(
  687. change_base_cmd->base, base_array_size,
  688. base_table, current_device_base);
  689. if (ret != 0) {
  690. CAM_ERR(CAM_CDM,
  691. "Get ioremap change base failed %x",
  692. change_base_cmd->base);
  693. break;
  694. }
  695. CAM_DBG(CAM_CDM, "Got ioremap for %x addr=%pK",
  696. change_base_cmd->base,
  697. current_device_base);
  698. cmd_buf_size -= (4 *
  699. cam_cdm_required_size_changebase());
  700. cmd_buf += cam_cdm_required_size_changebase();
  701. }
  702. break;
  703. default:
  704. CAM_ERR(CAM_CDM, "unsupported cdm_cmd_type type 0%x",
  705. cdm_cmd_type);
  706. ret = -EINVAL;
  707. break;
  708. }
  709. if (ret < 0)
  710. break;
  711. }
  712. return ret;
  713. }
  714. static long cam_cdm_util_dump_dmi_cmd(uint32_t *cmd_buf_addr,
  715. uint32_t *cmd_buf_addr_end)
  716. {
  717. long ret = 0;
  718. struct cdm_dmi_cmd *p_dmi_cmd;
  719. uint32_t *temp_ptr = cmd_buf_addr;
  720. p_dmi_cmd = (struct cdm_dmi_cmd *)cmd_buf_addr;
  721. temp_ptr += CDMCmdHeaderSizes[CAM_CDM_CMD_DMI];
  722. ret += CDMCmdHeaderSizes[CAM_CDM_CMD_DMI];
  723. if (temp_ptr > cmd_buf_addr_end)
  724. CAM_ERR(CAM_CDM,
  725. "Invalid cmd start addr:%pK end addr:%pK",
  726. temp_ptr, cmd_buf_addr_end);
  727. CAM_INFO(CAM_CDM,
  728. "DMI: LEN: %u DMIAddr: 0x%X DMISel: 0x%X LUT_addr: 0x%X",
  729. p_dmi_cmd->length, p_dmi_cmd->DMIAddr,
  730. p_dmi_cmd->DMISel, p_dmi_cmd->addr);
  731. return ret;
  732. }
  733. static long cam_cdm_util_dump_buff_indirect(uint32_t *cmd_buf_addr,
  734. uint32_t *cmd_buf_addr_end)
  735. {
  736. long ret = 0;
  737. struct cdm_indirect_cmd *p_indirect_cmd;
  738. uint32_t *temp_ptr = cmd_buf_addr;
  739. p_indirect_cmd = (struct cdm_indirect_cmd *)cmd_buf_addr;
  740. temp_ptr += CDMCmdHeaderSizes[CAM_CDM_CMD_BUFF_INDIRECT];
  741. ret += CDMCmdHeaderSizes[CAM_CDM_CMD_BUFF_INDIRECT];
  742. if (temp_ptr > cmd_buf_addr_end)
  743. CAM_ERR(CAM_CDM,
  744. "Invalid cmd start addr:%pK end addr:%pK",
  745. temp_ptr, cmd_buf_addr_end);
  746. CAM_INFO(CAM_CDM,
  747. "Buff Indirect: LEN: %u addr: 0x%X",
  748. p_indirect_cmd->length, p_indirect_cmd->addr);
  749. return ret;
  750. }
  751. static long cam_cdm_util_dump_reg_cont_cmd(uint32_t *cmd_buf_addr,
  752. uint32_t *cmd_buf_addr_end)
  753. {
  754. long ret = 0;
  755. struct cdm_regcontinuous_cmd *p_regcont_cmd;
  756. uint32_t *temp_ptr = cmd_buf_addr;
  757. int i = 0;
  758. p_regcont_cmd = (struct cdm_regcontinuous_cmd *)temp_ptr;
  759. temp_ptr += CDMCmdHeaderSizes[CAM_CDM_CMD_REG_CONT];
  760. ret += CDMCmdHeaderSizes[CAM_CDM_CMD_REG_CONT];
  761. CAM_INFO(CAM_CDM, "REG_CONT: COUNT: %u OFFSET: 0x%X",
  762. p_regcont_cmd->count, p_regcont_cmd->offset);
  763. for (i = 0; i < p_regcont_cmd->count; i++) {
  764. if (temp_ptr > cmd_buf_addr_end) {
  765. CAM_ERR(CAM_CDM,
  766. "Invalid cmd(%d) start addr:%pK end addr:%pK",
  767. i, temp_ptr, cmd_buf_addr_end);
  768. break;
  769. }
  770. CAM_INFO(CAM_CDM, "DATA_%d: 0x%X", i,
  771. *temp_ptr);
  772. temp_ptr++;
  773. ret++;
  774. }
  775. return ret;
  776. }
  777. static long cam_cdm_util_dump_reg_random_cmd(uint32_t *cmd_buf_addr,
  778. uint32_t *cmd_buf_addr_end)
  779. {
  780. struct cdm_regrandom_cmd *p_regrand_cmd;
  781. uint32_t *temp_ptr = cmd_buf_addr;
  782. long ret = 0;
  783. int i = 0;
  784. p_regrand_cmd = (struct cdm_regrandom_cmd *)temp_ptr;
  785. temp_ptr += CDMCmdHeaderSizes[CAM_CDM_CMD_REG_RANDOM];
  786. ret += CDMCmdHeaderSizes[CAM_CDM_CMD_REG_RANDOM];
  787. CAM_INFO(CAM_CDM, "REG_RAND: COUNT: %u",
  788. p_regrand_cmd->count);
  789. for (i = 0; i < p_regrand_cmd->count; i++) {
  790. if (temp_ptr > cmd_buf_addr_end) {
  791. CAM_ERR(CAM_CDM,
  792. "Invalid cmd(%d) start addr:%pK end addr:%pK",
  793. i, temp_ptr, cmd_buf_addr_end);
  794. break;
  795. }
  796. CAM_INFO(CAM_CDM, "OFFSET_%d: 0x%X DATA_%d: 0x%X",
  797. i, *temp_ptr & CAM_CDM_REG_OFFSET_MASK, i,
  798. *(temp_ptr + 1));
  799. temp_ptr += 2;
  800. ret += 2;
  801. }
  802. return ret;
  803. }
  804. static long cam_cdm_util_dump_gen_irq_cmd(uint32_t *cmd_buf_addr)
  805. {
  806. long ret = 0;
  807. ret += CDMCmdHeaderSizes[CAM_CDM_CMD_GEN_IRQ];
  808. CAM_INFO(CAM_CDM, "GEN_IRQ");
  809. return ret;
  810. }
  811. static long cam_cdm_util_dump_wait_event_cmd(uint32_t *cmd_buf_addr)
  812. {
  813. long ret = 0;
  814. ret += CDMCmdHeaderSizes[CAM_CDM_CMD_WAIT_EVENT];
  815. CAM_INFO(CAM_CDM, "WAIT_EVENT");
  816. return ret;
  817. }
  818. static long cam_cdm_util_dump_change_base_cmd(uint32_t *cmd_buf_addr,
  819. uint32_t *cmd_buf_addr_end)
  820. {
  821. long ret = 0;
  822. struct cdm_changebase_cmd *p_cbase_cmd;
  823. uint32_t *temp_ptr = cmd_buf_addr;
  824. if (temp_ptr > cmd_buf_addr_end) {
  825. CAM_ERR(CAM_CDM,
  826. "Invalid cmd start addr:%pK end addr:%pK",
  827. temp_ptr, cmd_buf_addr_end);
  828. return 0;
  829. }
  830. p_cbase_cmd = (struct cdm_changebase_cmd *)temp_ptr;
  831. temp_ptr += CDMCmdHeaderSizes[CAM_CDM_CMD_CHANGE_BASE];
  832. ret += CDMCmdHeaderSizes[CAM_CDM_CMD_CHANGE_BASE];
  833. CAM_INFO(CAM_CDM, "CHANGE_BASE: 0x%X, curr cmd addr: %pK",
  834. p_cbase_cmd->base, temp_ptr);
  835. return ret;
  836. }
  837. static long cam_cdm_util_dump_comp_wait_event_cmd(uint32_t *cmd_buf_addr)
  838. {
  839. long ret = 0;
  840. ret += CDMCmdHeaderSizes[CAM_CDM_CMD_COMP_WAIT];
  841. CAM_INFO(CAM_CDM, "WAIT_EVENT");
  842. return ret;
  843. }
  844. static long cam_cdm_util_dump_perf_ctrl_cmd(uint32_t *cmd_buf_addr)
  845. {
  846. long ret = 0;
  847. ret += CDMCmdHeaderSizes[CAM_CDM_CMD_PERF_CTRL];
  848. CAM_INFO(CAM_CDM, "PERF_CTRL");
  849. return ret;
  850. }
  851. bool cam_cdm_util_validate_cmd_buf(
  852. uint32_t *cmd_buf_start, uint32_t *cmd_buf_end)
  853. {
  854. uint32_t *buf_now = cmd_buf_start;
  855. uint32_t *buf_end = cmd_buf_end;
  856. uint32_t cmd = 0;
  857. int i = 0;
  858. struct cdm_regcontinuous_cmd *p_regcont_cmd = NULL;
  859. struct cdm_regrandom_cmd *p_regrand_cmd = NULL;
  860. if (!cmd_buf_start || !cmd_buf_end) {
  861. CAM_ERR(CAM_CDM, "Invalid args");
  862. return true;
  863. }
  864. do {
  865. cmd = *buf_now;
  866. cmd = cmd >> CAM_CDM_COMMAND_OFFSET;
  867. switch (cmd) {
  868. case CAM_CDM_CMD_DMI:
  869. case CAM_CDM_CMD_DMI_32:
  870. case CAM_CDM_CMD_DMI_64:
  871. if (buf_now > buf_end)
  872. return true;
  873. buf_now += CDMCmdHeaderSizes[CAM_CDM_CMD_DMI];
  874. break;
  875. case CAM_CDM_CMD_REG_CONT:
  876. p_regcont_cmd = (struct cdm_regcontinuous_cmd *)buf_now;
  877. buf_now += CDMCmdHeaderSizes[CAM_CDM_CMD_REG_CONT];
  878. for (i = 0; i < p_regcont_cmd->count; i++) {
  879. if (buf_now > buf_end)
  880. return true;
  881. buf_now++;
  882. }
  883. break;
  884. case CAM_CDM_CMD_REG_RANDOM:
  885. p_regrand_cmd = (struct cdm_regrandom_cmd *)buf_now;
  886. buf_now += CDMCmdHeaderSizes[CAM_CDM_CMD_REG_RANDOM];
  887. for (i = 0; i < p_regrand_cmd->count; i++) {
  888. if (buf_now > buf_end)
  889. return true;
  890. buf_now += 2;
  891. }
  892. break;
  893. case CAM_CDM_CMD_BUFF_INDIRECT:
  894. buf_now += CDMCmdHeaderSizes[CAM_CDM_CMD_BUFF_INDIRECT];
  895. if (buf_now > buf_end)
  896. return true;
  897. break;
  898. case CAM_CDM_CMD_GEN_IRQ:
  899. buf_now += CDMCmdHeaderSizes[CAM_CDM_CMD_GEN_IRQ];
  900. break;
  901. case CAM_CDM_CMD_WAIT_EVENT:
  902. buf_now += CDMCmdHeaderSizes[CAM_CDM_CMD_WAIT_EVENT];
  903. break;
  904. case CAM_CDM_CMD_CHANGE_BASE:
  905. if (buf_now > buf_end)
  906. return true;
  907. buf_now += CDMCmdHeaderSizes[CAM_CDM_CMD_CHANGE_BASE];
  908. break;
  909. case CAM_CDM_CMD_PERF_CTRL:
  910. buf_now += CDMCmdHeaderSizes[CAM_CDM_CMD_PERF_CTRL];
  911. break;
  912. case CAM_CDM_CMD_COMP_WAIT:
  913. buf_now += CDMCmdHeaderSizes[CAM_CDM_CMD_COMP_WAIT];
  914. break;
  915. default:
  916. CAM_ERR(CAM_CDM, "Invalid CMD: 0x%x buf 0x%x",
  917. cmd, *buf_now);
  918. return true;
  919. }
  920. } while (buf_now < cmd_buf_end);
  921. return false;
  922. }
  923. void cam_cdm_util_dump_cmd_buf(
  924. uint32_t *cmd_buf_start, uint32_t *cmd_buf_end)
  925. {
  926. uint32_t *buf_now = cmd_buf_start;
  927. uint32_t *buf_end = cmd_buf_end;
  928. uint32_t cmd = 0;
  929. if (!cmd_buf_start || !cmd_buf_end) {
  930. CAM_ERR(CAM_CDM, "Invalid args");
  931. return;
  932. }
  933. do {
  934. cmd = *buf_now;
  935. cmd = cmd >> CAM_CDM_COMMAND_OFFSET;
  936. switch (cmd) {
  937. case CAM_CDM_CMD_DMI:
  938. case CAM_CDM_CMD_DMI_32:
  939. case CAM_CDM_CMD_DMI_64:
  940. buf_now += cam_cdm_util_dump_dmi_cmd(buf_now,
  941. buf_end);
  942. break;
  943. case CAM_CDM_CMD_REG_CONT:
  944. buf_now += cam_cdm_util_dump_reg_cont_cmd(buf_now,
  945. buf_end);
  946. break;
  947. case CAM_CDM_CMD_REG_RANDOM:
  948. buf_now += cam_cdm_util_dump_reg_random_cmd(buf_now,
  949. buf_end);
  950. break;
  951. case CAM_CDM_CMD_BUFF_INDIRECT:
  952. buf_now += cam_cdm_util_dump_buff_indirect(buf_now,
  953. buf_end);
  954. break;
  955. case CAM_CDM_CMD_GEN_IRQ:
  956. buf_now += cam_cdm_util_dump_gen_irq_cmd(buf_now);
  957. break;
  958. case CAM_CDM_CMD_WAIT_EVENT:
  959. buf_now += cam_cdm_util_dump_wait_event_cmd(buf_now);
  960. break;
  961. case CAM_CDM_CMD_CHANGE_BASE:
  962. buf_now += cam_cdm_util_dump_change_base_cmd(buf_now,
  963. buf_end);
  964. break;
  965. case CAM_CDM_CMD_PERF_CTRL:
  966. buf_now += cam_cdm_util_dump_perf_ctrl_cmd(buf_now);
  967. break;
  968. case CAM_CDM_CMD_COMP_WAIT:
  969. buf_now +=
  970. cam_cdm_util_dump_comp_wait_event_cmd(buf_now);
  971. break;
  972. default:
  973. CAM_ERR(CAM_CDM, "Invalid CMD: 0x%x buf 0x%x",
  974. cmd, *buf_now);
  975. buf_now++;
  976. break;
  977. }
  978. } while (buf_now < cmd_buf_end);
  979. }
  980. static uint32_t cam_cdm_util_dump_reg_cont_cmd_v2(
  981. uint32_t *cmd_buf_addr,
  982. struct cam_cdm_cmd_buf_dump_info *dump_info)
  983. {
  984. int i;
  985. long ret;
  986. uint8_t *dst;
  987. size_t remain_len;
  988. uint32_t *temp_ptr = cmd_buf_addr;
  989. uint32_t *addr, *start;
  990. uint32_t min_len;
  991. struct cdm_regcontinuous_cmd *p_regcont_cmd;
  992. struct cam_cdm_cmd_dump_header *hdr;
  993. p_regcont_cmd = (struct cdm_regcontinuous_cmd *)temp_ptr;
  994. temp_ptr += cam_cdm_get_cmd_header_size(CAM_CDM_CMD_REG_CONT);
  995. ret = cam_cdm_get_cmd_header_size(CAM_CDM_CMD_REG_CONT);
  996. min_len = (sizeof(uint32_t) * p_regcont_cmd->count) +
  997. sizeof(struct cam_cdm_cmd_dump_header) +
  998. (2 * sizeof(uint32_t));
  999. remain_len = dump_info->dst_max_size - dump_info->dst_offset;
  1000. if (remain_len < min_len) {
  1001. CAM_WARN_RATE_LIMIT(CAM_CDM,
  1002. "Dump buffer exhaust remain %zu min %u",
  1003. remain_len, min_len);
  1004. return ret;
  1005. }
  1006. dst = (char *)dump_info->dst_start + dump_info->dst_offset;
  1007. hdr = (struct cam_cdm_cmd_dump_header *)dst;
  1008. scnprintf(hdr->tag, CAM_CDM_CMD_TAG_MAX_LEN, "CDM_REG_CONT:");
  1009. hdr->word_size = sizeof(uint32_t);
  1010. addr = (uint32_t *)(dst + sizeof(struct cam_cdm_cmd_dump_header));
  1011. start = addr;
  1012. *addr++ = p_regcont_cmd->offset;
  1013. *addr++ = p_regcont_cmd->count;
  1014. for (i = 0; i < p_regcont_cmd->count; i++) {
  1015. *addr = *temp_ptr;
  1016. temp_ptr++;
  1017. addr++;
  1018. ret++;
  1019. }
  1020. hdr->size = hdr->word_size * (addr - start);
  1021. dump_info->dst_offset += hdr->size +
  1022. sizeof(struct cam_cdm_cmd_dump_header);
  1023. return ret;
  1024. }
  1025. static uint32_t cam_cdm_util_dump_reg_random_cmd_v2(
  1026. uint32_t *cmd_buf_addr,
  1027. struct cam_cdm_cmd_buf_dump_info *dump_info)
  1028. {
  1029. int i;
  1030. long ret;
  1031. uint8_t *dst;
  1032. uint32_t *temp_ptr = cmd_buf_addr;
  1033. uint32_t *addr, *start;
  1034. size_t remain_len;
  1035. uint32_t min_len;
  1036. struct cdm_regrandom_cmd *p_regrand_cmd;
  1037. struct cam_cdm_cmd_dump_header *hdr;
  1038. p_regrand_cmd = (struct cdm_regrandom_cmd *)temp_ptr;
  1039. temp_ptr += cam_cdm_get_cmd_header_size(CAM_CDM_CMD_REG_RANDOM);
  1040. ret = cam_cdm_get_cmd_header_size(CAM_CDM_CMD_REG_RANDOM);
  1041. min_len = (2 * sizeof(uint32_t) * p_regrand_cmd->count) +
  1042. sizeof(struct cam_cdm_cmd_dump_header) + sizeof(uint32_t);
  1043. remain_len = dump_info->dst_max_size - dump_info->dst_offset;
  1044. if (remain_len < min_len) {
  1045. CAM_WARN_RATE_LIMIT(CAM_CDM,
  1046. "Dump buffer exhaust remain %zu min %u",
  1047. remain_len, min_len);
  1048. return ret;
  1049. }
  1050. dst = (char *)dump_info->dst_start + dump_info->dst_offset;
  1051. hdr = (struct cam_cdm_cmd_dump_header *)dst;
  1052. scnprintf(hdr->tag, CAM_CDM_CMD_TAG_MAX_LEN, "CDM_REG_RANDOM:");
  1053. hdr->word_size = sizeof(uint32_t);
  1054. addr = (uint32_t *)(dst + sizeof(struct cam_cdm_cmd_dump_header));
  1055. start = addr;
  1056. *addr++ = p_regrand_cmd->count;
  1057. for (i = 0; i < p_regrand_cmd->count; i++) {
  1058. addr[0] = temp_ptr[0] & CAM_CDM_REG_OFFSET_MASK;
  1059. addr[1] = temp_ptr[1];
  1060. temp_ptr += 2;
  1061. addr += 2;
  1062. ret += 2;
  1063. }
  1064. hdr->size = hdr->word_size * (addr - start);
  1065. dump_info->dst_offset += hdr->size +
  1066. sizeof(struct cam_cdm_cmd_dump_header);
  1067. return ret;
  1068. }
  1069. int cam_cdm_util_dump_cmd_bufs_v2(
  1070. struct cam_cdm_cmd_buf_dump_info *dump_info)
  1071. {
  1072. uint32_t cmd;
  1073. uint32_t *buf_now;
  1074. int rc = 0;
  1075. if (!dump_info || !dump_info->src_start || !dump_info->src_end ||
  1076. !dump_info->dst_start) {
  1077. CAM_INFO(CAM_CDM, "Invalid args");
  1078. return -EINVAL;
  1079. }
  1080. buf_now = dump_info->src_start;
  1081. do {
  1082. if (dump_info->dst_offset >= dump_info->dst_max_size) {
  1083. CAM_WARN(CAM_CDM,
  1084. "Dump overshoot offset %zu size %zu",
  1085. dump_info->dst_offset,
  1086. dump_info->dst_max_size);
  1087. return -ENOSPC;
  1088. }
  1089. cmd = *buf_now;
  1090. cmd = cmd >> CAM_CDM_COMMAND_OFFSET;
  1091. switch (cmd) {
  1092. case CAM_CDM_CMD_DMI:
  1093. case CAM_CDM_CMD_DMI_32:
  1094. case CAM_CDM_CMD_DMI_64:
  1095. buf_now += cam_cdm_get_cmd_header_size(CAM_CDM_CMD_DMI);
  1096. break;
  1097. case CAM_CDM_CMD_REG_CONT:
  1098. buf_now += cam_cdm_util_dump_reg_cont_cmd_v2(buf_now,
  1099. dump_info);
  1100. break;
  1101. case CAM_CDM_CMD_REG_RANDOM:
  1102. buf_now += cam_cdm_util_dump_reg_random_cmd_v2(buf_now,
  1103. dump_info);
  1104. break;
  1105. case CAM_CDM_CMD_BUFF_INDIRECT:
  1106. buf_now += cam_cdm_get_cmd_header_size(
  1107. CAM_CDM_CMD_BUFF_INDIRECT);
  1108. break;
  1109. case CAM_CDM_CMD_GEN_IRQ:
  1110. buf_now += cam_cdm_get_cmd_header_size(
  1111. CAM_CDM_CMD_GEN_IRQ);
  1112. break;
  1113. case CAM_CDM_CMD_WAIT_EVENT:
  1114. buf_now += cam_cdm_get_cmd_header_size(
  1115. CAM_CDM_CMD_WAIT_EVENT);
  1116. break;
  1117. case CAM_CDM_CMD_CHANGE_BASE:
  1118. buf_now += cam_cdm_get_cmd_header_size(
  1119. CAM_CDM_CMD_CHANGE_BASE);
  1120. break;
  1121. case CAM_CDM_CMD_PERF_CTRL:
  1122. buf_now += cam_cdm_get_cmd_header_size(
  1123. CAM_CDM_CMD_PERF_CTRL);
  1124. break;
  1125. case CAM_CDM_CMD_COMP_WAIT:
  1126. buf_now += cam_cdm_get_cmd_header_size(
  1127. CAM_CDM_CMD_COMP_WAIT);
  1128. break;
  1129. default:
  1130. CAM_ERR(CAM_CDM, "Invalid CMD: 0x%x", cmd);
  1131. buf_now++;
  1132. break;
  1133. }
  1134. } while (buf_now < dump_info->src_end);
  1135. return rc;
  1136. }