rfal_analogConfig.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479
  1. /******************************************************************************
  2. * \attention
  3. *
  4. * <h2><center>&copy; COPYRIGHT 2020 STMicroelectronics</center></h2>
  5. *
  6. * Licensed under ST MYLIBERTY SOFTWARE LICENSE AGREEMENT (the "License");
  7. * You may not use this file except in compliance with the License.
  8. * You may obtain a copy of the License at:
  9. *
  10. * www.st.com/myliberty
  11. *
  12. * Unless required by applicable law or agreed to in writing, software
  13. * distributed under the License is distributed on an "AS IS" BASIS,
  14. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied,
  15. * AND SPECIFICALLY DISCLAIMING THE IMPLIED WARRANTIES OF MERCHANTABILITY,
  16. * FITNESS FOR A PARTICULAR PURPOSE, AND NON-INFRINGEMENT.
  17. * See the License for the specific language governing permissions and
  18. * limitations under the License.
  19. *
  20. ******************************************************************************/
  21. /*
  22. * PROJECT: ST25R391x firmware
  23. * Revision:
  24. * LANGUAGE: ISO C99
  25. */
  26. /*! \file rfal_analogConfig.c
  27. *
  28. * \author bkam
  29. *
  30. * \brief Funcitons to manage and set analog settings.
  31. *
  32. */
  33. /*
  34. ******************************************************************************
  35. * INCLUDES
  36. ******************************************************************************
  37. */
  38. #include "rfal_analogConfig.h"
  39. #include "rfal_chip.h"
  40. #include "st_errno.h"
  41. #include "platform.h"
  42. #include "utils.h"
  43. /* Check whether the Default Analog settings are to be used or custom ones */
  44. #ifdef RFAL_ANALOG_CONFIG_CUSTOM
  45. extern const uint8_t* rfalAnalogConfigCustomSettings;
  46. extern const uint16_t rfalAnalogConfigCustomSettingsLength;
  47. #else
  48. #include "rfal_analogConfigTbl.h"
  49. #endif
  50. /*
  51. ******************************************************************************
  52. * DEFINES
  53. ******************************************************************************
  54. */
  55. #define RFAL_TEST_REG 0x0080U /*!< Test Register indicator */
  56. /*
  57. ******************************************************************************
  58. * MACROS
  59. ******************************************************************************
  60. */
  61. /*
  62. ******************************************************************************
  63. * LOCAL DATA TYPES
  64. ******************************************************************************
  65. */
  66. #if RFAL_FEATURE_DYNAMIC_ANALOG_CONFIG
  67. static uint8_t gRfalAnalogConfig[RFAL_ANALOG_CONFIG_TBL_SIZE]; /*!< Analog Configuration Settings List */
  68. #endif /* RFAL_FEATURE_DYNAMIC_ANALOG_CONFIG */
  69. /*! Struct for Analog Config Look Up Table Update */
  70. typedef struct {
  71. const uint8_t *currentAnalogConfigTbl; /*!< Reference to start of current Analog Configuration */
  72. uint16_t configTblSize; /*!< Total size of Analog Configuration */
  73. bool ready; /*!< Indicate if Look Up Table is complete and ready for use */
  74. } rfalAnalogConfigMgmt;
  75. static rfalAnalogConfigMgmt gRfalAnalogConfigMgmt; /*!< Analog Configuration LUT management */
  76. /*
  77. ******************************************************************************
  78. * LOCAL TABLES
  79. ******************************************************************************
  80. */
  81. /*
  82. ******************************************************************************
  83. * LOCAL FUNCTION PROTOTYPES
  84. ******************************************************************************
  85. */
  86. static rfalAnalogConfigNum rfalAnalogConfigSearch( rfalAnalogConfigId configId, uint16_t *configOffset );
  87. #if RFAL_FEATURE_DYNAMIC_ANALOG_CONFIG
  88. static void rfalAnalogConfigPtrUpdate( const uint8_t* analogConfigTbl );
  89. #endif /* RFAL_FEATURE_DYNAMIC_ANALOG_CONFIG */
  90. /*
  91. ******************************************************************************
  92. * GLOBAL VARIABLE DEFINITIONS
  93. ******************************************************************************
  94. */
  95. /*
  96. ******************************************************************************
  97. * GLOBAL FUNCTIONS
  98. ******************************************************************************
  99. */
  100. void rfalAnalogConfigInitialize( void )
  101. {
  102. /* Use default Analog configuration settings in Flash by default. */
  103. /* Check whether the Default Analog settings are to be used or custom ones */
  104. #ifdef RFAL_ANALOG_CONFIG_CUSTOM
  105. gRfalAnalogConfigMgmt.currentAnalogConfigTbl = (const uint8_t *)&rfalAnalogConfigCustomSettings;
  106. gRfalAnalogConfigMgmt.configTblSize = rfalAnalogConfigCustomSettingsLength;
  107. #else
  108. gRfalAnalogConfigMgmt.currentAnalogConfigTbl = (const uint8_t *)&rfalAnalogConfigDefaultSettings;
  109. gRfalAnalogConfigMgmt.configTblSize = sizeof(rfalAnalogConfigDefaultSettings);
  110. #endif
  111. gRfalAnalogConfigMgmt.ready = true;
  112. } /* rfalAnalogConfigInitialize() */
  113. bool rfalAnalogConfigIsReady( void )
  114. {
  115. return gRfalAnalogConfigMgmt.ready;
  116. }
  117. ReturnCode rfalAnalogConfigListWriteRaw( const uint8_t *configTbl, uint16_t configTblSize )
  118. {
  119. #if RFAL_FEATURE_DYNAMIC_ANALOG_CONFIG
  120. /* Check if the Configuration Table exceed the Table size */
  121. if ( configTblSize >= RFAL_ANALOG_CONFIG_TBL_SIZE )
  122. {
  123. rfalAnalogConfigInitialize(); /* Revert to default Analog Configuration */
  124. return ERR_NOMEM;
  125. }
  126. /* Check for invalid parameters */
  127. if( (configTbl == NULL) || (configTblSize == 0U) )
  128. {
  129. return ERR_PARAM;
  130. }
  131. /* NOTE: Function does not check for the validity of the Table contents (conf IDs, conf sets, register address) */
  132. ST_MEMCPY( gRfalAnalogConfig, configTbl, configTblSize );
  133. /* Update the total size of configuration settings */
  134. gRfalAnalogConfigMgmt.configTblSize = configTblSize;
  135. rfalAnalogConfigPtrUpdate(gRfalAnalogConfig);
  136. return ERR_NONE;
  137. #else
  138. // If Analog Configuration Update is to be disabled
  139. NO_WARNING(configTbl);
  140. NO_WARNING(configTblSize);
  141. return ERR_REQUEST;
  142. #endif /* RFAL_FEATURE_DYNAMIC_ANALOG_CONFIG */
  143. }
  144. ReturnCode rfalAnalogConfigListWrite( uint8_t more, const rfalAnalogConfig *config )
  145. {
  146. #if RFAL_FEATURE_DYNAMIC_ANALOG_CONFIG
  147. rfalAnalogConfigId configId;
  148. rfalAnalogConfigNum numConfig;
  149. uint8_t configSize;
  150. if (true == gRfalAnalogConfigMgmt.ready)
  151. { /* First Update to the Configuration list. */
  152. gRfalAnalogConfigMgmt.ready = false; // invalidate the config List
  153. gRfalAnalogConfigMgmt.configTblSize = 0; // Clear the config List
  154. }
  155. configId = GETU16(config->id);
  156. /* Check validity of the Configuration ID. */
  157. if ( (RFAL_ANALOG_CONFIG_TECH_RFU <= RFAL_ANALOG_CONFIG_ID_GET_TECH(configId))
  158. ||((RFAL_ANALOG_CONFIG_BITRATE_6780 < RFAL_ANALOG_CONFIG_ID_GET_BITRATE(configId)) && (RFAL_ANALOG_CONFIG_BITRATE_1OF4 > RFAL_ANALOG_CONFIG_ID_GET_BITRATE(configId)))
  159. ||(RFAL_ANALOG_CONFIG_BITRATE_1OF256 < RFAL_ANALOG_CONFIG_ID_GET_BITRATE(configId))
  160. )
  161. {
  162. rfalAnalogConfigInitialize(); /* Revert to default Analog Configuration */
  163. return ERR_PARAM;
  164. }
  165. numConfig = config->num;
  166. configSize = (uint8_t)(sizeof(rfalAnalogConfigId) + sizeof(rfalAnalogConfigNum) + (numConfig * sizeof(rfalAnalogConfigRegAddrMaskVal)));
  167. /* Check if the Configuration Set exceed the Table size. */
  168. if ( RFAL_ANALOG_CONFIG_TBL_SIZE <= (gRfalAnalogConfigMgmt.configTblSize + configSize) )
  169. {
  170. rfalAnalogConfigInitialize(); /* Revert to default Analog Configuration */
  171. return ERR_NOMEM;
  172. }
  173. /* NOTE: Function does not check for the validity of the Register Address. */
  174. ST_MEMCPY(&gRfalAnalogConfig[gRfalAnalogConfigMgmt.configTblSize], (const uint8_t*)config, configSize);
  175. /* Increment the total size of configuration settings. */
  176. gRfalAnalogConfigMgmt.configTblSize += configSize;
  177. /* Check if it is the last Analog Configuration to load. */
  178. if (RFAL_ANALOG_CONFIG_UPDATE_LAST == more)
  179. { /* Update the Analog Configuration to the new settings. */
  180. rfalAnalogConfigPtrUpdate(gRfalAnalogConfig);
  181. }
  182. return ERR_NONE;
  183. #else
  184. // If Analog Configuration Update is to be disabled
  185. NO_WARNING(config);
  186. NO_WARNING(more);
  187. return ERR_DISABLED;
  188. #endif /* RFAL_FEATURE_DYNAMIC_ANALOG_CONFIG */
  189. } /* rfalAnalogConfigListUpdate() */
  190. ReturnCode rfalAnalogConfigListReadRaw( uint8_t *tblBuf, uint16_t tblBufLen, uint16_t *configTblSize )
  191. {
  192. /* Check if the the current table will fit into the given buffer */
  193. if( tblBufLen < gRfalAnalogConfigMgmt.configTblSize )
  194. {
  195. return ERR_NOMEM;
  196. }
  197. /* Check for invalid parameters */
  198. if( configTblSize == NULL )
  199. {
  200. return ERR_PARAM;
  201. }
  202. /* Copy the whole Table to the given buffer */
  203. if( gRfalAnalogConfigMgmt.configTblSize > 0U ) /* MISRA 21.18 */
  204. {
  205. ST_MEMCPY( tblBuf, gRfalAnalogConfigMgmt.currentAnalogConfigTbl, gRfalAnalogConfigMgmt.configTblSize );
  206. }
  207. *configTblSize = gRfalAnalogConfigMgmt.configTblSize;
  208. return ERR_NONE;
  209. }
  210. ReturnCode rfalAnalogConfigListRead( rfalAnalogConfigOffset *configOffset, uint8_t *more, rfalAnalogConfig *config, rfalAnalogConfigNum numConfig )
  211. {
  212. uint16_t configSize;
  213. rfalAnalogConfigOffset offset = *configOffset;
  214. rfalAnalogConfigNum numConfigSet;
  215. /* Check if the number of register-mask-value settings for the respective Configuration ID will fit into the buffer passed in. */
  216. if (gRfalAnalogConfigMgmt.currentAnalogConfigTbl[offset + sizeof(rfalAnalogConfigId)] > numConfig)
  217. {
  218. return ERR_NOMEM;
  219. }
  220. /* Get the number of Configuration set */
  221. numConfigSet = gRfalAnalogConfigMgmt.currentAnalogConfigTbl[offset + sizeof(rfalAnalogConfigId)];
  222. /* Pass Configuration Register-Mask-Value sets */
  223. configSize = (sizeof(rfalAnalogConfigId) + sizeof(rfalAnalogConfigNum) + (uint16_t)(numConfigSet * sizeof(rfalAnalogConfigRegAddrMaskVal)));
  224. ST_MEMCPY( (uint8_t*) config
  225. , &gRfalAnalogConfigMgmt.currentAnalogConfigTbl[offset]
  226. , configSize
  227. );
  228. *configOffset = offset + configSize;
  229. /* Check if it is the last Analog Configuration in the Table.*/
  230. *more = (uint8_t)((*configOffset >= gRfalAnalogConfigMgmt.configTblSize) ? RFAL_ANALOG_CONFIG_UPDATE_LAST
  231. : RFAL_ANALOG_CONFIG_UPDATE_MORE);
  232. return ERR_NONE;
  233. } /* rfalAnalogConfigListRead() */
  234. ReturnCode rfalSetAnalogConfig( rfalAnalogConfigId configId )
  235. {
  236. rfalAnalogConfigOffset configOffset = 0;
  237. rfalAnalogConfigNum numConfigSet;
  238. const rfalAnalogConfigRegAddrMaskVal *configTbl;
  239. ReturnCode retCode = ERR_NONE;
  240. rfalAnalogConfigNum i;
  241. if (true != gRfalAnalogConfigMgmt.ready)
  242. {
  243. return ERR_REQUEST;
  244. }
  245. /* Search LUT for the specific Configuration ID. */
  246. while(true)
  247. {
  248. numConfigSet = rfalAnalogConfigSearch(configId, &configOffset);
  249. if( RFAL_ANALOG_CONFIG_LUT_NOT_FOUND == numConfigSet )
  250. {
  251. break;
  252. }
  253. configTbl = (rfalAnalogConfigRegAddrMaskVal *)( (uint32_t)gRfalAnalogConfigMgmt.currentAnalogConfigTbl + (uint32_t)configOffset);
  254. /* Increment the offset to the next index to search from. */
  255. configOffset += (uint16_t)(numConfigSet * sizeof(rfalAnalogConfigRegAddrMaskVal));
  256. if ((gRfalAnalogConfigMgmt.configTblSize + 1U) < configOffset)
  257. { /* Error check make sure that the we do not access outside the configuration Table Size */
  258. return ERR_NOMEM;
  259. }
  260. for ( i = 0; i < numConfigSet; i++)
  261. {
  262. if( (GETU16(configTbl[i].addr) & RFAL_TEST_REG) != 0U )
  263. {
  264. EXIT_ON_ERR(retCode, rfalChipChangeTestRegBits( (GETU16(configTbl[i].addr) & ~RFAL_TEST_REG), configTbl[i].mask, configTbl[i].val) );
  265. }
  266. else
  267. {
  268. EXIT_ON_ERR(retCode, rfalChipChangeRegBits( GETU16(configTbl[i].addr), configTbl[i].mask, configTbl[i].val) );
  269. }
  270. }
  271. } /* while(found Analog Config Id) */
  272. return retCode;
  273. } /* rfalSetAnalogConfig() */
  274. uint16_t rfalAnalogConfigGenModeID( rfalMode md, rfalBitRate br, uint16_t dir )
  275. {
  276. uint16_t id;
  277. /* Assign Poll/Listen Mode */
  278. id = ((md >= RFAL_MODE_LISTEN_NFCA) ? RFAL_ANALOG_CONFIG_LISTEN : RFAL_ANALOG_CONFIG_POLL);
  279. /* Assign Technology */
  280. switch( md )
  281. {
  282. case RFAL_MODE_POLL_NFCA:
  283. case RFAL_MODE_POLL_NFCA_T1T:
  284. case RFAL_MODE_LISTEN_NFCA:
  285. id |= RFAL_ANALOG_CONFIG_TECH_NFCA;
  286. break;
  287. case RFAL_MODE_POLL_NFCB:
  288. case RFAL_MODE_POLL_B_PRIME:
  289. case RFAL_MODE_POLL_B_CTS:
  290. case RFAL_MODE_LISTEN_NFCB:
  291. id |= RFAL_ANALOG_CONFIG_TECH_NFCB;
  292. break;
  293. case RFAL_MODE_POLL_NFCF:
  294. case RFAL_MODE_LISTEN_NFCF:
  295. id |= RFAL_ANALOG_CONFIG_TECH_NFCF;
  296. break;
  297. case RFAL_MODE_POLL_NFCV:
  298. case RFAL_MODE_POLL_PICOPASS:
  299. id |= RFAL_ANALOG_CONFIG_TECH_NFCV;
  300. break;
  301. case RFAL_MODE_POLL_ACTIVE_P2P:
  302. case RFAL_MODE_LISTEN_ACTIVE_P2P:
  303. id |= RFAL_ANALOG_CONFIG_TECH_AP2P;
  304. break;
  305. default:
  306. id = RFAL_ANALOG_CONFIG_TECH_CHIP;
  307. break;
  308. }
  309. /* Assign Bitrate */
  310. id |= (((((uint16_t)(br) >= (uint16_t)RFAL_BR_52p97) ? (uint16_t)(br) : ((uint16_t)(br)+1U)) << RFAL_ANALOG_CONFIG_BITRATE_SHIFT) & RFAL_ANALOG_CONFIG_BITRATE_MASK);
  311. /* Assign Direction */
  312. id |= ((dir<<RFAL_ANALOG_CONFIG_DIRECTION_SHIFT) & RFAL_ANALOG_CONFIG_DIRECTION_MASK);
  313. return id;
  314. } /* rfalAnalogConfigGenModeID() */
  315. /*
  316. ******************************************************************************
  317. * LOCAL FUNCTIONS
  318. ******************************************************************************
  319. */
  320. /*!
  321. *****************************************************************************
  322. * \brief Update the link to Analog Configuration LUT
  323. *
  324. * Update the link to the Analog Configuration LUT for the subsequent search
  325. * of Analog Settings.
  326. *
  327. * \param[in] analogConfigTbl: reference to the start of the new Analog Configuration Table
  328. *
  329. *****************************************************************************
  330. */
  331. #if RFAL_FEATURE_DYNAMIC_ANALOG_CONFIG
  332. static void rfalAnalogConfigPtrUpdate( const uint8_t* analogConfigTbl )
  333. {
  334. gRfalAnalogConfigMgmt.currentAnalogConfigTbl = analogConfigTbl;
  335. gRfalAnalogConfigMgmt.ready = true;
  336. } /* rfalAnalogConfigPtrUpdate() */
  337. #endif /* RFAL_FEATURE_DYNAMIC_ANALOG_CONFIG */
  338. /*!
  339. *****************************************************************************
  340. * \brief Search the Analog Configuration LUT for a specific Configuration ID.
  341. *
  342. * Search the Analog Configuration LUT for the Configuration ID.
  343. *
  344. * \param[in] configId: Configuration ID to search for.
  345. * \param[in] configOffset: Configuration Offset in Table
  346. *
  347. * \return number of Configuration Sets
  348. * \return #RFAL_ANALOG_CONFIG_LUT_NOT_FOUND in case Configuration ID is not found.
  349. *****************************************************************************
  350. */
  351. static rfalAnalogConfigNum rfalAnalogConfigSearch( rfalAnalogConfigId configId, uint16_t *configOffset )
  352. {
  353. rfalAnalogConfigId foundConfigId;
  354. rfalAnalogConfigId configIdMaskVal;
  355. const uint8_t *configTbl;
  356. const uint8_t *currentConfigTbl;
  357. uint16_t i;
  358. currentConfigTbl = gRfalAnalogConfigMgmt.currentAnalogConfigTbl;
  359. configIdMaskVal = ((RFAL_ANALOG_CONFIG_POLL_LISTEN_MODE_MASK | RFAL_ANALOG_CONFIG_BITRATE_MASK)
  360. |((RFAL_ANALOG_CONFIG_TECH_CHIP == RFAL_ANALOG_CONFIG_ID_GET_TECH(configId)) ? (RFAL_ANALOG_CONFIG_TECH_MASK | RFAL_ANALOG_CONFIG_CHIP_SPECIFIC_MASK) : configId)
  361. |((RFAL_ANALOG_CONFIG_NO_DIRECTION == RFAL_ANALOG_CONFIG_ID_GET_DIRECTION(configId)) ? RFAL_ANALOG_CONFIG_DIRECTION_MASK : configId)
  362. );
  363. /* When specific ConfigIDs are to be used, override search mask */
  364. if( (RFAL_ANALOG_CONFIG_ID_GET_DIRECTION(configId) == RFAL_ANALOG_CONFIG_DPO) )
  365. {
  366. configIdMaskVal = (RFAL_ANALOG_CONFIG_POLL_LISTEN_MODE_MASK | RFAL_ANALOG_CONFIG_TECH_MASK | RFAL_ANALOG_CONFIG_BITRATE_MASK | RFAL_ANALOG_CONFIG_DIRECTION_MASK);
  367. }
  368. i = *configOffset;
  369. while (i < gRfalAnalogConfigMgmt.configTblSize)
  370. {
  371. configTbl = &currentConfigTbl[i];
  372. foundConfigId = GETU16(configTbl);
  373. if (configId == (foundConfigId & configIdMaskVal))
  374. {
  375. *configOffset = (uint16_t)(i + sizeof(rfalAnalogConfigId) + sizeof(rfalAnalogConfigNum));
  376. return configTbl[sizeof(rfalAnalogConfigId)];
  377. }
  378. /* If Config Id does not match, increment to next Configuration Id */
  379. i += (uint16_t)( sizeof(rfalAnalogConfigId) + sizeof(rfalAnalogConfigNum)
  380. + (configTbl[sizeof(rfalAnalogConfigId)] * sizeof(rfalAnalogConfigRegAddrMaskVal) )
  381. );
  382. } /* for */
  383. return RFAL_ANALOG_CONFIG_LUT_NOT_FOUND;
  384. } /* rfalAnalogConfigSearch() */