Now spi-nor framework is up so access the environment through en/spinor.c Signed-off-by: Suneel Garapati <suneelgli...@gmail.com> Signed-off-by: Jagan Teki <ja...@amarulasolutions.com> --- cmd/nvedit.c | 1 + env/Kconfig | 27 ++++++++++++ env/Makefile | 1 + env/env.c | 2 + env/spinor.c | 113 ++++++++++++++++++++++++++++++++++++++++++++++++++ include/environment.h | 1 + 6 files changed, 145 insertions(+) create mode 100644 env/spinor.c
diff --git a/cmd/nvedit.c b/cmd/nvedit.c index 4e79d03..2e5e88d 100644 --- a/cmd/nvedit.c +++ b/cmd/nvedit.c @@ -50,6 +50,7 @@ DECLARE_GLOBAL_DATA_PTR; !defined(CONFIG_ENV_IS_IN_ONENAND) && \ !defined(CONFIG_ENV_IS_IN_SATA) && \ !defined(CONFIG_ENV_IS_IN_SPI_FLASH) && \ + !defined(CONFIG_ENV_IS_IN_SPI_NOR) && \ !defined(CONFIG_ENV_IS_IN_REMOTE) && \ !defined(CONFIG_ENV_IS_IN_UBI) && \ !defined(CONFIG_ENV_IS_NOWHERE) diff --git a/env/Kconfig b/env/Kconfig index 2477bf8..1b694d5 100644 --- a/env/Kconfig +++ b/env/Kconfig @@ -329,6 +329,22 @@ config ENV_IS_IN_SPI_FLASH Define the SPI work mode. If not defined then use SPI_MODE_3. +config ENV_IS_IN_SPI_NOR + bool "Environment is in SPI-NOR flash" + depends on !CHAIN_OF_TRUST + help + Define this if you have a SPI NOR Flash memory device which you + want to use for the environment. + + - CONFIG_ENV_OFFSET: + - CONFIG_ENV_SIZE: + + These two #defines specify the offset and size of the + environment area within the SPI NOR Flash. CONFIG_ENV_OFFSET must be + aligned to an erase sector boundary. + + - CONFIG_ENV_SECT_SIZE: + config ENV_IS_IN_UBI bool "Environment in a UBI volume" depends on !CHAIN_OF_TRUST @@ -396,6 +412,17 @@ config ENV_FAT_FILE It's a string of the FAT file name. This file use to store the environment. +config ENV_SPI_NOR_DEVNUM + int "SPI-NOR device number for the environment" + depends on ENV_IS_IN_SPI_NOR + default 0 + help + Define this an integer of spi nor device number for the environment. + Since spi nor framework assign device numbers automatically by devicetree + node definition which start from 0, so assign 0 as default. + + If board need different devnum then, this can be change. + if ARCH_SUNXI config ENV_OFFSET diff --git a/env/Makefile b/env/Makefile index 7ce8231..21fe9a4 100644 --- a/env/Makefile +++ b/env/Makefile @@ -25,6 +25,7 @@ obj-$(CONFIG_ENV_IS_IN_NVRAM) += nvram.o obj-$(CONFIG_ENV_IS_IN_ONENAND) += onenand.o obj-$(CONFIG_ENV_IS_IN_SATA) += sata.o obj-$(CONFIG_ENV_IS_IN_SPI_FLASH) += sf.o +obj-$(CONFIG_ENV_IS_IN_SPI_NOR) += spinor.o obj-$(CONFIG_ENV_IS_IN_REMOTE) += remote.o obj-$(CONFIG_ENV_IS_IN_UBI) += ubi.o obj-$(CONFIG_ENV_IS_NOWHERE) += nowhere.o diff --git a/env/env.c b/env/env.c index 76a5608..be1672c 100644 --- a/env/env.c +++ b/env/env.c @@ -44,6 +44,8 @@ static enum env_location env_get_default_location(void) return ENVL_REMOTE; else if IS_ENABLED(CONFIG_ENV_IS_IN_SPI_FLASH) return ENVL_SPI_FLASH; + else if IS_ENABLED(CONFIG_ENV_IS_IN_SPI_NOR) + return ENVL_SPI_NOR; else if IS_ENABLED(CONFIG_ENV_IS_IN_UBI) return ENVL_UBI; else if IS_ENABLED(CONFIG_ENV_IS_NOWHERE) diff --git a/env/spinor.c b/env/spinor.c new file mode 100644 index 0000000..bc6fdfa --- /dev/null +++ b/env/spinor.c @@ -0,0 +1,113 @@ +/* + * SPI-NOR Environment. + * + * Copyright (C) 2017 Jagan Teki <ja...@openedev.com> + * + * SPDX-License-Identifier: GPL-2.0+ + */ + +#include <common.h> +#include <dm.h> +#include <environment.h> +#include <memalign.h> +#include <mtd.h> +#include <linux/mtd/mtd.h> +#include <linux/mtd/spi-nor.h> + +DECLARE_GLOBAL_DATA_PTR; + +static struct spi_nor *env_init_spinor(void) +{ + struct spi_nor *nor; + int devnum = CONFIG_ENV_SPI_NOR_DEVNUM; + + nor = find_spi_nor_device(devnum); + if (!nor) + return NULL; + + if (spi_nor_scan(nor)) + return NULL; + + return nor; +} + +static int env_spinor_save(void) +{ + ALLOC_CACHE_ALIGN_BUFFER(env_t, env_new, 1); + struct spi_nor *nor; + struct mtd_info *mtd; + struct erase_info instr; + loff_t len, sector; + int ret; + + nor = env_init_spinor(); + if (!nor) + return -EIO; + + ret = env_export(env_new); + if (ret) + return -EINVAL; + + mtd = spi_nor_get_mtd_info(nor); + if (!mtd) + return -EIO; + + sector = DIV_ROUND_UP(CONFIG_ENV_SIZE, CONFIG_ENV_SECT_SIZE); + len = sector * CONFIG_ENV_SECT_SIZE; + instr.mtd = mtd; + instr.addr = CONFIG_ENV_OFFSET; + instr.len = len; + instr.callback = 0; + + puts("erasing spinor flash...\n"); + ret = mtd_derase(mtd, &instr); + if (ret) + return -EIO; + + len = CONFIG_ENV_SIZE; + puts("writing spinor flash...\n"); + ret = mtd_dwrite(mtd, CONFIG_ENV_OFFSET, len, + (size_t *)&len, (u_char *)env_new); + if (ret) + return -EIO; + + puts("done\n"); + return ret; +} + +static int env_spinor_load(void) +{ + ALLOC_CACHE_ALIGN_BUFFER(char, buf, CONFIG_ENV_SIZE); + struct spi_nor *nor; + struct mtd_info *mtd; + loff_t offset, len; + int ret; + + nor = env_init_spinor(); + if (!nor) + return -EIO; + + offset = CONFIG_ENV_OFFSET; + len = CONFIG_ENV_SIZE; + + mtd = spi_nor_get_mtd_info(nor); + if (!mtd) + return -EIO; + + ret = mtd_dread(mtd, offset, len, (size_t *)&len, (uchar *)buf); + if (ret) + return -EIO; + + ret = env_import(buf, 1); + if (ret) + gd->env_valid = ENV_VALID; + + return ret; +} + +U_BOOT_ENV_LOCATION(spinor) = { + .location = ENVL_SPI_NOR, + ENV_NAME("SPI-NOR Flash") + .load = env_spinor_load, + .save = env_save_ptr(env_spinor_save), +}; diff --git a/include/environment.h b/include/environment.h index d29f82c..8e8be26 100644 --- a/include/environment.h +++ b/include/environment.h @@ -198,6 +198,7 @@ enum env_location { ENVL_ONENAND, ENVL_REMOTE, ENVL_SPI_FLASH, + ENVL_SPI_NOR, ENVL_UBI, ENVL_NOWHERE, -- 2.7.4 _______________________________________________ U-Boot mailing list U-Boot@lists.denx.de https://lists.denx.de/listinfo/u-boot