rmi_f34v7.c 29 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2016, Zodiac Inflight Innovations
  4. * Copyright (c) 2007-2016, Synaptics Incorporated
  5. * Copyright (C) 2012 Alexandra Chin <[email protected]>
  6. * Copyright (C) 2012 Scott Lin <[email protected]>
  7. */
  8. #include <linux/bitops.h>
  9. #include <linux/kernel.h>
  10. #include <linux/rmi.h>
  11. #include <linux/firmware.h>
  12. #include <linux/delay.h>
  13. #include <linux/slab.h>
  14. #include <linux/jiffies.h>
  15. #include <asm/unaligned.h>
  16. #include "rmi_driver.h"
  17. #include "rmi_f34.h"
  18. static int rmi_f34v7_read_flash_status(struct f34_data *f34)
  19. {
  20. u8 status;
  21. u8 command;
  22. int ret;
  23. ret = rmi_read_block(f34->fn->rmi_dev,
  24. f34->fn->fd.data_base_addr + V7_FLASH_STATUS_OFFSET,
  25. &status,
  26. sizeof(status));
  27. if (ret < 0) {
  28. rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
  29. "%s: Error %d reading flash status\n", __func__, ret);
  30. return ret;
  31. }
  32. f34->v7.in_bl_mode = status >> 7;
  33. f34->v7.flash_status = status & 0x1f;
  34. if (f34->v7.flash_status != 0x00) {
  35. dev_err(&f34->fn->dev, "%s: status=%d, command=0x%02x\n",
  36. __func__, f34->v7.flash_status, f34->v7.command);
  37. }
  38. ret = rmi_read_block(f34->fn->rmi_dev,
  39. f34->fn->fd.data_base_addr + V7_COMMAND_OFFSET,
  40. &command,
  41. sizeof(command));
  42. if (ret < 0) {
  43. dev_err(&f34->fn->dev, "%s: Failed to read flash command\n",
  44. __func__);
  45. return ret;
  46. }
  47. f34->v7.command = command;
  48. return 0;
  49. }
  50. static int rmi_f34v7_wait_for_idle(struct f34_data *f34, int timeout_ms)
  51. {
  52. unsigned long timeout;
  53. timeout = msecs_to_jiffies(timeout_ms);
  54. if (!wait_for_completion_timeout(&f34->v7.cmd_done, timeout)) {
  55. dev_warn(&f34->fn->dev, "%s: Timed out waiting for idle status\n",
  56. __func__);
  57. return -ETIMEDOUT;
  58. }
  59. return 0;
  60. }
  61. static int rmi_f34v7_check_command_status(struct f34_data *f34, int timeout_ms)
  62. {
  63. int ret;
  64. ret = rmi_f34v7_wait_for_idle(f34, timeout_ms);
  65. if (ret < 0)
  66. return ret;
  67. ret = rmi_f34v7_read_flash_status(f34);
  68. if (ret < 0)
  69. return ret;
  70. if (f34->v7.flash_status != 0x00)
  71. return -EIO;
  72. return 0;
  73. }
  74. static int rmi_f34v7_write_command_single_transaction(struct f34_data *f34,
  75. u8 cmd)
  76. {
  77. int ret;
  78. u8 base;
  79. struct f34v7_data_1_5 data_1_5;
  80. base = f34->fn->fd.data_base_addr;
  81. memset(&data_1_5, 0, sizeof(data_1_5));
  82. switch (cmd) {
  83. case v7_CMD_ERASE_ALL:
  84. data_1_5.partition_id = CORE_CODE_PARTITION;
  85. data_1_5.command = CMD_V7_ERASE_AP;
  86. break;
  87. case v7_CMD_ERASE_UI_FIRMWARE:
  88. data_1_5.partition_id = CORE_CODE_PARTITION;
  89. data_1_5.command = CMD_V7_ERASE;
  90. break;
  91. case v7_CMD_ERASE_BL_CONFIG:
  92. data_1_5.partition_id = GLOBAL_PARAMETERS_PARTITION;
  93. data_1_5.command = CMD_V7_ERASE;
  94. break;
  95. case v7_CMD_ERASE_UI_CONFIG:
  96. data_1_5.partition_id = CORE_CONFIG_PARTITION;
  97. data_1_5.command = CMD_V7_ERASE;
  98. break;
  99. case v7_CMD_ERASE_DISP_CONFIG:
  100. data_1_5.partition_id = DISPLAY_CONFIG_PARTITION;
  101. data_1_5.command = CMD_V7_ERASE;
  102. break;
  103. case v7_CMD_ERASE_FLASH_CONFIG:
  104. data_1_5.partition_id = FLASH_CONFIG_PARTITION;
  105. data_1_5.command = CMD_V7_ERASE;
  106. break;
  107. case v7_CMD_ERASE_GUEST_CODE:
  108. data_1_5.partition_id = GUEST_CODE_PARTITION;
  109. data_1_5.command = CMD_V7_ERASE;
  110. break;
  111. case v7_CMD_ENABLE_FLASH_PROG:
  112. data_1_5.partition_id = BOOTLOADER_PARTITION;
  113. data_1_5.command = CMD_V7_ENTER_BL;
  114. break;
  115. }
  116. data_1_5.payload[0] = f34->bootloader_id[0];
  117. data_1_5.payload[1] = f34->bootloader_id[1];
  118. ret = rmi_write_block(f34->fn->rmi_dev,
  119. base + V7_PARTITION_ID_OFFSET,
  120. &data_1_5, sizeof(data_1_5));
  121. if (ret < 0) {
  122. dev_err(&f34->fn->dev,
  123. "%s: Failed to write single transaction command\n",
  124. __func__);
  125. return ret;
  126. }
  127. return 0;
  128. }
  129. static int rmi_f34v7_write_command(struct f34_data *f34, u8 cmd)
  130. {
  131. int ret;
  132. u8 base;
  133. u8 command;
  134. base = f34->fn->fd.data_base_addr;
  135. switch (cmd) {
  136. case v7_CMD_WRITE_FW:
  137. case v7_CMD_WRITE_CONFIG:
  138. case v7_CMD_WRITE_GUEST_CODE:
  139. command = CMD_V7_WRITE;
  140. break;
  141. case v7_CMD_READ_CONFIG:
  142. command = CMD_V7_READ;
  143. break;
  144. case v7_CMD_ERASE_ALL:
  145. command = CMD_V7_ERASE_AP;
  146. break;
  147. case v7_CMD_ERASE_UI_FIRMWARE:
  148. case v7_CMD_ERASE_BL_CONFIG:
  149. case v7_CMD_ERASE_UI_CONFIG:
  150. case v7_CMD_ERASE_DISP_CONFIG:
  151. case v7_CMD_ERASE_FLASH_CONFIG:
  152. case v7_CMD_ERASE_GUEST_CODE:
  153. command = CMD_V7_ERASE;
  154. break;
  155. case v7_CMD_ENABLE_FLASH_PROG:
  156. command = CMD_V7_ENTER_BL;
  157. break;
  158. default:
  159. dev_err(&f34->fn->dev, "%s: Invalid command 0x%02x\n",
  160. __func__, cmd);
  161. return -EINVAL;
  162. }
  163. f34->v7.command = command;
  164. switch (cmd) {
  165. case v7_CMD_ERASE_ALL:
  166. case v7_CMD_ERASE_UI_FIRMWARE:
  167. case v7_CMD_ERASE_BL_CONFIG:
  168. case v7_CMD_ERASE_UI_CONFIG:
  169. case v7_CMD_ERASE_DISP_CONFIG:
  170. case v7_CMD_ERASE_FLASH_CONFIG:
  171. case v7_CMD_ERASE_GUEST_CODE:
  172. case v7_CMD_ENABLE_FLASH_PROG:
  173. ret = rmi_f34v7_write_command_single_transaction(f34, cmd);
  174. if (ret < 0)
  175. return ret;
  176. else
  177. return 0;
  178. default:
  179. break;
  180. }
  181. rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev, "%s: writing cmd %02X\n",
  182. __func__, command);
  183. ret = rmi_write_block(f34->fn->rmi_dev,
  184. base + V7_COMMAND_OFFSET,
  185. &command, sizeof(command));
  186. if (ret < 0) {
  187. dev_err(&f34->fn->dev, "%s: Failed to write flash command\n",
  188. __func__);
  189. return ret;
  190. }
  191. return 0;
  192. }
  193. static int rmi_f34v7_write_partition_id(struct f34_data *f34, u8 cmd)
  194. {
  195. int ret;
  196. u8 base;
  197. u8 partition;
  198. base = f34->fn->fd.data_base_addr;
  199. switch (cmd) {
  200. case v7_CMD_WRITE_FW:
  201. partition = CORE_CODE_PARTITION;
  202. break;
  203. case v7_CMD_WRITE_CONFIG:
  204. case v7_CMD_READ_CONFIG:
  205. if (f34->v7.config_area == v7_UI_CONFIG_AREA)
  206. partition = CORE_CONFIG_PARTITION;
  207. else if (f34->v7.config_area == v7_DP_CONFIG_AREA)
  208. partition = DISPLAY_CONFIG_PARTITION;
  209. else if (f34->v7.config_area == v7_PM_CONFIG_AREA)
  210. partition = GUEST_SERIALIZATION_PARTITION;
  211. else if (f34->v7.config_area == v7_BL_CONFIG_AREA)
  212. partition = GLOBAL_PARAMETERS_PARTITION;
  213. else if (f34->v7.config_area == v7_FLASH_CONFIG_AREA)
  214. partition = FLASH_CONFIG_PARTITION;
  215. break;
  216. case v7_CMD_WRITE_GUEST_CODE:
  217. partition = GUEST_CODE_PARTITION;
  218. break;
  219. case v7_CMD_ERASE_ALL:
  220. partition = CORE_CODE_PARTITION;
  221. break;
  222. case v7_CMD_ERASE_BL_CONFIG:
  223. partition = GLOBAL_PARAMETERS_PARTITION;
  224. break;
  225. case v7_CMD_ERASE_UI_CONFIG:
  226. partition = CORE_CONFIG_PARTITION;
  227. break;
  228. case v7_CMD_ERASE_DISP_CONFIG:
  229. partition = DISPLAY_CONFIG_PARTITION;
  230. break;
  231. case v7_CMD_ERASE_FLASH_CONFIG:
  232. partition = FLASH_CONFIG_PARTITION;
  233. break;
  234. case v7_CMD_ERASE_GUEST_CODE:
  235. partition = GUEST_CODE_PARTITION;
  236. break;
  237. case v7_CMD_ENABLE_FLASH_PROG:
  238. partition = BOOTLOADER_PARTITION;
  239. break;
  240. default:
  241. dev_err(&f34->fn->dev, "%s: Invalid command 0x%02x\n",
  242. __func__, cmd);
  243. return -EINVAL;
  244. }
  245. ret = rmi_write_block(f34->fn->rmi_dev,
  246. base + V7_PARTITION_ID_OFFSET,
  247. &partition, sizeof(partition));
  248. if (ret < 0) {
  249. dev_err(&f34->fn->dev, "%s: Failed to write partition ID\n",
  250. __func__);
  251. return ret;
  252. }
  253. return 0;
  254. }
  255. static int rmi_f34v7_read_partition_table(struct f34_data *f34)
  256. {
  257. int ret;
  258. unsigned long timeout;
  259. u8 base;
  260. __le16 length;
  261. u16 block_number = 0;
  262. base = f34->fn->fd.data_base_addr;
  263. f34->v7.config_area = v7_FLASH_CONFIG_AREA;
  264. ret = rmi_f34v7_write_partition_id(f34, v7_CMD_READ_CONFIG);
  265. if (ret < 0)
  266. return ret;
  267. ret = rmi_write_block(f34->fn->rmi_dev,
  268. base + V7_BLOCK_NUMBER_OFFSET,
  269. &block_number, sizeof(block_number));
  270. if (ret < 0) {
  271. dev_err(&f34->fn->dev, "%s: Failed to write block number\n",
  272. __func__);
  273. return ret;
  274. }
  275. put_unaligned_le16(f34->v7.flash_config_length, &length);
  276. ret = rmi_write_block(f34->fn->rmi_dev,
  277. base + V7_TRANSFER_LENGTH_OFFSET,
  278. &length, sizeof(length));
  279. if (ret < 0) {
  280. dev_err(&f34->fn->dev, "%s: Failed to write transfer length\n",
  281. __func__);
  282. return ret;
  283. }
  284. init_completion(&f34->v7.cmd_done);
  285. ret = rmi_f34v7_write_command(f34, v7_CMD_READ_CONFIG);
  286. if (ret < 0) {
  287. dev_err(&f34->fn->dev, "%s: Failed to write command\n",
  288. __func__);
  289. return ret;
  290. }
  291. /*
  292. * rmi_f34v7_check_command_status() can't be used here, as this
  293. * function is called before IRQs are available
  294. */
  295. timeout = msecs_to_jiffies(F34_WRITE_WAIT_MS);
  296. while (time_before(jiffies, timeout)) {
  297. usleep_range(5000, 6000);
  298. rmi_f34v7_read_flash_status(f34);
  299. if (f34->v7.command == v7_CMD_IDLE &&
  300. f34->v7.flash_status == 0x00) {
  301. break;
  302. }
  303. }
  304. ret = rmi_read_block(f34->fn->rmi_dev,
  305. base + V7_PAYLOAD_OFFSET,
  306. f34->v7.read_config_buf,
  307. f34->v7.partition_table_bytes);
  308. if (ret < 0) {
  309. dev_err(&f34->fn->dev, "%s: Failed to read block data\n",
  310. __func__);
  311. return ret;
  312. }
  313. return 0;
  314. }
  315. static void rmi_f34v7_parse_partition_table(struct f34_data *f34,
  316. const void *partition_table,
  317. struct block_count *blkcount,
  318. struct physical_address *phyaddr)
  319. {
  320. int i;
  321. int index;
  322. u16 partition_length;
  323. u16 physical_address;
  324. const struct partition_table *ptable;
  325. for (i = 0; i < f34->v7.partitions; i++) {
  326. index = i * 8 + 2;
  327. ptable = partition_table + index;
  328. partition_length = le16_to_cpu(ptable->partition_length);
  329. physical_address = le16_to_cpu(ptable->start_physical_address);
  330. rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
  331. "%s: Partition entry %d: %*ph\n",
  332. __func__, i, sizeof(struct partition_table), ptable);
  333. switch (ptable->partition_id & 0x1f) {
  334. case CORE_CODE_PARTITION:
  335. blkcount->ui_firmware = partition_length;
  336. phyaddr->ui_firmware = physical_address;
  337. rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
  338. "%s: Core code block count: %d\n",
  339. __func__, blkcount->ui_firmware);
  340. break;
  341. case CORE_CONFIG_PARTITION:
  342. blkcount->ui_config = partition_length;
  343. phyaddr->ui_config = physical_address;
  344. rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
  345. "%s: Core config block count: %d\n",
  346. __func__, blkcount->ui_config);
  347. break;
  348. case DISPLAY_CONFIG_PARTITION:
  349. blkcount->dp_config = partition_length;
  350. phyaddr->dp_config = physical_address;
  351. rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
  352. "%s: Display config block count: %d\n",
  353. __func__, blkcount->dp_config);
  354. break;
  355. case FLASH_CONFIG_PARTITION:
  356. blkcount->fl_config = partition_length;
  357. rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
  358. "%s: Flash config block count: %d\n",
  359. __func__, blkcount->fl_config);
  360. break;
  361. case GUEST_CODE_PARTITION:
  362. blkcount->guest_code = partition_length;
  363. phyaddr->guest_code = physical_address;
  364. rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
  365. "%s: Guest code block count: %d\n",
  366. __func__, blkcount->guest_code);
  367. break;
  368. case GUEST_SERIALIZATION_PARTITION:
  369. blkcount->pm_config = partition_length;
  370. rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
  371. "%s: Guest serialization block count: %d\n",
  372. __func__, blkcount->pm_config);
  373. break;
  374. case GLOBAL_PARAMETERS_PARTITION:
  375. blkcount->bl_config = partition_length;
  376. rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
  377. "%s: Global parameters block count: %d\n",
  378. __func__, blkcount->bl_config);
  379. break;
  380. case DEVICE_CONFIG_PARTITION:
  381. blkcount->lockdown = partition_length;
  382. rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
  383. "%s: Device config block count: %d\n",
  384. __func__, blkcount->lockdown);
  385. break;
  386. }
  387. }
  388. }
  389. static int rmi_f34v7_read_queries_bl_version(struct f34_data *f34)
  390. {
  391. int ret;
  392. u8 base;
  393. int offset;
  394. u8 query_0;
  395. struct f34v7_query_1_7 query_1_7;
  396. base = f34->fn->fd.query_base_addr;
  397. ret = rmi_read_block(f34->fn->rmi_dev,
  398. base,
  399. &query_0,
  400. sizeof(query_0));
  401. if (ret < 0) {
  402. dev_err(&f34->fn->dev,
  403. "%s: Failed to read query 0\n", __func__);
  404. return ret;
  405. }
  406. offset = (query_0 & 0x7) + 1;
  407. ret = rmi_read_block(f34->fn->rmi_dev,
  408. base + offset,
  409. &query_1_7,
  410. sizeof(query_1_7));
  411. if (ret < 0) {
  412. dev_err(&f34->fn->dev, "%s: Failed to read queries 1 to 7\n",
  413. __func__);
  414. return ret;
  415. }
  416. f34->bootloader_id[0] = query_1_7.bl_minor_revision;
  417. f34->bootloader_id[1] = query_1_7.bl_major_revision;
  418. rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev, "Bootloader V%d.%d\n",
  419. f34->bootloader_id[1], f34->bootloader_id[0]);
  420. return 0;
  421. }
  422. static int rmi_f34v7_read_queries(struct f34_data *f34)
  423. {
  424. int ret;
  425. int i;
  426. u8 base;
  427. int offset;
  428. u8 *ptable;
  429. u8 query_0;
  430. struct f34v7_query_1_7 query_1_7;
  431. base = f34->fn->fd.query_base_addr;
  432. ret = rmi_read_block(f34->fn->rmi_dev,
  433. base,
  434. &query_0,
  435. sizeof(query_0));
  436. if (ret < 0) {
  437. dev_err(&f34->fn->dev,
  438. "%s: Failed to read query 0\n", __func__);
  439. return ret;
  440. }
  441. offset = (query_0 & 0x07) + 1;
  442. ret = rmi_read_block(f34->fn->rmi_dev,
  443. base + offset,
  444. &query_1_7,
  445. sizeof(query_1_7));
  446. if (ret < 0) {
  447. dev_err(&f34->fn->dev, "%s: Failed to read queries 1 to 7\n",
  448. __func__);
  449. return ret;
  450. }
  451. f34->bootloader_id[0] = query_1_7.bl_minor_revision;
  452. f34->bootloader_id[1] = query_1_7.bl_major_revision;
  453. f34->v7.block_size = le16_to_cpu(query_1_7.block_size);
  454. f34->v7.flash_config_length =
  455. le16_to_cpu(query_1_7.flash_config_length);
  456. f34->v7.payload_length = le16_to_cpu(query_1_7.payload_length);
  457. rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev, "%s: f34->v7.block_size = %d\n",
  458. __func__, f34->v7.block_size);
  459. f34->v7.has_display_cfg = query_1_7.partition_support[1] & HAS_DISP_CFG;
  460. f34->v7.has_guest_code =
  461. query_1_7.partition_support[1] & HAS_GUEST_CODE;
  462. if (query_0 & HAS_CONFIG_ID) {
  463. u8 f34_ctrl[CONFIG_ID_SIZE];
  464. ret = rmi_read_block(f34->fn->rmi_dev,
  465. f34->fn->fd.control_base_addr,
  466. f34_ctrl,
  467. sizeof(f34_ctrl));
  468. if (ret)
  469. return ret;
  470. /* Eat leading zeros */
  471. for (i = 0; i < sizeof(f34_ctrl) - 1 && !f34_ctrl[i]; i++)
  472. /* Empty */;
  473. snprintf(f34->configuration_id, sizeof(f34->configuration_id),
  474. "%*phN", (int)sizeof(f34_ctrl) - i, f34_ctrl + i);
  475. rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev, "Configuration ID: %s\n",
  476. f34->configuration_id);
  477. }
  478. f34->v7.partitions = 0;
  479. for (i = 0; i < sizeof(query_1_7.partition_support); i++)
  480. f34->v7.partitions += hweight8(query_1_7.partition_support[i]);
  481. rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev, "%s: Supported partitions: %*ph\n",
  482. __func__, sizeof(query_1_7.partition_support),
  483. query_1_7.partition_support);
  484. f34->v7.partition_table_bytes = f34->v7.partitions * 8 + 2;
  485. f34->v7.read_config_buf = devm_kzalloc(&f34->fn->dev,
  486. f34->v7.partition_table_bytes,
  487. GFP_KERNEL);
  488. if (!f34->v7.read_config_buf) {
  489. f34->v7.read_config_buf_size = 0;
  490. return -ENOMEM;
  491. }
  492. f34->v7.read_config_buf_size = f34->v7.partition_table_bytes;
  493. ptable = f34->v7.read_config_buf;
  494. ret = rmi_f34v7_read_partition_table(f34);
  495. if (ret < 0) {
  496. dev_err(&f34->fn->dev, "%s: Failed to read partition table\n",
  497. __func__);
  498. return ret;
  499. }
  500. rmi_f34v7_parse_partition_table(f34, ptable,
  501. &f34->v7.blkcount, &f34->v7.phyaddr);
  502. return 0;
  503. }
  504. static int rmi_f34v7_check_bl_config_size(struct f34_data *f34)
  505. {
  506. u16 block_count;
  507. block_count = f34->v7.img.bl_config.size / f34->v7.block_size;
  508. f34->update_size += block_count;
  509. if (block_count != f34->v7.blkcount.bl_config) {
  510. dev_err(&f34->fn->dev, "Bootloader config size mismatch\n");
  511. return -EINVAL;
  512. }
  513. return 0;
  514. }
  515. static int rmi_f34v7_erase_all(struct f34_data *f34)
  516. {
  517. int ret;
  518. dev_info(&f34->fn->dev, "Erasing firmware...\n");
  519. init_completion(&f34->v7.cmd_done);
  520. ret = rmi_f34v7_write_command(f34, v7_CMD_ERASE_ALL);
  521. if (ret < 0)
  522. return ret;
  523. ret = rmi_f34v7_check_command_status(f34, F34_ERASE_WAIT_MS);
  524. if (ret < 0)
  525. return ret;
  526. return 0;
  527. }
  528. static int rmi_f34v7_read_blocks(struct f34_data *f34,
  529. u16 block_cnt, u8 command)
  530. {
  531. int ret;
  532. u8 base;
  533. __le16 length;
  534. u16 transfer;
  535. u16 max_transfer;
  536. u16 remaining = block_cnt;
  537. u16 block_number = 0;
  538. u16 index = 0;
  539. base = f34->fn->fd.data_base_addr;
  540. ret = rmi_f34v7_write_partition_id(f34, command);
  541. if (ret < 0)
  542. return ret;
  543. ret = rmi_write_block(f34->fn->rmi_dev,
  544. base + V7_BLOCK_NUMBER_OFFSET,
  545. &block_number, sizeof(block_number));
  546. if (ret < 0) {
  547. dev_err(&f34->fn->dev, "%s: Failed to write block number\n",
  548. __func__);
  549. return ret;
  550. }
  551. max_transfer = min(f34->v7.payload_length,
  552. (u16)(PAGE_SIZE / f34->v7.block_size));
  553. do {
  554. transfer = min(remaining, max_transfer);
  555. put_unaligned_le16(transfer, &length);
  556. ret = rmi_write_block(f34->fn->rmi_dev,
  557. base + V7_TRANSFER_LENGTH_OFFSET,
  558. &length, sizeof(length));
  559. if (ret < 0) {
  560. dev_err(&f34->fn->dev,
  561. "%s: Write transfer length fail (%d remaining)\n",
  562. __func__, remaining);
  563. return ret;
  564. }
  565. init_completion(&f34->v7.cmd_done);
  566. ret = rmi_f34v7_write_command(f34, command);
  567. if (ret < 0)
  568. return ret;
  569. ret = rmi_f34v7_check_command_status(f34, F34_ENABLE_WAIT_MS);
  570. if (ret < 0)
  571. return ret;
  572. ret = rmi_read_block(f34->fn->rmi_dev,
  573. base + V7_PAYLOAD_OFFSET,
  574. &f34->v7.read_config_buf[index],
  575. transfer * f34->v7.block_size);
  576. if (ret < 0) {
  577. dev_err(&f34->fn->dev,
  578. "%s: Read block failed (%d blks remaining)\n",
  579. __func__, remaining);
  580. return ret;
  581. }
  582. index += (transfer * f34->v7.block_size);
  583. remaining -= transfer;
  584. } while (remaining);
  585. return 0;
  586. }
  587. static int rmi_f34v7_write_f34v7_blocks(struct f34_data *f34,
  588. const void *block_ptr, u16 block_cnt,
  589. u8 command)
  590. {
  591. int ret;
  592. u8 base;
  593. __le16 length;
  594. u16 transfer;
  595. u16 max_transfer;
  596. u16 remaining = block_cnt;
  597. u16 block_number = 0;
  598. base = f34->fn->fd.data_base_addr;
  599. ret = rmi_f34v7_write_partition_id(f34, command);
  600. if (ret < 0)
  601. return ret;
  602. ret = rmi_write_block(f34->fn->rmi_dev,
  603. base + V7_BLOCK_NUMBER_OFFSET,
  604. &block_number, sizeof(block_number));
  605. if (ret < 0) {
  606. dev_err(&f34->fn->dev, "%s: Failed to write block number\n",
  607. __func__);
  608. return ret;
  609. }
  610. if (f34->v7.payload_length > (PAGE_SIZE / f34->v7.block_size))
  611. max_transfer = PAGE_SIZE / f34->v7.block_size;
  612. else
  613. max_transfer = f34->v7.payload_length;
  614. do {
  615. transfer = min(remaining, max_transfer);
  616. put_unaligned_le16(transfer, &length);
  617. init_completion(&f34->v7.cmd_done);
  618. ret = rmi_write_block(f34->fn->rmi_dev,
  619. base + V7_TRANSFER_LENGTH_OFFSET,
  620. &length, sizeof(length));
  621. if (ret < 0) {
  622. dev_err(&f34->fn->dev,
  623. "%s: Write transfer length fail (%d remaining)\n",
  624. __func__, remaining);
  625. return ret;
  626. }
  627. ret = rmi_f34v7_write_command(f34, command);
  628. if (ret < 0)
  629. return ret;
  630. ret = rmi_write_block(f34->fn->rmi_dev,
  631. base + V7_PAYLOAD_OFFSET,
  632. block_ptr, transfer * f34->v7.block_size);
  633. if (ret < 0) {
  634. dev_err(&f34->fn->dev,
  635. "%s: Failed writing data (%d blks remaining)\n",
  636. __func__, remaining);
  637. return ret;
  638. }
  639. ret = rmi_f34v7_check_command_status(f34, F34_ENABLE_WAIT_MS);
  640. if (ret < 0)
  641. return ret;
  642. block_ptr += (transfer * f34->v7.block_size);
  643. remaining -= transfer;
  644. f34->update_progress += transfer;
  645. f34->update_status = (f34->update_progress * 100) /
  646. f34->update_size;
  647. } while (remaining);
  648. return 0;
  649. }
  650. static int rmi_f34v7_write_config(struct f34_data *f34)
  651. {
  652. return rmi_f34v7_write_f34v7_blocks(f34, f34->v7.config_data,
  653. f34->v7.config_block_count,
  654. v7_CMD_WRITE_CONFIG);
  655. }
  656. static int rmi_f34v7_write_ui_config(struct f34_data *f34)
  657. {
  658. f34->v7.config_area = v7_UI_CONFIG_AREA;
  659. f34->v7.config_data = f34->v7.img.ui_config.data;
  660. f34->v7.config_size = f34->v7.img.ui_config.size;
  661. f34->v7.config_block_count = f34->v7.config_size / f34->v7.block_size;
  662. return rmi_f34v7_write_config(f34);
  663. }
  664. static int rmi_f34v7_write_dp_config(struct f34_data *f34)
  665. {
  666. f34->v7.config_area = v7_DP_CONFIG_AREA;
  667. f34->v7.config_data = f34->v7.img.dp_config.data;
  668. f34->v7.config_size = f34->v7.img.dp_config.size;
  669. f34->v7.config_block_count = f34->v7.config_size / f34->v7.block_size;
  670. return rmi_f34v7_write_config(f34);
  671. }
  672. static int rmi_f34v7_write_guest_code(struct f34_data *f34)
  673. {
  674. return rmi_f34v7_write_f34v7_blocks(f34, f34->v7.img.guest_code.data,
  675. f34->v7.img.guest_code.size /
  676. f34->v7.block_size,
  677. v7_CMD_WRITE_GUEST_CODE);
  678. }
  679. static int rmi_f34v7_write_flash_config(struct f34_data *f34)
  680. {
  681. int ret;
  682. f34->v7.config_area = v7_FLASH_CONFIG_AREA;
  683. f34->v7.config_data = f34->v7.img.fl_config.data;
  684. f34->v7.config_size = f34->v7.img.fl_config.size;
  685. f34->v7.config_block_count = f34->v7.config_size / f34->v7.block_size;
  686. if (f34->v7.config_block_count != f34->v7.blkcount.fl_config) {
  687. dev_err(&f34->fn->dev, "%s: Flash config size mismatch\n",
  688. __func__);
  689. return -EINVAL;
  690. }
  691. init_completion(&f34->v7.cmd_done);
  692. ret = rmi_f34v7_write_config(f34);
  693. if (ret < 0)
  694. return ret;
  695. return 0;
  696. }
  697. static int rmi_f34v7_write_partition_table(struct f34_data *f34)
  698. {
  699. u16 block_count;
  700. int ret;
  701. block_count = f34->v7.blkcount.bl_config;
  702. f34->v7.config_area = v7_BL_CONFIG_AREA;
  703. f34->v7.config_size = f34->v7.block_size * block_count;
  704. devm_kfree(&f34->fn->dev, f34->v7.read_config_buf);
  705. f34->v7.read_config_buf = devm_kzalloc(&f34->fn->dev,
  706. f34->v7.config_size, GFP_KERNEL);
  707. if (!f34->v7.read_config_buf) {
  708. f34->v7.read_config_buf_size = 0;
  709. return -ENOMEM;
  710. }
  711. f34->v7.read_config_buf_size = f34->v7.config_size;
  712. ret = rmi_f34v7_read_blocks(f34, block_count, v7_CMD_READ_CONFIG);
  713. if (ret < 0)
  714. return ret;
  715. ret = rmi_f34v7_write_flash_config(f34);
  716. if (ret < 0)
  717. return ret;
  718. f34->v7.config_area = v7_BL_CONFIG_AREA;
  719. f34->v7.config_data = f34->v7.read_config_buf;
  720. f34->v7.config_size = f34->v7.img.bl_config.size;
  721. f34->v7.config_block_count = f34->v7.config_size / f34->v7.block_size;
  722. ret = rmi_f34v7_write_config(f34);
  723. if (ret < 0)
  724. return ret;
  725. return 0;
  726. }
  727. static int rmi_f34v7_write_firmware(struct f34_data *f34)
  728. {
  729. u16 blk_count;
  730. blk_count = f34->v7.img.ui_firmware.size / f34->v7.block_size;
  731. return rmi_f34v7_write_f34v7_blocks(f34, f34->v7.img.ui_firmware.data,
  732. blk_count, v7_CMD_WRITE_FW);
  733. }
  734. static void rmi_f34v7_parse_img_header_10_bl_container(struct f34_data *f34,
  735. const void *image)
  736. {
  737. int i;
  738. int num_of_containers;
  739. unsigned int addr;
  740. unsigned int container_id;
  741. unsigned int length;
  742. const void *content;
  743. const struct container_descriptor *descriptor;
  744. num_of_containers = f34->v7.img.bootloader.size / 4 - 1;
  745. for (i = 1; i <= num_of_containers; i++) {
  746. addr = get_unaligned_le32(f34->v7.img.bootloader.data + i * 4);
  747. descriptor = image + addr;
  748. container_id = le16_to_cpu(descriptor->container_id);
  749. content = image + le32_to_cpu(descriptor->content_address);
  750. length = le32_to_cpu(descriptor->content_length);
  751. switch (container_id) {
  752. case BL_CONFIG_CONTAINER:
  753. case GLOBAL_PARAMETERS_CONTAINER:
  754. f34->v7.img.bl_config.data = content;
  755. f34->v7.img.bl_config.size = length;
  756. break;
  757. case BL_LOCKDOWN_INFO_CONTAINER:
  758. case DEVICE_CONFIG_CONTAINER:
  759. f34->v7.img.lockdown.data = content;
  760. f34->v7.img.lockdown.size = length;
  761. break;
  762. default:
  763. break;
  764. }
  765. }
  766. }
  767. static void rmi_f34v7_parse_image_header_10(struct f34_data *f34)
  768. {
  769. unsigned int i;
  770. unsigned int num_of_containers;
  771. unsigned int addr;
  772. unsigned int offset;
  773. unsigned int container_id;
  774. unsigned int length;
  775. const void *image = f34->v7.image;
  776. const u8 *content;
  777. const struct container_descriptor *descriptor;
  778. const struct image_header_10 *header = image;
  779. f34->v7.img.checksum = le32_to_cpu(header->checksum);
  780. rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev, "%s: f34->v7.img.checksum=%X\n",
  781. __func__, f34->v7.img.checksum);
  782. /* address of top level container */
  783. offset = le32_to_cpu(header->top_level_container_start_addr);
  784. descriptor = image + offset;
  785. /* address of top level container content */
  786. offset = le32_to_cpu(descriptor->content_address);
  787. num_of_containers = le32_to_cpu(descriptor->content_length) / 4;
  788. for (i = 0; i < num_of_containers; i++) {
  789. addr = get_unaligned_le32(image + offset);
  790. offset += 4;
  791. descriptor = image + addr;
  792. container_id = le16_to_cpu(descriptor->container_id);
  793. content = image + le32_to_cpu(descriptor->content_address);
  794. length = le32_to_cpu(descriptor->content_length);
  795. rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
  796. "%s: container_id=%d, length=%d\n", __func__,
  797. container_id, length);
  798. switch (container_id) {
  799. case UI_CONTAINER:
  800. case CORE_CODE_CONTAINER:
  801. f34->v7.img.ui_firmware.data = content;
  802. f34->v7.img.ui_firmware.size = length;
  803. break;
  804. case UI_CONFIG_CONTAINER:
  805. case CORE_CONFIG_CONTAINER:
  806. f34->v7.img.ui_config.data = content;
  807. f34->v7.img.ui_config.size = length;
  808. break;
  809. case BL_CONTAINER:
  810. f34->v7.img.bl_version = *content;
  811. f34->v7.img.bootloader.data = content;
  812. f34->v7.img.bootloader.size = length;
  813. rmi_f34v7_parse_img_header_10_bl_container(f34, image);
  814. break;
  815. case GUEST_CODE_CONTAINER:
  816. f34->v7.img.contains_guest_code = true;
  817. f34->v7.img.guest_code.data = content;
  818. f34->v7.img.guest_code.size = length;
  819. break;
  820. case DISPLAY_CONFIG_CONTAINER:
  821. f34->v7.img.contains_display_cfg = true;
  822. f34->v7.img.dp_config.data = content;
  823. f34->v7.img.dp_config.size = length;
  824. break;
  825. case FLASH_CONFIG_CONTAINER:
  826. f34->v7.img.contains_flash_config = true;
  827. f34->v7.img.fl_config.data = content;
  828. f34->v7.img.fl_config.size = length;
  829. break;
  830. case GENERAL_INFORMATION_CONTAINER:
  831. f34->v7.img.contains_firmware_id = true;
  832. f34->v7.img.firmware_id =
  833. get_unaligned_le32(content + 4);
  834. break;
  835. default:
  836. break;
  837. }
  838. }
  839. }
  840. static int rmi_f34v7_parse_image_info(struct f34_data *f34)
  841. {
  842. const struct image_header_10 *header = f34->v7.image;
  843. memset(&f34->v7.img, 0x00, sizeof(f34->v7.img));
  844. rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
  845. "%s: header->major_header_version = %d\n",
  846. __func__, header->major_header_version);
  847. switch (header->major_header_version) {
  848. case IMAGE_HEADER_VERSION_10:
  849. rmi_f34v7_parse_image_header_10(f34);
  850. break;
  851. default:
  852. dev_err(&f34->fn->dev, "Unsupported image file format %02X\n",
  853. header->major_header_version);
  854. return -EINVAL;
  855. }
  856. if (!f34->v7.img.contains_flash_config) {
  857. dev_err(&f34->fn->dev, "%s: No flash config in fw image\n",
  858. __func__);
  859. return -EINVAL;
  860. }
  861. rmi_f34v7_parse_partition_table(f34, f34->v7.img.fl_config.data,
  862. &f34->v7.img.blkcount, &f34->v7.img.phyaddr);
  863. return 0;
  864. }
  865. int rmi_f34v7_do_reflash(struct f34_data *f34, const struct firmware *fw)
  866. {
  867. int ret;
  868. f34->fn->rmi_dev->driver->set_irq_bits(f34->fn->rmi_dev,
  869. f34->fn->irq_mask);
  870. rmi_f34v7_read_queries_bl_version(f34);
  871. f34->v7.image = fw->data;
  872. f34->update_progress = 0;
  873. f34->update_size = 0;
  874. ret = rmi_f34v7_parse_image_info(f34);
  875. if (ret < 0)
  876. return ret;
  877. ret = rmi_f34v7_check_bl_config_size(f34);
  878. if (ret < 0)
  879. return ret;
  880. ret = rmi_f34v7_erase_all(f34);
  881. if (ret < 0)
  882. return ret;
  883. ret = rmi_f34v7_write_partition_table(f34);
  884. if (ret < 0)
  885. return ret;
  886. dev_info(&f34->fn->dev, "%s: Partition table programmed\n", __func__);
  887. /*
  888. * Reset to reload partition table - as the previous firmware has been
  889. * erased, we remain in bootloader mode.
  890. */
  891. ret = rmi_scan_pdt(f34->fn->rmi_dev, NULL, rmi_initial_reset);
  892. if (ret < 0)
  893. dev_warn(&f34->fn->dev, "RMI reset failed!\n");
  894. dev_info(&f34->fn->dev, "Writing firmware (%d bytes)...\n",
  895. f34->v7.img.ui_firmware.size);
  896. ret = rmi_f34v7_write_firmware(f34);
  897. if (ret < 0)
  898. return ret;
  899. dev_info(&f34->fn->dev, "Writing config (%d bytes)...\n",
  900. f34->v7.img.ui_config.size);
  901. f34->v7.config_area = v7_UI_CONFIG_AREA;
  902. ret = rmi_f34v7_write_ui_config(f34);
  903. if (ret < 0)
  904. return ret;
  905. if (f34->v7.has_display_cfg && f34->v7.img.contains_display_cfg) {
  906. dev_info(&f34->fn->dev, "Writing display config...\n");
  907. ret = rmi_f34v7_write_dp_config(f34);
  908. if (ret < 0)
  909. return ret;
  910. }
  911. if (f34->v7.has_guest_code && f34->v7.img.contains_guest_code) {
  912. dev_info(&f34->fn->dev, "Writing guest code...\n");
  913. ret = rmi_f34v7_write_guest_code(f34);
  914. if (ret < 0)
  915. return ret;
  916. }
  917. return 0;
  918. }
  919. static int rmi_f34v7_enter_flash_prog(struct f34_data *f34)
  920. {
  921. int ret;
  922. f34->fn->rmi_dev->driver->set_irq_bits(f34->fn->rmi_dev, f34->fn->irq_mask);
  923. ret = rmi_f34v7_read_flash_status(f34);
  924. if (ret < 0)
  925. return ret;
  926. if (f34->v7.in_bl_mode) {
  927. dev_info(&f34->fn->dev, "%s: Device in bootloader mode\n",
  928. __func__);
  929. return 0;
  930. }
  931. init_completion(&f34->v7.cmd_done);
  932. ret = rmi_f34v7_write_command(f34, v7_CMD_ENABLE_FLASH_PROG);
  933. if (ret < 0)
  934. return ret;
  935. ret = rmi_f34v7_check_command_status(f34, F34_ENABLE_WAIT_MS);
  936. if (ret < 0)
  937. return ret;
  938. return 0;
  939. }
  940. int rmi_f34v7_start_reflash(struct f34_data *f34, const struct firmware *fw)
  941. {
  942. int ret = 0;
  943. f34->v7.config_area = v7_UI_CONFIG_AREA;
  944. f34->v7.image = fw->data;
  945. ret = rmi_f34v7_parse_image_info(f34);
  946. if (ret < 0)
  947. return ret;
  948. dev_info(&f34->fn->dev, "Firmware image OK\n");
  949. return rmi_f34v7_enter_flash_prog(f34);
  950. }
  951. int rmi_f34v7_probe(struct f34_data *f34)
  952. {
  953. int ret;
  954. /* Read bootloader version */
  955. ret = rmi_read_block(f34->fn->rmi_dev,
  956. f34->fn->fd.query_base_addr + V7_BOOTLOADER_ID_OFFSET,
  957. f34->bootloader_id,
  958. sizeof(f34->bootloader_id));
  959. if (ret < 0) {
  960. dev_err(&f34->fn->dev, "%s: Failed to read bootloader ID\n",
  961. __func__);
  962. return ret;
  963. }
  964. if (f34->bootloader_id[1] == '5') {
  965. f34->bl_version = 5;
  966. } else if (f34->bootloader_id[1] == '6') {
  967. f34->bl_version = 6;
  968. } else if (f34->bootloader_id[1] == 7) {
  969. f34->bl_version = 7;
  970. } else if (f34->bootloader_id[1] == 8) {
  971. f34->bl_version = 8;
  972. } else {
  973. dev_err(&f34->fn->dev,
  974. "%s: Unrecognized bootloader version: %d (%c) %d (%c)\n",
  975. __func__,
  976. f34->bootloader_id[0], f34->bootloader_id[0],
  977. f34->bootloader_id[1], f34->bootloader_id[1]);
  978. return -EINVAL;
  979. }
  980. memset(&f34->v7.blkcount, 0x00, sizeof(f34->v7.blkcount));
  981. memset(&f34->v7.phyaddr, 0x00, sizeof(f34->v7.phyaddr));
  982. init_completion(&f34->v7.cmd_done);
  983. ret = rmi_f34v7_read_queries(f34);
  984. if (ret < 0)
  985. return ret;
  986. return 0;
  987. }