compensation.c 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302
  1. /********************************************************************************
  2. * USDK(ZS283A)
  3. * Module: SYSTEM
  4. * Copyright(c) 2003-2017 Actions Semiconductor,
  5. * All Rights Reserved.
  6. *
  7. * History:
  8. * <author> <time> <version > <desc>
  9. ********************************************************************************/
  10. /*!
  11. * \file compensation.c
  12. * \brief
  13. * \author
  14. * \version 1.0
  15. * \date
  16. *******************************************************************************/
  17. #include <init.h>
  18. #include <kernel.h>
  19. #include <string.h>
  20. #include <compensation.h>
  21. #include <soc_atp.h>
  22. #include <drivers/nvram_config.h>
  23. #include <soc.h>
  24. #include <stdlib.h>
  25. #include <stdio.h>
  26. #include <string.h>
  27. #define SYS_LOG_DOMAIN "comp"
  28. #ifndef SYS_LOG_LEVEL
  29. #define SYS_LOG_LEVEL CONFIG_SYS_LOG_DEFAULT_LEVEL
  30. #endif
  31. #include <logging/log.h>
  32. //2*8=16bit
  33. #define MAX_EFUSE_CAP_RECORD (CONFIG_COMPENSATION_FREQ_INDEX_NUM)
  34. typedef struct
  35. {
  36. uint32_t value[MAX_EFUSE_CAP_RECORD];
  37. } cap_value_t;
  38. #if defined(CONFIG_SOC_SERIES_WOODPECKERFPGA)
  39. #define HOSC_CALIB_SIM
  40. #endif
  41. #ifdef HOSC_CALIB_SIM
  42. static uint8_t hosc_val[MAX_EFUSE_CAP_RECORD];
  43. static int soc_get_hosc_calib_sim(int id, unsigned char *calib_val)
  44. {
  45. *calib_val = hosc_val[id];
  46. return 0;
  47. }
  48. static int soc_set_hosc_calib_sim(int id, unsigned char calib_val)
  49. {
  50. hosc_val[id] = calib_val;
  51. return 0;
  52. }
  53. #endif
  54. static uint32_t read_efuse_freq_value(uint32_t *read_cap_value, int *index)
  55. {
  56. int i;
  57. cap_value_t cap_value;
  58. for (i = 0; i < MAX_EFUSE_CAP_RECORD; i++) {
  59. #ifndef HOSC_CALIB_SIM
  60. soc_atp_get_hosc_calib(i, (unsigned int *)(&cap_value.value[i]));
  61. #else
  62. soc_get_hosc_calib_sim(i, (unsigned int *)(&cap_value.value[i]));
  63. #endif
  64. }
  65. // the latter value is the newer efuse and return the last value which modified.
  66. for (i = 0; i < MAX_EFUSE_CAP_RECORD; i++) {
  67. if (cap_value.value[i] == 0) {
  68. break;
  69. }
  70. }
  71. //efuse that never written
  72. if (i == 0) {
  73. *read_cap_value = 0xff;
  74. *index = 0xff;
  75. } else {
  76. *read_cap_value = cap_value.value[i - 1];
  77. *index = i - 1;
  78. }
  79. return 0;
  80. }
  81. static int32_t read_efuse_freq_compensation(uint32_t *cap_value, int *index)
  82. {
  83. uint32_t trim_cap_value;
  84. read_efuse_freq_value(&trim_cap_value, index);
  85. if (*index != 0xff) {
  86. *cap_value = trim_cap_value;
  87. }
  88. return 0;
  89. }
  90. static int32_t spi_nor_freq_compensation_param_read(uint32_t *trim_cap)
  91. {
  92. uint32_t val ;
  93. int ret = nvram_config_get("BT_CFO_VAL", (void *) &val, 4);
  94. if (ret > 0){
  95. printk("nvram cap_val=%d\n", val);
  96. *trim_cap = val;
  97. }
  98. return ret;
  99. }
  100. int32_t freq_compensation_read(uint32_t *trim_cap, uint32_t mode)
  101. {
  102. int ret_val, ret_val_nvram;
  103. int index;
  104. uint32_t trim_cap_value_bak = 0;
  105. ret_val_nvram = spi_nor_freq_compensation_param_read(&trim_cap_value_bak);
  106. #ifdef CONFIG_COMPENSATION_HOSC_CAP_NVRAM_PRIORITY
  107. if (ret_val_nvram > 0) {
  108. // update the norflash parameters
  109. *trim_cap = trim_cap_value_bak;
  110. // the parameter partition is valid
  111. ret_val = TRIM_CAP_READ_NO_ERROR;
  112. return ret_val;
  113. }
  114. #endif
  115. ret_val = read_efuse_freq_compensation(trim_cap, &index);
  116. if (index == 0xff) {
  117. if (ret_val_nvram > 0) {
  118. // update the norflash parameters
  119. *trim_cap = trim_cap_value_bak;
  120. } else {
  121. // update the default value
  122. *trim_cap = CONFIG_COMPENSATION_DEFAULT_HOSC_CAP;
  123. }
  124. // the parameter partition is valid.
  125. ret_val = TRIM_CAP_READ_NO_ERROR;
  126. } else if (index < (MAX_EFUSE_CAP_RECORD - 1)) {
  127. // if efuse value is already valid and there is also a space to write, return the frequency offset value.
  128. ret_val = TRIM_CAP_READ_NO_ERROR;
  129. } else {
  130. if (mode == RW_TRIM_CAP_EFUSE) {
  131. if (*trim_cap == 0xff) {
  132. // update the default value
  133. *trim_cap = CONFIG_COMPENSATION_DEFAULT_HOSC_CAP;
  134. }
  135. // the parameter partition is valid.
  136. ret_val = TRIM_CAP_READ_NO_ERROR;
  137. } else {
  138. if (ret_val_nvram > 0) {
  139. // update the norflash parameters
  140. *trim_cap = trim_cap_value_bak;
  141. }
  142. // the parameter partition is valid.
  143. ret_val = TRIM_CAP_READ_NO_ERROR;
  144. }
  145. }
  146. return ret_val;
  147. }
  148. #ifndef CONFIG_COMPENSATION_HOSC_CAP_NVRAM_PRIORITY
  149. static int32_t write_efuse_new_value(int new_value, int old_index)
  150. {
  151. int new_index;
  152. int ret_val;
  153. if (old_index != 0xff) {
  154. new_index = old_index + 1;
  155. } else {
  156. new_index = 0;
  157. }
  158. if (new_index < MAX_EFUSE_CAP_RECORD) {
  159. //ret_val = soc_set_hosc_calib_sim(new_index, new_value);
  160. ret_val = -2;
  161. } else {
  162. return -2;
  163. }
  164. return ret_val;
  165. }
  166. static int32_t write_efuse_freq_compensation(uint32_t *cap_value)
  167. {
  168. int trim_cap_value;
  169. int index;
  170. int old_cap_value;
  171. // only write the least 8bits
  172. trim_cap_value = (*cap_value) & 0xff;
  173. read_efuse_freq_value(&old_cap_value, &index);
  174. if (index != 0xff) {
  175. if (old_cap_value == trim_cap_value) {
  176. return 0;
  177. } else {
  178. // add a new frequency offset value
  179. //efuse has already full
  180. if (index == (MAX_EFUSE_CAP_RECORD - 1)) {
  181. return -2;
  182. } else {
  183. return write_efuse_new_value(trim_cap_value, index);
  184. }
  185. }
  186. } else {
  187. //efuse that never been written
  188. return write_efuse_new_value(trim_cap_value, index);
  189. }
  190. }
  191. #endif
  192. static int32_t spi_nor_freq_compensation_param_write(uint32_t *trim_cap)
  193. {
  194. #ifdef CONFIG_PROPERTY
  195. return nvram_config_set_factory("BT_CFO_VAL", trim_cap, 4);
  196. #else
  197. return 0;
  198. #endif
  199. }
  200. int32_t freq_compensation_write(uint32_t *trim_cap, uint32_t mode)
  201. {
  202. int ret_val;
  203. // total 2 symmetrical bytes
  204. *trim_cap &= 0xffff;
  205. ret_val = spi_nor_freq_compensation_param_write(trim_cap);
  206. #ifdef CONFIG_COMPENSATION_HOSC_CAP_NVRAM_PRIORITY
  207. // write norflash
  208. ret_val = spi_nor_freq_compensation_param_write(trim_cap);
  209. if (ret_val == 0) {
  210. ret_val = TRIM_CAP_WRITE_NO_ERROR;
  211. } else {
  212. ret_val = TRIM_CAP_WRITE_ERR_HW;
  213. }
  214. #else
  215. ret_val = write_efuse_freq_compensation(trim_cap);
  216. if (mode == RW_TRIM_CAP_SNOR) {
  217. //efuse has no space
  218. if (ret_val == -2) {
  219. //write norflash
  220. ret_val = spi_nor_freq_compensation_param_write(trim_cap);
  221. if (ret_val == 0) {
  222. ret_val = TRIM_CAP_WRITE_NO_ERROR;
  223. } else {
  224. ret_val = TRIM_CAP_WRITE_ERR_HW;
  225. }
  226. } else if (ret_val == -1) {
  227. ret_val = TRIM_CAP_WRITE_ERR_HW;
  228. } else {
  229. ret_val = TRIM_CAP_WRITE_NO_ERROR;
  230. }
  231. } else {
  232. //efuse has no space
  233. if (ret_val == -2) {
  234. ret_val = TRIM_CAP_WRITE_ERR_NO_RESOURSE;
  235. } else if (ret_val == -1) {
  236. ret_val = TRIM_CAP_WRITE_ERR_HW;
  237. } else {
  238. ret_val = TRIM_CAP_WRITE_NO_ERROR;
  239. }
  240. }
  241. #endif
  242. return ret_val;
  243. }
  244. int freq_compensation_get_cap(uint32_t *cap)
  245. {
  246. uint32_t cap_adjust = 0;
  247. int ret_val;
  248. ret_val = freq_compensation_read(&cap_adjust, RW_TRIM_CAP_SNOR);
  249. if (ret_val != TRIM_CAP_READ_NO_ERROR) {
  250. *cap = CONFIG_COMPENSATION_DEFAULT_HOSC_CAP;
  251. }
  252. else{
  253. *cap = cap_adjust;
  254. }
  255. return ret_val;
  256. }