furi_hal_version.c 8.4 KB

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