mirror of
https://github.com/google/pebble.git
synced 2025-06-08 03:03:11 +00:00
Import of the watch repository from Pebble
This commit is contained in:
commit
3b92768480
10334 changed files with 2564465 additions and 0 deletions
176
platform/snowy/boot/src/drivers/flash/s29vs.c
Normal file
176
platform/snowy/boot/src/drivers/flash/s29vs.c
Normal file
|
@ -0,0 +1,176 @@
|
|||
/*
|
||||
* Copyright 2024 Google LLC
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#include <stdbool.h>
|
||||
#include <stdint.h>
|
||||
|
||||
#include "drivers/flash.h"
|
||||
#include "drivers/flash/s29vs.h"
|
||||
#include "drivers/gpio.h"
|
||||
#include "drivers/periph_config.h"
|
||||
#include "stm32f4xx_gpio.h"
|
||||
#include "util/delay.h"
|
||||
|
||||
|
||||
//! @param sector_address The address of the start of the sector to write the command to.
|
||||
//! @param cmd The command to write.
|
||||
static void flash_s29vs_issue_command(FlashAddress sector_address, S29VSCommand cmd) {
|
||||
// The offset in the sector we write the first part of commands to. Note that this is a 16-bit
|
||||
// word aligned address as opposed to a byte address.
|
||||
static const uint32_t COMMAND_ADDRESS = 0x555;
|
||||
|
||||
((__IO uint16_t*) (FMC_BANK_1_BASE_ADDRESS + sector_address))[COMMAND_ADDRESS] = cmd;
|
||||
}
|
||||
|
||||
static uint16_t flash_s29vs_read_short(FlashAddress addr) {
|
||||
return *((__IO uint16_t*)(FMC_BANK_1_BASE_ADDRESS + addr));
|
||||
}
|
||||
|
||||
void flash_read_bytes(uint8_t* buffer, uint32_t start_addr, uint32_t buffer_size) {
|
||||
memcpy(buffer, (void*)(FMC_BANK_1_BASE_ADDRESS + start_addr), buffer_size);
|
||||
}
|
||||
|
||||
|
||||
static void flash_s29vs_software_reset(void) {
|
||||
flash_s29vs_issue_command(0, S29VSCommand_SoftwareReset);
|
||||
}
|
||||
|
||||
void flash_init(void) {
|
||||
gpio_use(GPIOB);
|
||||
gpio_use(GPIOD);
|
||||
gpio_use(GPIOE);
|
||||
|
||||
// Configure the reset pin (D2)
|
||||
GPIO_InitTypeDef gpio_init = {
|
||||
.GPIO_Pin = GPIO_Pin_2,
|
||||
.GPIO_Mode = GPIO_Mode_OUT,
|
||||
.GPIO_Speed = GPIO_Speed_100MHz,
|
||||
.GPIO_OType = GPIO_OType_PP,
|
||||
.GPIO_PuPd = GPIO_PuPd_NOPULL
|
||||
};
|
||||
GPIO_Init(GPIOD, &gpio_init);
|
||||
|
||||
GPIO_WriteBit(GPIOD, GPIO_Pin_2, Bit_SET);
|
||||
|
||||
// Configure pins relating to the FMC peripheral (30 pins!)
|
||||
|
||||
// B7 - FMC AVD - FMC Address Valid aka Latch
|
||||
// D0-D1, D8-D15, E2-15 - FMC A, AD - FMC Address and Address/Data lines
|
||||
// D2 - Reset - GPIO Reset line
|
||||
// D3 - FMC CLK
|
||||
// D4 - FMC OE - FMC Output Enable
|
||||
// D5 - FMC WE - FMC Write Enable
|
||||
// D6 - FMC RDY - FMC Ready line
|
||||
// D7 - FMC CE - FMC Chip Enable
|
||||
|
||||
gpio_init = (GPIO_InitTypeDef) {
|
||||
.GPIO_Mode = GPIO_Mode_AF,
|
||||
.GPIO_Speed = GPIO_Speed_100MHz,
|
||||
.GPIO_OType = GPIO_OType_PP,
|
||||
.GPIO_PuPd = GPIO_PuPd_NOPULL
|
||||
};
|
||||
|
||||
GPIO_PinAFConfig(GPIOB, GPIO_PinSource7, GPIO_AF_FMC);
|
||||
gpio_init.GPIO_Pin = GPIO_Pin_7;
|
||||
GPIO_Init(GPIOB, &gpio_init);
|
||||
|
||||
for (uint8_t pin_source = 0; pin_source < 16; ++pin_source) {
|
||||
if (pin_source == 2) {
|
||||
continue;
|
||||
}
|
||||
GPIO_PinAFConfig(GPIOD, pin_source, GPIO_AF_FMC);
|
||||
}
|
||||
gpio_init.GPIO_Pin = GPIO_Pin_All & (~GPIO_Pin_2);
|
||||
GPIO_Init(GPIOD, &gpio_init);
|
||||
|
||||
for (uint8_t pin_source = 2; pin_source < 16; ++pin_source) {
|
||||
GPIO_PinAFConfig(GPIOE, pin_source, GPIO_AF_FMC);
|
||||
}
|
||||
gpio_init.GPIO_Pin = GPIO_Pin_All & (~GPIO_Pin_0) & (~GPIO_Pin_1);
|
||||
GPIO_Init(GPIOE, &gpio_init);
|
||||
|
||||
// We have configured the pins, lets perform a full HW reset to put the chip
|
||||
// in a good state
|
||||
GPIO_WriteBit(GPIOD, GPIO_Pin_2, Bit_RESET);
|
||||
delay_us(10); // only needs to be 50ns according to data sheet
|
||||
GPIO_WriteBit(GPIOD, GPIO_Pin_2, Bit_SET);
|
||||
delay_us(30); // need 200ns + 10us before CE can be pulled low
|
||||
|
||||
RCC_AHB3PeriphClockCmd(RCC_AHB3Periph_FMC, ENABLE);
|
||||
|
||||
// Setup default config for async
|
||||
// Configure the FMC peripheral itself
|
||||
FMC_NORSRAMTimingInitTypeDef nor_timing_init = {
|
||||
// time between address write and address latch (AVD high)
|
||||
// tAAVDS on datasheet, min 4 ns
|
||||
//
|
||||
// AVD low time
|
||||
// tAVDP on datasheet, min 6 ns
|
||||
.FMC_AddressSetupTime = 1,
|
||||
|
||||
// time between AVD high (address is available) and OE low (memory can write)
|
||||
// tAVDO on the datasheet, min 4 ns
|
||||
.FMC_AddressHoldTime = 1,
|
||||
|
||||
// time between OE low (memory can write) and valid data being available
|
||||
// tOE on datasheet, max 15 ns
|
||||
// 13 cycles is the default configuration in the component's configuration register
|
||||
// Setup to 3 for async
|
||||
.FMC_DataSetupTime = 3,
|
||||
|
||||
// Time between chip selects
|
||||
// not on the datasheet, picked a random safe number
|
||||
.FMC_BusTurnAroundDuration = 1,
|
||||
|
||||
.FMC_CLKDivision = 15, // Not used for async NOR
|
||||
.FMC_DataLatency = 15, // Not used for async NOR
|
||||
.FMC_AccessMode = FMC_AccessMode_A // Only used for ExtendedMode == FMC_ExtendedMode_Enable, which we don't use
|
||||
};
|
||||
|
||||
FMC_NORSRAMInitTypeDef nor_init = {
|
||||
.FMC_Bank = FMC_Bank1_NORSRAM1,
|
||||
.FMC_DataAddressMux = FMC_DataAddressMux_Enable,
|
||||
.FMC_MemoryType = FMC_MemoryType_NOR,
|
||||
.FMC_MemoryDataWidth = FMC_NORSRAM_MemoryDataWidth_16b,
|
||||
.FMC_BurstAccessMode = FMC_BurstAccessMode_Disable,
|
||||
.FMC_AsynchronousWait = FMC_AsynchronousWait_Disable,
|
||||
.FMC_WaitSignalPolarity = FMC_WaitSignalPolarity_Low,
|
||||
.FMC_WrapMode = FMC_WrapMode_Disable,
|
||||
.FMC_WaitSignalActive = FMC_WaitSignalActive_BeforeWaitState,
|
||||
.FMC_WriteOperation = FMC_WriteOperation_Enable,
|
||||
.FMC_WaitSignal = FMC_WaitSignal_Enable,
|
||||
.FMC_ExtendedMode = FMC_ExtendedMode_Disable,
|
||||
.FMC_WriteBurst = FMC_WriteBurst_Disable,
|
||||
.FMC_ContinousClock = FMC_CClock_SyncOnly,
|
||||
.FMC_ReadWriteTimingStruct = &nor_timing_init
|
||||
};
|
||||
|
||||
FMC_NORSRAMInit(&nor_init);
|
||||
|
||||
// Re-enable NOR
|
||||
FMC_NORSRAMCmd(FMC_Bank1_NORSRAM1, ENABLE);
|
||||
}
|
||||
|
||||
bool flash_sanity_check(void) {
|
||||
// Check that the first words of the CFI table are 'Q' 'R' 'Y'.
|
||||
// This will work on any flash memory, regardless of the manufacturer.
|
||||
flash_s29vs_issue_command(0, S29VSCommand_CFIEntry);
|
||||
bool ok = (flash_s29vs_read_short(0x20) & 0xff) == 'Q';
|
||||
ok = ok && (flash_s29vs_read_short(0x22) & 0xff) == 'R';
|
||||
ok = ok && (flash_s29vs_read_short(0x24) & 0xff) == 'Y';
|
||||
flash_s29vs_software_reset();
|
||||
return ok;
|
||||
}
|
63
platform/snowy/boot/src/drivers/flash/s29vs.h
Normal file
63
platform/snowy/boot/src/drivers/flash/s29vs.h
Normal file
|
@ -0,0 +1,63 @@
|
|||
/*
|
||||
* Copyright 2024 Google LLC
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <stdint.h>
|
||||
|
||||
#include "drivers/flash.h"
|
||||
|
||||
//! An address in the flash address spac
|
||||
typedef uint32_t FlashAddress;
|
||||
|
||||
//! This is the memory mapped region that's mapped to the parallel flash.
|
||||
static const uintptr_t FMC_BANK_1_BASE_ADDRESS = 0x60000000;
|
||||
|
||||
//! This is the unit that we use for erasing
|
||||
static const uint32_t SECTOR_SIZE_BYTES = 0x20000; // 128kb
|
||||
//! This is the unit that we use for writing
|
||||
static const uint32_t PAGE_SIZE_BYTES = 64;
|
||||
|
||||
//! Different commands we can send to the flash
|
||||
typedef enum S29VSCommand {
|
||||
S29VSCommand_WriteBufferLoad = 0x25,
|
||||
S29VSCommand_BufferToFlash = 0x29,
|
||||
S29VSCommand_ReadStatusRegister = 0x70,
|
||||
S29VSCommand_ClearStatusRegister = 0x71,
|
||||
S29VSCommand_EraseSetup = 0x80,
|
||||
S29VSCommand_DeviceIDEntry = 0x90,
|
||||
S29VSCommand_CFIEntry = 0x98,
|
||||
S29VSCommand_ConfigureRegisterEntry = 0xD0,
|
||||
S29VSCommand_SoftwareReset = 0xF0
|
||||
} S29VSCommand;
|
||||
|
||||
//! Arguments to the S29VSCommand_EraseSetup command
|
||||
typedef enum S29VSCommandEraseAguments {
|
||||
S29VSCommandEraseAguments_ChipErase = 0x10,
|
||||
S29VSCommandEraseAguments_SectorErase = 0x30
|
||||
} S29VSCommandEraseAguments;
|
||||
|
||||
//! The bitset stored in the status register, see flash_s29vs_read_status_register
|
||||
typedef enum S29VSStatusBit {
|
||||
S29VSStatusBit_BankStatus = 0x00,
|
||||
S29VSStatusBit_SectorLockStatus = 0x01,
|
||||
S29VSStatusBit_ProgramSuspended = 0x02,
|
||||
// 0x04 is unused
|
||||
S29VSStatusBit_ProgramStatus = 0x10,
|
||||
S29VSStatusBit_EraseStatus = 0x20,
|
||||
S29VSStatusBit_EraseSuspended = 0x40,
|
||||
S29VSStatusBit_DeviceReady = 0x80,
|
||||
} S29VSStatusBit;
|
Loading…
Add table
Add a link
Reference in a new issue