102 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			102 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
// Copyright 2022 Nick Brassel (@tzarc)
 | 
						|
// SPDX-License-Identifier: GPL-2.0-or-later
 | 
						|
#include <stdbool.h>
 | 
						|
#include <hal.h>
 | 
						|
#include "util.h"
 | 
						|
#include "timer.h"
 | 
						|
#include "wear_leveling.h"
 | 
						|
#include "wear_leveling_internal.h"
 | 
						|
 | 
						|
#ifndef WEAR_LEVELING_EXTERNAL_FLASH_BULK_COUNT
 | 
						|
#    define WEAR_LEVELING_EXTERNAL_FLASH_BULK_COUNT 32
 | 
						|
#endif // WEAR_LEVELING_EXTERNAL_FLASH_BULK_COUNT
 | 
						|
 | 
						|
bool backing_store_init(void) {
 | 
						|
    bs_dprintf("Init\n");
 | 
						|
    flash_init();
 | 
						|
    return true;
 | 
						|
}
 | 
						|
 | 
						|
bool backing_store_unlock(void) {
 | 
						|
    bs_dprintf("Unlock\n");
 | 
						|
    // No-op -- handled by the flash driver as it is.
 | 
						|
    return true;
 | 
						|
}
 | 
						|
 | 
						|
bool backing_store_erase(void) {
 | 
						|
#ifdef WEAR_LEVELING_DEBUG_OUTPUT
 | 
						|
    uint32_t start = timer_read32();
 | 
						|
#endif
 | 
						|
 | 
						|
    bool ret = true;
 | 
						|
    for (int i = 0; i < (WEAR_LEVELING_EXTERNAL_FLASH_BLOCK_COUNT); ++i) {
 | 
						|
        flash_status_t status = flash_erase_block(((WEAR_LEVELING_EXTERNAL_FLASH_BLOCK_OFFSET) + i) * (EXTERNAL_FLASH_BLOCK_SIZE));
 | 
						|
        if (status != FLASH_STATUS_SUCCESS) {
 | 
						|
            ret = false;
 | 
						|
            break;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    bs_dprintf("Backing store erase took %ldms to complete\n", ((long)(timer_read32() - start)));
 | 
						|
    return ret;
 | 
						|
}
 | 
						|
 | 
						|
bool backing_store_write(uint32_t address, backing_store_int_t value) {
 | 
						|
    return backing_store_write_bulk(address, &value, 1);
 | 
						|
}
 | 
						|
 | 
						|
bool backing_store_lock(void) {
 | 
						|
    bs_dprintf("Lock  \n");
 | 
						|
    // No-op -- handled by the flash driver as it is.
 | 
						|
    return true;
 | 
						|
}
 | 
						|
 | 
						|
bool backing_store_read(uint32_t address, backing_store_int_t *value) {
 | 
						|
    return backing_store_read_bulk(address, value, 1);
 | 
						|
}
 | 
						|
 | 
						|
bool backing_store_read_bulk(uint32_t address, backing_store_int_t *values, size_t item_count) {
 | 
						|
    bs_dprintf("Read  ");
 | 
						|
    uint32_t       offset = (WEAR_LEVELING_EXTERNAL_FLASH_BLOCK_OFFSET) * (EXTERNAL_FLASH_BLOCK_SIZE) + address;
 | 
						|
    flash_status_t status = flash_read_block(offset, values, sizeof(backing_store_int_t) * item_count);
 | 
						|
    if (status == FLASH_STATUS_SUCCESS) {
 | 
						|
        for (size_t i = 0; i < item_count; ++i) {
 | 
						|
            values[i] = ~values[i];
 | 
						|
        }
 | 
						|
        wl_dump(offset, values, sizeof(backing_store_int_t) * item_count);
 | 
						|
    }
 | 
						|
    return status == FLASH_STATUS_SUCCESS;
 | 
						|
}
 | 
						|
 | 
						|
bool backing_store_write_bulk(uint32_t address, backing_store_int_t *values, size_t item_count) {
 | 
						|
    uint32_t            offset = (WEAR_LEVELING_EXTERNAL_FLASH_BLOCK_OFFSET) * (EXTERNAL_FLASH_BLOCK_SIZE) + address;
 | 
						|
    size_t              index  = 0;
 | 
						|
    backing_store_int_t temp[WEAR_LEVELING_EXTERNAL_FLASH_BULK_COUNT];
 | 
						|
    do {
 | 
						|
        // Copy out the block of data we want to transmit first
 | 
						|
        size_t this_loop = MIN(item_count, WEAR_LEVELING_EXTERNAL_FLASH_BULK_COUNT);
 | 
						|
        for (size_t i = 0; i < this_loop; ++i) {
 | 
						|
            temp[i] = values[index + i];
 | 
						|
        }
 | 
						|
 | 
						|
        bs_dprintf("Write ");
 | 
						|
        wl_dump(offset, temp, sizeof(backing_store_int_t) * this_loop);
 | 
						|
 | 
						|
        // Take the complement instead
 | 
						|
        for (size_t i = 0; i < this_loop; ++i) {
 | 
						|
            temp[i] = ~temp[i];
 | 
						|
        }
 | 
						|
 | 
						|
        // Write out the block
 | 
						|
        if (flash_write_block(offset, temp, sizeof(backing_store_int_t) * this_loop) != FLASH_STATUS_SUCCESS) {
 | 
						|
            return false;
 | 
						|
        }
 | 
						|
 | 
						|
        offset += this_loop * sizeof(backing_store_int_t);
 | 
						|
        index += this_loop;
 | 
						|
        item_count -= this_loop;
 | 
						|
    } while (item_count > 0);
 | 
						|
 | 
						|
    return true;
 | 
						|
}
 |