mirror of
https://github.com/DarkFlippers/unleashed-firmware
synced 2024-12-01 00:39:12 +00:00
e02040107b
* WIP on stripping fw * Compact FW build - use RAM_EXEC=1 COMPACT=1 DEBUG=0 * Fixed uninitialized storage struct; small fixes to compact fw * Flasher srv w/mocked flash ops * Fixed typos & accomodated FFF changes * Alternative fw startup branch * Working load & jmp to RAM fw * +manifest processing for stage loader; + crc verification for stage payload * Fixed questionable code & potential leaks * Lowered screen update rate; added radio stack update stubs; working dfu write * Console EP with manifest & stage validation * Added microtar lib; minor ui fixes for updater * Removed microtar * Removed mtar #2 * Added a better version of microtar * TAR archive api; LFS backup & restore core * Recursive backup/restore * LFS worker thread * Added system apps to loader - not visible in UI; full update process with restarts * Typo fix * Dropped BL & f6; tooling for updater WIP * Minor py fixes * Minor fixes to make it build after merge * Ported flash workaround from BL + fixed visuals * Minor cleanup * Chmod + loader app search fix * Python linter fix * Removed usb stuff & float read support for staged loader == -10% of binary size * Added backup/restore & update pb requests * Added stub impl to RPC for backup/restore/update commands * Reworked TAR to use borrowed Storage api; slightly reduced build size by removing `static string`; hidden update-related RPC behind defines * Moved backup&restore to storage * Fixed new message types * Backup/restore/update RPC impl * Moved furi_hal_crc to LL; minor fixes * CRC HAL rework to LL * Purging STM HAL * Brought back minimal DFU boot mode (no gui); additional crc state checks * Added splash screen, BROKEN usb function * Clock init rework WIP * Stripped graphics from DFU mode * Temp fix for unused static fun * WIP update picker - broken! * Fixed UI * Bumping version * Fixed RTC setup * Backup to update folder instead of ext root * Removed unused scenes & more usb remnants from staged loader * CI updates * Fixed update bundle name * Temporary restored USB handler * Attempt to prevent .text corruption * Comments on how I spent this Saturday * Added update file icon * Documentation updates * Moved common code to lib folder * Storage: more unit tests * Storage: blocking dir open, differentiate file and dir when freed. * Major refactoring; added input processing to updater to allow retrying on failures (not very useful prob). Added API for extraction of thread return value * Removed re-init check for manifest * Changed low-level path manipulation to toolbox/path.h; makefile cleanup; tiny fix in lint.py * Increased update worker stack size * Text fixes in backup CLI * Displaying number of update stages to run; removed timeout in handling errors * Bumping version * Added thread cleanup for spawner thread * Updated build targets to exclude firmware bundle from 'ALL' * Fixed makefile for update_package; skipping VCP init for update mode (ugly) * Switched github build from ALL to update_package * Added +x for dist_update.sh * Cli: add total heap size to "free" command * Moved (RAM) suffix to build version instead of git commit no. * DFU comment * Some fixes suggested by clang-tidy * Fixed recursive PREFIX macro * Makefile: gather all new rules in updater namespace. FuriHal: rename bootloader to boot, isr safe delays * Github: correct build target name in firmware build * FuriHal: move target switch to boot * Makefile: fix firmware flash * Furi, FuriHal: move kernel start to furi, early init * Drop bootloader related stuff * Drop cube. Drop bootloader linker script. * Renamed update_hl, moved constants to #defines * Moved update-related boot mode to separate bitfield * Reworked updater cli to single entry point; fixed crash on tar cleanup * Added Python replacement for dist shell scripts * Linter fixes for dist.py +x * Fixes for environment suffix * Dropped bash scripts * Added dirty build flag to version structure & interfaces * Version string escapes * Fixed flag logic in dist.py; added support for App instances being imported and not terminating the whole program * Fixed fw address in ReadMe.md * Rpc: fix crash on double screen start * Return back original boot behavior and fix jump to system bootloader * Cleanup code, add error sequence for RTC * Update firmware readme * FuriHal: drop boot, restructure RTC registers usage and add header register check * Furi goes first * Toolchain: add ccache support * Renamed update bundle dir Co-authored-by: DrZlo13 <who.just.the.doctor@gmail.com> Co-authored-by: あく <alleteam@gmail.com>
289 lines
9.3 KiB
C
289 lines
9.3 KiB
C
#include <furi_hal_version.h>
|
|
#include <furi_hal_rtc.h>
|
|
|
|
#include <furi.h>
|
|
#include <stm32wbxx.h>
|
|
#include <stm32wbxx_ll_rtc.h>
|
|
|
|
#include <stdio.h>
|
|
#include "ble.h"
|
|
|
|
#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();
|
|
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;
|
|
|
|
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;
|
|
|
|
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;
|
|
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";
|
|
}
|
|
|
|
const 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;
|
|
}
|
|
}
|
|
}
|
|
|
|
const uint8_t furi_hal_version_get_hw_version() {
|
|
return furi_hal_version.board_version;
|
|
}
|
|
|
|
const uint8_t furi_hal_version_get_hw_target() {
|
|
return furi_hal_version.board_target;
|
|
}
|
|
|
|
const uint8_t furi_hal_version_get_hw_body() {
|
|
return furi_hal_version.board_body;
|
|
}
|
|
|
|
const FuriHalVersionColor furi_hal_version_get_hw_color() {
|
|
return furi_hal_version.board_color;
|
|
}
|
|
|
|
const uint8_t furi_hal_version_get_hw_connect() {
|
|
return furi_hal_version.board_connect;
|
|
}
|
|
|
|
const FuriHalVersionRegion furi_hal_version_get_hw_region() {
|
|
return furi_hal_version.board_region;
|
|
}
|
|
|
|
const FuriHalVersionDisplay furi_hal_version_get_hw_display() {
|
|
return furi_hal_version.board_display;
|
|
}
|
|
|
|
const 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() {
|
|
return (const uint8_t*)UID64_BASE;
|
|
}
|