furi-hal-version.c 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238
  1. #include <furi-hal-version.h>
  2. #include <furi.h>
  3. #include <stm32wbxx.h>
  4. #include <stm32wbxx_ll_rtc.h>
  5. #include <stdio.h>
  6. #include "ble.h"
  7. #define FURI_HAL_VERSION_OTP_HEADER_MAGIC 0xBABE
  8. #define FURI_HAL_VERSION_OTP_ADDRESS OTP_AREA_BASE
  9. /** OTP Versions enum */
  10. typedef enum {
  11. FuriHalVersionOtpVersion0=0x00,
  12. FuriHalVersionOtpVersion1=0x01,
  13. FuriHalVersionOtpVersionEmpty=0xFFFFFFFE,
  14. FuriHalVersionOtpVersionUnknown=0xFFFFFFFF,
  15. } FuriHalVersionOtpVersion;
  16. /** OTP V0 Structure: prototypes and early EVT */
  17. typedef struct {
  18. uint8_t board_version;
  19. uint8_t board_target;
  20. uint8_t board_body;
  21. uint8_t board_connect;
  22. uint32_t header_timestamp;
  23. char name[FURI_HAL_VERSION_NAME_LENGTH];
  24. } FuriHalVersionOTPv0;
  25. /** OTP V1 Structure: late EVT, DVT, PVT, Production */
  26. typedef struct {
  27. /* First 64 bits: header */
  28. uint16_t header_magic;
  29. uint8_t header_version;
  30. uint8_t header_reserved;
  31. uint32_t header_timestamp;
  32. /* Second 64 bits: board info */
  33. uint8_t board_version; /** Board version */
  34. uint8_t board_target; /** Board target firmware */
  35. uint8_t board_body; /** Board body */
  36. uint8_t board_connect; /** Board interconnect */
  37. uint8_t board_color; /** Board color */
  38. uint8_t board_region; /** Board region */
  39. uint16_t board_reserved; /** Reserved for future use, 0x0000 */
  40. /* Third 64 bits: Unique Device Name */
  41. char name[FURI_HAL_VERSION_NAME_LENGTH]; /** Unique Device Name */
  42. } FuriHalVersionOTPv1;
  43. /** Represenation Model: */
  44. typedef struct {
  45. FuriHalVersionOtpVersion otp_version;
  46. uint32_t timestamp;
  47. uint8_t board_version; /** Board version */
  48. uint8_t board_target; /** Board target firmware */
  49. uint8_t board_body; /** Board body */
  50. uint8_t board_connect; /** Board interconnect */
  51. uint8_t board_color; /** Board color */
  52. uint8_t board_region; /** Board region */
  53. char name[FURI_HAL_VERSION_ARRAY_NAME_LENGTH]; /** \0 terminated name */
  54. char device_name[FURI_HAL_VERSION_DEVICE_NAME_LENGTH]; /** device name for special needs */
  55. uint8_t ble_mac[6];
  56. } FuriHalVersion;
  57. static FuriHalVersion furi_hal_version = {0};
  58. static FuriHalVersionOtpVersion furi_hal_version_get_otp_version() {
  59. if (*(uint64_t*)FURI_HAL_VERSION_OTP_ADDRESS == 0xFFFFFFFF) {
  60. return FuriHalVersionOtpVersionEmpty;
  61. } else {
  62. if (((FuriHalVersionOTPv1*)FURI_HAL_VERSION_OTP_ADDRESS)->header_magic == FURI_HAL_VERSION_OTP_HEADER_MAGIC) {
  63. return FuriHalVersionOtpVersion1;
  64. } else if (((FuriHalVersionOTPv0*)FURI_HAL_VERSION_OTP_ADDRESS)->board_version <= 10) {
  65. return FuriHalVersionOtpVersion0;
  66. } else {
  67. return FuriHalVersionOtpVersionUnknown;
  68. }
  69. }
  70. }
  71. static void furi_hal_version_set_name(const char* name) {
  72. if(name != NULL) {
  73. strlcpy(furi_hal_version.name, name, FURI_HAL_VERSION_ARRAY_NAME_LENGTH);
  74. snprintf(
  75. furi_hal_version.device_name,
  76. FURI_HAL_VERSION_DEVICE_NAME_LENGTH,
  77. "xFlipper %s",
  78. furi_hal_version.name);
  79. } else {
  80. snprintf(
  81. furi_hal_version.device_name,
  82. FURI_HAL_VERSION_DEVICE_NAME_LENGTH,
  83. "xFlipper");
  84. }
  85. furi_hal_version.device_name[0] = AD_TYPE_COMPLETE_LOCAL_NAME;
  86. // BLE Mac address
  87. uint32_t udn = LL_FLASH_GetUDN();
  88. uint32_t company_id = LL_FLASH_GetSTCompanyID();
  89. uint32_t device_id = LL_FLASH_GetDeviceID();
  90. furi_hal_version.ble_mac[0] = (uint8_t)(udn & 0x000000FF);
  91. furi_hal_version.ble_mac[1] = (uint8_t)( (udn & 0x0000FF00) >> 8 );
  92. furi_hal_version.ble_mac[2] = (uint8_t)( (udn & 0x00FF0000) >> 16 );
  93. furi_hal_version.ble_mac[3] = (uint8_t)device_id;
  94. furi_hal_version.ble_mac[4] = (uint8_t)(company_id & 0x000000FF);
  95. furi_hal_version.ble_mac[5] = (uint8_t)( (company_id & 0x0000FF00) >> 8 );
  96. }
  97. static void furi_hal_version_load_otp_default() {
  98. furi_hal_version_set_name(NULL);
  99. }
  100. static void furi_hal_version_load_otp_v0() {
  101. const FuriHalVersionOTPv0* otp = (FuriHalVersionOTPv0*)FURI_HAL_VERSION_OTP_ADDRESS;
  102. furi_hal_version.timestamp = otp->header_timestamp;
  103. furi_hal_version.board_version = otp->board_version;
  104. furi_hal_version.board_target = otp->board_target;
  105. furi_hal_version.board_body = otp->board_body;
  106. furi_hal_version.board_connect = otp->board_connect;
  107. furi_hal_version.board_color = 0;
  108. furi_hal_version.board_region = 0;
  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. void furi_hal_version_init() {
  123. furi_hal_version.otp_version = furi_hal_version_get_otp_version();
  124. switch(furi_hal_version.otp_version) {
  125. case FuriHalVersionOtpVersionUnknown:
  126. furi_hal_version_load_otp_default();
  127. break;
  128. case FuriHalVersionOtpVersionEmpty:
  129. furi_hal_version_load_otp_default();
  130. break;
  131. case FuriHalVersionOtpVersion0:
  132. furi_hal_version_load_otp_v0();
  133. break;
  134. case FuriHalVersionOtpVersion1:
  135. furi_hal_version_load_otp_v1();
  136. break;
  137. default: furi_crash(NULL);
  138. }
  139. FURI_LOG_I("FuriHalVersion", "Init OK");
  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 uint8_t furi_hal_version_get_hw_version() {
  148. return furi_hal_version.board_version;
  149. }
  150. const uint8_t furi_hal_version_get_hw_target() {
  151. return furi_hal_version.board_target;
  152. }
  153. const uint8_t furi_hal_version_get_hw_body() {
  154. return furi_hal_version.board_body;
  155. }
  156. const FuriHalVersionColor furi_hal_version_get_hw_color() {
  157. return furi_hal_version.board_color;
  158. }
  159. const uint8_t furi_hal_version_get_hw_connect() {
  160. return furi_hal_version.board_connect;
  161. }
  162. const FuriHalVersionRegion furi_hal_version_get_hw_region() {
  163. return furi_hal_version.board_region;
  164. }
  165. const uint32_t furi_hal_version_get_hw_timestamp() {
  166. return furi_hal_version.timestamp;
  167. }
  168. const char* furi_hal_version_get_name_ptr() {
  169. return *furi_hal_version.name == 0x00 ? NULL : furi_hal_version.name;
  170. }
  171. const char* furi_hal_version_get_device_name_ptr() {
  172. return furi_hal_version.device_name + 1;
  173. }
  174. const char* furi_hal_version_get_ble_local_device_name_ptr() {
  175. return furi_hal_version.device_name;
  176. }
  177. const uint8_t* furi_hal_version_get_ble_mac() {
  178. return furi_hal_version.ble_mac;
  179. }
  180. const struct Version* furi_hal_version_get_firmware_version(void) {
  181. return version_get();
  182. }
  183. const struct Version* furi_hal_version_get_boot_version(void) {
  184. #ifdef NO_BOOTLOADER
  185. return 0;
  186. #else
  187. /* Backup register which points to structure in flash memory */
  188. return (const struct Version*)LL_RTC_BAK_GetRegister(RTC, LL_RTC_BKP_DR1);
  189. #endif
  190. }
  191. size_t furi_hal_version_uid_size() {
  192. return 64/8;
  193. }
  194. const uint8_t* furi_hal_version_uid() {
  195. return (const uint8_t *)UID64_BASE;
  196. }