181 lines
6.2 KiB
C
181 lines
6.2 KiB
C
/*
|
|
* This file is part of the MicroPython project, http://micropython.org/
|
|
*
|
|
* The MIT License (MIT)
|
|
*
|
|
* Copyright (c) 2021 Scott Shawcroft for Adafruit Industries
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
* in the Software without restriction, including without limitation the rights
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
* THE SOFTWARE.
|
|
*/
|
|
|
|
#include "supervisor/flash.h"
|
|
#include "supervisor/internal_flash.h"
|
|
|
|
#include <string.h>
|
|
|
|
#include "shared-bindings/digitalio/DigitalInOut.h"
|
|
#include "shared-bindings/sdioio/SDCard.h"
|
|
|
|
STATIC sdioio_sdcard_obj_t sd;
|
|
|
|
STATIC uint32_t first_lba = 0;
|
|
STATIC uint32_t sector_count = 0;
|
|
|
|
STATIC bool inited = false;
|
|
|
|
void supervisor_flash_init(void) {
|
|
if (inited) {
|
|
return;
|
|
}
|
|
inited = true;
|
|
|
|
#if BCM_VERSION != 2711
|
|
const mcu_pin_obj_t *data_pins[4] = {
|
|
&pin_GPIO50,
|
|
&pin_GPIO51,
|
|
&pin_GPIO52,
|
|
&pin_GPIO53,
|
|
};
|
|
common_hal_sdioio_sdcard_construct(&sd,
|
|
&pin_GPIO48, &pin_GPIO49,
|
|
4, data_pins, 8000000);
|
|
#else
|
|
common_hal_sdioio_sdcard_construct(&sd,
|
|
NULL, NULL,
|
|
0, NULL, 8000000);
|
|
#endif
|
|
common_hal_sdioio_sdcard_never_reset(&sd);
|
|
|
|
uint32_t buffer[512 / sizeof(uint32_t)];
|
|
mp_buffer_info_t bufinfo;
|
|
bufinfo.len = 512;
|
|
bufinfo.buf = buffer;
|
|
common_hal_sdioio_sdcard_readblocks(&sd, 0, &bufinfo);
|
|
size_t partition_count = 0;
|
|
size_t first_free_sector = 0;
|
|
for (size_t i = 0; i < 4; i++) {
|
|
uint32_t part_first_lba = 0;
|
|
uint32_t part_sector_count = 0;
|
|
for (size_t j = 0; j < 16; j++) {
|
|
size_t offset = 446 + i * 16 + j;
|
|
uint8_t value = ((uint8_t *)buffer)[offset];
|
|
// Little endian
|
|
if (j >= 8 && j < 12) {
|
|
part_first_lba |= value << (8 * (j - 8));
|
|
} else if (j >= 12) {
|
|
part_sector_count |= value << (8 * (j - 12));
|
|
}
|
|
}
|
|
if (part_sector_count > 0) {
|
|
partition_count += 1;
|
|
first_free_sector = part_first_lba + part_sector_count;
|
|
}
|
|
// If the second partition exists, then use it (possibly after reformatting it.)
|
|
if (i == 1 && part_sector_count > 0 && part_first_lba > 0) {
|
|
sector_count = part_sector_count;
|
|
first_lba = part_first_lba;
|
|
} else if (i > 1 && part_sector_count > 0) {
|
|
sector_count = 0;
|
|
}
|
|
}
|
|
|
|
// Special case where most of the card is unused and the first and only
|
|
// partition is likely the CP boot partition. (It is 255MB starting at 1MB)
|
|
size_t sectors_per_mb = 2048;
|
|
if (partition_count == 1 && first_free_sector == 256 * sectors_per_mb) {
|
|
size_t total_sectors = common_hal_sdioio_sdcard_get_count(&sd);
|
|
// Align to a MB
|
|
total_sectors = total_sectors - (total_sectors % sectors_per_mb);
|
|
// Don't use the last MB because the imager erases it. GPT tables put a
|
|
// backup there sometimes.
|
|
total_sectors -= sectors_per_mb;
|
|
size_t new_sector_count = total_sectors - first_free_sector;
|
|
|
|
// Modify the MBR
|
|
size_t offset = 446 + 1 * 16;
|
|
uint8_t *part = ((uint8_t *)buffer) + offset;
|
|
part[0] = 0; // not boot
|
|
part[1] = 0xff; // Old unused CHS sizing
|
|
part[2] = 0xff;
|
|
part[3] = 0xff;
|
|
uint8_t type = 0x0C; // FAT32 with LBA
|
|
if (new_sector_count >= 0x4000000) {
|
|
type = 0x07; // Exfat when 32GB+
|
|
}
|
|
part[4] = type;
|
|
part[5] = 0xff; // Old unused CHS sizing
|
|
part[6] = 0xff;
|
|
part[7] = 0xff;
|
|
part[8] = first_free_sector & 0xff;
|
|
part[9] = (first_free_sector >> 8) & 0xff;
|
|
part[10] = (first_free_sector >> 16) & 0xff;
|
|
part[11] = (first_free_sector >> 24) & 0xff;
|
|
part[12] = new_sector_count & 0xff;
|
|
part[13] = (new_sector_count >> 8) & 0xff;
|
|
part[14] = (new_sector_count >> 16) & 0xff;
|
|
part[15] = (new_sector_count >> 24) & 0xff;
|
|
|
|
// Write back the modified MBR with the new partition. CircuitPython can
|
|
// detect a previously formatted FAT filesystem later if only the MBR and
|
|
// BOOT partition was updated.
|
|
if (common_hal_sdioio_sdcard_writeblocks(&sd, 0, &bufinfo) == 0) {
|
|
sector_count = new_sector_count;
|
|
first_lba = first_free_sector;
|
|
}
|
|
}
|
|
}
|
|
|
|
uint32_t supervisor_flash_get_block_size(void) {
|
|
return 512;
|
|
}
|
|
|
|
uint32_t supervisor_flash_get_block_count(void) {
|
|
return sector_count;
|
|
}
|
|
|
|
void port_internal_flash_flush(void) {
|
|
}
|
|
|
|
mp_uint_t supervisor_flash_read_blocks(uint8_t *dest, uint32_t block, uint32_t num_blocks) {
|
|
mp_buffer_info_t bufinfo;
|
|
bufinfo.len = 512;
|
|
for (size_t i = 0; i < num_blocks; i++) {
|
|
bufinfo.buf = dest + 512 * i;
|
|
if (common_hal_sdioio_sdcard_readblocks(&sd, first_lba + block + i, &bufinfo) != 0) {
|
|
return 1; // error
|
|
}
|
|
}
|
|
return 0; // success
|
|
}
|
|
|
|
mp_uint_t supervisor_flash_write_blocks(const uint8_t *src, uint32_t block, uint32_t num_blocks) {
|
|
mp_buffer_info_t bufinfo;
|
|
bufinfo.len = 512;
|
|
for (size_t i = 0; i < num_blocks; i++) {
|
|
bufinfo.buf = (uint8_t *)src + 512 * i;
|
|
if (common_hal_sdioio_sdcard_writeblocks(&sd, first_lba + block + i, &bufinfo) != 0) {
|
|
return 1; // error
|
|
}
|
|
}
|
|
return 0; // success
|
|
}
|
|
|
|
void supervisor_flash_release_cache(void) {
|
|
}
|