From patchwork Thu Jul 2 21:10:02 2020 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Simon Glass X-Patchwork-Id: 240666 List-Id: U-Boot discussion From: sjg at chromium.org (Simon Glass) Date: Thu, 2 Jul 2020 15:10:02 -0600 Subject: [RFC PATCH v2 1/3] dm: Driver and uclass changes for tiny-dm In-Reply-To: <20200702211004.1491489-1-sjg@chromium.org> References: <20200702211004.1491489-1-sjg@chromium.org> Message-ID: <20200702211004.1491489-2-sjg@chromium.org> This includes various changes to support tiny-dm in serial, ram, clock, spi, SPI flash, syscon and sysreset drivers. Signed-off-by: Simon Glass --- (no changes since v1) drivers/clk/Kconfig | 54 +++++ drivers/clk/Makefile | 4 +- drivers/clk/clk-uclass.c | 53 ++++- drivers/clk/rockchip/clk_rk3288.c | 106 +++++++--- drivers/mtd/spi/Kconfig | 18 ++ drivers/mtd/spi/sf-uclass.c | 76 +++++++ drivers/mtd/spi/sf_probe.c | 4 + drivers/mtd/spi/spi-nor-tiny.c | 166 +++++++++++++-- drivers/ram/Kconfig | 18 ++ drivers/ram/ram-uclass.c | 12 ++ drivers/ram/rockchip/sdram_rk3188.c | 2 +- drivers/ram/rockchip/sdram_rk322x.c | 2 +- drivers/ram/rockchip/sdram_rk3288.c | 231 ++++++++++++-------- drivers/ram/rockchip/sdram_rk3328.c | 2 +- drivers/ram/rockchip/sdram_rk3399.c | 2 +- drivers/reset/reset-rockchip.c | 4 +- drivers/serial/Kconfig | 38 ++++ drivers/serial/ns16550.c | 195 +++++++++++++++-- drivers/serial/sandbox.c | 59 ++++-- drivers/serial/serial-uclass.c | 77 +++++++ drivers/serial/serial_omap.c | 2 +- drivers/serial/serial_rockchip.c | 59 ++++++ drivers/spi/Kconfig | 18 ++ drivers/spi/Makefile | 2 + drivers/spi/rk_spi.c | 301 +++++++++++++++++++-------- drivers/spi/spi-uclass.c | 77 +++++++ drivers/sysreset/Kconfig | 18 ++ drivers/sysreset/sysreset-uclass.c | 124 ++++++----- drivers/sysreset/sysreset_rockchip.c | 61 +++++- include/asm-generic/global_data.h | 7 +- include/clk-uclass.h | 11 + include/clk.h | 32 ++- include/linux/mtd/mtd.h | 23 +- include/linux/mtd/spi-nor.h | 22 ++ include/log.h | 6 + include/malloc.h | 3 + include/ns16550.h | 7 +- include/ram.h | 25 +++ include/regmap.h | 4 +- include/serial.h | 45 +++- include/spi.h | 31 +++ include/spi_flash.h | 7 + include/spl.h | 8 +- include/syscon.h | 2 + include/sysreset.h | 9 + 45 files changed, 1682 insertions(+), 345 deletions(-) diff --git a/drivers/clk/Kconfig b/drivers/clk/Kconfig index 8b8b719999..4762505a5b 100644 --- a/drivers/clk/Kconfig +++ b/drivers/clk/Kconfig @@ -10,6 +10,19 @@ config CLK feed into other clocks in a tree structure, with multiplexers to choose the source for each clock. +config CLK_FIXED_RATE + bool "Enable fixed-rate clock support" + depends on CLK + default y + help + This enables support for a simple fixed-rate clock. The rate + is provided by the device tree and is set up when the device is + probed. Obviously the clock rate cannot be changed after the device + is set up. + + This also enables a clock with a fixed factor (divider and + multipler) of its parent clock. + config SPL_CLK bool "Enable clock support in SPL" depends on CLK && SPL && SPL_DM @@ -20,6 +33,28 @@ config SPL_CLK setting up clocks within SPL, and allows the same drivers to be used as U-Boot proper. +config SPL_CLK_FIXED_RATE + bool "Enable fixed-rate clock support in SPL" + depends on CLK_FIXED_RATE + default y + help + This enables support for a simple fixed-rate clock in SPL. The rate + is provided by the device tree and is set up when the device is + probed. Obviously the clock rate cannot be changed after the device + is set up. + + This also enables a clock with a fixed factor (divider and + multipler) of its parent clock. + +config SPL_TINY_CLK + bool "Support tiny clock drivers in SPL" + depends on SPL_TINY + default y if SPL_TINY_ONLY + help + In constrained environments the driver-model overhead of several KB + of code and data structures can be problematic. Enable this to use a + tiny implementation that only supports a single driver. + config TPL_CLK bool "Enable clock support in TPL" depends on CLK && TPL_DM @@ -30,6 +65,25 @@ config TPL_CLK setting up clocks within TPL, and allows the same drivers to be used as U-Boot proper. +config TPL_CLK_FIXED_RATE + bool "Enable fixed-rate clock support in TPL" + depends on TPL_CLK + default y + help + This enables support for a simple fixed-rate clock in TPL. The rate + is provided by the device tree and is set up when the device is + probed. Obviously the clock rate cannot be changed after the device + is set up. + +config TPL_TINY_CLK + bool "Support tiny clock drivers in TPL" + depends on TPL_TINY + default y if TPL_TINY_ONLY + help + In constrained environments the driver-model overhead of several KB + of code and data structures can be problematic. Enable this to use a + tiny implementation that only supports a single driver. + config CLK_BCM6345 bool "Clock controller driver for BCM6345" depends on CLK && ARCH_BMIPS diff --git a/drivers/clk/Makefile b/drivers/clk/Makefile index e01783391d..8704fece92 100644 --- a/drivers/clk/Makefile +++ b/drivers/clk/Makefile @@ -5,8 +5,8 @@ # obj-$(CONFIG_$(SPL_TPL_)CLK) += clk-uclass.o -obj-$(CONFIG_$(SPL_TPL_)CLK) += clk_fixed_rate.o -obj-$(CONFIG_$(SPL_TPL_)CLK) += clk_fixed_factor.o +obj-$(CONFIG_$(SPL_TPL_)CLK_FIXED_RATE) += clk_fixed_rate.o +obj-$(CONFIG_$(SPL_TPL_)CLK_FIXED_RATE) += clk_fixed_factor.o obj-$(CONFIG_$(SPL_TPL_)CLK_CCF) += clk.o clk-divider.o clk-mux.o clk-gate.o obj-$(CONFIG_$(SPL_TPL_)CLK_CCF) += clk-fixed-factor.o obj-$(CONFIG_$(SPL_TPL_)CLK_COMPOSITE_CCF) += clk-composite.o diff --git a/drivers/clk/clk-uclass.c b/drivers/clk/clk-uclass.c index d2a381490e..e702209126 100644 --- a/drivers/clk/clk-uclass.c +++ b/drivers/clk/clk-uclass.c @@ -27,6 +27,7 @@ static inline const struct clk_ops *clk_dev_ops(struct udevice *dev) #if CONFIG_IS_ENABLED(OF_CONTROL) # if CONFIG_IS_ENABLED(OF_PLATDATA) +# if !CONFIG_IS_ENABLED(TINY_CLK) int clk_get_by_driver_info(struct udevice *dev, struct phandle_1_arg *cells, struct clk *clk) { @@ -40,6 +41,21 @@ int clk_get_by_driver_info(struct udevice *dev, struct phandle_1_arg *cells, return 0; } +# else /* TINY CLK */ +int tiny_clk_get_by_driver_info(struct phandle_1_arg *cells, + struct tiny_clk *tclk) +{ + struct tinydev *tdev; + + tdev = tiny_dev_get(UCLASS_CLK, 0); + if (!tdev) + return -ENODEV; + tclk->tdev = tdev; + tclk->id = cells->arg[0]; + + return 0; +} +# endif # else static int clk_of_xlate_default(struct clk *clk, struct ofnode_phandle_args *args) @@ -727,7 +743,8 @@ void devm_clk_put(struct udevice *dev, struct clk *clk) WARN_ON(rc); } -int clk_uclass_post_probe(struct udevice *dev) +#if !CONFIG_IS_ENABLED(TINY_CLK) +static int clk_uclass_post_probe(struct udevice *dev) { /* * when a clock provider is probed. Call clk_set_defaults() @@ -745,3 +762,37 @@ UCLASS_DRIVER(clk) = { .name = "clk", .post_probe = clk_uclass_post_probe, }; +#else /* TINY_CLK */ +static inline const struct tiny_clk_ops *tiny_clk_dev_ops(struct tinydev *tdev) +{ + return (const struct tiny_clk_ops *)tdev->drv->ops; +} + +ulong tiny_clk_set_rate(struct tiny_clk *tclk, ulong rate) +{ + const struct tiny_clk_ops *ops; + + debug("%s(tclk=%p, rate=%lu)\n", __func__, tclk, rate); + if (!tiny_clk_valid(tclk)) + return 0; + ops = tiny_clk_dev_ops(tclk->tdev); + + if (!ops->set_rate) + return -ENOSYS; + + return ops->set_rate(tclk, rate); +} + +int tiny_clk_request(struct tinydev *tdev, struct tiny_clk *tclk) +{ + const struct tiny_clk_ops *ops; + + debug("%s(tdev=%p, tclk=%p)\n", __func__, tdev, tclk); + if (!tclk) + return 0; + ops = tiny_clk_dev_ops(tdev); + tclk->tdev = tdev; + + return 0; +} +#endif diff --git a/drivers/clk/rockchip/clk_rk3288.c b/drivers/clk/rockchip/clk_rk3288.c index a1dd642eef..f3e3a21be9 100644 --- a/drivers/clk/rockchip/clk_rk3288.c +++ b/drivers/clk/rockchip/clk_rk3288.c @@ -746,7 +746,7 @@ static ulong rockchip_saradc_set_clk(struct rockchip_cru *cru, uint hz) return rockchip_saradc_get_clk(cru); } -static ulong rk3288_clk_get_rate(struct clk *clk) +static __maybe_unused ulong rk3288_clk_get_rate(struct clk *clk) { struct rk3288_clk_priv *priv = dev_get_priv(clk->dev); ulong new_rate, gclk_rate; @@ -788,14 +788,14 @@ static ulong rk3288_clk_get_rate(struct clk *clk) return new_rate; } -static ulong rk3288_clk_set_rate(struct clk *clk, ulong rate) +static ulong rk3288_clk_set_rate_(ulong clk_id, struct rk3288_clk_priv *priv, + ulong rate) { - struct rk3288_clk_priv *priv = dev_get_priv(clk->dev); struct rockchip_cru *cru = priv->cru; ulong new_rate, gclk_rate; gclk_rate = rkclk_pll_get_rate(priv->cru, CLK_GENERAL); - switch (clk->id) { + switch (clk_id) { case PLL_APLL: /* We only support a fixed rate here */ if (rate != 1800000000) @@ -812,12 +812,12 @@ static ulong rk3288_clk_set_rate(struct clk *clk, ulong rate) case SCLK_EMMC: case SCLK_SDMMC: case SCLK_SDIO0: - new_rate = rockchip_mmc_set_clk(cru, gclk_rate, clk->id, rate); + new_rate = rockchip_mmc_set_clk(cru, gclk_rate, clk_id, rate); break; case SCLK_SPI0: case SCLK_SPI1: case SCLK_SPI2: - new_rate = rockchip_spi_set_clk(cru, gclk_rate, clk->id, rate); + new_rate = rockchip_spi_set_clk(cru, gclk_rate, clk_id, rate); break; #ifndef CONFIG_SPL_BUILD case SCLK_I2S0: @@ -828,7 +828,7 @@ static ulong rk3288_clk_set_rate(struct clk *clk, ulong rate) break; case DCLK_VOP0: case DCLK_VOP1: - new_rate = rockchip_vop_set_clk(cru, priv->grf, clk->id, rate); + new_rate = rockchip_vop_set_clk(cru, priv->grf, clk_id, rate); break; case SCLK_EDP_24M: /* clk_edp_24M source: 24M */ @@ -848,7 +848,7 @@ static ulong rk3288_clk_set_rate(struct clk *clk, ulong rate) div = CPLL_HZ / rate; assert((div - 1 < 64) && (div * rate == CPLL_HZ)); - switch (clk->id) { + switch (clk_id) { case ACLK_VOP0: rk_clrsetreg(&cru->cru_clksel_con[31], 3 << 6 | 0x1f << 0, @@ -946,28 +946,9 @@ static int __maybe_unused rk3288_clk_set_parent(struct clk *clk, struct clk *par return -ENOENT; } -static struct clk_ops rk3288_clk_ops = { - .get_rate = rk3288_clk_get_rate, - .set_rate = rk3288_clk_set_rate, -#if CONFIG_IS_ENABLED(OF_CONTROL) && !CONFIG_IS_ENABLED(OF_PLATDATA) - .set_parent = rk3288_clk_set_parent, -#endif -}; - -static int rk3288_clk_ofdata_to_platdata(struct udevice *dev) -{ -#if !CONFIG_IS_ENABLED(OF_PLATDATA) - struct rk3288_clk_priv *priv = dev_get_priv(dev); - - priv->cru = dev_read_addr_ptr(dev); -#endif - - return 0; -} - -static int rk3288_clk_probe(struct udevice *dev) +static int rk3288_clk_probe_(struct rk3288_clk_priv *priv, + struct rk3288_clk_plat *plat) { - struct rk3288_clk_priv *priv = dev_get_priv(dev); bool init_clocks = false; priv->grf = syscon_get_first_range(ROCKCHIP_SYSCON_GRF); @@ -975,8 +956,6 @@ static int rk3288_clk_probe(struct udevice *dev) return PTR_ERR(priv->grf); #ifdef CONFIG_SPL_BUILD #if CONFIG_IS_ENABLED(OF_PLATDATA) - struct rk3288_clk_plat *plat = dev_get_platdata(dev); - priv->cru = map_sysmem(plat->dtd.reg[0], plat->dtd.reg[1]); #endif init_clocks = true; @@ -1001,8 +980,44 @@ static int rk3288_clk_probe(struct udevice *dev) return 0; } +#if !CONFIG_IS_ENABLED(TINY_CLK) +static ulong rk3288_clk_set_rate(struct clk *clk, ulong rate) +{ + struct rk3288_clk_priv *priv = dev_get_priv(clk->dev); + + return rk3288_clk_set_rate_(clk->id, priv, rate); +} + +static struct clk_ops rk3288_clk_ops = { + .get_rate = rk3288_clk_get_rate, + .set_rate = rk3288_clk_set_rate, +#if CONFIG_IS_ENABLED(OF_CONTROL) && !CONFIG_IS_ENABLED(OF_PLATDATA) + .set_parent = rk3288_clk_set_parent, +#endif +}; + +static int rk3288_clk_ofdata_to_platdata(struct udevice *dev) +{ +#if !CONFIG_IS_ENABLED(OF_PLATDATA) + struct rk3288_clk_priv *priv = dev_get_priv(dev); + + priv->cru = dev_read_addr_ptr(dev); +#endif + + return 0; +} + +static int rk3288_clk_probe(struct udevice *dev) +{ + struct rk3288_clk_priv *priv = dev_get_priv(dev); + struct rk3288_clk_plat *plat = dev_get_platdata(dev); + + return rk3288_clk_probe_(priv, plat); +} + static int rk3288_clk_bind(struct udevice *dev) { +#if 0 int ret; struct udevice *sys_child; struct sysreset_reg *priv; @@ -1020,6 +1035,7 @@ static int rk3288_clk_bind(struct udevice *dev) cru_glb_srst_snd_value); sys_child->priv = priv; } +#endif #if CONFIG_IS_ENABLED(RESET_ROCKCHIP) ret = offsetof(struct rockchip_cru, cru_softrst_con[0]); @@ -1047,3 +1063,31 @@ U_BOOT_DRIVER(rockchip_rk3288_cru) = { .ofdata_to_platdata = rk3288_clk_ofdata_to_platdata, .probe = rk3288_clk_probe, }; +#else +static int rockchip_clk_tiny_probe(struct tinydev *tdev) +{ + struct rk3288_clk_plat *plat = tdev->dtplat; + struct rk3288_clk_priv *priv = tdev->priv; + + return rk3288_clk_probe_(priv, plat); +} + +static ulong tiny_rk3288_clk_set_rate(struct tiny_clk *tclk, ulong rate) +{ + struct rk3288_clk_priv *priv = tclk->tdev->priv; + + return rk3288_clk_set_rate_(tclk->id, priv, rate); +} + +struct tiny_clk_ops rockchip_clk_tiny_ops = { + .set_rate = tiny_rk3288_clk_set_rate, +}; + +U_BOOT_TINY_DRIVER(rockchip_rk3288_cru) = { + .uclass_id = UCLASS_CLK, + .probe = rockchip_clk_tiny_probe, + .ops = &rockchip_clk_tiny_ops, + DM_TINY_PRIV(, \ + sizeof(struct rk3288_clk_priv)) +}; +#endif diff --git a/drivers/mtd/spi/Kconfig b/drivers/mtd/spi/Kconfig index 018e8c597e..57c4b4cb7e 100644 --- a/drivers/mtd/spi/Kconfig +++ b/drivers/mtd/spi/Kconfig @@ -16,6 +16,24 @@ config DM_SPI_FLASH enabled together (it is not possible to use driver model for one and not the other). +config SPL_TINY_SPI_FLASH + bool "Support tiny SPI-flash drivers in SPL" + depends on SPL_TINY + default y if SPL_TINY_ONLY + help + In constrained environments the driver-model overhead of several KB + of code and data structures can be problematic. Enable this to use a + tiny implementation that only supports a single driver. + +config TPL_TINY_SPI_FLASH + bool "Support tiny SPI-flash drivers in TPL" + depends on TPL_TINY + default y if TPL_TINY_ONLY + help + In constrained environments the driver-model overhead of several KB + of code and data structures can be problematic. Enable this to use a + tiny implementation that only supports a single driver. + config SPI_FLASH_SANDBOX bool "Support sandbox SPI flash device" depends on SANDBOX && DM_SPI_FLASH diff --git a/drivers/mtd/spi/sf-uclass.c b/drivers/mtd/spi/sf-uclass.c index 09c11439b0..3929120787 100644 --- a/drivers/mtd/spi/sf-uclass.c +++ b/drivers/mtd/spi/sf-uclass.c @@ -3,8 +3,11 @@ * Copyright (c) 2014 Google, Inc */ +#define LOG_CATEGORY UCLASS_SPI_FLASH + #include #include +#include #include #include #include @@ -14,6 +17,7 @@ DECLARE_GLOBAL_DATA_PTR; +#if !CONFIG_IS_ENABLED(TINY_SPI_FLASH) int spi_flash_read_dm(struct udevice *dev, u32 offset, size_t len, void *buf) { return log_ret(sf_get_ops(dev)->read(dev, offset, len, buf)); @@ -102,3 +106,75 @@ UCLASS_DRIVER(spi_flash) = { .post_bind = spi_flash_post_bind, .per_device_auto_alloc_size = sizeof(struct spi_flash), }; +#else /* TINY_SPI_FLASH */ +static int tiny_sf_probe(struct tinydev *tdev) +{ + const struct dtd_jedec_spi_nor *dtplat = tdev->dtplat; + struct tiny_spi_nor *nor = tinydev_get_priv(tdev); + struct dm_spi_slave_platdata *slave_plat; + struct spi_slave *slave; + bool exists; + int ret; + + slave = tinydev_ensure_data(tdev, DEVDATAT_PARENT_PRIV, sizeof(*slave), + &exists); + if (!slave) + return log_msg_ret("slave", -ENOMEM); + if (!exists) { + slave->tdev = tdev; + slave->max_hz = dtplat->spi_max_frequency; + slave->wordlen = SPI_DEFAULT_WORDLEN; + /* Leave mode as the default 0 */ + nor->spi = slave; + nor->tdev = tdev; + log_debug("slave->max_hz=%d\n", slave->max_hz); + } + slave_plat = tinydev_ensure_data(tdev, DEVDATAT_PARENT_PLAT, + sizeof(*slave_plat), &exists); + if (!slave_plat) + return log_msg_ret("plat", -ENOMEM); + if (!exists) { + slave_plat->cs = dtplat->reg[0]; + slave_plat->max_hz = dtplat->spi_max_frequency; + /* Leave mode as the default 0 */ + } + + log_debug("start spi_nor_scan\n"); + ret = spi_nor_scan(nor); + if (ret) + return ret; + + return 0; +} + +static int tiny_sf_read(struct tinydev *tdev, u32 offset, size_t len, void *buf) +{ + return log_ret(tiny_spi_flash_read(tdev, offset, len, buf)); +} + +struct tiny_spi_flash_ops tiny_sf_ops = { + .read = tiny_sf_read, +}; + +U_BOOT_TINY_DRIVER(jedec_spi_nor) = { + .uclass_id = UCLASS_SPI_FLASH, + .probe = tiny_sf_probe, + .ops = &tiny_sf_ops, + DM_TINY_PRIV(, sizeof(struct tiny_spi_nor)) +}; + +int tiny_spi_flash_read(struct tinydev *tdev, u32 offset, size_t len, + void *buf) +{ + struct tiny_spi_nor *nor = tinydev_get_priv(tdev); + struct tiny_mtd_info *mtd = &nor->mtd; + size_t retlen; + int ret; + + ret = tiny_spi_nor_read(mtd, offset, len, &retlen, buf); + if (ret) + return log_ret(ret); + + return 0; +} +#endif diff --git a/drivers/mtd/spi/sf_probe.c b/drivers/mtd/spi/sf_probe.c index 475f6c31db..c5245e1131 100644 --- a/drivers/mtd/spi/sf_probe.c +++ b/drivers/mtd/spi/sf_probe.c @@ -17,6 +17,7 @@ #include "sf_internal.h" +#if !CONFIG_IS_ENABLED(TINY_SPI_FLASH) /** * spi_flash_probe_slave() - Probe for a SPI flash device on a bus * @@ -173,3 +174,6 @@ U_BOOT_DRIVER(jedec_spi_nor) = { U_BOOT_DRIVER_ALIAS(jedec_spi_nor, spansion_m25p16) #endif /* CONFIG_DM_SPI_FLASH */ +#else /* TINY_SPI_FLASH */ +#endif + diff --git a/drivers/mtd/spi/spi-nor-tiny.c b/drivers/mtd/spi/spi-nor-tiny.c index 9f676c649d..397fb39a9d 100644 --- a/drivers/mtd/spi/spi-nor-tiny.c +++ b/drivers/mtd/spi/spi-nor-tiny.c @@ -9,6 +9,9 @@ * Synced from Linux v4.19 */ +// #define LOG_DEBUG +#define LOG_CATEGORY UCLASS_SPI_FLASH + #include #include #include @@ -36,6 +39,138 @@ #define DEFAULT_READY_WAIT_JIFFIES (40UL * HZ) +#if CONFIG_IS_ENABLED(TINY_SPI) + +#define spi_nor tiny_spi_nor +#define mtd_info tiny_mtd_info + +/** + * spi_mem_adjust_op_size() - Adjust the data size of a SPI mem operation to + * match controller limitations + * @tdev: the SPI device (its parent being a bus) + * @op: the operation to adjust + * + * Some controllers have FIFO limitations and must split a data transfer + * operation into multiple ones, others require a specific alignment for + * optimized accesses. This function allows SPI mem drivers to split a single + * operation into multiple sub-operations when required. + * + * Return: a negative error code if the controller can't properly adjust @op, + * 0 otherwise. Note that @op->data.nbytes will be updated if @op + * can't be handled in a single step. + */ +static int spi_mem_adjust_op_size_(struct tinydev *tdev, struct spi_mem_op *op) +{ + struct tinydev *bus = tinydev_get_parent(tdev); + struct tiny_spi_ops *ops = tiny_spi_get_ops(bus); + + if (ops->adjust_op_size) + return ops->adjust_op_size(tdev, op); + + return 0; +} + +static int spi_mem_exec_op_(struct tinydev *tdev, const struct spi_mem_op *op) +{ + struct tinydev *bus = tinydev_get_parent(tdev); + struct tiny_spi_ops *ops = tiny_spi_get_ops(bus); + unsigned int pos = 0; + const u8 *tx_buf = NULL; + u8 *rx_buf = NULL; + int op_len; + u32 flag; + int ret; + int i; + + log_debug("bus=%s\n", bus->name); + ret = tiny_spi_claim_bus(tdev); + if (ret < 0) + return ret; + + if (ops->exec_op) { + ret = ops->exec_op(tdev, op); + + /* + * Some controllers only optimize specific paths (typically the + * read path) and expect the core to use the regular SPI + * interface in other cases. + */ + if (!ret || ret != -ENOTSUPP) { + tiny_spi_release_bus(tdev); + return ret; + } + } + + if (op->data.nbytes) { + if (op->data.dir == SPI_MEM_DATA_IN) + rx_buf = op->data.buf.in; + else + tx_buf = op->data.buf.out; + } + + op_len = sizeof(op->cmd.opcode) + op->addr.nbytes + op->dummy.nbytes; + + /* + * Avoid using malloc() here so that we can use this code in SPL where + * simple malloc may be used. That implementation does not allow free() + * so repeated calls to this code can exhaust the space. + * + * The value of op_len is small, since it does not include the actual + * data being sent, only the op-code and address. In fact, it should be + * possible to just use a small fixed value here instead of op_len. + */ + u8 op_buf[op_len]; + + op_buf[pos++] = op->cmd.opcode; + + if (op->addr.nbytes) { + for (i = 0; i < op->addr.nbytes; i++) + op_buf[pos + i] = op->addr.val >> + (8 * (op->addr.nbytes - i - 1)); + + pos += op->addr.nbytes; + } + + if (op->dummy.nbytes) + memset(op_buf + pos, 0xff, op->dummy.nbytes); + + /* 1st transfer: opcode + address + dummy cycles */ + flag = SPI_XFER_BEGIN; + /* Make sure to set END bit if no tx or rx data messages follow */ + if (!tx_buf && !rx_buf) + flag |= SPI_XFER_END; + + ret = tiny_spi_xfer(tdev, op_len * 8, op_buf, NULL, flag); + if (ret) + return ret; + + /* 2nd transfer: rx or tx data path */ + if (tx_buf || rx_buf) { + ret = tiny_spi_xfer(tdev, op->data.nbytes * 8, tx_buf, rx_buf, + SPI_XFER_END); + if (ret) + return ret; + } + + tiny_spi_release_bus(tdev); + + for (i = 0; i < pos; i++) + debug("%02x ", op_buf[i]); + debug("| [%dB %s] ", + tx_buf || rx_buf ? op->data.nbytes : 0, + tx_buf || rx_buf ? (tx_buf ? "out" : "in") : "-"); + for (i = 0; i < op->data.nbytes; i++) + debug("%02x ", tx_buf ? tx_buf[i] : rx_buf[i]); + debug("[ret %d]\n", ret); + + if (ret < 0) + return ret; + + return 0; +} + +#endif /* TINY_SPI */ + static int spi_nor_read_write_reg(struct spi_nor *nor, struct spi_mem_op *op, void *buf) { @@ -43,7 +178,8 @@ static int spi_nor_read_write_reg(struct spi_nor *nor, struct spi_mem_op op->data.buf.in = buf; else op->data.buf.out = buf; - return spi_mem_exec_op(nor->spi, op); + + return spi_mem_exec_op_(nor->tdev, op); } static int spi_nor_read_reg(struct spi_nor *nor, u8 code, u8 *val, int len) @@ -94,11 +230,11 @@ static ssize_t spi_nor_read_data(struct spi_nor *nor, loff_t from, size_t len, while (remaining) { op.data.nbytes = remaining < UINT_MAX ? remaining : UINT_MAX; - ret = spi_mem_adjust_op_size(nor->spi, &op); + ret = spi_mem_adjust_op_size_(nor->tdev, &op); if (ret) return ret; - ret = spi_mem_exec_op(nor->spi, &op); + ret = spi_mem_exec_op_(nor->tdev, &op); if (ret) return ret; @@ -351,7 +487,8 @@ static int spi_nor_wait_till_ready(struct spi_nor *nor) * Erase an address range on the nor chip. The address range may extend * one or more erase sectors. Return an error is there is a problem erasing. */ -static int spi_nor_erase(struct mtd_info *mtd, struct erase_info *instr) +__maybe_unused static int spi_nor_erase(struct mtd_info *mtd, + struct erase_info *instr) { return -ENOTSUPP; } @@ -380,8 +517,8 @@ static const struct flash_info *spi_nor_read_id(struct spi_nor *nor) return ERR_PTR(-ENODEV); } -static int spi_nor_read(struct mtd_info *mtd, loff_t from, size_t len, - size_t *retlen, u_char *buf) +int tiny_spi_nor_read(struct mtd_info *mtd, loff_t from, size_t len, + size_t *retlen, u_char *buf) { struct spi_nor *nor = mtd_to_spi_nor(mtd); int ret; @@ -416,8 +553,8 @@ read_err: * FLASH_PAGESIZE chunks. The address range may be any size provided * it is within the physical boundaries. */ -static int spi_nor_write(struct mtd_info *mtd, loff_t to, size_t len, - size_t *retlen, const u_char *buf) +__maybe_unused static int spi_nor_write(struct mtd_info *mtd, loff_t to, + size_t len, size_t *retlen, const u_char *buf) { return -ENOTSUPP; } @@ -723,10 +860,14 @@ int spi_nor_scan(struct spi_nor *nor) struct spi_slave *spi = nor->spi; int ret; + log_debug("start\n"); + /* Reset SPI protocol for all commands. */ - nor->reg_proto = SNOR_PROTO_1_1_1; nor->read_proto = SNOR_PROTO_1_1_1; +#if !CONFIG_IS_ENABLED(TINY_SPI) + nor->reg_proto = SNOR_PROTO_1_1_1; nor->write_proto = SNOR_PROTO_1_1_1; +#endif if (spi->mode & SPI_RX_QUAD) hwcaps.mask |= SNOR_HWCAPS_READ_1_1_4; @@ -739,16 +880,17 @@ int spi_nor_scan(struct spi_nor *nor) if (ret) return ret; - mtd->name = "spi-flash"; mtd->priv = nor; + mtd->size = info->sector_size * info->n_sectors; +#if !CONFIG_IS_ENABLED(TINY_SPI) + mtd->name = "spi-flash"; mtd->type = MTD_NORFLASH; mtd->writesize = 1; mtd->flags = MTD_CAP_NORFLASH; - mtd->size = info->sector_size * info->n_sectors; mtd->_erase = spi_nor_erase; mtd->_read = spi_nor_read; mtd->_write = spi_nor_write; - +#endif nor->size = mtd->size; if (info->flags & USE_FSR) diff --git a/drivers/ram/Kconfig b/drivers/ram/Kconfig index 7e6e981897..0c6d811219 100644 --- a/drivers/ram/Kconfig +++ b/drivers/ram/Kconfig @@ -17,6 +17,15 @@ config SPL_RAM SPL, enable this option. It might provide a cleaner interface to setting up RAM (e.g. SDRAM / DDR) within SPL. +config SPL_TINY_RAM + bool "Support tiny RAM drivers in SPL" + depends on SPL_RAM + default y if SPL_TINY_ONLY + help + In constrained environments the driver-model overhead of several KB + of code and data structures can be problematic. Enable this to use a + tiny implementation that only supports a single driver. + config TPL_RAM bool "Enable RAM support in TPL" depends on RAM @@ -26,6 +35,15 @@ config TPL_RAM TPL, enable this option. It might provide a cleaner interface to setting up RAM (e.g. SDRAM / DDR) within TPL. +config TPL_TINY_RAM + bool "Support tiny RAM drivers in TPL" + depends on TPL_RAM + default y if TPL_TINY_ONLY + help + In constrained environments the driver-model overhead of several KB + of code and data structures can be problematic. Enable this to use a + tiny implementation that only supports a single driver. + config STM32_SDRAM bool "Enable STM32 SDRAM support" depends on RAM diff --git a/drivers/ram/ram-uclass.c b/drivers/ram/ram-uclass.c index f4d387fed1..608eab9cd4 100644 --- a/drivers/ram/ram-uclass.c +++ b/drivers/ram/ram-uclass.c @@ -11,6 +11,7 @@ #include #include +#if !CONFIG_IS_ENABLED(TINY_RAM) int ram_get_info(struct udevice *dev, struct ram_info *info) { struct ram_ops *ops = ram_get_ops(dev); @@ -25,3 +26,14 @@ UCLASS_DRIVER(ram) = { .id = UCLASS_RAM, .name = "ram", }; +#else /* TINY_RAM */ +int tiny_ram_get_info(struct tinydev *tdev, struct ram_info *info) +{ + struct tiny_ram_ops *ops = tiny_ram_get_ops(tdev); + + if (IS_ENABLED(CONFIG_TINY_CHECK) && !ops->get_info) + return -ENOSYS; + + return ops->get_info(tdev, info); +} +#endif diff --git a/drivers/ram/rockchip/sdram_rk3188.c b/drivers/ram/rockchip/sdram_rk3188.c index 06f9eba1a5..5d94862e28 100644 --- a/drivers/ram/rockchip/sdram_rk3188.c +++ b/drivers/ram/rockchip/sdram_rk3188.c @@ -866,7 +866,7 @@ static int conv_of_platdata(struct udevice *dev) memcpy(&plat->base, of_plat->rockchip_sdram_params, sizeof(plat->base)); /* rk3188 supports dual-channel, set default channel num to 2 */ plat->num_channels = 1; - ret = regmap_init_mem_platdata(dev, of_plat->reg, + ret = regmap_init_mem_platdata(of_plat->reg, ARRAY_SIZE(of_plat->reg) / 2, &plat->map); if (ret) diff --git a/drivers/ram/rockchip/sdram_rk322x.c b/drivers/ram/rockchip/sdram_rk322x.c index 094693ce24..16f4eb0df0 100644 --- a/drivers/ram/rockchip/sdram_rk322x.c +++ b/drivers/ram/rockchip/sdram_rk322x.c @@ -767,7 +767,7 @@ static int conv_of_platdata(struct udevice *dev) memcpy(&plat->base, of_plat->rockchip_sdram_params, sizeof(plat->base)); plat->num_channels = 1; - ret = regmap_init_mem_platdata(dev, of_plat->reg, + ret = regmap_init_mem_platdata(of_plat->reg, ARRAY_SIZE(of_plat->reg) / 2, &plat->map); if (ret) diff --git a/drivers/ram/rockchip/sdram_rk3288.c b/drivers/ram/rockchip/sdram_rk3288.c index 26e8d059b5..83bc62caf7 100644 --- a/drivers/ram/rockchip/sdram_rk3288.c +++ b/drivers/ram/rockchip/sdram_rk3288.c @@ -30,27 +30,10 @@ #include #include -struct chan_info { - struct rk3288_ddr_pctl *pctl; - struct rk3288_ddr_publ *publ; - struct rk3288_msch *msch; -}; - -struct dram_info { - struct chan_info chan[2]; - struct ram_info info; - struct clk ddr_clk; - struct rockchip_cru *cru; - struct rk3288_grf *grf; - struct rk3288_sgrf *sgrf; - struct rk3288_pmu *pmu; - bool is_veyron; -}; +struct dtd_rockchip_rk3288_dmc; struct rk3288_sdram_params { -#if CONFIG_IS_ENABLED(OF_PLATDATA) - struct dtd_rockchip_rk3288_dmc of_plat; -#endif + IF_OF_PLATDATA(struct dtd_rockchip_rk3288_dmc of_plat;) struct rk3288_sdram_channel ch[2]; struct rk3288_sdram_pctl_timing pctl_timing; struct rk3288_sdram_phy_timing phy_timing; @@ -85,6 +68,11 @@ const int ddrconf_table[] = { #if defined(CONFIG_TPL_BUILD) || \ (!defined(CONFIG_TPL) && defined(CONFIG_SPL_BUILD)) +#define DO_SDRAM_INIT 1 +#else +#define DO_SDRAM_INIT 0 +#endif + static void copy_to_reg(u32 *dest, const u32 *src, u32 n) { int i; @@ -291,7 +279,7 @@ static void pctl_cfg(int channel, struct rk3288_ddr_pctl *pctl, setbits_le32(&pctl->scfg, 1); } -static void phy_cfg(const struct chan_info *chan, int channel, +static void phy_cfg(const struct rk_chan_info *chan, int channel, struct rk3288_sdram_params *sdram_params) { struct rk3288_ddr_publ *publ = chan->publ; @@ -436,7 +424,7 @@ static void move_to_config_state(struct rk3288_ddr_publ *publ, } } -static void set_bandwidth_ratio(const struct chan_info *chan, int channel, +static void set_bandwidth_ratio(const struct rk_chan_info *chan, int channel, u32 n, struct rk3288_grf *grf) { struct rk3288_ddr_pctl *pctl = chan->pctl; @@ -474,7 +462,7 @@ static void set_bandwidth_ratio(const struct chan_info *chan, int channel, setbits_le32(&pctl->dfistcfg0, 1 << 2); } -static int data_training(const struct chan_info *chan, int channel, +static int data_training(const struct rk_chan_info *chan, int channel, struct rk3288_sdram_params *sdram_params) { unsigned int j; @@ -537,7 +525,7 @@ static int data_training(const struct chan_info *chan, int channel, return ret; } -static void move_to_access_state(const struct chan_info *chan) +static void move_to_access_state(const struct rk_chan_info *chan) { struct rk3288_ddr_publ *publ = chan->publ; struct rk3288_ddr_pctl *pctl = chan->pctl; @@ -577,7 +565,7 @@ static void move_to_access_state(const struct chan_info *chan) } } -static void dram_cfg_rbc(const struct chan_info *chan, u32 chnum, +static void dram_cfg_rbc(const struct rk_chan_info *chan, u32 chnum, struct rk3288_sdram_params *sdram_params) { struct rk3288_ddr_publ *publ = chan->publ; @@ -591,7 +579,7 @@ static void dram_cfg_rbc(const struct chan_info *chan, u32 chnum, writel(sdram_params->base.ddrconfig, &chan->msch->ddrconf); } -static void dram_all_config(const struct dram_info *dram, +static void dram_all_config(const struct rk_dram_info *dram, struct rk3288_sdram_params *sdram_params) { unsigned int chan; @@ -619,12 +607,12 @@ static void dram_all_config(const struct dram_info *dram, rk_clrsetreg(&dram->sgrf->soc_con2, 0x1f, sdram_params->base.stride); } -static int sdram_rank_bw_detect(struct dram_info *dram, int channel, +static int sdram_rank_bw_detect(struct rk_dram_info *dram, int channel, struct rk3288_sdram_params *sdram_params) { int reg; int need_trainig = 0; - const struct chan_info *chan = &dram->chan[channel]; + const struct rk_chan_info *chan = &dram->chan[channel]; struct rk3288_ddr_publ *publ = chan->publ; if (data_training(chan, channel, sdram_params) < 0) { @@ -672,12 +660,12 @@ static int sdram_rank_bw_detect(struct dram_info *dram, int channel, return 0; } -static int sdram_col_row_detect(struct dram_info *dram, int channel, +static int sdram_col_row_detect(struct rk_dram_info *dram, int channel, struct rk3288_sdram_params *sdram_params) { int row, col; unsigned int addr; - const struct chan_info *chan = &dram->chan[channel]; + const struct rk_chan_info *chan = &dram->chan[channel]; struct rk3288_ddr_pctl *pctl = chan->pctl; struct rk3288_ddr_publ *publ = chan->publ; int ret = 0; @@ -782,7 +770,7 @@ static int sdram_get_stride(struct rk3288_sdram_params *sdram_params) return ret; } -static int sdram_init(struct dram_info *dram, +static int sdram_init(struct rk_dram_info *dram, struct rk3288_sdram_params *sdram_params) { int channel; @@ -799,7 +787,11 @@ static int sdram_init(struct dram_info *dram, } debug("ddr clk dpll\n"); - ret = clk_set_rate(&dram->ddr_clk, sdram_params->base.ddr_freq); + if (!CONFIG_IS_ENABLED(TINY_CLK)) + ret = clk_set_rate(&dram->ddr_clk, sdram_params->base.ddr_freq); + else + ret = tiny_clk_set_rate(&dram->tiny_ddr_clk, + sdram_params->base.ddr_freq); debug("ret=%d\n", ret); if (ret) { debug("Could not set DDR clock\n"); @@ -807,7 +799,7 @@ static int sdram_init(struct dram_info *dram, } for (channel = 0; channel < 2; channel++) { - const struct chan_info *chan = &dram->chan[channel]; + const struct rk_chan_info *chan = &dram->chan[channel]; struct rk3288_ddr_pctl *pctl = chan->pctl; struct rk3288_ddr_publ *publ = chan->publ; @@ -927,8 +919,7 @@ error: hang(); } -# ifdef CONFIG_ROCKCHIP_FAST_SPL -static int veyron_init(struct dram_info *priv) +static int veyron_init(struct rk_dram_info *priv) { struct udevice *pmic; int ret; @@ -951,32 +942,31 @@ static int veyron_init(struct dram_info *priv) return 0; } -# endif -static int setup_sdram(struct udevice *dev) +static int setup_sdram(struct rk_dram_info *priv, + struct rk3288_sdram_params *params) { - struct dram_info *priv = dev_get_priv(dev); - struct rk3288_sdram_params *params = dev_get_platdata(dev); - -# ifdef CONFIG_ROCKCHIP_FAST_SPL - if (priv->is_veyron) { + if (IS_ENABLED(CONFIG_ROCKCHIP_FAST_SPL) && priv->is_veyron) { int ret; ret = veyron_init(priv); if (ret) return ret; } -# endif return sdram_init(priv, params); } +#if !CONFIG_IS_ENABLED(TINY_RAM) static int rk3288_dmc_ofdata_to_platdata(struct udevice *dev) { #if !CONFIG_IS_ENABLED(OF_PLATDATA) struct rk3288_sdram_params *params = dev_get_platdata(dev); int ret; + if (!DO_SDRAM_INIT) + return 0; + /* Rk3288 supports dual-channel, set default channel num to 2 */ params->num_channels = 2; ret = dev_read_u32_array(dev, "rockchip,pctl-timing", @@ -1000,11 +990,12 @@ static int rk3288_dmc_ofdata_to_platdata(struct udevice *dev) debug("%s: Cannot read rockchip,sdram-params\n", __func__); return -EINVAL; } -#ifdef CONFIG_ROCKCHIP_FAST_SPL - struct dram_info *priv = dev_get_priv(dev); + if (IS_ENABLED(CONFIG_ROCKCHIP_FAST_SPL)) { + struct rk_dram_info *priv = dev_get_priv(dev); - priv->is_veyron = !fdt_node_check_compatible(blob, 0, "google,veyron"); -#endif + priv->is_veyron = !fdt_node_check_compatible(gd->fdt_blob, 0, + "google,veyron"); + } ret = regmap_init_mem(dev_ofnode(dev), ¶ms->map); if (ret) return ret; @@ -1012,13 +1003,12 @@ static int rk3288_dmc_ofdata_to_platdata(struct udevice *dev) return 0; } -#endif /* CONFIG_SPL_BUILD */ +#endif /* !TINY_RAM */ -#if CONFIG_IS_ENABLED(OF_PLATDATA) -static int conv_of_platdata(struct udevice *dev) +static int conv_of_platdata(struct rk3288_sdram_params *plat, + struct dtd_rockchip_rk3288_dmc *of_plat) { - struct rk3288_sdram_params *plat = dev_get_platdata(dev); - struct dtd_rockchip_rk3288_dmc *of_plat = &plat->of_plat; +#if CONFIG_IS_ENABLED(OF_PLATDATA) int ret; memcpy(&plat->pctl_timing, of_plat->rockchip_pctl_timing, @@ -1028,35 +1018,22 @@ static int conv_of_platdata(struct udevice *dev) memcpy(&plat->base, of_plat->rockchip_sdram_params, sizeof(plat->base)); /* Rk3288 supports dual-channel, set default channel num to 2 */ plat->num_channels = 2; - ret = regmap_init_mem_platdata(dev, of_plat->reg, + ret = regmap_init_mem_platdata(of_plat->reg, ARRAY_SIZE(of_plat->reg) / 2, &plat->map); if (ret) return ret; +#endif return 0; } -#endif -static int rk3288_dmc_probe(struct udevice *dev) +static int complete_probe(struct rk3288_sdram_params *plat, + struct rk_dram_info *priv) { -#if defined(CONFIG_TPL_BUILD) || \ - (!defined(CONFIG_TPL) && defined(CONFIG_SPL_BUILD)) - struct rk3288_sdram_params *plat = dev_get_platdata(dev); - struct udevice *dev_clk; struct regmap *map; int ret; -#endif - struct dram_info *priv = dev_get_priv(dev); - priv->pmu = syscon_get_first_range(ROCKCHIP_SYSCON_PMU); -#if defined(CONFIG_TPL_BUILD) || \ - (!defined(CONFIG_TPL) && defined(CONFIG_SPL_BUILD)) -#if CONFIG_IS_ENABLED(OF_PLATDATA) - ret = conv_of_platdata(dev); - if (ret) - return ret; -#endif map = syscon_get_regmap_by_driver_data(ROCKCHIP_SYSCON_NOC); if (IS_ERR(map)) return PTR_ERR(map); @@ -1072,32 +1049,69 @@ static int rk3288_dmc_probe(struct udevice *dev) priv->chan[1].pctl = regmap_get_range(plat->map, 2); priv->chan[1].publ = regmap_get_range(plat->map, 3); - ret = rockchip_get_clk(&dev_clk); - if (ret) - return ret; - priv->ddr_clk.id = CLK_DDR; - ret = clk_request(dev_clk, &priv->ddr_clk); - if (ret) - return ret; + if (!CONFIG_IS_ENABLED(TINY_CLK)) { + struct udevice *dev_clk; + + ret = rockchip_get_clk(&dev_clk); + if (ret) + return ret; + priv->ddr_clk.id = CLK_DDR; + ret = clk_request(dev_clk, &priv->ddr_clk); + if (ret) + return ret; + } else { + struct tinydev *tdev_clk; + + tdev_clk = tiny_rockchip_get_clk(); + if (!tdev_clk) + return -ENODEV; + priv->tiny_ddr_clk.id = CLK_DDR; + ret = tiny_clk_request(tdev_clk, &priv->tiny_ddr_clk); + if (ret) + return ret; + } priv->cru = rockchip_get_cru(); if (IS_ERR(priv->cru)) return PTR_ERR(priv->cru); - ret = setup_sdram(dev); + ret = setup_sdram(priv, plat); if (ret) return ret; -#else - priv->info.base = CONFIG_SYS_SDRAM_BASE; - priv->info.size = rockchip_sdram_size( - (phys_addr_t)&priv->pmu->sys_reg[2]); -#endif + + return 0; +} + +#if !CONFIG_IS_ENABLED(TINY_RAM) +static int rk3288_dmc_probe(struct udevice *dev) +{ + struct rk3288_sdram_params *plat = dev_get_platdata(dev); + int ret; + struct rk_dram_info *priv = dev_get_priv(dev); + + priv->pmu = syscon_get_first_range(ROCKCHIP_SYSCON_PMU); + if (DO_SDRAM_INIT) { + if (CONFIG_IS_ENABLED(OF_PLATDATA)) { + ret = conv_of_platdata(plat, + CONFIG_IS_ENABLED(OF_PLATDATA, (&plat->of_plat), + (NULL))); + if (ret) + return log_msg_ret("plat", ret); + } + ret = complete_probe(plat, priv); + if (ret) + return log_msg_ret("complete", ret); + } else { + priv->info.base = CONFIG_SYS_SDRAM_BASE; + priv->info.size = rockchip_sdram_size( + (phys_addr_t)&priv->pmu->sys_reg[2]); + } return 0; } static int rk3288_dmc_get_info(struct udevice *dev, struct ram_info *info) { - struct dram_info *priv = dev_get_priv(dev); + struct rk_dram_info *priv = dev_get_priv(dev); *info = priv->info; @@ -1118,14 +1132,55 @@ U_BOOT_DRIVER(rockchip_rk3288_dmc) = { .id = UCLASS_RAM, .of_match = rk3288_dmc_ids, .ops = &rk3288_dmc_ops, -#if defined(CONFIG_TPL_BUILD) || \ - (!defined(CONFIG_TPL) && defined(CONFIG_SPL_BUILD)) .ofdata_to_platdata = rk3288_dmc_ofdata_to_platdata, -#endif .probe = rk3288_dmc_probe, - .priv_auto_alloc_size = sizeof(struct dram_info), -#if defined(CONFIG_TPL_BUILD) || \ - (!defined(CONFIG_TPL) && defined(CONFIG_SPL_BUILD)) + .priv_auto_alloc_size = sizeof(struct rk_dram_info), +#if DO_SDRAM_INIT .platdata_auto_alloc_size = sizeof(struct rk3288_sdram_params), #endif }; +#else /* TINY_RAM */ +static int tiny_rk3288_dmc_probe(struct tinydev *tdev) +{ + struct rk_dram_info *priv = tinydev_get_priv(tdev); + int ret; + + priv->pmu = syscon_get_first_range(ROCKCHIP_SYSCON_PMU); + if (DO_SDRAM_INIT) { + struct rk3288_sdram_params plat; + + ret = conv_of_platdata(&plat, tdev->dtplat); + if (ret) + return log_msg_ret("plat", ret); + ret = complete_probe(&plat, priv); + if (ret) + return log_msg_ret("complete", ret); + } else { + priv->info.base = CONFIG_SYS_SDRAM_BASE; + priv->info.size = rockchip_sdram_size( + (phys_addr_t)&priv->pmu->sys_reg[2]); + } + + return 0; +} +static int tiny_rk3288_dmc_get_info(struct tinydev *dev, struct ram_info *info) +{ + struct rk_dram_info *priv = tinydev_get_priv(dev); + + *info = priv->info; + + return 0; +} + +static struct tiny_ram_ops tiny_rk3288_dmc_ops = { + .get_info = tiny_rk3288_dmc_get_info, +}; + +U_BOOT_TINY_DRIVER(rockchip_rk3288_dmc) = { + .uclass_id = UCLASS_RAM, + .ops = &tiny_rk3288_dmc_ops, + .probe = tiny_rk3288_dmc_probe, + DM_TINY_PRIV(, \ + sizeof(struct rk_dram_info)) +}; +#endif diff --git a/drivers/ram/rockchip/sdram_rk3328.c b/drivers/ram/rockchip/sdram_rk3328.c index 98c7feb6cf..184af5c861 100644 --- a/drivers/ram/rockchip/sdram_rk3328.c +++ b/drivers/ram/rockchip/sdram_rk3328.c @@ -54,7 +54,7 @@ static int conv_of_platdata(struct udevice *dev) struct dtd_rockchip_rk3328_dmc *dtplat = &plat->dtplat; int ret; - ret = regmap_init_mem_platdata(dev, dtplat->reg, + ret = regmap_init_mem_platdata(dtplat->reg, ARRAY_SIZE(dtplat->reg) / 2, &plat->map); if (ret) diff --git a/drivers/ram/rockchip/sdram_rk3399.c b/drivers/ram/rockchip/sdram_rk3399.c index 0fe2cedc52..111a8856f6 100644 --- a/drivers/ram/rockchip/sdram_rk3399.c +++ b/drivers/ram/rockchip/sdram_rk3399.c @@ -3061,7 +3061,7 @@ static int conv_of_platdata(struct udevice *dev) struct dtd_rockchip_rk3399_dmc *dtplat = &plat->dtplat; int ret; - ret = regmap_init_mem_platdata(dev, dtplat->reg, + ret = regmap_init_mem_platdata(dtplat->reg, ARRAY_SIZE(dtplat->reg) / 2, &plat->map); if (ret) diff --git a/drivers/reset/reset-rockchip.c b/drivers/reset/reset-rockchip.c index 8092555650..689f38405f 100644 --- a/drivers/reset/reset-rockchip.c +++ b/drivers/reset/reset-rockchip.c @@ -112,8 +112,8 @@ int rockchip_reset_bind(struct udevice *pdev, u32 reg_offset, u32 reg_number) struct rockchip_reset_priv *priv; int ret; - ret = device_bind_driver_to_node(pdev, "rockchip_reset", "reset", - dev_ofnode(pdev), &rst_dev); + ret = device_bind_driver_to_node(pdev, "rockchip_reset", "reset", + dev_ofnode(pdev), &rst_dev); if (ret) { debug("Warning: No rockchip reset driver: ret=%d\n", ret); return ret; diff --git a/drivers/serial/Kconfig b/drivers/serial/Kconfig index bc7f42bbf3..8d3591b235 100644 --- a/drivers/serial/Kconfig +++ b/drivers/serial/Kconfig @@ -108,6 +108,24 @@ config DM_SERIAL implements serial_putc() etc. The uclass interface is defined in include/serial.h. +config SPL_TINY_SERIAL + bool "Support tiny serial drivers in SPL" + depends on SPL_TINY + default y if SPL_TINY_ONLY + help + In constrained environments the driver-model overhead of several KB + of code and data structures can be problematic. Enable this to use a + tiny implementation that only supports a single driver. + +config TPL_TINY_SERIAL + bool "Support tiny serial drivers in TPL" + depends on TPL_TINY + default y if TPL_TINY_ONLY + help + In constrained environments the driver-model overhead of several KB + of code and data structures can be problematic. Enable this to use a + tiny implementation that only supports a single driver. + config SERIAL_RX_BUFFER bool "Enable RX buffer for serial input" depends on DM_SERIAL @@ -641,7 +659,9 @@ config SYS_NS16550 config NS16550_DYNAMIC bool "Allow NS16550 to be configured at runtime" + depends on SYS_NS16550 default y if SYS_COREBOOT || SYS_SLIMBOOTLOADER + default y if SPL_TINY_SERIAL || TPL_TINY_SERIAL help Enable this option to allow device-tree control of the driver. @@ -660,6 +680,24 @@ config NS16550_DYNAMIC UARTs in a system. This option avoids this problem at the cost of a slightly increased code size. +config NS16550_SUPPORT_IO + bool "Support I/O access in the ns16550 driver" + default y if X86 + help + This enables I/O access in this driver, as wells the normal + memory-mapped access. This is used on some architectures, such as + x86. Note that I/O access is not supported on some more modern + architectures, such as ARM. + +config NS16550_SUPPORT_ENDIAN + bool "Support endian-aware access in the ns16550 driver" + default y if POWERPC + help + This enables in_be32() and the like in this driver, as wells the + normal memory-mapped access. This is used on some architectures, such + as PowerPC. Note that this access is not used on some architectures, + such as ARM. + config INTEL_MID_SERIAL bool "Intel MID platform UART support" depends on DM_SERIAL && OF_CONTROL diff --git a/drivers/serial/ns16550.c b/drivers/serial/ns16550.c index 20340a83f8..eb8f1f7166 100644 --- a/drivers/serial/ns16550.c +++ b/drivers/serial/ns16550.c @@ -88,10 +88,12 @@ static inline int serial_in_shift(void *addr, int shift) static void serial_out_dynamic(struct ns16550_platdata *plat, u8 *addr, int value) { - if (plat->flags & NS16550_FLAG_IO) { + if (IS_ENABLED(CONFIG_NS16550_SUPPORT_IO) && + (plat->flags & NS16550_FLAG_IO)) { outb(value, addr); } else if (plat->reg_width == 4) { - if (plat->flags & NS16550_FLAG_ENDIAN) { + if (IS_ENABLED(CONFIG_NS16550_SUPPORT_ENDIAN) && + (plat->flags & NS16550_FLAG_ENDIAN)) { if (plat->flags & NS16550_FLAG_BE) out_be32(addr, value); else @@ -99,7 +101,8 @@ static void serial_out_dynamic(struct ns16550_platdata *plat, u8 *addr, } else { writel(value, addr); } - } else if (plat->flags & NS16550_FLAG_BE) { + } else if (IS_ENABLED(CONFIG_NS16550_SUPPORT_ENDIAN) && + (plat->flags & NS16550_FLAG_BE)) { writeb(value, addr + (1 << plat->reg_shift) - 1); } else { writeb(value, addr); @@ -108,10 +111,12 @@ static void serial_out_dynamic(struct ns16550_platdata *plat, u8 *addr, static int serial_in_dynamic(struct ns16550_platdata *plat, u8 *addr) { - if (plat->flags & NS16550_FLAG_IO) { + if (IS_ENABLED(CONFIG_NS16550_SUPPORT_IO) && + (plat->flags & NS16550_FLAG_IO)) { return inb(addr); } else if (plat->reg_width == 4) { - if (plat->flags & NS16550_FLAG_ENDIAN) { + if (IS_ENABLED(CONFIG_NS16550_SUPPORT_ENDIAN) && + (plat->flags & NS16550_FLAG_ENDIAN)) { if (plat->flags & NS16550_FLAG_BE) return in_be32(addr); else @@ -119,7 +124,8 @@ static int serial_in_dynamic(struct ns16550_platdata *plat, u8 *addr) } else { return readl(addr); } - } else if (plat->flags & NS16550_FLAG_BE) { + } else if (IS_ENABLED(CONFIG_NS16550_SUPPORT_ENDIAN) && + (plat->flags & NS16550_FLAG_BE)) { return readb(addr + (1 << plat->reg_shift) - 1); } else { return readb(addr); @@ -138,13 +144,27 @@ static inline int serial_in_dynamic(struct ns16550_platdata *plat, u8 *addr) #endif /* CONFIG_NS16550_DYNAMIC */ +static u8 *ns16550_get_addr(struct ns16550_platdata *plat, int offset) +{ + offset *= 1 << plat->reg_shift; + + return (u8 *)plat->base + offset + plat->reg_offset; +} + +int ns16550_calc_divisor(int clock, int baudrate) +{ + const unsigned int mode_x_div = 16; + + return DIV_ROUND_CLOSEST(clock, mode_x_div * baudrate); +} + +#if !CONFIG_IS_ENABLED(TINY_SERIAL) static void ns16550_writeb(NS16550_t port, int offset, int value) { struct ns16550_platdata *plat = port->plat; unsigned char *addr; - offset *= 1 << plat->reg_shift; - addr = (unsigned char *)plat->base + offset + plat->reg_offset; + addr = ns16550_get_addr(plat, offset); if (IS_ENABLED(CONFIG_NS16550_DYNAMIC)) serial_out_dynamic(plat, addr, value); @@ -157,8 +177,7 @@ static int ns16550_readb(NS16550_t port, int offset) struct ns16550_platdata *plat = port->plat; unsigned char *addr; - offset *= 1 << plat->reg_shift; - addr = (unsigned char *)plat->base + offset + plat->reg_offset; + addr = ns16550_get_addr(plat, offset); if (IS_ENABLED(CONFIG_NS16550_DYNAMIC)) return serial_in_dynamic(plat, addr); @@ -181,13 +200,6 @@ static u32 ns16550_getfcr(NS16550_t port) ns16550_readb(com_port, \ (unsigned char *)addr - (unsigned char *)com_port) -int ns16550_calc_divisor(NS16550_t port, int clock, int baudrate) -{ - const unsigned int mode_x_div = 16; - - return DIV_ROUND_CLOSEST(clock, mode_x_div * baudrate); -} - static void NS16550_setbrg(NS16550_t com_port, int baud_divisor) { /* to keep serial format, read lcr before writing BKSE */ @@ -309,7 +321,7 @@ static inline void _debug_uart_init(void) * feasible. The better fix is to move all users of this driver to * driver model. */ - baud_divisor = ns16550_calc_divisor(com_port, CONFIG_DEBUG_UART_CLOCK, + baud_divisor = ns16550_calc_divisor(CONFIG_DEBUG_UART_CLOCK, CONFIG_BAUDRATE); serial_dout(&com_port->ier, CONFIG_SYS_NS16550_IER); serial_dout(&com_port->mcr, UART_MCRVAL); @@ -396,7 +408,7 @@ static int ns16550_serial_setbrg(struct udevice *dev, int baudrate) struct ns16550_platdata *plat = com_port->plat; int clock_divisor; - clock_divisor = ns16550_calc_divisor(com_port, plat->clock, baudrate); + clock_divisor = ns16550_calc_divisor(plat->clock, baudrate); NS16550_setbrg(com_port, clock_divisor); @@ -601,3 +613,148 @@ U_BOOT_DRIVER_ALIAS(ns16550_serial, rockchip_rk3368_uart) U_BOOT_DRIVER_ALIAS(ns16550_serial, ti_da830_uart) #endif #endif /* SERIAL_PRESENT */ + +#else /* TINY_SERIAL */ + +static void serial_out_reg(struct ns16550_platdata *plat, int offset, int value) +{ + unsigned char *addr = ns16550_get_addr(plat, offset); + + serial_out_dynamic(plat, addr, value); +} + +static int serial_in_reg(struct ns16550_platdata *plat, int offset) +{ + unsigned char *addr = ns16550_get_addr(plat, offset); + + return serial_in_dynamic(plat, addr); +} + +#define ns16550_reg(field) offsetof(struct NS16550, field) + +int ns16550_tiny_probe_plat(struct ns16550_platdata *plat) +{ + while (!(serial_in_reg(plat, ns16550_reg(lsr)) & UART_LSR_TEMT)) + ; + + serial_out_reg(plat, ns16550_reg(ier), CONFIG_SYS_NS16550_IER); + serial_out_reg(plat, ns16550_reg(mcr), UART_MCRVAL); + serial_out_reg(plat, ns16550_reg(fcr), plat->fcr); + + /* initialise serial config to 8N1 before writing baudrate */ + serial_out_reg(plat, ns16550_reg(lcr), UART_LCRVAL); + + return 0; +} + +int ns16550_tiny_setbrg(struct ns16550_platdata *plat, int baud_rate) +{ + int baud_divisor; + + baud_divisor = ns16550_calc_divisor(plat->clock, baud_rate); + serial_out_reg(plat, ns16550_reg(lcr), UART_LCR_BKSE | UART_LCRVAL); + serial_out_reg(plat, ns16550_reg(dll), baud_divisor & 0xff); + serial_out_reg(plat, ns16550_reg(dlm), (baud_divisor >> 8) & 0xff); + serial_out_reg(plat, ns16550_reg(lcr), UART_LCRVAL); + + return 0; +} + +int ns16550_tiny_putc(struct ns16550_platdata *plat, const char ch) +{ + while (!(serial_in_reg(plat, ns16550_reg(lsr)) & UART_LSR_THRE)) + ; + serial_out_reg(plat, ns16550_reg(thr), ch); + + return 0; +} + +#ifdef CONFIG_DEBUG_UART_NS16550 + +#include + +static inline void _debug_uart_init(void) +{ + struct NS16550 *com_port = (struct NS16550 *)CONFIG_DEBUG_UART_BASE; + int baud_divisor; + + /* + * We copy the code from above because it is already horribly messy. + * Trying to refactor to nicely remove the duplication doesn't seem + * feasible. The better fix is to move all users of this driver to + * driver model. + */ + baud_divisor = ns16550_calc_divisor(CONFIG_DEBUG_UART_CLOCK, + CONFIG_BAUDRATE); + serial_dout(&com_port->ier, CONFIG_SYS_NS16550_IER); + serial_dout(&com_port->mcr, UART_MCRVAL); + serial_dout(&com_port->fcr, UART_FCR_DEFVAL); + + serial_dout(&com_port->lcr, UART_LCR_BKSE | UART_LCRVAL); + serial_dout(&com_port->dll, baud_divisor & 0xff); + serial_dout(&com_port->dlm, (baud_divisor >> 8) & 0xff); + serial_dout(&com_port->lcr, UART_LCRVAL); +} + +static inline int NS16550_read_baud_divisor(struct NS16550 *com_port) +{ + int ret; + + serial_dout(&com_port->lcr, UART_LCR_BKSE | UART_LCRVAL); + ret = serial_din(&com_port->dll) & 0xff; + ret |= (serial_din(&com_port->dlm) & 0xff) << 8; + serial_dout(&com_port->lcr, UART_LCRVAL); + + return ret; +} + +static inline void _debug_uart_putc(int ch) +{ + struct NS16550 *com_port = (struct NS16550 *)CONFIG_DEBUG_UART_BASE; + + while (!(serial_din(&com_port->lsr) & UART_LSR_THRE)) { +#ifdef CONFIG_DEBUG_UART_NS16550_CHECK_ENABLED + if (!NS16550_read_baud_divisor(com_port)) + return; +#endif + } + serial_dout(&com_port->thr, ch); +} +#if 0 +static inline void _debug_uart_init(void) +{ + struct ns16550_platdata plat; + + plat.base = CONFIG_DEBUG_UART_BASE; + plat.reg_shift = 1 << CONFIG_DEBUG_UART_SHIFT; + plat.reg_width = 1; + plat.reg_offset = 0; + plat.clock = CONFIG_DEBUG_UART_CLOCK; + plat.fcr = UART_FCR_DEFVAL; + plat.flags = 0; + ns16550_tiny_probe_plat(&plat); + ns16550_tiny_setbrg(&plat, CONFIG_BAUDRATE); +} + +static inline void _debug_uart_putc(int ch) +{ + struct ns16550_platdata plat; + + plat.base = CONFIG_DEBUG_UART_BASE; + plat.reg_shift = 1 << CONFIG_DEBUG_UART_SHIFT; + plat.reg_width = 1; + plat.reg_offset = 0; + plat.clock = CONFIG_DEBUG_UART_CLOCK; + plat.fcr = UART_FCR_DEFVAL; + plat.flags = 0; + while (!(serial_in_reg(&plat, ns16550_reg(lsr)) & UART_LSR_THRE)) + ; + serial_out_reg(&plat, ns16550_reg(thr), ch); +} +#endif + +DEBUG_UART_FUNCS + +#endif + +#endif /* TINY_SERIAL */ diff --git a/drivers/serial/sandbox.c b/drivers/serial/sandbox.c index bae9ed5178..68d808c559 100644 --- a/drivers/serial/sandbox.c +++ b/drivers/serial/sandbox.c @@ -19,6 +19,8 @@ #include #include +#if !CONFIG_IS_ENABLED(TINY_SERIAL) + DECLARE_GLOBAL_DATA_PTR; struct sandbox_serial_platdata { @@ -133,23 +135,6 @@ static int sandbox_serial_getc(struct udevice *dev) return membuff_getbyte(&priv->buf); } -#ifdef CONFIG_DEBUG_UART_SANDBOX - -#include - -static inline void _debug_uart_init(void) -{ -} - -static inline void _debug_uart_putc(int ch) -{ - os_putc(ch); -} - -DEBUG_UART_FUNCS - -#endif /* CONFIG_DEBUG_UART_SANDBOX */ - static int sandbox_serial_getconfig(struct udevice *dev, uint *serial_config) { uint config = SERIAL_DEFAULT_CONFIG; @@ -258,3 +243,43 @@ U_BOOT_DEVICE(serial_sandbox_non_fdt) = { .name = "sandbox_serial", .platdata = &platdata_non_fdt, }; + +#else /* TINY_SERIAL */ + +static int sandbox_serial_tiny_putc(struct tinydev *tdev, const char ch) +{ + os_putc(ch); + + return 0; +} + +struct tiny_serial_ops sandbox_serial_tiny_ops = { + .probe = sandbox_serial_tiny_probe, + .setbrg = sandbox_serial_tiny_setbrg, + .putc = sandbox_serial_tiny_putc, +}; + +U_BOOT_TINY_DRIVER(sandbox_serial) = { + .uclass_id = UCLASS_SERIAL, + .probe = sandbox_serial_tiny_probe, + .ops = &sandbox_serial_tiny_ops, +}; + +#endif + +#ifdef CONFIG_DEBUG_UART_SANDBOX + +#include + +static inline void _debug_uart_init(void) +{ +} + +static inline void _debug_uart_putc(int ch) +{ + os_putc(ch); +} + +DEBUG_UART_FUNCS + +#endif /* CONFIG_DEBUG_UART_SANDBOX */ diff --git a/drivers/serial/serial-uclass.c b/drivers/serial/serial-uclass.c index ed25d5cec8..32be645775 100644 --- a/drivers/serial/serial-uclass.c +++ b/drivers/serial/serial-uclass.c @@ -4,9 +4,11 @@ */ #include +#include #include #include #include +#include #include #include #include @@ -19,6 +21,8 @@ DECLARE_GLOBAL_DATA_PTR; +#if !CONFIG_IS_ENABLED(TINY_SERIAL) + /* * Table with supported baudrates (defined in config_xyz.h) */ @@ -507,3 +511,76 @@ UCLASS_DRIVER(serial) = { .per_device_auto_alloc_size = sizeof(struct serial_dev_priv), }; #endif + +#else /* TINY_SERIAL */ + +int serial_init(void) +{ + struct tinydev *tdev; + int ret; + + tdev = tiny_dev_find(UCLASS_SERIAL, 0); + if (!tdev) { + if (IS_ENABLED(CONFIG_REQUIRE_SERIAL_CONSOLE)) + panic_str("No serial"); + return -ENODEV; + } + ret = tiny_dev_probe(tdev); + if (ret) + return log_msg_ret("probe", ret); + ret = tiny_serial_setbrg(tdev, gd->baudrate); + if (ret) + return log_msg_ret("brg", ret); + gd->tiny_serial = tdev; + gd->flags |= GD_FLG_SERIAL_READY; + + return 0; +} + +int serial_getc(void) +{ + return -ENOSYS; +} + +void serial_putc(const char ch) +{ + struct tinydev *tdev = gd->tiny_serial; + struct tiny_serial_ops *ops; + + /* + * If we don't have a serial port or the driver is broken, try the + * debug UART. This helps with debugging the serial driver in early + * bringup, and adds very little to code size. + */ + if (!tdev) + goto err; + ops = tiny_serial_get_ops(tdev); + if (IS_ENABLED(CONFIG_TINY_CHECK) && !ops->putc) + goto err; + if (ch == '\n') + ops->putc(tdev, '\r'); + ops->putc(tdev, ch); + + return; +err: + if (IS_ENABLED(DEBUG_UART)) + printch(ch); +} + +void serial_puts(const char *str) +{ + for (const char *s = str; *s; s++) + serial_putc(*s); +} + +int tiny_serial_setbrg(struct tinydev *tdev, int baudrate) +{ + const struct tiny_serial_ops *ops = tiny_serial_get_ops(tdev); + + if (IS_ENABLED(CONFIG_TINY_CHECK) && !ops) + return -ENOSYS; + + return ops->setbrg(tdev, baudrate); +} + +#endif diff --git a/drivers/serial/serial_omap.c b/drivers/serial/serial_omap.c index ee7a18e5c5..d98e52832e 100644 --- a/drivers/serial/serial_omap.c +++ b/drivers/serial/serial_omap.c @@ -69,7 +69,7 @@ static inline void _debug_uart_init(void) struct NS16550 *com_port = (struct NS16550 *)CONFIG_DEBUG_UART_BASE; int baud_divisor; - baud_divisor = ns16550_calc_divisor(com_port, CONFIG_DEBUG_UART_CLOCK, + baud_divisor = ns16550_calc_divisor(CONFIG_DEBUG_UART_CLOCK, CONFIG_BAUDRATE); serial_dout(&com_port->ier, CONFIG_SYS_NS16550_IER); serial_dout(&com_port->mdr1, 0x7); diff --git a/drivers/serial/serial_rockchip.c b/drivers/serial/serial_rockchip.c index b1718f72d1..b47ff7142c 100644 --- a/drivers/serial/serial_rockchip.c +++ b/drivers/serial/serial_rockchip.c @@ -7,6 +7,7 @@ #include #include #include +#include #include #include #include @@ -25,6 +26,7 @@ struct rockchip_uart_platdata { struct dtd_rockchip_rk3288_uart *dtplat, s_dtplat; #endif +#if !CONFIG_IS_ENABLED(TINY_SERIAL) static int rockchip_serial_probe(struct udevice *dev) { struct rockchip_uart_platdata *plat = dev_get_platdata(dev); @@ -49,12 +51,69 @@ U_BOOT_DRIVER(rockchip_rk3188_uart) = { .flags = DM_FLAG_PRE_RELOC, }; +static const struct udevice_id rockchip_serial_ids[] = { + { .compatible = "rockchip,rk3288-uart" }, + { }, +}; + U_BOOT_DRIVER(rockchip_rk3288_uart) = { .name = "rockchip_rk3288_uart", .id = UCLASS_SERIAL, + .of_match = rockchip_serial_ids, .priv_auto_alloc_size = sizeof(struct NS16550), .platdata_auto_alloc_size = sizeof(struct rockchip_uart_platdata), .probe = rockchip_serial_probe, .ops = &ns16550_serial_ops, .flags = DM_FLAG_PRE_RELOC, }; +#else /* TINY_SERIAL */ + +static int rockchip_serial_tiny_probe(struct tinydev *tdev) +{ + struct dtd_rockchip_rk3288_uart *dtplat = tdev->dtplat; + struct ns16550_platdata *plat = tdev->priv; + int ret; + + /* Create some new platform data for the standard driver */ + plat->base = dtplat->reg[0]; + plat->reg_shift = dtplat->reg_shift; + plat->reg_width = dtplat->reg_io_width; + plat->clock = dtplat->clock_frequency; + plat->fcr = UART_FCR_DEFVAL; + + log_debug("plat=%p, base=%lx, offset=%x, width=%x, shift=%x, clock=%d, flags=%x\n", + plat, plat->base, plat->reg_offset, plat->reg_width, + plat->reg_shift, plat->clock, plat->flags); + ret = ns16550_tiny_probe_plat(plat); + if (ret) + return log_ret(ret); + + return 0; +} + +static int rockchip_serial_tiny_setbrg(struct tinydev *tdev, int baudrate) +{ + struct ns16550_platdata *plat = tdev->priv; + + return ns16550_tiny_setbrg(plat, baudrate); +} + +static int rockchip_serial_tiny_putc(struct tinydev *tdev, const char ch) +{ + struct ns16550_platdata *plat = tdev->priv; + + return ns16550_tiny_putc(plat, ch); +} + +struct tiny_serial_ops rockchip_serial_tiny_ops = { + .setbrg = rockchip_serial_tiny_setbrg, + .putc = rockchip_serial_tiny_putc, +}; + +U_BOOT_TINY_DRIVER(rockchip_rk3288_uart) = { + .uclass_id = UCLASS_SERIAL, + .probe = rockchip_serial_tiny_probe, + .ops = &rockchip_serial_tiny_ops, + DM_TINY_PRIV(, sizeof(struct ns16550_platdata)) +}; +#endif diff --git a/drivers/spi/Kconfig b/drivers/spi/Kconfig index 09b9cb17d8..6fffbde710 100644 --- a/drivers/spi/Kconfig +++ b/drivers/spi/Kconfig @@ -42,6 +42,24 @@ config SPI_MEM if DM_SPI +config SPL_TINY_SPI + bool "Support tiny SPI drivers in SPL" + depends on SPL_TINY + default y if SPL_TINY_ONLY + help + In constrained environments the driver-model overhead of several KB + of code and data structures can be problematic. Enable this to use a + tiny implementation that only supports a single driver. + +config TPL_TINY_SPI + bool "Support tiny SPI drivers in TPL" + depends on TPL_TINY + default y if TPL_TINY_ONLY + help + In constrained environments the driver-model overhead of several KB + of code and data structures can be problematic. Enable this to use a + tiny implementation that only supports a single driver. + config ALTERA_SPI bool "Altera SPI driver" help diff --git a/drivers/spi/Makefile b/drivers/spi/Makefile index 4e7461771f..332c6f3d5c 100644 --- a/drivers/spi/Makefile +++ b/drivers/spi/Makefile @@ -9,7 +9,9 @@ obj-y += spi-uclass.o obj-$(CONFIG_CADENCE_QSPI) += cadence_qspi.o cadence_qspi_apb.o obj-$(CONFIG_SANDBOX) += spi-emul-uclass.o obj-$(CONFIG_SOFT_SPI) += soft_spi.o +ifeq ($(CONFIG_$(SPL_TPL_)TINY_SPI),) obj-$(CONFIG_SPI_MEM) += spi-mem.o +endif obj-$(CONFIG_TI_QSPI) += ti_qspi.o else obj-y += spi.o diff --git a/drivers/spi/rk_spi.c b/drivers/spi/rk_spi.c index 6cadeac56a..cf5b9ecfc4 100644 --- a/drivers/spi/rk_spi.c +++ b/drivers/spi/rk_spi.c @@ -10,6 +10,8 @@ * Peter, Software Engineering, . */ +#define LOG_CATEGORY UCLASS_SPI + #include #include #include @@ -97,7 +99,7 @@ static void rkspi_set_clk(struct rockchip_spi_priv *priv, uint speed) /* Round up to the next even 16bit number */ clk_div = (clk_div + 1) & 0xfffe; - debug("spi speed %u, div %u\n", speed, clk_div); + log_debug("spi speed %u, div %u\n", speed, clk_div); clrsetbits_le32(&priv->regs->baudr, 0xffff, clk_div); priv->last_speed_hz = speed; @@ -118,10 +120,8 @@ static int rkspi_wait_till_not_busy(struct rockchip_spi *regs) return 0; } -static void spi_cs_activate(struct udevice *dev, uint cs) +static void spi_cs_activate_bus(struct rockchip_spi_priv *priv, uint cs) { - struct udevice *bus = dev->parent; - struct rockchip_spi_priv *priv = dev_get_priv(bus); struct rockchip_spi *regs = priv->regs; /* If it's too soon to do another transaction, wait */ @@ -143,10 +143,8 @@ static void spi_cs_activate(struct udevice *dev, uint cs) udelay(priv->activate_delay_us); } -static void spi_cs_deactivate(struct udevice *dev, uint cs) +static void spi_cs_deactivate_bus(struct rockchip_spi_priv *priv, uint cs) { - struct udevice *bus = dev->parent; - struct rockchip_spi_priv *priv = dev_get_priv(bus); struct rockchip_spi *regs = priv->regs; debug("deactivate cs%u\n", cs); @@ -157,54 +155,6 @@ static void spi_cs_deactivate(struct udevice *dev, uint cs) priv->last_transaction_us = timer_get_us(); } -#if CONFIG_IS_ENABLED(OF_PLATDATA) -static int conv_of_platdata(struct udevice *dev) -{ - struct rockchip_spi_platdata *plat = dev->platdata; - struct dtd_rockchip_rk3288_spi *dtplat = &plat->of_plat; - struct rockchip_spi_priv *priv = dev_get_priv(dev); - int ret; - - priv->base = dtplat->reg[0]; - priv->frequency = 20000000; - ret = clk_get_by_driver_info(dev, dtplat->clocks, &priv->clk); - if (ret < 0) - return ret; - dev->req_seq = 0; - - return 0; -} -#endif - -static int rockchip_spi_ofdata_to_platdata(struct udevice *bus) -{ -#if !CONFIG_IS_ENABLED(OF_PLATDATA) - struct rockchip_spi_priv *priv = dev_get_priv(bus); - int ret; - - priv->base = dev_read_addr(bus); - - ret = clk_get_by_index(bus, 0, &priv->clk); - if (ret < 0) { - debug("%s: Could not get clock for %s: %d\n", __func__, - bus->name, ret); - return ret; - } - - priv->frequency = - dev_read_u32_default(bus, "spi-max-frequency", 50000000); - priv->deactivate_delay_us = - dev_read_u32_default(bus, "spi-deactivate-delay", 0); - priv->activate_delay_us = - dev_read_u32_default(bus, "spi-activate-delay", 0); - - debug("%s: base=%x, max-frequency=%d, deactivate_delay=%d\n", - __func__, (uint)priv->base, priv->frequency, - priv->deactivate_delay_us); -#endif - - return 0; -} static int rockchip_spi_calc_modclk(ulong max_freq) { @@ -234,17 +184,10 @@ static int rockchip_spi_calc_modclk(ulong max_freq) return gpll_hz / div; } -static int rockchip_spi_probe(struct udevice *bus) +static int rockchip_spi_probe_(struct rockchip_spi_priv *priv) { - struct rockchip_spi_priv *priv = dev_get_priv(bus); - int ret; + int ret, rate; - debug("%s: probe\n", __func__); -#if CONFIG_IS_ENABLED(OF_PLATDATA) - ret = conv_of_platdata(bus); - if (ret) - return ret; -#endif priv->regs = (struct rockchip_spi *)priv->base; priv->last_transaction_us = timer_get_us(); @@ -255,24 +198,30 @@ static int rockchip_spi_probe(struct udevice *bus) priv->max_freq = ROCKCHIP_SPI_MAX_RATE; /* Find a module-input clock that fits with the max_freq setting */ - ret = clk_set_rate(&priv->clk, - rockchip_spi_calc_modclk(priv->max_freq)); + log_debug("priv->max_freq=%d, modclk=%d\n", priv->max_freq, + rockchip_spi_calc_modclk(priv->max_freq)); + rate = rockchip_spi_calc_modclk(priv->max_freq); + if (!CONFIG_IS_ENABLED(TINY_SPI)) { + log_debug("clk=%s, id=%ld\n", priv->clk.dev->name, + priv->clk.id); + ret = clk_set_rate(&priv->clk, rate); + } else { + log_debug("clk=%s, id=%ld\n", priv->tiny_clk.tdev->name, + priv->tiny_clk.id); + ret = tiny_clk_set_rate(&priv->tiny_clk, rate); + } if (ret < 0) { debug("%s: Failed to set clock: %d\n", __func__, ret); - return ret; + return log_ret(ret); } priv->input_rate = ret; - if (dev_get_driver_data(bus) == RK_SPI_RK33XX) - priv->master_manages_fifo = true; debug("%s: rate = %u\n", __func__, priv->input_rate); return 0; } -static int rockchip_spi_claim_bus(struct udevice *dev) +static int rockchip_spi_claim_bus_(struct rockchip_spi_priv *priv) { - struct udevice *bus = dev->parent; - struct rockchip_spi_priv *priv = dev_get_priv(bus); struct rockchip_spi *regs = priv->regs; uint ctrlr0; @@ -323,21 +272,16 @@ static int rockchip_spi_claim_bus(struct udevice *dev) return 0; } -static int rockchip_spi_release_bus(struct udevice *dev) +static int rockchip_spi_release_bus_(struct rockchip_spi_priv *priv) { - struct udevice *bus = dev->parent; - struct rockchip_spi_priv *priv = dev_get_priv(bus); - rkspi_enable_chip(priv->regs, false); return 0; } -static inline int rockchip_spi_16bit_reader(struct udevice *dev, - u8 **din, int *len) +static int rockchip_spi_16bit_reader(struct rockchip_spi_priv *priv, u8 **din, + int *len) { - struct udevice *bus = dev->parent; - struct rockchip_spi_priv *priv = dev_get_priv(bus); struct rockchip_spi *regs = priv->regs; const u32 saved_ctrlr0 = readl(®s->ctrlr0); #if defined(DEBUG) @@ -359,7 +303,6 @@ static inline int rockchip_spi_16bit_reader(struct udevice *dev, if (priv->master_manages_fifo) max_chunk_size = ROCKCHIP_SPI_MAX_TRANLEN; - // rockchip_spi_configure(dev, mode, size) rkspi_enable_chip(regs, false); clrsetbits_le32(®s->ctrlr0, TMOD_MASK << TMOD_SHIFT, @@ -376,6 +319,7 @@ static inline int rockchip_spi_16bit_reader(struct udevice *dev, while (frames) { u32 chunk_size = min(frames, max_chunk_size); + log_debug("frames=%u\n", frames); frames -= chunk_size; writew(chunk_size - 1, ®s->ctrlr1); @@ -392,6 +336,7 @@ static inline int rockchip_spi_16bit_reader(struct udevice *dev, *in++ = val & 0xff; *in++ = val >> 8; } + log_debug("chunk_size=%u\n", chunk_size); } while (chunk_size); rkspi_enable_chip(regs, false); @@ -405,16 +350,14 @@ static inline int rockchip_spi_16bit_reader(struct udevice *dev, #endif /* Restore the original transfer setup and return error-free. */ writel(saved_ctrlr0, ®s->ctrlr0); + return 0; } -static int rockchip_spi_xfer(struct udevice *dev, unsigned int bitlen, - const void *dout, void *din, unsigned long flags) +static int rockchip_spi_xfer_(struct rockchip_spi_priv *priv, uint bitlen, + const void *dout, void *din, ulong flags, uint cs) { - struct udevice *bus = dev->parent; - struct rockchip_spi_priv *priv = dev_get_priv(bus); struct rockchip_spi *regs = priv->regs; - struct dm_spi_slave_platdata *slave_plat = dev_get_parent_platdata(dev); int len = bitlen >> 3; const u8 *out = dout; u8 *in = din; @@ -428,7 +371,7 @@ static int rockchip_spi_xfer(struct udevice *dev, unsigned int bitlen, /* Assert CS before transfer */ if (flags & SPI_XFER_BEGIN) - spi_cs_activate(dev, slave_plat->cs); + spi_cs_activate_bus(priv, cs); /* * To ensure fast loading of firmware images (e.g. full U-Boot @@ -437,12 +380,13 @@ static int rockchip_spi_xfer(struct udevice *dev, unsigned int bitlen, * FIFO element. */ if (!out) - ret = rockchip_spi_16bit_reader(dev, &in, &len); + ret = rockchip_spi_16bit_reader(priv, &in, &len); /* This is the original 8bit reader/writer code */ while (len > 0) { int todo = min(len, ROCKCHIP_SPI_MAX_TRANLEN); + log_debug("todo=%d\n", todo); rkspi_enable_chip(regs, false); writel(todo - 1, ®s->ctrlr1); rkspi_enable_chip(regs, true); @@ -481,13 +425,43 @@ static int rockchip_spi_xfer(struct udevice *dev, unsigned int bitlen, /* Deassert CS after transfer */ if (flags & SPI_XFER_END) - spi_cs_deactivate(dev, slave_plat->cs); + spi_cs_deactivate_bus(priv, cs); rkspi_enable_chip(regs, false); return ret; } +#if !CONFIG_IS_ENABLED(TINY_SPI) +static int rockchip_spi_claim_bus(struct udevice *dev) +{ + struct udevice *bus = dev_get_parent(dev); + struct rockchip_spi_priv *priv = dev_get_priv(bus); + + return rockchip_spi_claim_bus_(priv); +} + +static int rockchip_spi_release_bus(struct udevice *dev) +{ + struct udevice *bus = dev_get_parent(dev); + struct rockchip_spi_priv *priv = dev_get_priv(bus); + + rockchip_spi_release_bus_(priv); + + return 0; +} + +static int rockchip_spi_xfer(struct udevice *dev, uint bitlen, + const void *dout, void *din, ulong flags) +{ + struct dm_spi_slave_platdata *slave_plat = dev_get_parent_platdata(dev); + struct udevice *bus = dev_get_parent(dev); + struct rockchip_spi_priv *priv = dev_get_priv(bus); + + return rockchip_spi_xfer_(priv, bitlen, dout, din, flags, + slave_plat->cs); +} + static int rockchip_spi_set_speed(struct udevice *bus, uint speed) { struct rockchip_spi_priv *priv = dev_get_priv(bus); @@ -510,6 +484,72 @@ static int rockchip_spi_set_mode(struct udevice *bus, uint mode) return 0; } +static int conv_of_platdata(struct udevice *dev) +{ +#if CONFIG_IS_ENABLED(OF_PLATDATA) + struct rockchip_spi_platdata *plat = dev->platdata; + struct dtd_rockchip_rk3288_spi *dtplat = &plat->of_plat; + struct rockchip_spi_priv *priv = dev_get_priv(dev); + int ret; + + priv->base = dtplat->reg[0]; + priv->frequency = 20000000; + ret = clk_get_by_driver_info(dev, dtplat->clocks, &priv->clk); + if (ret < 0) + return log_ret(ret); + dev->req_seq = 0; +#endif + + return 0; +} + +static int rockchip_spi_probe(struct udevice *bus) +{ + struct rockchip_spi_priv *priv = dev_get_priv(bus); + int ret; + + debug("%s: probe\n", __func__); + if (CONFIG_IS_ENABLED(OF_PLATDATA)) { + ret = conv_of_platdata(bus); + if (ret) + return log_ret(ret); + } + if (dev_get_driver_data(bus) == RK_SPI_RK33XX) + priv->master_manages_fifo = true; + + return rockchip_spi_probe_(priv); +} + +static int rockchip_spi_ofdata_to_platdata(struct udevice *bus) +{ +#if !CONFIG_IS_ENABLED(OF_PLATDATA) + struct rockchip_spi_priv *priv = dev_get_priv(bus); + int ret; + + priv->base = dev_read_addr(bus); + + ret = clk_get_by_index(bus, 0, &priv->clk); + if (ret < 0) { + debug("%s: Could not get clock for %s: %d\n", __func__, + bus->name, ret); + return ret; + } + + priv->frequency = + dev_read_u32_default(bus, "spi-max-frequency", 50000000); + priv->deactivate_delay_us = + dev_read_u32_default(bus, "spi-deactivate-delay", 0); + priv->activate_delay_us = + dev_read_u32_default(bus, "spi-activate-delay", 0); + + debug("%s: base=%x, max-frequency=%d, deactivate_delay=%d\n", + __func__, (uint)priv->base, priv->frequency, + priv->deactivate_delay_us); +#endif + + return 0; +} + static const struct dm_spi_ops rockchip_spi_ops = { .claim_bus = rockchip_spi_claim_bus, .release_bus = rockchip_spi_release_bus, @@ -542,4 +582,87 @@ U_BOOT_DRIVER(rockchip_rk3288_spi) = { .probe = rockchip_spi_probe, }; +#else /* TINY_SPI */ +static int rockchip_tiny_spi_claim_bus(struct tinydev *tdev) +{ + struct tinydev *tbus = tinydev_get_parent(tdev); + struct rockchip_spi_priv *priv = tinydev_get_priv(tbus); + + return rockchip_spi_claim_bus_(priv); +} + +static int rockchip_tiny_spi_release_bus(struct tinydev *tdev) +{ + struct tinydev *tbus = tinydev_get_parent(tdev); + struct rockchip_spi_priv *priv = tinydev_get_priv(tbus); + + rockchip_spi_release_bus_(priv); + + return 0; +} + +static int rockchip_tiny_set_speed_mode(struct tinydev *tbus, uint speed_hz, + uint mode) +{ + struct rockchip_spi_priv *priv = tinydev_get_priv(tbus); + + /* Clamp to the maximum frequency specified in the DTS */ + if (speed_hz > priv->max_freq) + speed_hz = priv->max_freq; + + priv->speed_hz = speed_hz; + priv->mode = mode; + + return 0; +} + +static int rockchip_tiny_spi_xfer(struct tinydev *tdev, uint bitlen, + const void *dout, void *din, ulong flags) +{ + log_debug("xfer\n"); + struct tinydev *tbus = tinydev_get_parent(tdev); + struct rockchip_spi_priv *priv = tinydev_get_priv(tbus); + struct dm_spi_slave_platdata *slave_plat; + + slave_plat = tinydev_get_data(tdev, DEVDATAT_PARENT_PLAT); + log_debug("priv=%p, slave_plat=%p, cs=%d\n", priv, slave_plat, + slave_plat->cs); + + return rockchip_spi_xfer_(priv, bitlen, dout, din, flags, + slave_plat->cs); +} + +static int rockchip_spi_tiny_probe(struct tinydev *tdev) +{ + log_debug("start\n"); + struct rockchip_spi_priv *priv = tinydev_get_priv(tdev); + struct dtd_rockchip_rk3288_spi *dtplat = tdev->dtplat; + int ret; + + priv->base = dtplat->reg[0]; + priv->frequency = 20000000; + ret = tiny_clk_get_by_driver_info(dtplat->clocks, &priv->tiny_clk); + if (ret < 0) + return log_ret(ret); + log_debug("priv->base=%lx\n", priv->base); + + return rockchip_spi_probe_(priv); +} + +static struct tiny_spi_ops rockchip_spi_tiny_ops = { + .claim_bus = rockchip_tiny_spi_claim_bus, + .release_bus = rockchip_tiny_spi_release_bus, + .set_speed_mode = rockchip_tiny_set_speed_mode, + .xfer = rockchip_tiny_spi_xfer, +}; + +U_BOOT_TINY_DRIVER(rockchip_rk3288_spi) = { + .uclass_id = UCLASS_SPI, + .probe = rockchip_spi_tiny_probe, + .ops = &rockchip_spi_tiny_ops, + DM_TINY_PRIV(, \ + sizeof(struct rockchip_spi_priv)) +}; +#endif + U_BOOT_DRIVER_ALIAS(rockchip_rk3288_spi, rockchip_rk3368_spi) diff --git a/drivers/spi/spi-uclass.c b/drivers/spi/spi-uclass.c index cffd9cf0b0..1d143a417d 100644 --- a/drivers/spi/spi-uclass.c +++ b/drivers/spi/spi-uclass.c @@ -3,6 +3,8 @@ * Copyright (c) 2014 Google, Inc */ +#define LOG_CATEGORY UCLASS_SPI + #include #include #include @@ -18,6 +20,8 @@ DECLARE_GLOBAL_DATA_PTR; #define SPI_DEFAULT_SPEED_HZ 100000 +#if !CONFIG_IS_ENABLED(TINY_SPI) + static int spi_set_speed_mode(struct udevice *bus, int speed, int mode) { struct dm_spi_ops *ops; @@ -520,3 +524,76 @@ U_BOOT_DRIVER(spi_generic_drv) = { .name = "spi_generic_drv", .id = UCLASS_SPI_GENERIC, }; +#else /* TINY_SPI */ +int tiny_spi_claim_bus(struct tinydev *tdev) +{ + log_debug("claim\n"); + struct tinydev *bus = tinydev_get_parent(tdev); + struct tiny_spi_ops *ops = tiny_spi_get_ops(bus); + struct spi_slave *slave = tinydev_get_data(tdev, DEVDATAT_PARENT_PRIV); + int speed = 0; + int ret; + + log_debug("bus=%s\n", bus->name); + log_debug("slave=%p\n", slave); + speed = slave->max_hz; + if (!speed) + speed = SPI_DEFAULT_SPEED_HZ; + log_debug("speed=%d\n", speed); + if (speed != slave->speed) { + int ret = tiny_spi_set_speed_mode(bus, speed, slave->mode); + + if (ret) + return log_msg_ret("speed", ret); + slave->speed = speed; + } + + if (ops->claim_bus) { + ret = ops->claim_bus(tdev); + if (ret) + return log_msg_ret("claim", ret); + } + + return 0; +} + +int tiny_spi_release_bus(struct tinydev *tdev) +{ + log_debug("release\n"); + struct tinydev *bus = tinydev_get_parent(tdev); + struct tiny_spi_ops *ops = tiny_spi_get_ops(bus); + int ret; + + if (ops->release_bus) { + ret = ops->release_bus(tdev); + if (ret) + return log_ret(ret); + } + + return 0; +} + +int tiny_spi_xfer(struct tinydev *tdev, unsigned int bitlen, + const void *dout, void *din, unsigned long flags) +{ + log_debug("xfer\n"); + struct tinydev *bus = tinydev_get_parent(tdev); + struct tiny_spi_ops *ops = tiny_spi_get_ops(bus); + + if (!ops->xfer) + return -ENOSYS; + + return ops->xfer(tdev, bitlen, dout, din, flags); +} + +int tiny_spi_set_speed_mode(struct tinydev *bus, uint hz, uint mode) +{ + struct tiny_spi_ops *ops = tiny_spi_get_ops(bus); + + if (!ops->set_speed_mode) + return -ENOSYS; + + return ops->set_speed_mode(bus, hz, mode); +} + +#endif /* TINY_SPI */ diff --git a/drivers/sysreset/Kconfig b/drivers/sysreset/Kconfig index 4be7433404..7b566dc2b4 100644 --- a/drivers/sysreset/Kconfig +++ b/drivers/sysreset/Kconfig @@ -22,6 +22,15 @@ config SPL_SYSRESET to effect a reset. The uclass will try all available drivers when reset_walk() is called. +config SPL_TINY_SYSRESET + bool "Support tiny sysreset drivers in SPL" + depends on SPL_TINY + default y if SPL_TINY_ONLY + help + In constrained environments the driver-model overhead of several KB + of code and data structures can be problematic. Enable this to use a + tiny implementation that only supports a single driver. + config TPL_SYSRESET bool "Enable support for system reset drivers in TPL mode" depends on SYSRESET && TPL_DM @@ -31,6 +40,15 @@ config TPL_SYSRESET to effect a reset. The uclass will try all available drivers when reset_walk() is called. +config TPL_TINY_SYSRESET + bool "Support tiny sysreset drivers in TPL" + depends on TPL_TINY + default y if TPL_TINY_ONLY + help + In constrained environments the driver-model overhead of several KB + of code and data structures can be problematic. Enable this to use a + tiny implementation that only supports a single driver. + if SYSRESET if CMD_POWEROFF diff --git a/drivers/sysreset/sysreset-uclass.c b/drivers/sysreset/sysreset-uclass.c index 995240f0cb..73561dce3e 100644 --- a/drivers/sysreset/sysreset-uclass.c +++ b/drivers/sysreset/sysreset-uclass.c @@ -21,6 +21,7 @@ #include #include +#if !CONFIG_IS_ENABLED(TINY_SYSRESET) int sysreset_request(struct udevice *dev, enum sysreset_t type) { struct sysreset_ops *ops = sysreset_get_ops(dev); @@ -51,25 +52,6 @@ int sysreset_get_last(struct udevice *dev) return ops->get_last(dev); } -int sysreset_walk(enum sysreset_t type) -{ - struct udevice *dev; - int ret = -ENOSYS; - - while (ret != -EINPROGRESS && type < SYSRESET_COUNT) { - for (uclass_first_device(UCLASS_SYSRESET, &dev); - dev; - uclass_next_device(&dev)) { - ret = sysreset_request(dev, type); - if (ret == -EINPROGRESS) - break; - } - type++; - } - - return ret; -} - int sysreset_get_last_walk(void) { struct udevice *dev; @@ -90,39 +72,6 @@ int sysreset_get_last_walk(void) return value; } -void sysreset_walk_halt(enum sysreset_t type) -{ - int ret; - - ret = sysreset_walk(type); - - /* Wait for the reset to take effect */ - if (ret == -EINPROGRESS) - mdelay(100); - - /* Still no reset? Give up */ - log_err("System reset not supported on this platform\n"); - hang(); -} - -/** - * reset_cpu() - calls sysreset_walk(SYSRESET_WARM) - */ -void reset_cpu(ulong addr) -{ - sysreset_walk_halt(SYSRESET_WARM); -} - - -int do_reset(struct cmd_tbl *cmdtp, int flag, int argc, char *const argv[]) -{ - printf("resetting ...\n"); - - sysreset_walk_halt(SYSRESET_COLD); - - return 0; -} - #if IS_ENABLED(CONFIG_SYSRESET_CMD_POWEROFF) int do_poweroff(struct cmd_tbl *cmdtp, int flag, int argc, char *const argv[]) { @@ -161,3 +110,74 @@ UCLASS_DRIVER(sysreset) = { .name = "sysreset", .post_bind = sysreset_post_bind, }; +#else +int tiny_sysreset_request(struct tinydev *tdev, enum sysreset_t type) +{ + struct tiny_sysreset_ops *ops = tiny_sysreset_get_ops(tdev); + + if (!ops->request) + return -ENOSYS; + + return ops->request(tdev, type); +} +#endif + +int sysreset_walk(enum sysreset_t type) +{ + int ret = -ENOSYS; + + while (ret != -EINPROGRESS && type < SYSRESET_COUNT) { + if (!CONFIG_IS_ENABLED(TINY_SYSRESET)) { + struct udevice *dev; + + for (uclass_first_device(UCLASS_SYSRESET, &dev); + dev; + uclass_next_device(&dev)) { + ret = sysreset_request(dev, type); + if (ret == -EINPROGRESS) + break; + } + } else { + struct tinydev *tdev; + + tdev = tiny_dev_get(UCLASS_SYSRESET, 0); + if (tdev) + ret = tiny_sysreset_request(tdev, type); + } + type++; + } + + return ret; +} + +void sysreset_walk_halt(enum sysreset_t type) +{ + int ret; + + ret = sysreset_walk(type); + + /* Wait for the reset to take effect */ + if (ret == -EINPROGRESS) + mdelay(100); + + /* Still no reset? Give up */ + log_err("System reset not supported on this platform\n"); + hang(); +} + +/** + * reset_cpu() - calls sysreset_walk(SYSRESET_WARM) + */ +void reset_cpu(ulong addr) +{ + sysreset_walk_halt(SYSRESET_WARM); +} + +int do_reset(struct cmd_tbl *cmdtp, int flag, int argc, char *const argv[]) +{ + printf("resetting ...\n"); + + sysreset_walk_halt(SYSRESET_COLD); + + return 0; +} diff --git a/drivers/sysreset/sysreset_rockchip.c b/drivers/sysreset/sysreset_rockchip.c index 0fc6b683f2..195ebc229a 100644 --- a/drivers/sysreset/sysreset_rockchip.c +++ b/drivers/sysreset/sysreset_rockchip.c @@ -3,6 +3,8 @@ * (C) Copyright 2017 Rockchip Electronics Co., Ltd */ +#define LOG_CATEGORY UCLASS_SYSRESET + #include #include #include @@ -13,9 +15,9 @@ #include #include -int rockchip_sysreset_request(struct udevice *dev, enum sysreset_t type) +static int rockchip_sysreset_request_(struct sysreset_reg *priv, + enum sysreset_t type) { - struct sysreset_reg *offset = dev_get_priv(dev); unsigned long cru_base = (unsigned long)rockchip_get_cru(); if (IS_ERR_VALUE(cru_base)) @@ -23,10 +25,10 @@ int rockchip_sysreset_request(struct udevice *dev, enum sysreset_t type) switch (type) { case SYSRESET_WARM: - writel(0xeca8, cru_base + offset->glb_srst_snd_value); + writel(0xeca8, cru_base + priv->glb_srst_snd_value); break; case SYSRESET_COLD: - writel(0xfdb9, cru_base + offset->glb_srst_fst_value); + writel(0xfdb9, cru_base + priv->glb_srst_fst_value); break; default: return -EPROTONOSUPPORT; @@ -35,12 +37,57 @@ int rockchip_sysreset_request(struct udevice *dev, enum sysreset_t type) return -EINPROGRESS; } -static struct sysreset_ops rockchip_sysreset = { +#if !CONFIG_IS_ENABLED(TINY_SYSRESET) +int rockchip_sysreset_request(struct udevice *dev, enum sysreset_t type) +{ + struct sysreset_reg *priv = dev_get_priv(dev); + + return rockchip_sysreset_request_(priv, type); +} + +static int rockchip_sysreset_probe(struct udevice *dev) +{ + return rockchip_cru_setup_sysreset(dev); +} + +static struct sysreset_ops rockchip_sysreset_ops = { .request = rockchip_sysreset_request, }; -U_BOOT_DRIVER(sysreset_rockchip) = { +static const struct udevice_id rockchip_sysreset_ids[] = { + { .compatible = "rockchip,sysreset" }, + { } +}; + +U_BOOT_DRIVER(rockchip_sysreset) = { .name = "rockchip_sysreset", .id = UCLASS_SYSRESET, - .ops = &rockchip_sysreset, + .of_match = rockchip_sysreset_ids, + .ops = &rockchip_sysreset_ops, + .probe = rockchip_sysreset_probe, + .priv_auto_alloc_size = sizeof(struct sysreset_reg), +}; +#else +int rockchip_sysreset_tiny_request(struct tinydev *tdev, enum sysreset_t type) +{ + struct sysreset_reg *priv = tinydev_get_priv(tdev); + + return rockchip_sysreset_request_(priv, type); +} + +static int rockchip_sysreset_tiny_probe(struct tinydev *tdev) +{ + return rockchip_cru_setup_tiny_sysreset(tdev); +} + +static struct tiny_sysreset_ops rockchip_sysreset_tiny_ops = { + .request = rockchip_sysreset_tiny_request, +}; + +U_BOOT_TINY_DRIVER(rockchip_sysreset) = { + .uclass_id = UCLASS_SYSRESET, + .probe = rockchip_sysreset_tiny_probe, + .ops = &rockchip_sysreset_tiny_ops, + DM_TINY_PRIV(, sizeof(struct sysreset_reg)) }; +#endif diff --git a/include/asm-generic/global_data.h b/include/asm-generic/global_data.h index 8c78792cc9..d6f6c3080f 100644 --- a/include/asm-generic/global_data.h +++ b/include/asm-generic/global_data.h @@ -23,6 +23,7 @@ #include #include #include +#include typedef struct global_data { struct bd_info *bd; @@ -68,6 +69,9 @@ typedef struct global_data { struct udevice *dm_root_f; /* Pre-relocation root instance */ struct list_head uclass_root; /* Head of core tree */ #endif +#if CONFIG_IS_ENABLED(TINY) + struct tinydev_info tinydev_info; +#endif #ifdef CONFIG_TIMER struct udevice *timer; /* Timer instance for Driver Model */ #endif @@ -95,7 +99,7 @@ typedef struct global_data { #if CONFIG_VAL(SYS_MALLOC_F_LEN) unsigned long malloc_base; /* base address of early malloc() */ unsigned long malloc_limit; /* limit address */ - unsigned long malloc_ptr; /* current address */ + unsigned long malloc_ptr; /* offset of next byte to allocate */ #endif #ifdef CONFIG_PCI struct pci_controller *hose; /* PCI hose for early use */ @@ -105,6 +109,7 @@ typedef struct global_data { int pcidelay_done; #endif struct udevice *cur_serial_dev; /* current serial device */ + struct tinydev *tiny_serial; struct arch_global_data arch; /* architecture-specific data */ #ifdef CONFIG_CONSOLE_RECORD struct membuff console_out; /* console output */ diff --git a/include/clk-uclass.h b/include/clk-uclass.h index dac42dab36..b652975195 100644 --- a/include/clk-uclass.h +++ b/include/clk-uclass.h @@ -100,4 +100,15 @@ struct clk_ops { int (*disable)(struct clk *clk); }; +struct tiny_clk_ops { + /** + * set_rate() - Set current clock rate. + * + * @tclk: The clock to manipulate. + * @rate: New clock rate in Hz. + * @return new rate, or -ve error code. + */ + ulong (*set_rate)(struct tiny_clk *tclk, ulong rate); +}; + #endif diff --git a/include/clk.h b/include/clk.h index a62e2efa2c..f3348a3b9e 100644 --- a/include/clk.h +++ b/include/clk.h @@ -63,9 +63,8 @@ struct clk { long long rate; /* in HZ */ u32 flags; int enable_count; - /* - * Written by of_xlate. In the future, we might add more fields here. - */ + + /* Written by of_xlate. In the future, we might add more fields here */ unsigned long id; unsigned long data; }; @@ -87,11 +86,22 @@ struct clk_bulk { unsigned int count; }; +struct tiny_clk { + struct tinydev *tdev; + long long rate; /* in HZ */ + + /* Written by of_xlate. In the future, we might add more fields here */ + unsigned long id; +}; + #if CONFIG_IS_ENABLED(OF_CONTROL) && CONFIG_IS_ENABLED(CLK) struct phandle_1_arg; int clk_get_by_driver_info(struct udevice *dev, struct phandle_1_arg *cells, struct clk *clk); +int tiny_clk_get_by_driver_info(struct phandle_1_arg *cells, + struct tiny_clk *tclk); + /** * clk_get_by_index - Get/request a clock by integer index. * @@ -455,6 +465,10 @@ int clk_get_by_id(ulong id, struct clk **clkp); */ bool clk_dev_binded(struct clk *clk); +int tiny_clk_request(struct tinydev *tdev, struct tiny_clk *tclk); + +ulong tiny_clk_set_rate(struct tiny_clk *tclk, ulong rate); + #else /* CONFIG_IS_ENABLED(CLK) */ static inline int clk_request(struct udevice *dev, struct clk *clk) @@ -526,6 +540,7 @@ static inline bool clk_dev_binded(struct clk *clk) { return false; } + #endif /* CONFIG_IS_ENABLED(CLK) */ /** @@ -539,6 +554,17 @@ static inline bool clk_valid(struct clk *clk) return clk && !!clk->dev; } +/** + * tiny_clk_valid() - check if clk is valid + * + * @tiny_clk: the clock to check + * @return true if valid, or false + */ +static inline bool tiny_clk_valid(struct tiny_clk *tclk) +{ + return tclk && !!tclk->tdev; +} + int soc_clk_dump(void); #endif diff --git a/include/linux/mtd/mtd.h b/include/linux/mtd/mtd.h index 1b9151714c..4a715a966a 100644 --- a/include/linux/mtd/mtd.h +++ b/include/linux/mtd/mtd.h @@ -332,6 +332,13 @@ struct mtd_info { }; #if IS_ENABLED(CONFIG_DM) +struct tiny_mtd_info { + uint64_t size; // Total size of the MTD +// int (*_read) (struct mtd_info *mtd, loff_t from, size_t len, +// size_t *retlen, u_char *buf); + void *priv; +}; + static inline void mtd_set_of_node(struct mtd_info *mtd, const struct device_node *np) { @@ -354,7 +361,7 @@ static inline const struct device_node *mtd_get_of_node(struct mtd_info *mtd) { return NULL; } -#endif +#endif /* DM */ static inline bool mtd_is_partition(const struct mtd_info *mtd) { @@ -402,7 +409,7 @@ int mtd_erase(struct mtd_info *mtd, struct erase_info *instr); int mtd_point(struct mtd_info *mtd, loff_t from, size_t len, size_t *retlen, void **virt, resource_size_t *phys); int mtd_unpoint(struct mtd_info *mtd, loff_t from, size_t len); -#endif +#endif /* __UBOOT__ */ unsigned long mtd_get_unmapped_area(struct mtd_info *mtd, unsigned long len, unsigned long offset, unsigned long flags); int mtd_read(struct mtd_info *mtd, loff_t from, size_t len, size_t *retlen, @@ -430,7 +437,7 @@ int mtd_lock_user_prot_reg(struct mtd_info *mtd, loff_t from, size_t len); #ifndef __UBOOT__ int mtd_writev(struct mtd_info *mtd, const struct kvec *vecs, unsigned long count, loff_t to, size_t *retlen); -#endif +#endif /* __UBOOT__ */ static inline void mtd_sync(struct mtd_info *mtd) { @@ -456,7 +463,7 @@ static inline void mtd_resume(struct mtd_info *mtd) if (mtd->_resume) mtd->_resume(mtd); } -#endif +#endif /* __UBOOT__ */ static inline uint32_t mtd_div_by_eb(uint64_t sz, struct mtd_info *mtd) { @@ -533,7 +540,7 @@ struct mtd_notifier { extern void register_mtd_user (struct mtd_notifier *new); extern int unregister_mtd_user (struct mtd_notifier *old); -#endif +#endif /* __UBOOT__ */ void *mtd_kmalloc_up_to(const struct mtd_info *mtd, size_t *size); #ifdef CONFIG_MTD_PARTITIONS @@ -544,7 +551,7 @@ static inline void mtd_erase_callback(struct erase_info *instr) if (instr->callback) instr->callback(instr); } -#endif +#endif /* CONFIG_MTD_PARTITIONS */ static inline int mtd_is_bitflip(int err) { return err == -EUCLEAN; @@ -580,7 +587,7 @@ static inline int del_mtd_partitions(struct mtd_info *mtd) { return 0; } -#endif +#endif /* CONFIG_MTD_PARTITIONS */ struct mtd_info *__mtd_next_device(int i); #define mtd_for_each_device(mtd) \ @@ -598,5 +605,5 @@ bool mtd_dev_list_updated(void); int mtd_search_alternate_name(const char *mtdname, char *altname, unsigned int max_len); -#endif +#endif /* __UBOOT__ */ #endif /* __MTD_MTD_H__ */ diff --git a/include/linux/mtd/spi-nor.h b/include/linux/mtd/spi-nor.h index 233fdc341a..81b61e47bc 100644 --- a/include/linux/mtd/spi-nor.h +++ b/include/linux/mtd/spi-nor.h @@ -352,6 +352,24 @@ struct spi_nor { u32 erase_size; }; +struct tiny_spi_nor { + struct tinydev *tdev; /* SPI device */ + struct tiny_mtd_info mtd; + struct spi_slave *spi; + const struct flash_info *info; + u8 addr_width; + u8 read_opcode; + u8 read_dummy; + enum spi_nor_protocol read_proto; + u32 flags; + u8 cmd_buf[SPI_NOR_MAX_CMD_SIZE]; + void *priv; + u32 size; +}; + +int tiny_spi_nor_read(struct tiny_mtd_info *mtd, loff_t from, size_t len, + size_t *retlen, u_char *buf); + static inline void spi_nor_set_flash_node(struct spi_nor *nor, const struct device_node *np) { @@ -435,6 +453,10 @@ struct spi_nor_hwcaps { * * Return: 0 for success, others for failure. */ +#if !CONFIG_IS_ENABLED(TINY_SPI) int spi_nor_scan(struct spi_nor *nor); +#else +int spi_nor_scan(struct tiny_spi_nor *nor); +#endif #endif diff --git a/include/log.h b/include/log.h index 63052f74eb..a417333153 100644 --- a/include/log.h +++ b/include/log.h @@ -16,6 +16,7 @@ #include struct cmd_tbl; +struct global_data; /** Log levels supported, ranging from most to least important */ enum log_level_t { @@ -60,6 +61,7 @@ enum log_category_t { LOGC_DEVRES, /* Device resources (devres_... functions) */ /* Advanced Configuration and Power Interface (ACPI) */ LOGC_ACPI, + LOGC_TINYDEV, /* Tiny devices (struct tinydev) */ LOGC_COUNT, /* Number of log categories */ LOGC_END, /* Sentinel value for a list of log categories */ @@ -484,4 +486,8 @@ static inline int log_get_default_format(void) (IS_ENABLED(CONFIG_LOGF_FUNC) ? BIT(LOGF_FUNC) : 0); } +void log_check(const char *msg); + +void log_fixup_for_gd_move(struct global_data *new_gd); + #endif diff --git a/include/malloc.h b/include/malloc.h index f66c2e8617..566c2d3ee7 100644 --- a/include/malloc.h +++ b/include/malloc.h @@ -935,6 +935,9 @@ int initf_malloc(void); void *malloc_simple(size_t size); void *memalign_simple(size_t alignment, size_t bytes); +uint malloc_ptr_to_ofs(void *ptr); +void *malloc_ofs_to_ptr(uint offset); + #pragma GCC visibility push(hidden) # if __STD_C diff --git a/include/ns16550.h b/include/ns16550.h index 18c9077755..1d5c311bcd 100644 --- a/include/ns16550.h +++ b/include/ns16550.h @@ -233,12 +233,11 @@ void NS16550_reinit(NS16550_t com_port, int baud_divisor); * Given the UART input clock and required baudrate, calculate the divisor * that should be used. * - * @port: UART port * @clock: UART input clock speed in Hz * @baudrate: Required baud rate * @return baud rate divisor that should be used */ -int ns16550_calc_divisor(NS16550_t port, int clock, int baudrate); +int ns16550_calc_divisor(int clock, int baudrate); /** * ns16550_serial_ofdata_to_platdata() - convert DT to platform data @@ -266,3 +265,7 @@ int ns16550_serial_probe(struct udevice *dev); * These should be used by the client driver for the driver's 'ops' member */ extern const struct dm_serial_ops ns16550_serial_ops; + +int ns16550_tiny_probe_plat(struct ns16550_platdata *plat); +int ns16550_tiny_setbrg(struct ns16550_platdata *plat, int baud_rate); +int ns16550_tiny_putc(struct ns16550_platdata *plat, const char ch); diff --git a/include/ram.h b/include/ram.h index 67e22d76c9..434e65a85a 100644 --- a/include/ram.h +++ b/include/ram.h @@ -34,4 +34,29 @@ struct ram_ops { */ int ram_get_info(struct udevice *dev, struct ram_info *info); +/** + * struct tiny_ram_ops - Operations for tiny RAM devices + */ +struct tiny_ram_ops { + /** + * get_info() - Get basic memory info + * + * @dev: Device to check (UCLASS_RAM) + * @info: Place to put info + * @return 0 if OK, -ve on error + */ + int (*get_info)(struct tinydev *dev, struct ram_info *info); +}; + +#define tiny_ram_get_ops(dev) ((struct tiny_ram_ops *)(dev)->drv->ops) + +/** + * tiny_ram_get_info() - Get information about a RAM device + * + * @dev: Device to check (UCLASS_RAM) + * @info: Returns RAM info + * @return 0 if OK, -ve on error + */ +int tiny_ram_get_info(struct tinydev *tdev, struct ram_info *info); + #endif diff --git a/include/regmap.h b/include/regmap.h index 30183c5e71..01c3e62317 100644 --- a/include/regmap.h +++ b/include/regmap.h @@ -318,7 +318,6 @@ int regmap_init_mem(ofnode node, struct regmap **mapp); * regmap_init_mem_platdata() - Set up a new memory register map for * of-platdata * - * @dev: Device that uses this map * @reg: List of address, size pairs * @count: Number of pairs (e.g. 1 if the regmap has a single entry) * @mapp: Returns allocated map @@ -330,8 +329,7 @@ int regmap_init_mem(ofnode node, struct regmap **mapp); * Use regmap_uninit() to free it. * */ -int regmap_init_mem_platdata(struct udevice *dev, fdt_val_t *reg, int count, - struct regmap **mapp); +int regmap_init_mem_platdata(fdt_val_t *reg, int count, struct regmap **mapp); int regmap_init_mem_index(ofnode node, struct regmap **mapp, int index); diff --git a/include/serial.h b/include/serial.h index 5e384db8ee..d53cb26432 100644 --- a/include/serial.h +++ b/include/serial.h @@ -9,6 +9,7 @@ void serial_initialize(void); #ifdef CONFIG_USB_TTY struct stdio_dev; +struct tinydev; int usbtty_getc(struct stdio_dev *dev); void usbtty_putc(struct stdio_dev *dev, const char c); @@ -115,7 +116,7 @@ struct serial_device_info { #define SERIAL_DEFAULT_CLOCK (16 * 115200) /** - * struct struct dm_serial_ops - Driver model serial operations + * struct dm_serial_ops - Driver model serial operations * * The uclass interface is implemented by all serial devices which use * driver model. @@ -243,6 +244,48 @@ struct serial_dev_priv { /* Access the serial operations for a device */ #define serial_get_ops(dev) ((struct dm_serial_ops *)(dev)->driver->ops) +/** + * struct tiny_serial_ops - Tiny operations support for serial + * + * This interface is optional for serial drivers and depends on + * CONFIG_SPL/TPL_TINY_SERIAL + */ +struct tiny_serial_ops { + /** + * setbrg() - Set up the baud rate generator + * + * Adjust baud rate divisors to set up a new baud rate for this + * device. Not all devices will support all rates. If the rate + * cannot be supported, the driver is free to select the nearest + * available rate. or return -EINVAL if this is not possible. + * + * @dev: Device pointer + * @baudrate: New baud rate to use + * @return 0 if OK, -ve on error + */ + int (*setbrg)(struct tinydev *tdev, int baudrate); + /** + * putc() - Write a character + * + * @dev: Device pointer + * @ch: character to write + * @return 0 if OK, -ve on error + */ + int (*putc)(struct tinydev *tdev, const char ch); +}; + +#define tiny_serial_get_ops(dev) ((struct tiny_serial_ops *)(dev)->drv->ops) + +/** + * tiny_serial_setbrg() - Set the baud rate + * + * Set the baud rate for a tiny-serial device + * + * @tdev: Tiny device + * @baudrate: Baud rate to set (e.g. 115200) + */ +int tiny_serial_setbrg(struct tinydev *tdev, int baudrate); + /** * serial_getconfig() - Get the uart configuration * (parity, 5/6/7/8 bits word length, stop bits) diff --git a/include/spi.h b/include/spi.h index 9b4fb8dc0b..df96fcf307 100644 --- a/include/spi.h +++ b/include/spi.h @@ -12,6 +12,8 @@ #include #include +struct spi_mem_op; + /* SPI mode flags */ #define SPI_CPHA BIT(0) /* clock phase */ #define SPI_CPOL BIT(1) /* clock polarity */ @@ -133,6 +135,7 @@ enum spi_polarity { struct spi_slave { #if CONFIG_IS_ENABLED(DM_SPI) struct udevice *dev; /* struct spi_slave is dev->parentdata */ + struct tinydev *tdev; uint max_hz; uint speed; #else @@ -542,6 +545,33 @@ struct dm_spi_emul_ops { const void *dout, void *din, unsigned long flags); }; +struct tiny_spi_ops { + int (*claim_bus)(struct tinydev *tdev); + int (*release_bus)(struct tinydev *tdev); + int (*xfer)(struct tinydev *tdev, uint bitlen, const void *dout, + void *din, ulong flags); + /** + * Set transfer speed and mode + * This sets a new speed to be applied for next tiny_spi_xfer(). + * @bus: The SPI bus + * @hz: The transfer speed + * @return 0 if OK, -ve on error + */ + int (*set_speed_mode)(struct tinydev *tbus, uint hz, uint mode); + + int (*adjust_op_size)(struct tinydev *tdev, struct spi_mem_op *op); + bool (*supports_op)(struct tinydev *tdev, + const struct spi_mem_op *op); + int (*exec_op)(struct tinydev *tdev, + const struct spi_mem_op *op); +}; + +int tiny_spi_claim_bus(struct tinydev *tdev); +int tiny_spi_release_bus(struct tinydev *tdev); +int tiny_spi_xfer(struct tinydev *tdev, uint bitlen, const void *dout, + void *din, ulong flags); +int tiny_spi_set_speed_mode(struct tinydev *bus, uint hz, uint mode); + /** * spi_find_bus_and_cs() - Find bus and slave devices by number * @@ -717,6 +747,7 @@ int dm_spi_get_mmap(struct udevice *dev, ulong *map_basep, uint *map_sizep, /* Access the operations for a SPI device */ #define spi_get_ops(dev) ((struct dm_spi_ops *)(dev)->driver->ops) #define spi_emul_get_ops(dev) ((struct dm_spi_emul_ops *)(dev)->driver->ops) +#define tiny_spi_get_ops(tdev) ((struct tiny_spi_ops *)(tdev)->drv->ops) #endif /* CONFIG_DM_SPI */ #endif /* _SPI_H_ */ diff --git a/include/spi_flash.h b/include/spi_flash.h index b336619487..773bc8eea2 100644 --- a/include/spi_flash.h +++ b/include/spi_flash.h @@ -178,4 +178,11 @@ static inline int spi_flash_protect(struct spi_flash *flash, u32 ofs, u32 len, return flash->flash_unlock(flash, ofs, len); } +struct tiny_spi_flash_ops { + int (*read)(struct tinydev *tdev, u32 offset, size_t len, void *buf); +}; + +int tiny_spi_flash_read(struct tinydev *tdev, u32 offset, size_t len, + void *buf); + #endif /* _SPI_FLASH_H_ */ diff --git a/include/spl.h b/include/spl.h index b31c9bb4ab..02b32b5a30 100644 --- a/include/spl.h +++ b/include/spl.h @@ -158,19 +158,23 @@ struct spl_image_info { /* * Information required to load data from a device * - * @dev: Pointer to the device, e.g. struct mmc * + * @dev: Pointer to the device (NULL if using tdev) + * @tdev: Pointer to the tiny device (NULL if using dev) * @priv: Private data for the device * @bl_len: Block length for reading in bytes * @filename: Name of the fit image file. * @read: Function to call to read from the device + * @legacy_dev: Pointer to the device, e.g. struct mmc * */ struct spl_load_info { - void *dev; + struct udevice *dev; + struct tinydev *tdev; void *priv; int bl_len; const char *filename; ulong (*read)(struct spl_load_info *load, ulong sector, ulong count, void *buf); + void *legacy_dev; /* Do not use */ }; /* diff --git a/include/syscon.h b/include/syscon.h index 3df96e3276..86a3fac1a1 100644 --- a/include/syscon.h +++ b/include/syscon.h @@ -102,4 +102,6 @@ void *syscon_get_first_range(ulong driver_data); */ struct regmap *syscon_node_to_regmap(ofnode node); +int tiny_syscon_setup(struct tinydev *tdev); + #endif diff --git a/include/sysreset.h b/include/sysreset.h index 61295e3fcb..46f727cf1b 100644 --- a/include/sysreset.h +++ b/include/sysreset.h @@ -50,6 +50,13 @@ struct sysreset_ops { #define sysreset_get_ops(dev) ((struct sysreset_ops *)(dev)->driver->ops) +struct tiny_sysreset_ops { + int (*request)(struct tinydev *tdev, enum sysreset_t type); +}; + +#define tiny_sysreset_get_ops(dev) \ + ((struct tiny_sysreset_ops *)(dev)->drv->ops) + /** * sysreset_request() - request a sysreset * @@ -116,4 +123,6 @@ void sysreset_walk_halt(enum sysreset_t type); */ void reset_cpu(ulong addr); +int tiny_sysreset_request(struct tinydev *tdev, enum sysreset_t type); + #endif