mirror of
https://github.com/Fishwaldo/opensbi.git
synced 2025-03-15 19:31:32 +00:00
lib: utils/regmap: Add simple FDT based regmap framework
We add a simple FDT based regmap framework which is built on top of generic regmap library. The phandle of FDT regmap DT node is treated as unique regmap ID. The FDT based regmap drivers will be probed on-demand from fdt_regmap_get_by_phandle() and fdt_regmap_get() called by the regmap client drivers. Signed-off-by: Anup Patel <apatel@ventanamicro.com> Reviewed-by: Xiang W <wxjstz@126.com>
This commit is contained in:
parent
14a35b0e0e
commit
8e97275d97
6 changed files with 138 additions and 1 deletions
31
include/sbi_utils/regmap/fdt_regmap.h
Normal file
31
include/sbi_utils/regmap/fdt_regmap.h
Normal file
|
@ -0,0 +1,31 @@
|
|||
/*
|
||||
* SPDX-License-Identifier: BSD-2-Clause
|
||||
*
|
||||
* Copyright (c) 2023 Ventana Micro Systems Inc.
|
||||
*
|
||||
* Authors:
|
||||
* Anup Patel <apatel@ventanamicro.com>
|
||||
*/
|
||||
|
||||
#ifndef __FDT_REGMAP_H__
|
||||
#define __FDT_REGMAP_H__
|
||||
|
||||
#include <sbi_utils/regmap/regmap.h>
|
||||
|
||||
struct fdt_phandle_args;
|
||||
|
||||
/** FDT based regmap driver */
|
||||
struct fdt_regmap {
|
||||
const struct fdt_match *match_table;
|
||||
int (*init)(void *fdt, int nodeoff, u32 phandle,
|
||||
const struct fdt_match *match);
|
||||
};
|
||||
|
||||
/** Get regmap instance based on phandle */
|
||||
int fdt_regmap_get_by_phandle(void *fdt, u32 phandle,
|
||||
struct regmap **out_rmap);
|
||||
|
||||
/** Get regmap instance based on "regmap' property of the specified DT node */
|
||||
int fdt_regmap_get(void *fdt, int nodeoff, struct regmap **out_rmap);
|
||||
|
||||
#endif
|
|
@ -2,6 +2,12 @@
|
|||
|
||||
menu "Regmap Support"
|
||||
|
||||
config FDT_REGMAP
|
||||
bool "FDT based regmap drivers"
|
||||
depends on FDT
|
||||
select REGMAP
|
||||
default n
|
||||
|
||||
config REGMAP
|
||||
bool "Regmap support"
|
||||
default n
|
||||
|
|
94
lib/utils/regmap/fdt_regmap.c
Normal file
94
lib/utils/regmap/fdt_regmap.c
Normal file
|
@ -0,0 +1,94 @@
|
|||
/*
|
||||
* SPDX-License-Identifier: BSD-2-Clause
|
||||
*
|
||||
* Copyright (c) 2023 Ventana Micro Systems Inc.
|
||||
*
|
||||
* Authors:
|
||||
* Anup Patel <apatel@ventanamicro.com>
|
||||
*/
|
||||
|
||||
#include <libfdt.h>
|
||||
#include <sbi/sbi_error.h>
|
||||
#include <sbi_utils/fdt/fdt_helper.h>
|
||||
#include <sbi_utils/regmap/fdt_regmap.h>
|
||||
|
||||
/* List of FDT regmap drivers generated at compile time */
|
||||
extern struct fdt_regmap *fdt_regmap_drivers[];
|
||||
extern unsigned long fdt_regmap_drivers_size;
|
||||
|
||||
static int fdt_regmap_init(void *fdt, int nodeoff, u32 phandle)
|
||||
{
|
||||
int pos, rc;
|
||||
struct fdt_regmap *drv;
|
||||
const struct fdt_match *match;
|
||||
|
||||
/* Try all I2C drivers one-by-one */
|
||||
for (pos = 0; pos < fdt_regmap_drivers_size; pos++) {
|
||||
drv = fdt_regmap_drivers[pos];
|
||||
match = fdt_match_node(fdt, nodeoff, drv->match_table);
|
||||
if (match && drv->init) {
|
||||
rc = drv->init(fdt, nodeoff, phandle, match);
|
||||
if (rc == SBI_ENODEV)
|
||||
continue;
|
||||
if (rc)
|
||||
return rc;
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
return SBI_ENOSYS;
|
||||
}
|
||||
|
||||
static int fdt_regmap_find(void *fdt, int nodeoff, u32 phandle,
|
||||
struct regmap **out_rmap)
|
||||
{
|
||||
int rc;
|
||||
struct regmap *rmap = regmap_find(phandle);
|
||||
|
||||
if (!rmap) {
|
||||
/* Regmap not found so initialize matching driver */
|
||||
rc = fdt_regmap_init(fdt, nodeoff, phandle);
|
||||
if (rc)
|
||||
return rc;
|
||||
|
||||
/* Try to find regmap again */
|
||||
rmap = regmap_find(phandle);
|
||||
if (!rmap)
|
||||
return SBI_ENOSYS;
|
||||
}
|
||||
|
||||
if (out_rmap)
|
||||
*out_rmap = rmap;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int fdt_regmap_get_by_phandle(void *fdt, u32 phandle,
|
||||
struct regmap **out_rmap)
|
||||
{
|
||||
int pnodeoff;
|
||||
|
||||
if (!fdt || !out_rmap)
|
||||
return SBI_EINVAL;
|
||||
|
||||
pnodeoff = fdt_node_offset_by_phandle(fdt, phandle);
|
||||
if (pnodeoff < 0)
|
||||
return pnodeoff;
|
||||
|
||||
return fdt_regmap_find(fdt, pnodeoff, phandle, out_rmap);
|
||||
}
|
||||
|
||||
int fdt_regmap_get(void *fdt, int nodeoff, struct regmap **out_rmap)
|
||||
{
|
||||
int len;
|
||||
const fdt32_t *val;
|
||||
|
||||
if (!fdt || (nodeoff < 0) || !out_rmap)
|
||||
return SBI_EINVAL;
|
||||
|
||||
val = fdt_getprop(fdt, nodeoff, "regmap", &len);
|
||||
if (!val)
|
||||
return SBI_ENOENT;
|
||||
|
||||
return fdt_regmap_get_by_phandle(fdt, fdt32_to_cpu(*val), out_rmap);
|
||||
}
|
3
lib/utils/regmap/fdt_regmap_drivers.carray
Normal file
3
lib/utils/regmap/fdt_regmap_drivers.carray
Normal file
|
@ -0,0 +1,3 @@
|
|||
HEADER: sbi_utils/regmap/fdt_regmap.h
|
||||
TYPE: struct fdt_regmap
|
||||
NAME: fdt_regmap_drivers
|
|
@ -7,4 +7,7 @@
|
|||
# Anup Patel <apatel@ventanamicro.com>
|
||||
#
|
||||
|
||||
libsbiutils-objs-$(CONFIG_FDT_REGMAP) += regmap/fdt_regmap.o
|
||||
libsbiutils-objs-$(CONFIG_FDT_REGMAP) += regmap/fdt_regmap_drivers.o
|
||||
|
||||
libsbiutils-objs-$(CONFIG_REGMAP) += regmap/regmap.o
|
||||
|
|
|
@ -18,7 +18,7 @@ CONFIG_FDT_IRQCHIP=y
|
|||
CONFIG_FDT_IRQCHIP_APLIC=y
|
||||
CONFIG_FDT_IRQCHIP_IMSIC=y
|
||||
CONFIG_FDT_IRQCHIP_PLIC=y
|
||||
CONFIG_REGMAP=y
|
||||
CONFIG_FDT_REGMAP=y
|
||||
CONFIG_FDT_RESET=y
|
||||
CONFIG_FDT_RESET_ATCWDT200=y
|
||||
CONFIG_FDT_RESET_GPIO=y
|
||||
|
|
Loading…
Add table
Reference in a new issue