ipa_dt_replacement.c 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2018-2019, The Linux Foundation. All rights reserved.
  4. */
  5. #include <linux/errno.h>
  6. #include <linux/string.h>
  7. #include <linux/ioport.h>
  8. #include <linux/msm_ipa.h>
  9. #include "ipa_i.h"
  10. #include "ipa_emulation_stubs.h"
  11. # undef strsame
  12. # define strsame(x, y) \
  13. (!strcmp((x), (y)))
  14. /*
  15. * The following enum values used to index tables below.
  16. */
  17. enum dtsi_index_e {
  18. DTSI_INDEX_3_5_1 = 0,
  19. DTSI_INDEX_4_0 = 1,
  20. DTSI_INDEX_4_5 = 2,
  21. };
  22. struct dtsi_replacement_u32 {
  23. char *key;
  24. u32 value;
  25. };
  26. struct dtsi_replacement_u32_table {
  27. struct dtsi_replacement_u32 *p_table;
  28. u32 num_entries;
  29. };
  30. struct dtsi_replacement_bool {
  31. char *key;
  32. bool value;
  33. };
  34. struct dtsi_replacement_bool_table {
  35. struct dtsi_replacement_bool *p_table;
  36. u32 num_entries;
  37. };
  38. struct dtsi_replacement_u32_array {
  39. char *key;
  40. u32 *p_value;
  41. u32 num_elements;
  42. };
  43. struct dtsi_replacement_u32_array_table {
  44. struct dtsi_replacement_u32_array *p_table;
  45. u32 num_entries;
  46. };
  47. struct dtsi_replacement_resource_table {
  48. struct resource *p_table;
  49. u32 num_entries;
  50. };
  51. /*
  52. * Any of the data below with _4_5 in the name represent data taken
  53. * from the 4.5 dtsi file.
  54. *
  55. * Any of the data below with _4_0 in the name represent data taken
  56. * from the 4.0 dtsi file.
  57. *
  58. * Any of the data below with _3_5_1 in the name represent data taken
  59. * from the 3.5.1 dtsi file.
  60. */
  61. static struct dtsi_replacement_bool ipa3_plat_drv_bool_4_5[] = {
  62. {"qcom,use-ipa-tethering-bridge", true},
  63. {"qcom,modem-cfg-emb-pipe-flt", true},
  64. {"qcom,ipa-wdi2", false},
  65. {"qcom,use-64-bit-dma-mask", false},
  66. {"qcom,bandwidth-vote-for-ipa", true},
  67. {"qcom,skip-uc-pipe-reset", false},
  68. {"qcom,tethered-flow-control", false},
  69. {"qcom,use-rg10-limitation-mitigation", false},
  70. {"qcom,do-not-use-ch-gsi-20", false},
  71. {"qcom,use-ipa-pm", true},
  72. {"qcom,register-collection-on-crash", true},
  73. {"qcom,testbus-collection-on-crash", true},
  74. {"qcom,non-tn-collection-on-crash", true},
  75. };
  76. static struct dtsi_replacement_bool ipa3_plat_drv_bool_4_0[] = {
  77. {"qcom,use-ipa-tethering-bridge", true},
  78. {"qcom,modem-cfg-emb-pipe-flt", true},
  79. {"qcom,ipa-wdi2", true},
  80. {"qcom,use-64-bit-dma-mask", false},
  81. {"qcom,bandwidth-vote-for-ipa", false},
  82. {"qcom,skip-uc-pipe-reset", false},
  83. {"qcom,tethered-flow-control", true},
  84. {"qcom,use-rg10-limitation-mitigation", false},
  85. {"qcom,do-not-use-ch-gsi-20", false},
  86. {"qcom,use-ipa-pm", false},
  87. {"qcom,register-collection-on-crash", true},
  88. {"qcom,testbus-collection-on-crash", true},
  89. {"qcom,non-tn-collection-on-crash", true},
  90. };
  91. static struct dtsi_replacement_bool ipa3_plat_drv_bool_3_5_1[] = {
  92. {"qcom,use-ipa-tethering-bridge", true},
  93. {"qcom,modem-cfg-emb-pipe-flt", true},
  94. {"qcom,ipa-wdi2", true},
  95. {"qcom,use-64-bit-dma-mask", false},
  96. {"qcom,bandwidth-vote-for-ipa", true},
  97. {"qcom,skip-uc-pipe-reset", false},
  98. {"qcom,tethered-flow-control", false},
  99. {"qcom,use-rg10-limitation-mitigation", false},
  100. {"qcom,do-not-use-ch-gsi-20", false},
  101. {"qcom,use-ipa-pm", false},
  102. {"qcom,register-collection-on-crash", true},
  103. {"qcom,testbus-collection-on-crash", true},
  104. {"qcom,non-tn-collection-on-crash", true},
  105. };
  106. static struct dtsi_replacement_bool_table
  107. ipa3_plat_drv_bool_table[] = {
  108. { ipa3_plat_drv_bool_3_5_1,
  109. ARRAY_SIZE(ipa3_plat_drv_bool_3_5_1) },
  110. { ipa3_plat_drv_bool_4_0,
  111. ARRAY_SIZE(ipa3_plat_drv_bool_4_0) },
  112. { ipa3_plat_drv_bool_4_5,
  113. ARRAY_SIZE(ipa3_plat_drv_bool_4_5) },
  114. };
  115. static struct dtsi_replacement_u32 ipa3_plat_drv_u32_4_5[] = {
  116. {"qcom,ipa-hw-ver", IPA_HW_v4_5},
  117. {"qcom,ipa-hw-mode", 3},
  118. {"qcom,wan-rx-ring-size", 192},
  119. {"qcom,lan-rx-ring-size", 192},
  120. {"qcom,ee", 0},
  121. {"qcom,msm-bus,num-cases", 5},
  122. {"emulator-bar0-offset", 0x01C00000},
  123. {"qcom,entire-ipa-block-size", 0x00100000},
  124. };
  125. static struct dtsi_replacement_u32 ipa3_plat_drv_u32_4_0[] = {
  126. {"qcom,ipa-hw-ver", IPA_HW_v4_0},
  127. {"qcom,ipa-hw-mode", 3},
  128. {"qcom,wan-rx-ring-size", 192},
  129. {"qcom,lan-rx-ring-size", 192},
  130. {"qcom,ee", 0},
  131. {"emulator-bar0-offset", 0x01C00000},
  132. {"qcom,entire-ipa-block-size", 0x00100000},
  133. };
  134. static struct dtsi_replacement_u32 ipa3_plat_drv_u32_3_5_1[] = {
  135. {"qcom,ipa-hw-ver", IPA_HW_v3_5_1},
  136. {"qcom,ipa-hw-mode", 3},
  137. {"qcom,wan-rx-ring-size", 192},
  138. {"qcom,lan-rx-ring-size", 192},
  139. {"qcom,ee", 0},
  140. {"emulator-bar0-offset", 0x01C00000},
  141. {"qcom,entire-ipa-block-size", 0x00100000},
  142. };
  143. static struct dtsi_replacement_u32_table ipa3_plat_drv_u32_table[] = {
  144. { ipa3_plat_drv_u32_3_5_1,
  145. ARRAY_SIZE(ipa3_plat_drv_u32_3_5_1) },
  146. { ipa3_plat_drv_u32_4_0,
  147. ARRAY_SIZE(ipa3_plat_drv_u32_4_0) },
  148. { ipa3_plat_drv_u32_4_5,
  149. ARRAY_SIZE(ipa3_plat_drv_u32_4_5) },
  150. };
  151. static u32 mhi_event_ring_id_limits_array_4_5[] = {
  152. 9, 10
  153. };
  154. static u32 mhi_event_ring_id_limits_array_4_0[] = {
  155. 9, 10
  156. };
  157. static u32 mhi_event_ring_id_limits_array_3_5_1[] = {
  158. IPA_MHI_GSI_EVENT_RING_ID_START, IPA_MHI_GSI_EVENT_RING_ID_END
  159. };
  160. static u32 ipa_tz_unlock_reg_array_4_5[] = {
  161. 0x04043583c, 0x00001000
  162. };
  163. static u32 ipa_throughput_thresh_array_4_5[] = {
  164. 310, 600, 1000
  165. };
  166. static u32 ipa_tz_unlock_reg_array_4_0[] = {
  167. 0x04043583c, 0x00001000
  168. };
  169. static u32 ipa_tz_unlock_reg_array_3_5_1[] = {
  170. 0x04043583c, 0x00001000
  171. };
  172. struct dtsi_replacement_u32_array ipa3_plat_drv_u32_array_4_5[] = {
  173. {"qcom,mhi-event-ring-id-limits",
  174. mhi_event_ring_id_limits_array_4_5,
  175. ARRAY_SIZE(mhi_event_ring_id_limits_array_4_5) },
  176. {"qcom,ipa-tz-unlock-reg",
  177. ipa_tz_unlock_reg_array_4_5,
  178. ARRAY_SIZE(ipa_tz_unlock_reg_array_4_5) },
  179. {"qcom,throughput-threshold",
  180. ipa_throughput_thresh_array_4_5,
  181. ARRAY_SIZE(ipa_throughput_thresh_array_4_5) },
  182. };
  183. struct dtsi_replacement_u32_array ipa3_plat_drv_u32_array_4_0[] = {
  184. {"qcom,mhi-event-ring-id-limits",
  185. mhi_event_ring_id_limits_array_4_0,
  186. ARRAY_SIZE(mhi_event_ring_id_limits_array_4_0) },
  187. {"qcom,ipa-tz-unlock-reg",
  188. ipa_tz_unlock_reg_array_4_0,
  189. ARRAY_SIZE(ipa_tz_unlock_reg_array_4_0) },
  190. };
  191. struct dtsi_replacement_u32_array ipa3_plat_drv_u32_array_3_5_1[] = {
  192. {"qcom,mhi-event-ring-id-limits",
  193. mhi_event_ring_id_limits_array_3_5_1,
  194. ARRAY_SIZE(mhi_event_ring_id_limits_array_3_5_1) },
  195. {"qcom,ipa-tz-unlock-reg",
  196. ipa_tz_unlock_reg_array_3_5_1,
  197. ARRAY_SIZE(ipa_tz_unlock_reg_array_3_5_1) },
  198. };
  199. struct dtsi_replacement_u32_array_table
  200. ipa3_plat_drv_u32_array_table[] = {
  201. { ipa3_plat_drv_u32_array_3_5_1,
  202. ARRAY_SIZE(ipa3_plat_drv_u32_array_3_5_1) },
  203. { ipa3_plat_drv_u32_array_4_0,
  204. ARRAY_SIZE(ipa3_plat_drv_u32_array_4_0) },
  205. { ipa3_plat_drv_u32_array_4_5,
  206. ARRAY_SIZE(ipa3_plat_drv_u32_array_4_5) },
  207. };
  208. #define INTCTRL_OFFSET 0x083C0000
  209. #define INTCTRL_SIZE 0x00000110
  210. #define IPA_BASE_OFFSET_4_5 0x01e00000
  211. #define IPA_BASE_SIZE_4_5 0x000c0000
  212. #define GSI_BASE_OFFSET_4_5 0x01e04000
  213. #define GSI_BASE_SIZE_4_5 0x00023000
  214. struct resource ipa3_plat_drv_resource_4_5[] = {
  215. /*
  216. * PLEASE NOTE: The following offset values below ("ipa-base",
  217. * "gsi-base", and "intctrl-base") are used to calculate
  218. * offsets relative to the PCI BAR0 address provided by the
  219. * PCI probe. After their use to calculate the offsets, they
  220. * are not used again, since PCI ultimately dictates where
  221. * things live.
  222. */
  223. {
  224. IPA_BASE_OFFSET_4_5,
  225. (IPA_BASE_OFFSET_4_5 + IPA_BASE_SIZE_4_5),
  226. "ipa-base",
  227. IORESOURCE_MEM,
  228. 0,
  229. NULL,
  230. NULL,
  231. NULL
  232. },
  233. {
  234. GSI_BASE_OFFSET_4_5,
  235. (GSI_BASE_OFFSET_4_5 + GSI_BASE_SIZE_4_5),
  236. "gsi-base",
  237. IORESOURCE_MEM,
  238. 0,
  239. NULL,
  240. NULL,
  241. NULL
  242. },
  243. /*
  244. * The following entry is germane only to the emulator
  245. * environment. It is needed to locate the emulator's PCI
  246. * interrupt controller...
  247. */
  248. {
  249. INTCTRL_OFFSET,
  250. (INTCTRL_OFFSET + INTCTRL_SIZE),
  251. "intctrl-base",
  252. IORESOURCE_MEM,
  253. 0,
  254. NULL,
  255. NULL,
  256. NULL
  257. },
  258. {
  259. IPA_PIPE_MEM_START_OFST,
  260. (IPA_PIPE_MEM_START_OFST + IPA_PIPE_MEM_SIZE),
  261. "ipa-pipe-mem",
  262. IORESOURCE_MEM,
  263. 0,
  264. NULL,
  265. NULL,
  266. NULL
  267. },
  268. {
  269. 0,
  270. 0,
  271. "gsi-irq",
  272. IORESOURCE_IRQ,
  273. 0,
  274. NULL,
  275. NULL,
  276. NULL
  277. },
  278. {
  279. 0,
  280. 0,
  281. "ipa-irq",
  282. IORESOURCE_IRQ,
  283. 0,
  284. NULL,
  285. NULL,
  286. NULL
  287. },
  288. };
  289. #define IPA_BASE_OFFSET_4_0 0x01e00000
  290. #define IPA_BASE_SIZE_4_0 0x00034000
  291. #define GSI_BASE_OFFSET_4_0 0x01e04000
  292. #define GSI_BASE_SIZE_4_0 0x00028000
  293. struct resource ipa3_plat_drv_resource_4_0[] = {
  294. /*
  295. * PLEASE NOTE: The following offset values below ("ipa-base",
  296. * "gsi-base", and "intctrl-base") are used to calculate
  297. * offsets relative to the PCI BAR0 address provided by the
  298. * PCI probe. After their use to calculate the offsets, they
  299. * are not used again, since PCI ultimately dictates where
  300. * things live.
  301. */
  302. {
  303. IPA_BASE_OFFSET_4_0,
  304. (IPA_BASE_OFFSET_4_0 + IPA_BASE_SIZE_4_0),
  305. "ipa-base",
  306. IORESOURCE_MEM,
  307. 0,
  308. NULL,
  309. NULL,
  310. NULL
  311. },
  312. {
  313. GSI_BASE_OFFSET_4_0,
  314. (GSI_BASE_OFFSET_4_0 + GSI_BASE_SIZE_4_0),
  315. "gsi-base",
  316. IORESOURCE_MEM,
  317. 0,
  318. NULL,
  319. NULL,
  320. NULL
  321. },
  322. /*
  323. * The following entry is germane only to the emulator
  324. * environment. It is needed to locate the emulator's PCI
  325. * interrupt controller...
  326. */
  327. {
  328. INTCTRL_OFFSET,
  329. (INTCTRL_OFFSET + INTCTRL_SIZE),
  330. "intctrl-base",
  331. IORESOURCE_MEM,
  332. 0,
  333. NULL,
  334. NULL,
  335. NULL
  336. },
  337. {
  338. IPA_PIPE_MEM_START_OFST,
  339. (IPA_PIPE_MEM_START_OFST + IPA_PIPE_MEM_SIZE),
  340. "ipa-pipe-mem",
  341. IORESOURCE_MEM,
  342. 0,
  343. NULL,
  344. NULL,
  345. NULL
  346. },
  347. {
  348. 0,
  349. 0,
  350. "gsi-irq",
  351. IORESOURCE_IRQ,
  352. 0,
  353. NULL,
  354. NULL,
  355. NULL
  356. },
  357. {
  358. 0,
  359. 0,
  360. "ipa-irq",
  361. IORESOURCE_IRQ,
  362. 0,
  363. NULL,
  364. NULL,
  365. NULL
  366. },
  367. };
  368. #define IPA_BASE_OFFSET_3_5_1 0x01e00000
  369. #define IPA_BASE_SIZE_3_5_1 0x00034000
  370. #define GSI_BASE_OFFSET_3_5_1 0x01e04000
  371. #define GSI_BASE_SIZE_3_5_1 0x0002c000
  372. struct resource ipa3_plat_drv_resource_3_5_1[] = {
  373. /*
  374. * PLEASE NOTE: The following offset values below ("ipa-base",
  375. * "gsi-base", and "intctrl-base") are used to calculate
  376. * offsets relative to the PCI BAR0 address provided by the
  377. * PCI probe. After their use to calculate the offsets, they
  378. * are not used again, since PCI ultimately dictates where
  379. * things live.
  380. */
  381. {
  382. IPA_BASE_OFFSET_3_5_1,
  383. (IPA_BASE_OFFSET_3_5_1 + IPA_BASE_SIZE_3_5_1),
  384. "ipa-base",
  385. IORESOURCE_MEM,
  386. 0,
  387. NULL,
  388. NULL,
  389. NULL
  390. },
  391. {
  392. GSI_BASE_OFFSET_3_5_1,
  393. (GSI_BASE_OFFSET_3_5_1 + GSI_BASE_SIZE_3_5_1),
  394. "gsi-base",
  395. IORESOURCE_MEM,
  396. 0,
  397. NULL,
  398. NULL,
  399. NULL
  400. },
  401. /*
  402. * The following entry is germane only to the emulator
  403. * environment. It is needed to locate the emulator's PCI
  404. * interrupt controller...
  405. */
  406. {
  407. INTCTRL_OFFSET,
  408. (INTCTRL_OFFSET + INTCTRL_SIZE),
  409. "intctrl-base",
  410. IORESOURCE_MEM,
  411. 0,
  412. NULL,
  413. NULL,
  414. NULL
  415. },
  416. {
  417. IPA_PIPE_MEM_START_OFST,
  418. (IPA_PIPE_MEM_START_OFST + IPA_PIPE_MEM_SIZE),
  419. "ipa-pipe-mem",
  420. IORESOURCE_MEM,
  421. 0,
  422. NULL,
  423. NULL,
  424. NULL
  425. },
  426. {
  427. 0,
  428. 0,
  429. "gsi-irq",
  430. IORESOURCE_IRQ,
  431. 0,
  432. NULL,
  433. NULL,
  434. NULL
  435. },
  436. {
  437. 0,
  438. 0,
  439. "ipa-irq",
  440. IORESOURCE_IRQ,
  441. 0,
  442. NULL,
  443. NULL,
  444. NULL
  445. },
  446. };
  447. struct dtsi_replacement_resource_table
  448. ipa3_plat_drv_resource_table[] = {
  449. { ipa3_plat_drv_resource_3_5_1,
  450. ARRAY_SIZE(ipa3_plat_drv_resource_3_5_1) },
  451. { ipa3_plat_drv_resource_4_0,
  452. ARRAY_SIZE(ipa3_plat_drv_resource_4_0) },
  453. { ipa3_plat_drv_resource_4_5,
  454. ARRAY_SIZE(ipa3_plat_drv_resource_4_5) },
  455. };
  456. /*
  457. * The following code uses the data above...
  458. */
  459. static u32 emulator_type_to_index(void)
  460. {
  461. /*
  462. * Use the input parameter to the IPA driver loadable module,
  463. * which specifies the type of hardware the driver is running
  464. * on.
  465. */
  466. u32 index = DTSI_INDEX_4_0;
  467. uint emulation_type = ipa3_get_emulation_type();
  468. switch (emulation_type) {
  469. case IPA_HW_v3_5_1:
  470. index = DTSI_INDEX_3_5_1;
  471. break;
  472. case IPA_HW_v4_0:
  473. index = DTSI_INDEX_4_0;
  474. break;
  475. case IPA_HW_v4_5:
  476. index = DTSI_INDEX_4_5;
  477. break;
  478. default:
  479. break;
  480. }
  481. IPADBG("emulation_type(%u) emulation_index(%u)\n",
  482. emulation_type, index);
  483. return index;
  484. }
  485. /* From include/linux/of.h */
  486. /**
  487. * emulator_of_property_read_bool - Find from a property
  488. * @np: device node from which the property value is to be read.
  489. * @propname: name of the property to be searched.
  490. *
  491. * Search for a property in a device node.
  492. * Returns true if the property exists false otherwise.
  493. */
  494. bool emulator_of_property_read_bool(
  495. const struct device_node *np,
  496. const char *propname)
  497. {
  498. u16 i;
  499. u32 index;
  500. struct dtsi_replacement_bool *ipa3_plat_drv_boolP;
  501. /*
  502. * Get the index for the type of hardware we're running on.
  503. * This is used as a table index.
  504. */
  505. index = emulator_type_to_index();
  506. if (index >= ARRAY_SIZE(ipa3_plat_drv_bool_table)) {
  507. IPADBG(
  508. "Did not find ipa3_plat_drv_bool_table for index %u\n",
  509. index);
  510. return false;
  511. }
  512. ipa3_plat_drv_boolP =
  513. ipa3_plat_drv_bool_table[index].p_table;
  514. for (i = 0;
  515. i < ipa3_plat_drv_bool_table[index].num_entries;
  516. i++) {
  517. if (strsame(ipa3_plat_drv_boolP[i].key, propname)) {
  518. IPADBG(
  519. "Found value %u for propname %s index %u\n",
  520. ipa3_plat_drv_boolP[i].value,
  521. propname,
  522. index);
  523. return ipa3_plat_drv_boolP[i].value;
  524. }
  525. }
  526. IPADBG("Did not find match for propname %s index %u\n",
  527. propname,
  528. index);
  529. return false;
  530. }
  531. /* From include/linux/of.h */
  532. int emulator_of_property_read_u32(
  533. const struct device_node *np,
  534. const char *propname,
  535. u32 *out_value)
  536. {
  537. u16 i;
  538. u32 index;
  539. struct dtsi_replacement_u32 *ipa3_plat_drv_u32P;
  540. /*
  541. * Get the index for the type of hardware we're running on.
  542. * This is used as a table index.
  543. */
  544. index = emulator_type_to_index();
  545. if (index >= ARRAY_SIZE(ipa3_plat_drv_u32_table)) {
  546. IPADBG(
  547. "Did not find ipa3_plat_drv_u32_table for index %u\n",
  548. index);
  549. return false;
  550. }
  551. ipa3_plat_drv_u32P =
  552. ipa3_plat_drv_u32_table[index].p_table;
  553. for (i = 0;
  554. i < ipa3_plat_drv_u32_table[index].num_entries;
  555. i++) {
  556. if (strsame(ipa3_plat_drv_u32P[i].key, propname)) {
  557. *out_value = ipa3_plat_drv_u32P[i].value;
  558. IPADBG(
  559. "Found value %u for propname %s index %u\n",
  560. ipa3_plat_drv_u32P[i].value,
  561. propname,
  562. index);
  563. return 0;
  564. }
  565. }
  566. IPADBG("Did not find match for propname %s index %u\n",
  567. propname,
  568. index);
  569. return -EINVAL;
  570. }
  571. /* From include/linux/of.h */
  572. /**
  573. * emulator_of_property_read_u32_array - Find and read an array of 32
  574. * bit integers from a property.
  575. *
  576. * @np: device node from which the property value is to be read.
  577. * @propname: name of the property to be searched.
  578. * @out_values: pointer to return value, modified only if return value is 0.
  579. * @sz: number of array elements to read
  580. *
  581. * Search for a property in a device node and read 32-bit value(s) from
  582. * it. Returns 0 on success, -EINVAL if the property does not exist,
  583. * -ENODATA if property does not have a value, and -EOVERFLOW if the
  584. * property data isn't large enough.
  585. *
  586. * The out_values is modified only if a valid u32 value can be decoded.
  587. */
  588. int emulator_of_property_read_u32_array(
  589. const struct device_node *np,
  590. const char *propname,
  591. u32 *out_values,
  592. size_t sz)
  593. {
  594. u16 i;
  595. u32 index;
  596. struct dtsi_replacement_u32_array *u32_arrayP;
  597. /*
  598. * Get the index for the type of hardware we're running on.
  599. * This is used as a table index.
  600. */
  601. index = emulator_type_to_index();
  602. if (index >= ARRAY_SIZE(ipa3_plat_drv_u32_array_table)) {
  603. IPADBG(
  604. "Did not find ipa3_plat_drv_u32_array_table for index %u\n",
  605. index);
  606. return false;
  607. }
  608. u32_arrayP =
  609. ipa3_plat_drv_u32_array_table[index].p_table;
  610. for (i = 0;
  611. i < ipa3_plat_drv_u32_array_table[index].num_entries;
  612. i++) {
  613. if (strsame(
  614. u32_arrayP[i].key, propname)) {
  615. u32 num_elements =
  616. u32_arrayP[i].num_elements;
  617. u32 *p_element =
  618. &u32_arrayP[i].p_value[0];
  619. size_t j = 0;
  620. if (num_elements > sz) {
  621. IPAERR(
  622. "Found array of %u values for propname %s; only room for %u elements in copy buffer\n",
  623. num_elements,
  624. propname,
  625. (unsigned int) sz);
  626. return -EOVERFLOW;
  627. }
  628. while (j++ < num_elements)
  629. *out_values++ = *p_element++;
  630. IPADBG(
  631. "Found array of values starting with %u for propname %s index %u\n",
  632. u32_arrayP[i].p_value[0],
  633. propname,
  634. index);
  635. return 0;
  636. }
  637. }
  638. IPADBG("Did not find match for propname %s index %u\n",
  639. propname,
  640. index);
  641. return -EINVAL;
  642. }
  643. /* From drivers/base/platform.c */
  644. /**
  645. * emulator_platform_get_resource_byname - get a resource for a device by name
  646. * @dev: platform device
  647. * @type: resource type
  648. * @name: resource name
  649. */
  650. struct resource *emulator_platform_get_resource_byname(
  651. struct platform_device *dev,
  652. unsigned int type,
  653. const char *name)
  654. {
  655. u16 i;
  656. u32 index;
  657. struct resource *ipa3_plat_drv_resourceP;
  658. /*
  659. * Get the index for the type of hardware we're running on.
  660. * This is used as a table index.
  661. */
  662. index = emulator_type_to_index();
  663. if (index >= ARRAY_SIZE(ipa3_plat_drv_resource_table)) {
  664. IPADBG(
  665. "Did not find ipa3_plat_drv_resource_table for index %u\n",
  666. index);
  667. return false;
  668. }
  669. ipa3_plat_drv_resourceP =
  670. ipa3_plat_drv_resource_table[index].p_table;
  671. for (i = 0;
  672. i < ipa3_plat_drv_resource_table[index].num_entries;
  673. i++) {
  674. struct resource *r = &ipa3_plat_drv_resourceP[i];
  675. if (type == resource_type(r) && strsame(r->name, name)) {
  676. IPADBG(
  677. "Found start 0x%x size %u for name %s index %u\n",
  678. (unsigned int) (r->start),
  679. (unsigned int) (resource_size(r)),
  680. name,
  681. index);
  682. return r;
  683. }
  684. }
  685. IPADBG("Did not find match for name %s index %u\n",
  686. name,
  687. index);
  688. return NULL;
  689. }
  690. /* From drivers/of/base.c */
  691. /**
  692. * emulator_of_property_count_elems_of_size - Count the number of
  693. * elements in a property
  694. *
  695. * @np: device node from which the property value is to
  696. * be read. Not used.
  697. * @propname: name of the property to be searched.
  698. * @elem_size: size of the individual element
  699. *
  700. * Search for a property and count the number of elements of size
  701. * elem_size in it. Returns number of elements on success, -EINVAL if
  702. * the property does not exist or its length does not match a multiple
  703. * of elem_size and -ENODATA if the property does not have a value.
  704. */
  705. int emulator_of_property_count_elems_of_size(
  706. const struct device_node *np,
  707. const char *propname,
  708. int elem_size)
  709. {
  710. u32 index;
  711. /*
  712. * Get the index for the type of hardware we're running on.
  713. * This is used as a table index.
  714. */
  715. index = emulator_type_to_index();
  716. /*
  717. * Use elem_size to determine which table to search for the
  718. * specified property name
  719. */
  720. if (elem_size == sizeof(u32)) {
  721. u16 i;
  722. struct dtsi_replacement_u32_array *u32_arrayP;
  723. if (index >= ARRAY_SIZE(ipa3_plat_drv_u32_array_table)) {
  724. IPADBG(
  725. "Did not find ipa3_plat_drv_u32_array_table for index %u\n",
  726. index);
  727. return false;
  728. }
  729. u32_arrayP =
  730. ipa3_plat_drv_u32_array_table[index].p_table;
  731. for (i = 0;
  732. i < ipa3_plat_drv_u32_array_table[index].num_entries;
  733. i++) {
  734. if (strsame(u32_arrayP[i].key, propname)) {
  735. if (u32_arrayP[i].p_value == NULL) {
  736. IPADBG(
  737. "Found no elements for propname %s index %u\n",
  738. propname,
  739. index);
  740. return -ENODATA;
  741. }
  742. IPADBG(
  743. "Found %u elements for propname %s index %u\n",
  744. u32_arrayP[i].num_elements,
  745. propname,
  746. index);
  747. return u32_arrayP[i].num_elements;
  748. }
  749. }
  750. IPADBG(
  751. "Found no match in table with elem_size %d for propname %s index %u\n",
  752. elem_size,
  753. propname,
  754. index);
  755. return -EINVAL;
  756. }
  757. IPAERR(
  758. "Found no tables with element size %u to search for propname %s index %u\n",
  759. elem_size,
  760. propname,
  761. index);
  762. return -EINVAL;
  763. }
  764. int emulator_of_property_read_variable_u32_array(
  765. const struct device_node *np,
  766. const char *propname,
  767. u32 *out_values,
  768. size_t sz_min,
  769. size_t sz_max)
  770. {
  771. return emulator_of_property_read_u32_array(
  772. np, propname, out_values, sz_max);
  773. }
  774. resource_size_t emulator_resource_size(const struct resource *res)
  775. {
  776. return res->end - res->start + 1;
  777. }