ipa_uc_mhi.c 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2015-2019, The Linux Foundation. All rights reserved.
  4. */
  5. #include <linux/ipa.h>
  6. #include "ipa_i.h"
  7. /* MHI uC interface definitions */
  8. #define IPA_HW_INTERFACE_MHI_VERSION 0x0004
  9. #define IPA_HW_MAX_NUMBER_OF_CHANNELS 2
  10. #define IPA_HW_MAX_NUMBER_OF_EVENTRINGS 2
  11. #define IPA_HW_MAX_CHANNEL_HANDLE (IPA_HW_MAX_NUMBER_OF_CHANNELS-1)
  12. /**
  13. * Values that represent the MHI commands from CPU to IPA HW.
  14. * @IPA_CPU_2_HW_CMD_MHI_INIT: Initialize HW to be ready for MHI processing.
  15. * Once operation was completed HW shall respond with
  16. * IPA_HW_2_CPU_RESPONSE_CMD_COMPLETED.
  17. * @IPA_CPU_2_HW_CMD_MHI_INIT_CHANNEL: Initialize specific channel to be ready
  18. * to serve MHI transfers. Once initialization was completed HW shall
  19. * respond with IPA_HW_2_CPU_RESPONSE_MHI_CHANGE_CHANNEL_STATE.
  20. * IPA_HW_MHI_CHANNEL_STATE_ENABLE
  21. * @IPA_CPU_2_HW_CMD_MHI_UPDATE_MSI: Update MHI MSI interrupts data.
  22. * Once operation was completed HW shall respond with
  23. * IPA_HW_2_CPU_RESPONSE_CMD_COMPLETED.
  24. * @IPA_CPU_2_HW_CMD_MHI_CHANGE_CHANNEL_STATE: Change specific channel
  25. * processing state following host request. Once operation was completed
  26. * HW shall respond with IPA_HW_2_CPU_RESPONSE_MHI_CHANGE_CHANNEL_STATE.
  27. * @IPA_CPU_2_HW_CMD_MHI_DL_UL_SYNC_INFO: Info related to DL UL syncronization.
  28. * @IPA_CPU_2_HW_CMD_MHI_STOP_EVENT_UPDATE: Cmd to stop event ring processing.
  29. */
  30. enum ipa_cpu_2_hw_mhi_commands {
  31. IPA_CPU_2_HW_CMD_MHI_INIT
  32. = FEATURE_ENUM_VAL(IPA_HW_FEATURE_MHI, 0),
  33. IPA_CPU_2_HW_CMD_MHI_INIT_CHANNEL
  34. = FEATURE_ENUM_VAL(IPA_HW_FEATURE_MHI, 1),
  35. IPA_CPU_2_HW_CMD_MHI_UPDATE_MSI
  36. = FEATURE_ENUM_VAL(IPA_HW_FEATURE_MHI, 2),
  37. IPA_CPU_2_HW_CMD_MHI_CHANGE_CHANNEL_STATE
  38. = FEATURE_ENUM_VAL(IPA_HW_FEATURE_MHI, 3),
  39. IPA_CPU_2_HW_CMD_MHI_DL_UL_SYNC_INFO
  40. = FEATURE_ENUM_VAL(IPA_HW_FEATURE_MHI, 4),
  41. IPA_CPU_2_HW_CMD_MHI_STOP_EVENT_UPDATE
  42. = FEATURE_ENUM_VAL(IPA_HW_FEATURE_MHI, 5)
  43. };
  44. /**
  45. * Values that represent MHI related HW responses to CPU commands.
  46. * @IPA_HW_2_CPU_RESPONSE_MHI_CHANGE_CHANNEL_STATE: Response to
  47. * IPA_CPU_2_HW_CMD_MHI_INIT_CHANNEL or
  48. * IPA_CPU_2_HW_CMD_MHI_CHANGE_CHANNEL_STATE commands.
  49. */
  50. enum ipa_hw_2_cpu_mhi_responses {
  51. IPA_HW_2_CPU_RESPONSE_MHI_CHANGE_CHANNEL_STATE
  52. = FEATURE_ENUM_VAL(IPA_HW_FEATURE_MHI, 0),
  53. };
  54. /**
  55. * Values that represent MHI related HW event to be sent to CPU.
  56. * @IPA_HW_2_CPU_EVENT_MHI_CHANNEL_ERROR: Event specify the device detected an
  57. * error in an element from the transfer ring associated with the channel
  58. * @IPA_HW_2_CPU_EVENT_MHI_CHANNEL_WAKE_UP_REQUEST: Event specify a transport
  59. * interrupt was asserted when MHI engine is suspended
  60. */
  61. enum ipa_hw_2_cpu_mhi_events {
  62. IPA_HW_2_CPU_EVENT_MHI_CHANNEL_ERROR
  63. = FEATURE_ENUM_VAL(IPA_HW_FEATURE_MHI, 0),
  64. IPA_HW_2_CPU_EVENT_MHI_CHANNEL_WAKE_UP_REQUEST
  65. = FEATURE_ENUM_VAL(IPA_HW_FEATURE_MHI, 1),
  66. };
  67. /**
  68. * Channel error types.
  69. * @IPA_HW_CHANNEL_ERROR_NONE: No error persists.
  70. * @IPA_HW_CHANNEL_INVALID_RE_ERROR: Invalid Ring Element was detected
  71. */
  72. enum ipa_hw_channel_errors {
  73. IPA_HW_CHANNEL_ERROR_NONE,
  74. IPA_HW_CHANNEL_INVALID_RE_ERROR
  75. };
  76. /**
  77. * MHI error types.
  78. * @IPA_HW_INVALID_MMIO_ERROR: Invalid data read from MMIO space
  79. * @IPA_HW_INVALID_CHANNEL_ERROR: Invalid data read from channel context array
  80. * @IPA_HW_INVALID_EVENT_ERROR: Invalid data read from event ring context array
  81. * @IPA_HW_NO_ED_IN_RING_ERROR: No event descriptors are available to report on
  82. * secondary event ring
  83. * @IPA_HW_LINK_ERROR: Link error
  84. */
  85. enum ipa_hw_mhi_errors {
  86. IPA_HW_INVALID_MMIO_ERROR
  87. = FEATURE_ENUM_VAL(IPA_HW_FEATURE_MHI, 0),
  88. IPA_HW_INVALID_CHANNEL_ERROR
  89. = FEATURE_ENUM_VAL(IPA_HW_FEATURE_MHI, 1),
  90. IPA_HW_INVALID_EVENT_ERROR
  91. = FEATURE_ENUM_VAL(IPA_HW_FEATURE_MHI, 2),
  92. IPA_HW_NO_ED_IN_RING_ERROR
  93. = FEATURE_ENUM_VAL(IPA_HW_FEATURE_MHI, 4),
  94. IPA_HW_LINK_ERROR
  95. = FEATURE_ENUM_VAL(IPA_HW_FEATURE_MHI, 5),
  96. };
  97. /**
  98. * Structure referring to the common and MHI section of 128B shared memory
  99. * located in offset zero of SW Partition in IPA SRAM.
  100. * The shared memory is used for communication between IPA HW and CPU.
  101. * @common: common section in IPA SRAM
  102. * @interfaceVersionMhi: The MHI interface version as reported by HW
  103. * @mhiState: Overall MHI state
  104. * @reserved_2B: reserved
  105. * @mhiCnl0State: State of MHI channel 0.
  106. * The state carries information regarding the error type.
  107. * See IPA_HW_MHI_CHANNEL_STATES.
  108. * @mhiCnl0State: State of MHI channel 1.
  109. * @mhiCnl0State: State of MHI channel 2.
  110. * @mhiCnl0State: State of MHI channel 3
  111. * @mhiCnl0State: State of MHI channel 4.
  112. * @mhiCnl0State: State of MHI channel 5.
  113. * @mhiCnl0State: State of MHI channel 6.
  114. * @mhiCnl0State: State of MHI channel 7.
  115. * @reserved_37_34: reserved
  116. * @reserved_3B_38: reserved
  117. * @reserved_3F_3C: reserved
  118. */
  119. struct IpaHwSharedMemMhiMapping_t {
  120. struct IpaHwSharedMemCommonMapping_t common;
  121. u16 interfaceVersionMhi;
  122. u8 mhiState;
  123. u8 reserved_2B;
  124. u8 mhiCnl0State;
  125. u8 mhiCnl1State;
  126. u8 mhiCnl2State;
  127. u8 mhiCnl3State;
  128. u8 mhiCnl4State;
  129. u8 mhiCnl5State;
  130. u8 mhiCnl6State;
  131. u8 mhiCnl7State;
  132. u32 reserved_37_34;
  133. u32 reserved_3B_38;
  134. u32 reserved_3F_3C;
  135. };
  136. /**
  137. * Structure holding the parameters for IPA_CPU_2_HW_CMD_MHI_INIT command.
  138. * Parameters are sent as pointer thus should be reside in address accessible
  139. * to HW.
  140. * @msiAddress: The MSI base (in device space) used for asserting the interrupt
  141. * (MSI) associated with the event ring
  142. * mmioBaseAddress: The address (in device space) of MMIO structure in
  143. * host space
  144. * deviceMhiCtrlBaseAddress: Base address of the memory region in the device
  145. * address space where the MHI control data structures are allocated by
  146. * the host, including channel context array, event context array,
  147. * and rings. This value is used for host/device address translation.
  148. * deviceMhiDataBaseAddress: Base address of the memory region in the device
  149. * address space where the MHI data buffers are allocated by the host.
  150. * This value is used for host/device address translation.
  151. * firstChannelIndex: First channel ID. Doorbell 0 is mapped to this channel
  152. * firstEventRingIndex: First event ring ID. Doorbell 16 is mapped to this
  153. * event ring.
  154. */
  155. struct IpaHwMhiInitCmdData_t {
  156. u32 msiAddress;
  157. u32 mmioBaseAddress;
  158. u32 deviceMhiCtrlBaseAddress;
  159. u32 deviceMhiDataBaseAddress;
  160. u32 firstChannelIndex;
  161. u32 firstEventRingIndex;
  162. };
  163. /**
  164. * Structure holding the parameters for IPA_CPU_2_HW_CMD_MHI_INIT_CHANNEL
  165. * command. Parameters are sent as 32b immediate parameters.
  166. * @hannelHandle: The channel identifier as allocated by driver.
  167. * value is within the range 0 to IPA_HW_MAX_CHANNEL_HANDLE
  168. * @contexArrayIndex: Unique index for channels, between 0 and 255. The index is
  169. * used as an index in channel context array structures.
  170. * @bamPipeId: The IPA pipe number for pipe dedicated for this channel
  171. * @channelDirection: The direction of the channel as defined in the channel
  172. * type field (CHTYPE) in the channel context data structure.
  173. * @reserved: reserved.
  174. */
  175. union IpaHwMhiInitChannelCmdData_t {
  176. struct IpaHwMhiInitChannelCmdParams_t {
  177. u32 channelHandle:8;
  178. u32 contexArrayIndex:8;
  179. u32 bamPipeId:6;
  180. u32 channelDirection:2;
  181. u32 reserved:8;
  182. } params;
  183. u32 raw32b;
  184. };
  185. /**
  186. * Structure holding the parameters for IPA_CPU_2_HW_CMD_MHI_UPDATE_MSI command.
  187. * @msiAddress_low: The MSI lower base addr (in device space) used for asserting
  188. * the interrupt (MSI) associated with the event ring.
  189. * @msiAddress_hi: The MSI higher base addr (in device space) used for asserting
  190. * the interrupt (MSI) associated with the event ring.
  191. * @msiMask: Mask indicating number of messages assigned by the host to device
  192. * @msiData: Data Pattern to use when generating the MSI
  193. */
  194. struct IpaHwMhiMsiCmdData_t {
  195. u32 msiAddress_low;
  196. u32 msiAddress_hi;
  197. u32 msiMask;
  198. u32 msiData;
  199. };
  200. /**
  201. * Structure holding the parameters for
  202. * IPA_CPU_2_HW_CMD_MHI_CHANGE_CHANNEL_STATE command.
  203. * Parameters are sent as 32b immediate parameters.
  204. * @requestedState: The requested channel state as was indicated from Host.
  205. * Use IPA_HW_MHI_CHANNEL_STATES to specify the requested state
  206. * @channelHandle: The channel identifier as allocated by driver.
  207. * value is within the range 0 to IPA_HW_MAX_CHANNEL_HANDLE
  208. * @LPTransitionRejected: Indication that low power state transition was
  209. * rejected
  210. * @reserved: reserved
  211. */
  212. union IpaHwMhiChangeChannelStateCmdData_t {
  213. struct IpaHwMhiChangeChannelStateCmdParams_t {
  214. u32 requestedState:8;
  215. u32 channelHandle:8;
  216. u32 LPTransitionRejected:8;
  217. u32 reserved:8;
  218. } params;
  219. u32 raw32b;
  220. };
  221. /**
  222. * Structure holding the parameters for
  223. * IPA_CPU_2_HW_CMD_MHI_STOP_EVENT_UPDATE command.
  224. * Parameters are sent as 32b immediate parameters.
  225. * @channelHandle: The channel identifier as allocated by driver.
  226. * value is within the range 0 to IPA_HW_MAX_CHANNEL_HANDLE
  227. * @reserved: reserved
  228. */
  229. union IpaHwMhiStopEventUpdateData_t {
  230. struct IpaHwMhiStopEventUpdateDataParams_t {
  231. u32 channelHandle:8;
  232. u32 reserved:24;
  233. } params;
  234. u32 raw32b;
  235. };
  236. /**
  237. * Structure holding the parameters for
  238. * IPA_HW_2_CPU_RESPONSE_MHI_CHANGE_CHANNEL_STATE response.
  239. * Parameters are sent as 32b immediate parameters.
  240. * @state: The new channel state. In case state is not as requested this is
  241. * error indication for the last command
  242. * @channelHandle: The channel identifier
  243. * @additonalParams: For stop: the number of pending transport descriptors
  244. * currently queued
  245. */
  246. union IpaHwMhiChangeChannelStateResponseData_t {
  247. struct IpaHwMhiChangeChannelStateResponseParams_t {
  248. u32 state:8;
  249. u32 channelHandle:8;
  250. u32 additonalParams:16;
  251. } params;
  252. u32 raw32b;
  253. };
  254. /**
  255. * Structure holding the parameters for
  256. * IPA_HW_2_CPU_EVENT_MHI_CHANNEL_ERROR event.
  257. * Parameters are sent as 32b immediate parameters.
  258. * @errorType: Type of error - IPA_HW_CHANNEL_ERRORS
  259. * @channelHandle: The channel identifier as allocated by driver.
  260. * value is within the range 0 to IPA_HW_MAX_CHANNEL_HANDLE
  261. * @reserved: reserved
  262. */
  263. union IpaHwMhiChannelErrorEventData_t {
  264. struct IpaHwMhiChannelErrorEventParams_t {
  265. u32 errorType:8;
  266. u32 channelHandle:8;
  267. u32 reserved:16;
  268. } params;
  269. u32 raw32b;
  270. };
  271. /**
  272. * Structure holding the parameters for
  273. * IPA_HW_2_CPU_EVENT_MHI_CHANNEL_WAKE_UP_REQUEST event.
  274. * Parameters are sent as 32b immediate parameters.
  275. * @channelHandle: The channel identifier as allocated by driver.
  276. * value is within the range 0 to IPA_HW_MAX_CHANNEL_HANDLE
  277. * @reserved: reserved
  278. */
  279. union IpaHwMhiChannelWakeupEventData_t {
  280. struct IpaHwMhiChannelWakeupEventParams_t {
  281. u32 channelHandle:8;
  282. u32 reserved:24;
  283. } params;
  284. u32 raw32b;
  285. };
  286. /**
  287. * Structure holding the MHI Common statistics
  288. * @numULDLSync: Number of times UL activity trigged due to DL activity
  289. * @numULTimerExpired: Number of times UL Accm Timer expired
  290. */
  291. struct IpaHwStatsMhiCmnInfoData_t {
  292. u32 numULDLSync;
  293. u32 numULTimerExpired;
  294. u32 numChEvCtxWpRead;
  295. u32 reserved;
  296. };
  297. /**
  298. * Structure holding the MHI Channel statistics
  299. * @doorbellInt: The number of doorbell int
  300. * @reProccesed: The number of ring elements processed
  301. * @bamFifoFull: Number of times Bam Fifo got full
  302. * @bamFifoEmpty: Number of times Bam Fifo got empty
  303. * @bamFifoUsageHigh: Number of times Bam fifo usage went above 75%
  304. * @bamFifoUsageLow: Number of times Bam fifo usage went below 25%
  305. * @bamInt: Number of BAM Interrupts
  306. * @ringFull: Number of times Transfer Ring got full
  307. * @ringEmpty: umber of times Transfer Ring got empty
  308. * @ringUsageHigh: Number of times Transfer Ring usage went above 75%
  309. * @ringUsageLow: Number of times Transfer Ring usage went below 25%
  310. * @delayedMsi: Number of times device triggered MSI to host after
  311. * Interrupt Moderation Timer expiry
  312. * @immediateMsi: Number of times device triggered MSI to host immediately
  313. * @thresholdMsi: Number of times device triggered MSI due to max pending
  314. * events threshold reached
  315. * @numSuspend: Number of times channel was suspended
  316. * @numResume: Number of times channel was suspended
  317. * @num_OOB: Number of times we indicated that we are OOB
  318. * @num_OOB_timer_expiry: Number of times we indicated that we are OOB
  319. * after timer expiry
  320. * @num_OOB_moderation_timer_start: Number of times we started timer after
  321. * sending OOB and hitting OOB again before we processed threshold
  322. * number of packets
  323. * @num_db_mode_evt: Number of times we indicated that we are in Doorbell mode
  324. */
  325. struct IpaHwStatsMhiCnlInfoData_t {
  326. u32 doorbellInt;
  327. u32 reProccesed;
  328. u32 bamFifoFull;
  329. u32 bamFifoEmpty;
  330. u32 bamFifoUsageHigh;
  331. u32 bamFifoUsageLow;
  332. u32 bamInt;
  333. u32 ringFull;
  334. u32 ringEmpty;
  335. u32 ringUsageHigh;
  336. u32 ringUsageLow;
  337. u32 delayedMsi;
  338. u32 immediateMsi;
  339. u32 thresholdMsi;
  340. u32 numSuspend;
  341. u32 numResume;
  342. u32 num_OOB;
  343. u32 num_OOB_timer_expiry;
  344. u32 num_OOB_moderation_timer_start;
  345. u32 num_db_mode_evt;
  346. };
  347. /**
  348. * Structure holding the MHI statistics
  349. * @mhiCmnStats: Stats pertaining to MHI
  350. * @mhiCnlStats: Stats pertaining to each channel
  351. */
  352. struct IpaHwStatsMhiInfoData_t {
  353. struct IpaHwStatsMhiCmnInfoData_t mhiCmnStats;
  354. struct IpaHwStatsMhiCnlInfoData_t mhiCnlStats[
  355. IPA_HW_MAX_NUMBER_OF_CHANNELS];
  356. };
  357. /**
  358. * Structure holding the MHI Common Config info
  359. * @isDlUlSyncEnabled: Flag to indicate if DL-UL synchronization is enabled
  360. * @UlAccmVal: Out Channel(UL) accumulation time in ms when DL UL Sync is
  361. * enabled
  362. * @ulMsiEventThreshold: Threshold at which HW fires MSI to host for UL events
  363. * @dlMsiEventThreshold: Threshold at which HW fires MSI to host for DL events
  364. */
  365. struct IpaHwConfigMhiCmnInfoData_t {
  366. u8 isDlUlSyncEnabled;
  367. u8 UlAccmVal;
  368. u8 ulMsiEventThreshold;
  369. u8 dlMsiEventThreshold;
  370. };
  371. /**
  372. * Structure holding the parameters for MSI info data
  373. * @msiAddress_low: The MSI lower base addr (in device space) used for asserting
  374. * the interrupt (MSI) associated with the event ring.
  375. * @msiAddress_hi: The MSI higher base addr (in device space) used for asserting
  376. * the interrupt (MSI) associated with the event ring.
  377. * @msiMask: Mask indicating number of messages assigned by the host to device
  378. * @msiData: Data Pattern to use when generating the MSI
  379. */
  380. struct IpaHwConfigMhiMsiInfoData_t {
  381. u32 msiAddress_low;
  382. u32 msiAddress_hi;
  383. u32 msiMask;
  384. u32 msiData;
  385. };
  386. /**
  387. * Structure holding the MHI Channel Config info
  388. * @transferRingSize: The Transfer Ring size in terms of Ring Elements
  389. * @transferRingIndex: The Transfer Ring channel number as defined by host
  390. * @eventRingIndex: The Event Ring Index associated with this Transfer Ring
  391. * @bamPipeIndex: The BAM Pipe associated with this channel
  392. * @isOutChannel: Indication for the direction of channel
  393. * @reserved_0: Reserved byte for maintaining 4byte alignment
  394. * @reserved_1: Reserved byte for maintaining 4byte alignment
  395. */
  396. struct IpaHwConfigMhiCnlInfoData_t {
  397. u16 transferRingSize;
  398. u8 transferRingIndex;
  399. u8 eventRingIndex;
  400. u8 bamPipeIndex;
  401. u8 isOutChannel;
  402. u8 reserved_0;
  403. u8 reserved_1;
  404. };
  405. /**
  406. * Structure holding the MHI Event Config info
  407. * @msiVec: msi vector to invoke MSI interrupt
  408. * @intmodtValue: Interrupt moderation timer (in milliseconds)
  409. * @eventRingSize: The Event Ring size in terms of Ring Elements
  410. * @eventRingIndex: The Event Ring number as defined by host
  411. * @reserved_0: Reserved byte for maintaining 4byte alignment
  412. * @reserved_1: Reserved byte for maintaining 4byte alignment
  413. * @reserved_2: Reserved byte for maintaining 4byte alignment
  414. */
  415. struct IpaHwConfigMhiEventInfoData_t {
  416. u32 msiVec;
  417. u16 intmodtValue;
  418. u16 eventRingSize;
  419. u8 eventRingIndex;
  420. u8 reserved_0;
  421. u8 reserved_1;
  422. u8 reserved_2;
  423. };
  424. /**
  425. * Structure holding the MHI Config info
  426. * @mhiCmnCfg: Common Config pertaining to MHI
  427. * @mhiMsiCfg: Config pertaining to MSI config
  428. * @mhiCnlCfg: Config pertaining to each channel
  429. * @mhiEvtCfg: Config pertaining to each event Ring
  430. */
  431. struct IpaHwConfigMhiInfoData_t {
  432. struct IpaHwConfigMhiCmnInfoData_t mhiCmnCfg;
  433. struct IpaHwConfigMhiMsiInfoData_t mhiMsiCfg;
  434. struct IpaHwConfigMhiCnlInfoData_t mhiCnlCfg[
  435. IPA_HW_MAX_NUMBER_OF_CHANNELS];
  436. struct IpaHwConfigMhiEventInfoData_t mhiEvtCfg[
  437. IPA_HW_MAX_NUMBER_OF_EVENTRINGS];
  438. };
  439. struct ipa3_uc_mhi_ctx {
  440. u8 expected_responseOp;
  441. u32 expected_responseParams;
  442. void (*ready_cb)(void);
  443. void (*wakeup_request_cb)(void);
  444. u32 mhi_uc_stats_ofst;
  445. struct IpaHwStatsMhiInfoData_t *mhi_uc_stats_mmio;
  446. };
  447. #define PRINT_COMMON_STATS(x) \
  448. (nBytes += scnprintf(&dbg_buff[nBytes], size - nBytes, \
  449. #x "=0x%x\n", ipa3_uc_mhi_ctx->mhi_uc_stats_mmio->mhiCmnStats.x))
  450. #define PRINT_CHANNEL_STATS(ch, x) \
  451. (nBytes += scnprintf(&dbg_buff[nBytes], size - nBytes, \
  452. #x "=0x%x\n", ipa3_uc_mhi_ctx->mhi_uc_stats_mmio->mhiCnlStats[ch].x))
  453. struct ipa3_uc_mhi_ctx *ipa3_uc_mhi_ctx;
  454. static int ipa3_uc_mhi_response_hdlr(struct IpaHwSharedMemCommonMapping_t
  455. *uc_sram_mmio, u32 *uc_status)
  456. {
  457. IPADBG("responseOp=%d\n", uc_sram_mmio->responseOp);
  458. if (uc_sram_mmio->responseOp == ipa3_uc_mhi_ctx->expected_responseOp &&
  459. uc_sram_mmio->responseParams ==
  460. ipa3_uc_mhi_ctx->expected_responseParams) {
  461. *uc_status = 0;
  462. return 0;
  463. }
  464. return -EINVAL;
  465. }
  466. static void ipa3_uc_mhi_event_hdlr(struct IpaHwSharedMemCommonMapping_t
  467. *uc_sram_mmio)
  468. {
  469. if (ipa3_ctx->uc_ctx.uc_sram_mmio->eventOp ==
  470. IPA_HW_2_CPU_EVENT_MHI_CHANNEL_ERROR) {
  471. union IpaHwMhiChannelErrorEventData_t evt;
  472. IPAERR("Channel error\n");
  473. evt.raw32b = uc_sram_mmio->eventParams;
  474. IPAERR("errorType=%d channelHandle=%d reserved=%d\n",
  475. evt.params.errorType, evt.params.channelHandle,
  476. evt.params.reserved);
  477. } else if (ipa3_ctx->uc_ctx.uc_sram_mmio->eventOp ==
  478. IPA_HW_2_CPU_EVENT_MHI_CHANNEL_WAKE_UP_REQUEST) {
  479. union IpaHwMhiChannelWakeupEventData_t evt;
  480. IPADBG("WakeUp channel request\n");
  481. evt.raw32b = uc_sram_mmio->eventParams;
  482. IPADBG("channelHandle=%d reserved=%d\n",
  483. evt.params.channelHandle, evt.params.reserved);
  484. ipa3_uc_mhi_ctx->wakeup_request_cb();
  485. }
  486. }
  487. static void ipa3_uc_mhi_event_log_info_hdlr(
  488. struct IpaHwEventLogInfoData_t *uc_event_top_mmio)
  489. {
  490. struct Ipa3HwEventInfoData_t *evt_info_ptr;
  491. u32 size;
  492. if ((uc_event_top_mmio->protocolMask & (1 << IPA_HW_FEATURE_MHI))
  493. == 0) {
  494. IPAERR("MHI feature missing 0x%x\n",
  495. uc_event_top_mmio->protocolMask);
  496. return;
  497. }
  498. evt_info_ptr = &uc_event_top_mmio->statsInfo;
  499. size = evt_info_ptr->featureInfo[IPA_HW_FEATURE_MHI].params.size;
  500. if (size != sizeof(struct IpaHwStatsMhiInfoData_t)) {
  501. IPAERR("mhi stats sz invalid exp=%zu is=%u\n",
  502. sizeof(struct IpaHwStatsMhiInfoData_t),
  503. size);
  504. return;
  505. }
  506. ipa3_uc_mhi_ctx->mhi_uc_stats_ofst =
  507. evt_info_ptr->baseAddrOffset +
  508. evt_info_ptr->featureInfo[IPA_HW_FEATURE_MHI].params.offset;
  509. IPAERR("MHI stats ofst=0x%x\n", ipa3_uc_mhi_ctx->mhi_uc_stats_ofst);
  510. if (ipa3_uc_mhi_ctx->mhi_uc_stats_ofst +
  511. sizeof(struct IpaHwStatsMhiInfoData_t) >=
  512. ipa3_ctx->ctrl->ipa_reg_base_ofst +
  513. ipahal_get_reg_n_ofst(IPA_SW_AREA_RAM_DIRECT_ACCESS_n, 0) +
  514. ipa3_ctx->smem_sz) {
  515. IPAERR("uc_mhi_stats 0x%x outside SRAM\n",
  516. ipa3_uc_mhi_ctx->mhi_uc_stats_ofst);
  517. return;
  518. }
  519. ipa3_uc_mhi_ctx->mhi_uc_stats_mmio =
  520. ioremap(ipa3_ctx->ipa_wrapper_base +
  521. ipa3_uc_mhi_ctx->mhi_uc_stats_ofst,
  522. sizeof(struct IpaHwStatsMhiInfoData_t));
  523. if (!ipa3_uc_mhi_ctx->mhi_uc_stats_mmio) {
  524. IPAERR("fail to ioremap uc mhi stats\n");
  525. return;
  526. }
  527. }
  528. int ipa3_uc_mhi_init(void (*ready_cb)(void), void (*wakeup_request_cb)(void))
  529. {
  530. struct ipa3_uc_hdlrs hdlrs;
  531. if (ipa3_uc_mhi_ctx) {
  532. IPAERR("Already initialized\n");
  533. return -EFAULT;
  534. }
  535. ipa3_uc_mhi_ctx = kzalloc(sizeof(*ipa3_uc_mhi_ctx), GFP_KERNEL);
  536. if (!ipa3_uc_mhi_ctx) {
  537. IPAERR("no mem\n");
  538. return -ENOMEM;
  539. }
  540. ipa3_uc_mhi_ctx->ready_cb = ready_cb;
  541. ipa3_uc_mhi_ctx->wakeup_request_cb = wakeup_request_cb;
  542. memset(&hdlrs, 0, sizeof(hdlrs));
  543. hdlrs.ipa_uc_loaded_hdlr = ipa3_uc_mhi_ctx->ready_cb;
  544. hdlrs.ipa3_uc_response_hdlr = ipa3_uc_mhi_response_hdlr;
  545. hdlrs.ipa_uc_event_hdlr = ipa3_uc_mhi_event_hdlr;
  546. hdlrs.ipa_uc_event_log_info_hdlr = ipa3_uc_mhi_event_log_info_hdlr;
  547. ipa3_uc_register_handlers(IPA_HW_FEATURE_MHI, &hdlrs);
  548. IPADBG("Done\n");
  549. return 0;
  550. }
  551. void ipa3_uc_mhi_cleanup(void)
  552. {
  553. struct ipa3_uc_hdlrs null_hdlrs = { 0 };
  554. IPADBG("Enter\n");
  555. if (!ipa3_uc_mhi_ctx) {
  556. IPAERR("ipa3_uc_mhi_ctx is not initialized\n");
  557. return;
  558. }
  559. ipa3_uc_register_handlers(IPA_HW_FEATURE_MHI, &null_hdlrs);
  560. kfree(ipa3_uc_mhi_ctx);
  561. ipa3_uc_mhi_ctx = NULL;
  562. IPADBG("Done\n");
  563. }
  564. int ipa3_uc_mhi_init_engine(struct ipa_mhi_msi_info *msi, u32 mmio_addr,
  565. u32 host_ctrl_addr, u32 host_data_addr, u32 first_ch_idx,
  566. u32 first_evt_idx)
  567. {
  568. int res;
  569. struct ipa_mem_buffer mem;
  570. struct IpaHwMhiInitCmdData_t *init_cmd_data;
  571. struct IpaHwMhiMsiCmdData_t *msi_cmd;
  572. if (!ipa3_uc_mhi_ctx) {
  573. IPAERR("Not initialized\n");
  574. return -EFAULT;
  575. }
  576. IPA_ACTIVE_CLIENTS_INC_SIMPLE();
  577. res = ipa3_uc_update_hw_flags(0);
  578. if (res) {
  579. IPAERR("ipa3_uc_update_hw_flags failed %d\n", res);
  580. goto disable_clks;
  581. }
  582. mem.size = sizeof(*init_cmd_data);
  583. mem.base = dma_alloc_coherent(ipa3_ctx->pdev, mem.size, &mem.phys_base,
  584. GFP_KERNEL);
  585. if (!mem.base) {
  586. IPAERR("fail to alloc DMA buff of size %d\n", mem.size);
  587. res = -ENOMEM;
  588. goto disable_clks;
  589. }
  590. init_cmd_data = (struct IpaHwMhiInitCmdData_t *)mem.base;
  591. init_cmd_data->msiAddress = msi->addr_low;
  592. init_cmd_data->mmioBaseAddress = mmio_addr;
  593. init_cmd_data->deviceMhiCtrlBaseAddress = host_ctrl_addr;
  594. init_cmd_data->deviceMhiDataBaseAddress = host_data_addr;
  595. init_cmd_data->firstChannelIndex = first_ch_idx;
  596. init_cmd_data->firstEventRingIndex = first_evt_idx;
  597. res = ipa3_uc_send_cmd((u32)mem.phys_base, IPA_CPU_2_HW_CMD_MHI_INIT, 0,
  598. false, HZ);
  599. if (res) {
  600. IPAERR("ipa3_uc_send_cmd failed %d\n", res);
  601. dma_free_coherent(ipa3_ctx->pdev, mem.size, mem.base,
  602. mem.phys_base);
  603. goto disable_clks;
  604. }
  605. dma_free_coherent(ipa3_ctx->pdev, mem.size, mem.base, mem.phys_base);
  606. mem.size = sizeof(*msi_cmd);
  607. mem.base = dma_alloc_coherent(ipa3_ctx->pdev, mem.size, &mem.phys_base,
  608. GFP_KERNEL);
  609. if (!mem.base) {
  610. IPAERR("fail to alloc DMA buff of size %d\n", mem.size);
  611. res = -ENOMEM;
  612. goto disable_clks;
  613. }
  614. msi_cmd = (struct IpaHwMhiMsiCmdData_t *)mem.base;
  615. msi_cmd->msiAddress_hi = msi->addr_hi;
  616. msi_cmd->msiAddress_low = msi->addr_low;
  617. msi_cmd->msiData = msi->data;
  618. msi_cmd->msiMask = msi->mask;
  619. res = ipa3_uc_send_cmd((u32)mem.phys_base,
  620. IPA_CPU_2_HW_CMD_MHI_UPDATE_MSI, 0, false, HZ);
  621. if (res) {
  622. IPAERR("ipa3_uc_send_cmd failed %d\n", res);
  623. dma_free_coherent(ipa3_ctx->pdev, mem.size, mem.base,
  624. mem.phys_base);
  625. goto disable_clks;
  626. }
  627. dma_free_coherent(ipa3_ctx->pdev, mem.size, mem.base, mem.phys_base);
  628. res = 0;
  629. disable_clks:
  630. IPA_ACTIVE_CLIENTS_DEC_SIMPLE();
  631. return res;
  632. }
  633. int ipa3_uc_mhi_init_channel(int ipa_ep_idx, int channelHandle,
  634. int contexArrayIndex, int channelDirection)
  635. {
  636. int res;
  637. union IpaHwMhiInitChannelCmdData_t init_cmd;
  638. union IpaHwMhiChangeChannelStateResponseData_t uc_rsp;
  639. if (!ipa3_uc_mhi_ctx) {
  640. IPAERR("Not initialized\n");
  641. return -EFAULT;
  642. }
  643. if (ipa_ep_idx < 0 || ipa_ep_idx >= ipa3_ctx->ipa_num_pipes) {
  644. IPAERR("Invalid ipa_ep_idx.\n");
  645. return -EINVAL;
  646. }
  647. IPA_ACTIVE_CLIENTS_INC_SIMPLE();
  648. memset(&uc_rsp, 0, sizeof(uc_rsp));
  649. uc_rsp.params.state = IPA_HW_MHI_CHANNEL_STATE_RUN;
  650. uc_rsp.params.channelHandle = channelHandle;
  651. ipa3_uc_mhi_ctx->expected_responseOp =
  652. IPA_HW_2_CPU_RESPONSE_MHI_CHANGE_CHANNEL_STATE;
  653. ipa3_uc_mhi_ctx->expected_responseParams = uc_rsp.raw32b;
  654. memset(&init_cmd, 0, sizeof(init_cmd));
  655. init_cmd.params.channelHandle = channelHandle;
  656. init_cmd.params.contexArrayIndex = contexArrayIndex;
  657. init_cmd.params.bamPipeId = ipa_ep_idx;
  658. init_cmd.params.channelDirection = channelDirection;
  659. res = ipa3_uc_send_cmd(init_cmd.raw32b,
  660. IPA_CPU_2_HW_CMD_MHI_INIT_CHANNEL, 0, false, HZ);
  661. if (res) {
  662. IPAERR("ipa3_uc_send_cmd failed %d\n", res);
  663. goto disable_clks;
  664. }
  665. res = 0;
  666. disable_clks:
  667. IPA_ACTIVE_CLIENTS_DEC_SIMPLE();
  668. return res;
  669. }
  670. int ipa3_uc_mhi_reset_channel(int channelHandle)
  671. {
  672. union IpaHwMhiChangeChannelStateCmdData_t cmd;
  673. union IpaHwMhiChangeChannelStateResponseData_t uc_rsp;
  674. int res;
  675. if (!ipa3_uc_mhi_ctx) {
  676. IPAERR("Not initialized\n");
  677. return -EFAULT;
  678. }
  679. IPA_ACTIVE_CLIENTS_INC_SIMPLE();
  680. memset(&uc_rsp, 0, sizeof(uc_rsp));
  681. uc_rsp.params.state = IPA_HW_MHI_CHANNEL_STATE_DISABLE;
  682. uc_rsp.params.channelHandle = channelHandle;
  683. ipa3_uc_mhi_ctx->expected_responseOp =
  684. IPA_HW_2_CPU_RESPONSE_MHI_CHANGE_CHANNEL_STATE;
  685. ipa3_uc_mhi_ctx->expected_responseParams = uc_rsp.raw32b;
  686. memset(&cmd, 0, sizeof(cmd));
  687. cmd.params.requestedState = IPA_HW_MHI_CHANNEL_STATE_DISABLE;
  688. cmd.params.channelHandle = channelHandle;
  689. res = ipa3_uc_send_cmd(cmd.raw32b,
  690. IPA_CPU_2_HW_CMD_MHI_CHANGE_CHANNEL_STATE, 0, false, HZ);
  691. if (res) {
  692. IPAERR("ipa3_uc_send_cmd failed %d\n", res);
  693. goto disable_clks;
  694. }
  695. res = 0;
  696. disable_clks:
  697. IPA_ACTIVE_CLIENTS_DEC_SIMPLE();
  698. return res;
  699. }
  700. int ipa3_uc_mhi_suspend_channel(int channelHandle)
  701. {
  702. union IpaHwMhiChangeChannelStateCmdData_t cmd;
  703. union IpaHwMhiChangeChannelStateResponseData_t uc_rsp;
  704. int res;
  705. if (!ipa3_uc_mhi_ctx) {
  706. IPAERR("Not initialized\n");
  707. return -EFAULT;
  708. }
  709. IPA_ACTIVE_CLIENTS_INC_SIMPLE();
  710. memset(&uc_rsp, 0, sizeof(uc_rsp));
  711. uc_rsp.params.state = IPA_HW_MHI_CHANNEL_STATE_SUSPEND;
  712. uc_rsp.params.channelHandle = channelHandle;
  713. ipa3_uc_mhi_ctx->expected_responseOp =
  714. IPA_HW_2_CPU_RESPONSE_MHI_CHANGE_CHANNEL_STATE;
  715. ipa3_uc_mhi_ctx->expected_responseParams = uc_rsp.raw32b;
  716. memset(&cmd, 0, sizeof(cmd));
  717. cmd.params.requestedState = IPA_HW_MHI_CHANNEL_STATE_SUSPEND;
  718. cmd.params.channelHandle = channelHandle;
  719. res = ipa3_uc_send_cmd(cmd.raw32b,
  720. IPA_CPU_2_HW_CMD_MHI_CHANGE_CHANNEL_STATE, 0, false, HZ);
  721. if (res) {
  722. IPAERR("ipa3_uc_send_cmd failed %d\n", res);
  723. goto disable_clks;
  724. }
  725. res = 0;
  726. disable_clks:
  727. IPA_ACTIVE_CLIENTS_DEC_SIMPLE();
  728. return res;
  729. }
  730. int ipa3_uc_mhi_resume_channel(int channelHandle, bool LPTransitionRejected)
  731. {
  732. union IpaHwMhiChangeChannelStateCmdData_t cmd;
  733. union IpaHwMhiChangeChannelStateResponseData_t uc_rsp;
  734. int res;
  735. if (!ipa3_uc_mhi_ctx) {
  736. IPAERR("Not initialized\n");
  737. return -EFAULT;
  738. }
  739. IPA_ACTIVE_CLIENTS_INC_SIMPLE();
  740. memset(&uc_rsp, 0, sizeof(uc_rsp));
  741. uc_rsp.params.state = IPA_HW_MHI_CHANNEL_STATE_RUN;
  742. uc_rsp.params.channelHandle = channelHandle;
  743. ipa3_uc_mhi_ctx->expected_responseOp =
  744. IPA_HW_2_CPU_RESPONSE_MHI_CHANGE_CHANNEL_STATE;
  745. ipa3_uc_mhi_ctx->expected_responseParams = uc_rsp.raw32b;
  746. memset(&cmd, 0, sizeof(cmd));
  747. cmd.params.requestedState = IPA_HW_MHI_CHANNEL_STATE_RUN;
  748. cmd.params.channelHandle = channelHandle;
  749. cmd.params.LPTransitionRejected = LPTransitionRejected;
  750. res = ipa3_uc_send_cmd(cmd.raw32b,
  751. IPA_CPU_2_HW_CMD_MHI_CHANGE_CHANNEL_STATE, 0, false, HZ);
  752. if (res) {
  753. IPAERR("ipa3_uc_send_cmd failed %d\n", res);
  754. goto disable_clks;
  755. }
  756. res = 0;
  757. disable_clks:
  758. IPA_ACTIVE_CLIENTS_DEC_SIMPLE();
  759. return res;
  760. }
  761. int ipa3_uc_mhi_stop_event_update_channel(int channelHandle)
  762. {
  763. union IpaHwMhiStopEventUpdateData_t cmd;
  764. int res;
  765. if (!ipa3_uc_mhi_ctx) {
  766. IPAERR("Not initialized\n");
  767. return -EFAULT;
  768. }
  769. IPA_ACTIVE_CLIENTS_INC_SIMPLE();
  770. memset(&cmd, 0, sizeof(cmd));
  771. cmd.params.channelHandle = channelHandle;
  772. ipa3_uc_mhi_ctx->expected_responseOp =
  773. IPA_CPU_2_HW_CMD_MHI_STOP_EVENT_UPDATE;
  774. ipa3_uc_mhi_ctx->expected_responseParams = cmd.raw32b;
  775. res = ipa3_uc_send_cmd(cmd.raw32b,
  776. IPA_CPU_2_HW_CMD_MHI_STOP_EVENT_UPDATE, 0, false, HZ);
  777. if (res) {
  778. IPAERR("ipa3_uc_send_cmd failed %d\n", res);
  779. goto disable_clks;
  780. }
  781. res = 0;
  782. disable_clks:
  783. IPA_ACTIVE_CLIENTS_DEC_SIMPLE();
  784. return res;
  785. }
  786. int ipa3_uc_mhi_send_dl_ul_sync_info(union IpaHwMhiDlUlSyncCmdData_t *cmd)
  787. {
  788. int res;
  789. if (!ipa3_uc_mhi_ctx) {
  790. IPAERR("Not initialized\n");
  791. return -EFAULT;
  792. }
  793. IPADBG("isDlUlSyncEnabled=0x%x UlAccmVal=0x%x\n",
  794. cmd->params.isDlUlSyncEnabled, cmd->params.UlAccmVal);
  795. IPADBG("ulMsiEventThreshold=0x%x dlMsiEventThreshold=0x%x\n",
  796. cmd->params.ulMsiEventThreshold,
  797. cmd->params.dlMsiEventThreshold);
  798. IPA_ACTIVE_CLIENTS_INC_SIMPLE();
  799. res = ipa3_uc_send_cmd(cmd->raw32b,
  800. IPA_CPU_2_HW_CMD_MHI_DL_UL_SYNC_INFO, 0, false, HZ);
  801. if (res) {
  802. IPAERR("ipa3_uc_send_cmd failed %d\n", res);
  803. goto disable_clks;
  804. }
  805. res = 0;
  806. disable_clks:
  807. IPA_ACTIVE_CLIENTS_DEC_SIMPLE();
  808. return res;
  809. }
  810. int ipa3_uc_mhi_print_stats(char *dbg_buff, int size)
  811. {
  812. int nBytes = 0;
  813. int i;
  814. if (!ipa3_uc_mhi_ctx->mhi_uc_stats_mmio) {
  815. IPAERR("MHI uc stats is not valid\n");
  816. return 0;
  817. }
  818. nBytes += scnprintf(&dbg_buff[nBytes], size - nBytes,
  819. "Common Stats:\n");
  820. PRINT_COMMON_STATS(numULDLSync);
  821. PRINT_COMMON_STATS(numULTimerExpired);
  822. PRINT_COMMON_STATS(numChEvCtxWpRead);
  823. for (i = 0; i < IPA_HW_MAX_NUMBER_OF_CHANNELS; i++) {
  824. nBytes += scnprintf(&dbg_buff[nBytes], size - nBytes,
  825. "Channel %d Stats:\n", i);
  826. PRINT_CHANNEL_STATS(i, doorbellInt);
  827. PRINT_CHANNEL_STATS(i, reProccesed);
  828. PRINT_CHANNEL_STATS(i, bamFifoFull);
  829. PRINT_CHANNEL_STATS(i, bamFifoEmpty);
  830. PRINT_CHANNEL_STATS(i, bamFifoUsageHigh);
  831. PRINT_CHANNEL_STATS(i, bamFifoUsageLow);
  832. PRINT_CHANNEL_STATS(i, bamInt);
  833. PRINT_CHANNEL_STATS(i, ringFull);
  834. PRINT_CHANNEL_STATS(i, ringEmpty);
  835. PRINT_CHANNEL_STATS(i, ringUsageHigh);
  836. PRINT_CHANNEL_STATS(i, ringUsageLow);
  837. PRINT_CHANNEL_STATS(i, delayedMsi);
  838. PRINT_CHANNEL_STATS(i, immediateMsi);
  839. PRINT_CHANNEL_STATS(i, thresholdMsi);
  840. PRINT_CHANNEL_STATS(i, numSuspend);
  841. PRINT_CHANNEL_STATS(i, numResume);
  842. PRINT_CHANNEL_STATS(i, num_OOB);
  843. PRINT_CHANNEL_STATS(i, num_OOB_timer_expiry);
  844. PRINT_CHANNEL_STATS(i, num_OOB_moderation_timer_start);
  845. PRINT_CHANNEL_STATS(i, num_db_mode_evt);
  846. }
  847. return nBytes;
  848. }