* Core2, SRAM2: use ob, provide safety gap * thread: comment about critical section and scheduler state
		
			
				
	
	
		
			126 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			126 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
#include <furi_hal.h>
 | 
						|
#include <furi_hal_memory.h>
 | 
						|
#include <furi_hal_rtc.h>
 | 
						|
 | 
						|
#define TAG "FuriHalMemory"
 | 
						|
 | 
						|
// STM(TM) Copro(TM) bug(TM) workaround size
 | 
						|
#define RAM2B_COPRO_GAP_SIZE_KB 2
 | 
						|
 | 
						|
typedef enum {
 | 
						|
    SRAM_A,
 | 
						|
    SRAM_B,
 | 
						|
    SRAM_MAX,
 | 
						|
} SRAM;
 | 
						|
 | 
						|
typedef struct {
 | 
						|
    void* start;
 | 
						|
    uint32_t size;
 | 
						|
} FuriHalMemoryRegion;
 | 
						|
 | 
						|
typedef struct {
 | 
						|
    FuriHalMemoryRegion region[SRAM_MAX];
 | 
						|
} FuriHalMemory;
 | 
						|
 | 
						|
static FuriHalMemory* furi_hal_memory = NULL;
 | 
						|
 | 
						|
extern const void __sram2a_start__;
 | 
						|
extern const void __sram2a_free__;
 | 
						|
extern const void __sram2b_start__;
 | 
						|
 | 
						|
void furi_hal_memory_init() {
 | 
						|
    if(furi_hal_rtc_get_boot_mode() != FuriHalRtcBootModeNormal) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    FuriHalMemory* memory = malloc(sizeof(FuriHalMemory));
 | 
						|
 | 
						|
    uint32_t sbrsa = (FLASH->SRRVR & FLASH_SRRVR_SBRSA_Msk) >> FLASH_SRRVR_SBRSA_Pos;
 | 
						|
    uint32_t snbrsa = (FLASH->SRRVR & FLASH_SRRVR_SNBRSA_Msk) >> FLASH_SRRVR_SNBRSA_Pos;
 | 
						|
 | 
						|
    uint32_t sram2a_busy_size = (uint32_t)&__sram2a_free__ - (uint32_t)&__sram2a_start__;
 | 
						|
    uint32_t sram2a_unprotected_size = (sbrsa)*1024;
 | 
						|
    uint32_t sram2b_unprotected_size = (snbrsa)*1024;
 | 
						|
 | 
						|
    // STM(TM) Copro(TM) bug(TM) workaround
 | 
						|
    sram2b_unprotected_size -= 1024 * RAM2B_COPRO_GAP_SIZE_KB;
 | 
						|
 | 
						|
    memory->region[SRAM_A].start = (uint8_t*)&__sram2a_free__;
 | 
						|
    memory->region[SRAM_B].start = (uint8_t*)&__sram2b_start__;
 | 
						|
 | 
						|
    if(sram2a_unprotected_size > sram2a_busy_size) {
 | 
						|
        memory->region[SRAM_A].size = sram2a_unprotected_size - sram2a_busy_size;
 | 
						|
    } else {
 | 
						|
        memory->region[SRAM_A].size = 0;
 | 
						|
    }
 | 
						|
    memory->region[SRAM_B].size = sram2b_unprotected_size;
 | 
						|
 | 
						|
    FURI_LOG_I(
 | 
						|
        TAG, "SRAM2A: 0x%p, %lu", memory->region[SRAM_A].start, memory->region[SRAM_A].size);
 | 
						|
    FURI_LOG_I(
 | 
						|
        TAG, "SRAM2B: 0x%p, %lu", memory->region[SRAM_B].start, memory->region[SRAM_B].size);
 | 
						|
 | 
						|
    if((memory->region[SRAM_A].size > 0) || (memory->region[SRAM_B].size > 0)) {
 | 
						|
        if((memory->region[SRAM_A].size > 0)) {
 | 
						|
            FURI_LOG_I(TAG, "SRAM2A clear");
 | 
						|
            memset(memory->region[SRAM_A].start, 0, memory->region[SRAM_A].size);
 | 
						|
        }
 | 
						|
        if((memory->region[SRAM_B].size > 0)) {
 | 
						|
            FURI_LOG_I(TAG, "SRAM2B clear");
 | 
						|
            memset(memory->region[SRAM_B].start, 0, memory->region[SRAM_B].size);
 | 
						|
        }
 | 
						|
        furi_hal_memory = memory;
 | 
						|
        FURI_LOG_I(TAG, "Enabled");
 | 
						|
    } else {
 | 
						|
        free(memory);
 | 
						|
        FURI_LOG_E(TAG, "No SRAM2 available");
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void* furi_hal_memory_alloc(size_t size) {
 | 
						|
    if(FURI_IS_IRQ_MODE()) {
 | 
						|
        furi_crash("memmgt in ISR");
 | 
						|
    }
 | 
						|
 | 
						|
    if(furi_hal_memory == NULL) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
 | 
						|
    void* allocated_memory = NULL;
 | 
						|
    FURI_CRITICAL_ENTER();
 | 
						|
    for(int i = 0; i < SRAM_MAX; i++) {
 | 
						|
        if(furi_hal_memory->region[i].size >= size) {
 | 
						|
            void* ptr = furi_hal_memory->region[i].start;
 | 
						|
            furi_hal_memory->region[i].start += size;
 | 
						|
            furi_hal_memory->region[i].size -= size;
 | 
						|
            allocated_memory = ptr;
 | 
						|
            break;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    FURI_CRITICAL_EXIT();
 | 
						|
 | 
						|
    return allocated_memory;
 | 
						|
}
 | 
						|
 | 
						|
size_t furi_hal_memory_get_free() {
 | 
						|
    if(furi_hal_memory == NULL) return 0;
 | 
						|
 | 
						|
    size_t free = 0;
 | 
						|
    for(int i = 0; i < SRAM_MAX; i++) {
 | 
						|
        free += furi_hal_memory->region[i].size;
 | 
						|
    }
 | 
						|
    return free;
 | 
						|
}
 | 
						|
 | 
						|
size_t furi_hal_memory_max_pool_block() {
 | 
						|
    if(furi_hal_memory == NULL) return 0;
 | 
						|
 | 
						|
    size_t max = 0;
 | 
						|
    for(int i = 0; i < SRAM_MAX; i++) {
 | 
						|
        if(furi_hal_memory->region[i].size > max) {
 | 
						|
            max = furi_hal_memory->region[i].size;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    return max;
 | 
						|
}
 |