2015-01-12 14:42:48 +01:00
|
|
|
/*
|
2017-04-27 15:42:22 +02:00
|
|
|
* Copyright (C) 2014-2017 Eistec AB
|
2015-01-12 14:42:48 +01:00
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser General
|
|
|
|
* Public License v2.1. See the file LICENSE in the top level directory for more
|
|
|
|
* details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2017-04-13 17:08:13 +02:00
|
|
|
* @ingroup boards_mulle
|
2015-01-12 14:42:48 +01:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Board specific implementations for the Mulle board
|
|
|
|
*
|
2015-09-20 13:47:39 +02:00
|
|
|
* @author Joakim Nohlgård <joakim.nohlgard@eistec.se>
|
2015-01-12 14:42:48 +01:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stddef.h> /* for NULL */
|
|
|
|
#include <stdio.h>
|
|
|
|
#include "board.h"
|
|
|
|
#include "cpu.h"
|
|
|
|
#include "mcg.h"
|
|
|
|
#include "periph/gpio.h"
|
2015-07-22 14:13:30 +02:00
|
|
|
#include "periph/rtt.h"
|
2015-01-28 13:51:45 +01:00
|
|
|
#include "periph/spi.h"
|
|
|
|
#include "nvram-spi.h"
|
2016-07-12 08:00:10 +02:00
|
|
|
#include "nvram.h"
|
2015-10-22 07:54:25 +02:00
|
|
|
#include "xtimer.h"
|
2016-07-12 08:00:10 +02:00
|
|
|
#include "vfs.h"
|
|
|
|
#include "fs/devfs.h"
|
2016-07-15 19:10:14 +02:00
|
|
|
#include "mtd_spi_nor.h"
|
2015-01-28 13:51:45 +01:00
|
|
|
|
|
|
|
static nvram_t mulle_nvram_dev;
|
|
|
|
nvram_t *mulle_nvram = &mulle_nvram_dev;
|
|
|
|
static nvram_spi_params_t nvram_spi_params = {
|
|
|
|
.spi = MULLE_NVRAM_SPI_DEV,
|
2016-11-08 18:19:12 +01:00
|
|
|
.clk = MULLE_NVRAM_SPI_CLK,
|
2015-01-28 13:51:45 +01:00
|
|
|
.cs = MULLE_NVRAM_SPI_CS,
|
|
|
|
.address_count = MULLE_NVRAM_SPI_ADDRESS_COUNT,
|
|
|
|
};
|
2015-01-12 14:42:48 +01:00
|
|
|
|
2016-07-12 08:00:10 +02:00
|
|
|
static devfs_t mulle_nvram_devfs = {
|
2016-07-15 19:10:14 +02:00
|
|
|
.path = "/fram0",
|
2016-07-12 08:00:10 +02:00
|
|
|
.f_op = &nvram_vfs_ops,
|
|
|
|
.private_data = &mulle_nvram_dev,
|
|
|
|
};
|
|
|
|
|
2020-02-10 20:40:41 +01:00
|
|
|
static const mtd_spi_nor_params_t mulle_nor_params = {
|
|
|
|
.opcode = &mtd_spi_nor_opcode_default,
|
2020-02-10 22:29:08 +01:00
|
|
|
.wait_chip_erase = 16LU * US_PER_SEC,
|
2020-10-25 23:23:48 +01:00
|
|
|
.wait_sector_erase = 10LU * US_PER_MS,
|
2020-02-10 22:29:08 +01:00
|
|
|
.wait_32k_erase = 20LU * US_PER_MS,
|
2020-03-05 10:09:13 +01:00
|
|
|
.wait_chip_wake_up = 1LU * US_PER_MS,
|
2020-02-10 20:40:41 +01:00
|
|
|
.spi = MULLE_NOR_SPI_DEV,
|
|
|
|
.addr_width = 3,
|
|
|
|
.mode = SPI_MODE_3,
|
2020-11-02 02:42:45 +01:00
|
|
|
.cs = MULLE_NOR_SPI_CS,
|
|
|
|
.wp = GPIO_UNDEF,
|
|
|
|
.hold = GPIO_UNDEF,
|
2020-02-10 20:40:41 +01:00
|
|
|
.clk = SPI_CLK_10MHZ,
|
|
|
|
};
|
|
|
|
|
2016-07-15 19:10:14 +02:00
|
|
|
static mtd_spi_nor_t mulle_nor_dev = {
|
|
|
|
.base = {
|
|
|
|
.driver = &mtd_spi_nor_driver,
|
|
|
|
.page_size = 256,
|
|
|
|
.pages_per_sector = 256,
|
|
|
|
.sector_count = 32,
|
|
|
|
},
|
2020-02-10 20:40:41 +01:00
|
|
|
.params = &mulle_nor_params,
|
2016-07-15 19:10:14 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
mtd_dev_t *mtd0 = (mtd_dev_t *)&mulle_nor_dev;
|
|
|
|
|
|
|
|
static devfs_t mulle_nor_devfs = {
|
|
|
|
.path = "/mtd0",
|
|
|
|
.f_op = &mtd_vfs_ops,
|
|
|
|
.private_data = &mulle_nor_dev,
|
|
|
|
};
|
|
|
|
|
2015-01-12 14:42:48 +01:00
|
|
|
/** @brief Initialize the GPIO pins controlling the power switches. */
|
|
|
|
static inline void power_pins_init(void);
|
|
|
|
|
2015-01-28 13:51:45 +01:00
|
|
|
static void increase_boot_count(void);
|
|
|
|
static int mulle_nvram_init(void);
|
2015-01-12 14:42:48 +01:00
|
|
|
|
2016-07-15 19:10:14 +02:00
|
|
|
int mulle_nor_init(void);
|
|
|
|
|
2015-01-12 14:42:48 +01:00
|
|
|
void board_init(void)
|
|
|
|
{
|
2015-01-28 13:51:45 +01:00
|
|
|
int status;
|
2016-03-11 18:04:26 +01:00
|
|
|
|
|
|
|
/* initialize the boards LEDs */
|
2016-02-29 17:37:44 +01:00
|
|
|
gpio_init(LED0_PIN, GPIO_OUT);
|
|
|
|
gpio_init(LED1_PIN, GPIO_OUT);
|
|
|
|
gpio_init(LED2_PIN, GPIO_OUT);
|
2015-01-12 14:42:48 +01:00
|
|
|
|
2015-07-22 14:13:30 +02:00
|
|
|
/* Initialize power control pins */
|
|
|
|
power_pins_init();
|
|
|
|
|
|
|
|
/* Turn on Vperiph for peripherals */
|
2015-01-28 13:51:45 +01:00
|
|
|
/*
|
|
|
|
* By turning on Vperiph first, and before waiting for the clocks to
|
|
|
|
* stabilize, we will have used enough time to have let the FRAM start up
|
|
|
|
* properly when we want to access it later without having to add any extra
|
|
|
|
* delays.
|
|
|
|
*/
|
2015-07-22 14:13:30 +02:00
|
|
|
gpio_set(MULLE_POWER_VPERIPH);
|
|
|
|
|
|
|
|
/* Turn on AVDD for reading voltages */
|
|
|
|
gpio_set(MULLE_POWER_AVDD);
|
|
|
|
|
2015-01-28 13:51:45 +01:00
|
|
|
|
2015-10-22 07:54:25 +02:00
|
|
|
/* NVRAM requires xtimer for timing */
|
|
|
|
xtimer_init();
|
|
|
|
|
2015-01-28 13:51:45 +01:00
|
|
|
/* Initialize NVRAM */
|
|
|
|
status = mulle_nvram_init();
|
|
|
|
if (status == 0) {
|
|
|
|
/* Increment boot counter */
|
|
|
|
increase_boot_count();
|
|
|
|
}
|
2016-07-15 19:10:14 +02:00
|
|
|
|
|
|
|
/* Initialize NOR flash */
|
|
|
|
mulle_nor_init();
|
2015-01-12 14:42:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void power_pins_init(void)
|
|
|
|
{
|
2016-02-29 17:37:44 +01:00
|
|
|
gpio_init(MULLE_POWER_AVDD, GPIO_OUT);
|
|
|
|
gpio_init(MULLE_POWER_VPERIPH, GPIO_OUT);
|
|
|
|
gpio_init(MULLE_POWER_VSEC, GPIO_OUT);
|
2015-01-12 14:42:48 +01:00
|
|
|
gpio_clear(MULLE_POWER_AVDD);
|
|
|
|
gpio_clear(MULLE_POWER_VPERIPH);
|
|
|
|
gpio_clear(MULLE_POWER_VSEC);
|
|
|
|
}
|
|
|
|
|
2015-01-28 13:51:45 +01:00
|
|
|
static int mulle_nvram_init(void)
|
|
|
|
{
|
|
|
|
union {
|
|
|
|
uint32_t u32;
|
|
|
|
uint8_t u8[sizeof(uint32_t)];
|
|
|
|
} rec;
|
|
|
|
rec.u32 = 0;
|
|
|
|
|
|
|
|
if (nvram_spi_init(mulle_nvram, &nvram_spi_params, MULLE_NVRAM_CAPACITY) != 0) {
|
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mulle_nvram->read(mulle_nvram, &rec.u8[0], MULLE_NVRAM_MAGIC, sizeof(rec.u32)) != sizeof(rec.u32)) {
|
|
|
|
return -3;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rec.u32 != MULLE_NVRAM_MAGIC_EXPECTED) {
|
|
|
|
int i;
|
|
|
|
union {
|
|
|
|
uint64_t u64;
|
|
|
|
uint8_t u8[sizeof(uint64_t)];
|
|
|
|
} zero;
|
|
|
|
zero.u64 = 0;
|
|
|
|
for (i = 0; i < MULLE_NVRAM_CAPACITY; i += sizeof(zero)) {
|
|
|
|
if (mulle_nvram->write(mulle_nvram, &zero.u8[0], i, sizeof(zero.u64)) != sizeof(zero.u64)) {
|
|
|
|
return -4;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rec.u32 = MULLE_NVRAM_MAGIC_EXPECTED;
|
|
|
|
if (mulle_nvram->write(mulle_nvram, &rec.u8[0], MULLE_NVRAM_MAGIC, sizeof(rec.u32)) != sizeof(rec.u32)) {
|
|
|
|
return -5;
|
|
|
|
}
|
|
|
|
}
|
2016-07-12 08:00:10 +02:00
|
|
|
|
|
|
|
/* Register DevFS node */
|
2020-12-08 14:13:15 +01:00
|
|
|
return devfs_register(&mulle_nvram_devfs);
|
2015-01-28 13:51:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void increase_boot_count(void)
|
|
|
|
{
|
|
|
|
union {
|
|
|
|
uint32_t u32;
|
|
|
|
uint8_t u8[sizeof(uint32_t)];
|
|
|
|
} rec;
|
|
|
|
rec.u32 = 0;
|
|
|
|
if (mulle_nvram->read(mulle_nvram, &rec.u8[0], MULLE_NVRAM_BOOT_COUNT, sizeof(rec.u32)) != sizeof(rec.u32)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
++rec.u32;
|
|
|
|
mulle_nvram->write(mulle_nvram, &rec.u8[0], MULLE_NVRAM_BOOT_COUNT, sizeof(rec.u32));
|
|
|
|
}
|
2016-07-15 19:10:14 +02:00
|
|
|
|
|
|
|
int mulle_nor_init(void)
|
|
|
|
{
|
|
|
|
int res = mtd_init(mtd0);
|
|
|
|
|
|
|
|
if (res >= 0) {
|
|
|
|
/* Register DevFS node */
|
2020-12-08 14:13:15 +01:00
|
|
|
res = devfs_register(&mulle_nor_devfs);
|
2016-07-15 19:10:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|