#include #include #include #include #include #include #include #define TAG "FuriHalVersion" #define FURI_HAL_VERSION_OTP_HEADER_MAGIC 0xBABE #define FURI_HAL_VERSION_OTP_ADDRESS OTP_AREA_BASE /** OTP V0 Structure: prototypes and early EVT */ typedef struct { uint8_t board_version; uint8_t board_target; uint8_t board_body; uint8_t board_connect; uint32_t header_timestamp; char name[FURI_HAL_VERSION_NAME_LENGTH]; } FuriHalVersionOTPv0; /** OTP V1 Structure: late EVT, DVT */ typedef struct { /* First 64 bits: header */ uint16_t header_magic; uint8_t header_version; uint8_t header_reserved; uint32_t header_timestamp; /* Second 64 bits: board info */ uint8_t board_version; /** Board version */ uint8_t board_target; /** Board target firmware */ uint8_t board_body; /** Board body */ uint8_t board_connect; /** Board interconnect */ uint8_t board_color; /** Board color */ uint8_t board_region; /** Board region */ uint16_t board_reserved; /** Reserved for future use, 0x0000 */ /* Third 64 bits: Unique Device Name */ char name[FURI_HAL_VERSION_NAME_LENGTH]; /** Unique Device Name */ } FuriHalVersionOTPv1; /** OTP V2 Structure: DVT2, PVT, Production */ typedef struct { /* Early First 64 bits: header */ uint16_t header_magic; uint8_t header_version; uint8_t header_reserved; uint32_t header_timestamp; /* Early Second 64 bits: board info */ uint8_t board_version; /** Board version */ uint8_t board_target; /** Board target firmware */ uint8_t board_body; /** Board body */ uint8_t board_connect; /** Board interconnect */ uint8_t board_display; /** Board display */ uint8_t board_reserved2_0; /** Reserved for future use, 0x00 */ uint16_t board_reserved2_1; /** Reserved for future use, 0x0000 */ /* Late Third 64 bits: device info */ uint8_t board_color; /** Board color */ uint8_t board_region; /** Board region */ uint16_t board_reserved3_0; /** Reserved for future use, 0x0000 */ uint32_t board_reserved3_1; /** Reserved for future use, 0x00000000 */ /* Late Fourth 64 bits: Unique Device Name */ char name[FURI_HAL_VERSION_NAME_LENGTH]; /** Unique Device Name */ } FuriHalVersionOTPv2; /** Represenation Model: */ typedef struct { uint32_t timestamp; uint8_t board_version; /** Board version */ uint8_t board_target; /** Board target firmware */ uint8_t board_body; /** Board body */ uint8_t board_connect; /** Board interconnect */ uint8_t board_color; /** Board color */ uint8_t board_region; /** Board region */ uint8_t board_display; /** Board display */ char name[FURI_HAL_VERSION_ARRAY_NAME_LENGTH]; /** \0 terminated name */ char device_name[FURI_HAL_VERSION_DEVICE_NAME_LENGTH]; /** device name for special needs */ uint8_t ble_mac[6]; } FuriHalVersion; static FuriHalVersion furi_hal_version = {0}; static void furi_hal_version_set_name(const char* name) { if(name != NULL) { strlcpy(furi_hal_version.name, name, FURI_HAL_VERSION_ARRAY_NAME_LENGTH); snprintf( furi_hal_version.device_name, FURI_HAL_VERSION_DEVICE_NAME_LENGTH, "xFlipper %s", furi_hal_version.name); } else { snprintf(furi_hal_version.device_name, FURI_HAL_VERSION_DEVICE_NAME_LENGTH, "xFlipper"); } furi_hal_version.device_name[0] = AD_TYPE_COMPLETE_LOCAL_NAME; // BLE Mac address uint32_t udn = LL_FLASH_GetUDN(); if(version_get_custom_name(NULL) != NULL) { udn = *((uint32_t*)version_get_custom_name(NULL)); } uint32_t company_id = LL_FLASH_GetSTCompanyID(); uint32_t device_id = LL_FLASH_GetDeviceID(); furi_hal_version.ble_mac[0] = (uint8_t)(udn & 0x000000FF); furi_hal_version.ble_mac[1] = (uint8_t)((udn & 0x0000FF00) >> 8); furi_hal_version.ble_mac[2] = (uint8_t)((udn & 0x00FF0000) >> 16); furi_hal_version.ble_mac[3] = (uint8_t)device_id; furi_hal_version.ble_mac[4] = (uint8_t)(company_id & 0x000000FF); furi_hal_version.ble_mac[5] = (uint8_t)((company_id & 0x0000FF00) >> 8); } static void furi_hal_version_load_otp_default() { furi_hal_version_set_name(NULL); } static void furi_hal_version_load_otp_v0() { const FuriHalVersionOTPv0* otp = (FuriHalVersionOTPv0*)FURI_HAL_VERSION_OTP_ADDRESS; furi_hal_version.timestamp = otp->header_timestamp; furi_hal_version.board_version = otp->board_version; furi_hal_version.board_target = otp->board_target; furi_hal_version.board_body = otp->board_body; furi_hal_version.board_connect = otp->board_connect; if(version_get_custom_name(NULL) != NULL) { furi_hal_version_set_name(version_get_custom_name(NULL)); } else { furi_hal_version_set_name(otp->name); } } static void furi_hal_version_load_otp_v1() { const FuriHalVersionOTPv1* otp = (FuriHalVersionOTPv1*)FURI_HAL_VERSION_OTP_ADDRESS; furi_hal_version.timestamp = otp->header_timestamp; furi_hal_version.board_version = otp->board_version; furi_hal_version.board_target = otp->board_target; furi_hal_version.board_body = otp->board_body; furi_hal_version.board_connect = otp->board_connect; furi_hal_version.board_color = otp->board_color; furi_hal_version.board_region = otp->board_region; if(version_get_custom_name(NULL) != NULL) { furi_hal_version_set_name(version_get_custom_name(NULL)); } else { furi_hal_version_set_name(otp->name); } } static void furi_hal_version_load_otp_v2() { const FuriHalVersionOTPv2* otp = (FuriHalVersionOTPv2*)FURI_HAL_VERSION_OTP_ADDRESS; // 1st block, programmed afer baking furi_hal_version.timestamp = otp->header_timestamp; // 2nd block, programmed afer baking furi_hal_version.board_version = otp->board_version; furi_hal_version.board_target = otp->board_target; furi_hal_version.board_body = otp->board_body; furi_hal_version.board_connect = otp->board_connect; furi_hal_version.board_display = otp->board_display; // 3rd and 4th blocks, programmed on FATP stage if(otp->board_color != 0xFF) { furi_hal_version.board_color = otp->board_color; furi_hal_version.board_region = otp->board_region; if(version_get_custom_name(NULL) != NULL) { furi_hal_version_set_name(version_get_custom_name(NULL)); } else { furi_hal_version_set_name(otp->name); } } else { furi_hal_version.board_color = 0; furi_hal_version.board_region = 0; furi_hal_version_set_name(NULL); } } void furi_hal_version_init() { switch(furi_hal_version_get_otp_version()) { case FuriHalVersionOtpVersionUnknown: furi_hal_version_load_otp_default(); break; case FuriHalVersionOtpVersionEmpty: furi_hal_version_load_otp_default(); break; case FuriHalVersionOtpVersion0: furi_hal_version_load_otp_v0(); break; case FuriHalVersionOtpVersion1: furi_hal_version_load_otp_v1(); break; case FuriHalVersionOtpVersion2: furi_hal_version_load_otp_v2(); break; default: furi_crash(NULL); } furi_hal_rtc_set_register(FuriHalRtcRegisterVersion, (uint32_t)version_get()); FURI_LOG_I(TAG, "Init OK"); } bool furi_hal_version_do_i_belong_here() { return furi_hal_version_get_hw_target() == 7; } const char* furi_hal_version_get_model_name() { return "Flipper Zero"; } FuriHalVersionOtpVersion furi_hal_version_get_otp_version() { if(*(uint64_t*)FURI_HAL_VERSION_OTP_ADDRESS == 0xFFFFFFFF) { return FuriHalVersionOtpVersionEmpty; } else { if(((FuriHalVersionOTPv1*)FURI_HAL_VERSION_OTP_ADDRESS)->header_magic == FURI_HAL_VERSION_OTP_HEADER_MAGIC) { // Version 1+ uint8_t version = ((FuriHalVersionOTPv1*)FURI_HAL_VERSION_OTP_ADDRESS)->header_version; if(version >= FuriHalVersionOtpVersion1 && version <= FuriHalVersionOtpVersion2) { return version; } else { return FuriHalVersionOtpVersionUnknown; } } else if(((FuriHalVersionOTPv0*)FURI_HAL_VERSION_OTP_ADDRESS)->board_version <= 10) { // Version 0 return FuriHalVersionOtpVersion0; } else { // Version Unknown return FuriHalVersionOtpVersionUnknown; } } } uint8_t furi_hal_version_get_hw_version() { return furi_hal_version.board_version; } uint8_t furi_hal_version_get_hw_target() { return furi_hal_version.board_target; } uint8_t furi_hal_version_get_hw_body() { return furi_hal_version.board_body; } FuriHalVersionColor furi_hal_version_get_hw_color() { return furi_hal_version.board_color; } uint8_t furi_hal_version_get_hw_connect() { return furi_hal_version.board_connect; } FuriHalVersionRegion furi_hal_version_get_hw_region() { return FuriHalVersionRegionUnknown; } const char* furi_hal_version_get_hw_region_name() { switch(furi_hal_version_get_hw_region()) { case FuriHalVersionRegionUnknown: return "R00"; case FuriHalVersionRegionEuRu: return "R01"; case FuriHalVersionRegionUsCaAu: return "R02"; case FuriHalVersionRegionJp: return "R03"; case FuriHalVersionRegionWorld: return "R04"; } return "R??"; } FuriHalVersionDisplay furi_hal_version_get_hw_display() { return furi_hal_version.board_display; } uint32_t furi_hal_version_get_hw_timestamp() { return furi_hal_version.timestamp; } const char* furi_hal_version_get_name_ptr() { return *furi_hal_version.name == 0x00 ? NULL : furi_hal_version.name; } const char* furi_hal_version_get_device_name_ptr() { return furi_hal_version.device_name + 1; } const char* furi_hal_version_get_ble_local_device_name_ptr() { return furi_hal_version.device_name; } const uint8_t* furi_hal_version_get_ble_mac() { return furi_hal_version.ble_mac; } const struct Version* furi_hal_version_get_firmware_version(void) { return version_get(); } size_t furi_hal_version_uid_size() { return 64 / 8; } const uint8_t* furi_hal_version_uid() { if(version_get_custom_name(NULL) != NULL) { return (const uint8_t*)&(*((uint32_t*)version_get_custom_name(NULL))); } return (const uint8_t*)UID64_BASE; }