furi_hal_version.c 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305
  1. #include <furi_hal_version.h>
  2. #include <furi_hal_rtc.h>
  3. #include <furi.h>
  4. #include <stm32wbxx.h>
  5. #include <stm32wbxx_ll_rtc.h>
  6. #include <stdio.h>
  7. #include <ble/ble.h>
  8. #define TAG "FuriHalVersion"
  9. #define FURI_HAL_VERSION_OTP_HEADER_MAGIC 0xBABE
  10. #define FURI_HAL_VERSION_OTP_ADDRESS OTP_AREA_BASE
  11. /** OTP V0 Structure: prototypes and early EVT */
  12. typedef struct {
  13. uint8_t board_version;
  14. uint8_t board_target;
  15. uint8_t board_body;
  16. uint8_t board_connect;
  17. uint32_t header_timestamp;
  18. char name[FURI_HAL_VERSION_NAME_LENGTH];
  19. } FuriHalVersionOTPv0;
  20. /** OTP V1 Structure: late EVT, DVT */
  21. typedef struct {
  22. /* First 64 bits: header */
  23. uint16_t header_magic;
  24. uint8_t header_version;
  25. uint8_t header_reserved;
  26. uint32_t header_timestamp;
  27. /* Second 64 bits: board info */
  28. uint8_t board_version; /** Board version */
  29. uint8_t board_target; /** Board target firmware */
  30. uint8_t board_body; /** Board body */
  31. uint8_t board_connect; /** Board interconnect */
  32. uint8_t board_color; /** Board color */
  33. uint8_t board_region; /** Board region */
  34. uint16_t board_reserved; /** Reserved for future use, 0x0000 */
  35. /* Third 64 bits: Unique Device Name */
  36. char name[FURI_HAL_VERSION_NAME_LENGTH]; /** Unique Device Name */
  37. } FuriHalVersionOTPv1;
  38. /** OTP V2 Structure: DVT2, PVT, Production */
  39. typedef struct {
  40. /* Early First 64 bits: header */
  41. uint16_t header_magic;
  42. uint8_t header_version;
  43. uint8_t header_reserved;
  44. uint32_t header_timestamp;
  45. /* Early Second 64 bits: board info */
  46. uint8_t board_version; /** Board version */
  47. uint8_t board_target; /** Board target firmware */
  48. uint8_t board_body; /** Board body */
  49. uint8_t board_connect; /** Board interconnect */
  50. uint8_t board_display; /** Board display */
  51. uint8_t board_reserved2_0; /** Reserved for future use, 0x00 */
  52. uint16_t board_reserved2_1; /** Reserved for future use, 0x0000 */
  53. /* Late Third 64 bits: device info */
  54. uint8_t board_color; /** Board color */
  55. uint8_t board_region; /** Board region */
  56. uint16_t board_reserved3_0; /** Reserved for future use, 0x0000 */
  57. uint32_t board_reserved3_1; /** Reserved for future use, 0x00000000 */
  58. /* Late Fourth 64 bits: Unique Device Name */
  59. char name[FURI_HAL_VERSION_NAME_LENGTH]; /** Unique Device Name */
  60. } FuriHalVersionOTPv2;
  61. /** Represenation Model: */
  62. typedef struct {
  63. uint32_t timestamp;
  64. uint8_t board_version; /** Board version */
  65. uint8_t board_target; /** Board target firmware */
  66. uint8_t board_body; /** Board body */
  67. uint8_t board_connect; /** Board interconnect */
  68. uint8_t board_color; /** Board color */
  69. uint8_t board_region; /** Board region */
  70. uint8_t board_display; /** Board display */
  71. char name[FURI_HAL_VERSION_ARRAY_NAME_LENGTH]; /** \0 terminated name */
  72. char device_name[FURI_HAL_VERSION_DEVICE_NAME_LENGTH]; /** device name for special needs */
  73. uint8_t ble_mac[6];
  74. } FuriHalVersion;
  75. static FuriHalVersion furi_hal_version = {0};
  76. static void furi_hal_version_set_name(const char* name) {
  77. if(name != NULL) {
  78. strlcpy(furi_hal_version.name, name, FURI_HAL_VERSION_ARRAY_NAME_LENGTH);
  79. snprintf(
  80. furi_hal_version.device_name,
  81. FURI_HAL_VERSION_DEVICE_NAME_LENGTH,
  82. "xFlipper %s",
  83. furi_hal_version.name);
  84. } else {
  85. snprintf(furi_hal_version.device_name, FURI_HAL_VERSION_DEVICE_NAME_LENGTH, "xFlipper");
  86. }
  87. furi_hal_version.device_name[0] = AD_TYPE_COMPLETE_LOCAL_NAME;
  88. // BLE Mac address
  89. uint32_t udn = LL_FLASH_GetUDN();
  90. uint32_t company_id = LL_FLASH_GetSTCompanyID();
  91. uint32_t device_id = LL_FLASH_GetDeviceID();
  92. furi_hal_version.ble_mac[0] = (uint8_t)(udn & 0x000000FF);
  93. furi_hal_version.ble_mac[1] = (uint8_t)((udn & 0x0000FF00) >> 8);
  94. furi_hal_version.ble_mac[2] = (uint8_t)((udn & 0x00FF0000) >> 16);
  95. furi_hal_version.ble_mac[3] = (uint8_t)device_id;
  96. furi_hal_version.ble_mac[4] = (uint8_t)(company_id & 0x000000FF);
  97. furi_hal_version.ble_mac[5] = (uint8_t)((company_id & 0x0000FF00) >> 8);
  98. }
  99. static void furi_hal_version_load_otp_default() {
  100. furi_hal_version_set_name(NULL);
  101. }
  102. static void furi_hal_version_load_otp_v0() {
  103. const FuriHalVersionOTPv0* otp = (FuriHalVersionOTPv0*)FURI_HAL_VERSION_OTP_ADDRESS;
  104. furi_hal_version.timestamp = otp->header_timestamp;
  105. furi_hal_version.board_version = otp->board_version;
  106. furi_hal_version.board_target = otp->board_target;
  107. furi_hal_version.board_body = otp->board_body;
  108. furi_hal_version.board_connect = otp->board_connect;
  109. furi_hal_version_set_name(otp->name);
  110. }
  111. static void furi_hal_version_load_otp_v1() {
  112. const FuriHalVersionOTPv1* otp = (FuriHalVersionOTPv1*)FURI_HAL_VERSION_OTP_ADDRESS;
  113. furi_hal_version.timestamp = otp->header_timestamp;
  114. furi_hal_version.board_version = otp->board_version;
  115. furi_hal_version.board_target = otp->board_target;
  116. furi_hal_version.board_body = otp->board_body;
  117. furi_hal_version.board_connect = otp->board_connect;
  118. furi_hal_version.board_color = otp->board_color;
  119. furi_hal_version.board_region = otp->board_region;
  120. furi_hal_version_set_name(otp->name);
  121. }
  122. static void furi_hal_version_load_otp_v2() {
  123. const FuriHalVersionOTPv2* otp = (FuriHalVersionOTPv2*)FURI_HAL_VERSION_OTP_ADDRESS;
  124. // 1st block, programmed afer baking
  125. furi_hal_version.timestamp = otp->header_timestamp;
  126. // 2nd block, programmed afer baking
  127. furi_hal_version.board_version = otp->board_version;
  128. furi_hal_version.board_target = otp->board_target;
  129. furi_hal_version.board_body = otp->board_body;
  130. furi_hal_version.board_connect = otp->board_connect;
  131. furi_hal_version.board_display = otp->board_display;
  132. // 3rd and 4th blocks, programmed on FATP stage
  133. if(otp->board_color != 0xFF) {
  134. furi_hal_version.board_color = otp->board_color;
  135. furi_hal_version.board_region = otp->board_region;
  136. furi_hal_version_set_name(otp->name);
  137. } else {
  138. furi_hal_version.board_color = 0;
  139. furi_hal_version.board_region = 0;
  140. furi_hal_version_set_name(NULL);
  141. }
  142. }
  143. void furi_hal_version_init() {
  144. switch(furi_hal_version_get_otp_version()) {
  145. case FuriHalVersionOtpVersionUnknown:
  146. furi_hal_version_load_otp_default();
  147. break;
  148. case FuriHalVersionOtpVersionEmpty:
  149. furi_hal_version_load_otp_default();
  150. break;
  151. case FuriHalVersionOtpVersion0:
  152. furi_hal_version_load_otp_v0();
  153. break;
  154. case FuriHalVersionOtpVersion1:
  155. furi_hal_version_load_otp_v1();
  156. break;
  157. case FuriHalVersionOtpVersion2:
  158. furi_hal_version_load_otp_v2();
  159. break;
  160. default:
  161. furi_crash(NULL);
  162. }
  163. furi_hal_rtc_set_register(FuriHalRtcRegisterVersion, (uint32_t)version_get());
  164. FURI_LOG_I(TAG, "Init OK");
  165. }
  166. bool furi_hal_version_do_i_belong_here() {
  167. return furi_hal_version_get_hw_target() == 7;
  168. }
  169. const char* furi_hal_version_get_model_name() {
  170. return "Flipper Zero";
  171. }
  172. FuriHalVersionOtpVersion furi_hal_version_get_otp_version() {
  173. if(*(uint64_t*)FURI_HAL_VERSION_OTP_ADDRESS == 0xFFFFFFFF) {
  174. return FuriHalVersionOtpVersionEmpty;
  175. } else {
  176. if(((FuriHalVersionOTPv1*)FURI_HAL_VERSION_OTP_ADDRESS)->header_magic ==
  177. FURI_HAL_VERSION_OTP_HEADER_MAGIC) {
  178. // Version 1+
  179. uint8_t version = ((FuriHalVersionOTPv1*)FURI_HAL_VERSION_OTP_ADDRESS)->header_version;
  180. if(version >= FuriHalVersionOtpVersion1 && version <= FuriHalVersionOtpVersion2) {
  181. return version;
  182. } else {
  183. return FuriHalVersionOtpVersionUnknown;
  184. }
  185. } else if(((FuriHalVersionOTPv0*)FURI_HAL_VERSION_OTP_ADDRESS)->board_version <= 10) {
  186. // Version 0
  187. return FuriHalVersionOtpVersion0;
  188. } else {
  189. // Version Unknown
  190. return FuriHalVersionOtpVersionUnknown;
  191. }
  192. }
  193. }
  194. uint8_t furi_hal_version_get_hw_version() {
  195. return furi_hal_version.board_version;
  196. }
  197. uint8_t furi_hal_version_get_hw_target() {
  198. return furi_hal_version.board_target;
  199. }
  200. uint8_t furi_hal_version_get_hw_body() {
  201. return furi_hal_version.board_body;
  202. }
  203. FuriHalVersionColor furi_hal_version_get_hw_color() {
  204. return furi_hal_version.board_color;
  205. }
  206. uint8_t furi_hal_version_get_hw_connect() {
  207. return furi_hal_version.board_connect;
  208. }
  209. FuriHalVersionRegion furi_hal_version_get_hw_region() {
  210. return furi_hal_version.board_region;
  211. }
  212. const char* furi_hal_version_get_hw_region_name() {
  213. switch(furi_hal_version_get_hw_region()) {
  214. case FuriHalVersionRegionUnknown:
  215. return "R00";
  216. case FuriHalVersionRegionEuRu:
  217. return "R01";
  218. case FuriHalVersionRegionUsCaAu:
  219. return "R02";
  220. case FuriHalVersionRegionJp:
  221. return "R03";
  222. case FuriHalVersionRegionWorld:
  223. return "R04";
  224. }
  225. return "R??";
  226. }
  227. FuriHalVersionDisplay furi_hal_version_get_hw_display() {
  228. return furi_hal_version.board_display;
  229. }
  230. uint32_t furi_hal_version_get_hw_timestamp() {
  231. return furi_hal_version.timestamp;
  232. }
  233. const char* furi_hal_version_get_name_ptr() {
  234. return *furi_hal_version.name == 0x00 ? NULL : furi_hal_version.name;
  235. }
  236. const char* furi_hal_version_get_device_name_ptr() {
  237. return furi_hal_version.device_name + 1;
  238. }
  239. const char* furi_hal_version_get_ble_local_device_name_ptr() {
  240. return furi_hal_version.device_name;
  241. }
  242. const uint8_t* furi_hal_version_get_ble_mac() {
  243. return furi_hal_version.ble_mac;
  244. }
  245. const struct Version* furi_hal_version_get_firmware_version(void) {
  246. return version_get();
  247. }
  248. size_t furi_hal_version_uid_size() {
  249. return 64 / 8;
  250. }
  251. const uint8_t* furi_hal_version_uid() {
  252. return (const uint8_t*)UID64_BASE;
  253. }