circuitpython/ports/broadcom/supervisor/internal_flash.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) {
}