bus/dpaa: add OF parser for device scanning
This layer is used by Bus driver's scan function. Devices are parsed using OF parser and added to DPAA device list. Signed-off-by: Geoff Thorpe <geoff.thorpe@nxp.com> Signed-off-by: Shreyansh Jain <shreyansh.jain@nxp.com>
This commit is contained in:
parent
39f373cf01
commit
2183c6f69d
@ -38,7 +38,11 @@ LIB = librte_bus_dpaa.a
|
||||
|
||||
CFLAGS := -I$(SRCDIR) $(CFLAGS)
|
||||
CFLAGS += -O3 $(WERROR_FLAGS)
|
||||
CFLAGS += -Wno-pointer-arith
|
||||
CFLAGS += -Wno-cast-qual
|
||||
CFLAGS += -D _GNU_SOURCE
|
||||
CFLAGS += -I$(RTE_BUS_DPAA)/
|
||||
CFLAGS += -I$(RTE_BUS_DPAA)/include
|
||||
CFLAGS += -I$(RTE_SDK)/lib/librte_eal/linuxapp/eal
|
||||
CFLAGS += -I$(RTE_SDK)/lib/librte_eal/common/include
|
||||
|
||||
@ -52,6 +56,9 @@ LIBABIVER := 1
|
||||
SRCS-$(CONFIG_RTE_LIBRTE_DPAA_BUS) += \
|
||||
dpaa_bus.c
|
||||
|
||||
SRCS-$(CONFIG_RTE_LIBRTE_DPAA_BUS) += \
|
||||
base/fman/of.c \
|
||||
|
||||
# Link Pthread
|
||||
LDLIBS += -lpthread
|
||||
|
||||
|
576
drivers/bus/dpaa/base/fman/of.c
Normal file
576
drivers/bus/dpaa/base/fman/of.c
Normal file
@ -0,0 +1,576 @@
|
||||
/*-
|
||||
* This file is provided under a dual BSD/GPLv2 license. When using or
|
||||
* redistributing this file, you may do so under either license.
|
||||
*
|
||||
* BSD LICENSE
|
||||
*
|
||||
* Copyright 2010-2016 Freescale Semiconductor Inc.
|
||||
* Copyright 2017 NXP.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
* modification, are permitted provided that the following conditions are met:
|
||||
* * Redistributions of source code must retain the above copyright
|
||||
* notice, this list of conditions and the following disclaimer.
|
||||
* * Redistributions in binary form must reproduce the above copyright
|
||||
* notice, this list of conditions and the following disclaimer in the
|
||||
* documentation and/or other materials provided with the distribution.
|
||||
* * Neither the name of the above-listed copyright holders nor the
|
||||
* names of any contributors may be used to endorse or promote products
|
||||
* derived from this software without specific prior written permission.
|
||||
*
|
||||
* GPL LICENSE SUMMARY
|
||||
*
|
||||
* ALTERNATIVELY, this software may be distributed under the terms of the
|
||||
* GNU General Public License ("GPL") as published by the Free Software
|
||||
* Foundation, either version 2 of that License or (at your option) any
|
||||
* later version.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||||
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE
|
||||
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
||||
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
||||
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
||||
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
||||
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
||||
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
||||
* POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
#include <of.h>
|
||||
#include <rte_dpaa_logs.h>
|
||||
|
||||
static int alive;
|
||||
static struct dt_dir root_dir;
|
||||
static const char *base_dir;
|
||||
static COMPAT_LIST_HEAD(linear);
|
||||
|
||||
static int
|
||||
of_open_dir(const char *relative_path, struct dirent ***d)
|
||||
{
|
||||
int ret;
|
||||
char full_path[PATH_MAX];
|
||||
|
||||
snprintf(full_path, PATH_MAX, "%s/%s", base_dir, relative_path);
|
||||
ret = scandir(full_path, d, 0, versionsort);
|
||||
if (ret < 0)
|
||||
DPAA_BUS_LOG(ERR, "Failed to open directory %s",
|
||||
full_path);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void
|
||||
of_close_dir(struct dirent **d, int num)
|
||||
{
|
||||
while (num--)
|
||||
free(d[num]);
|
||||
free(d);
|
||||
}
|
||||
|
||||
static int
|
||||
of_open_file(const char *relative_path)
|
||||
{
|
||||
int ret;
|
||||
char full_path[PATH_MAX];
|
||||
|
||||
snprintf(full_path, PATH_MAX, "%s/%s", base_dir, relative_path);
|
||||
ret = open(full_path, O_RDONLY);
|
||||
if (ret < 0)
|
||||
DPAA_BUS_LOG(ERR, "Failed to open directory %s",
|
||||
full_path);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void
|
||||
process_file(struct dirent *dent, struct dt_dir *parent)
|
||||
{
|
||||
int fd;
|
||||
struct dt_file *f = malloc(sizeof(*f));
|
||||
|
||||
if (!f) {
|
||||
DPAA_BUS_LOG(DEBUG, "Unable to allocate memory for file node");
|
||||
return;
|
||||
}
|
||||
f->node.is_file = 1;
|
||||
snprintf(f->node.node.name, NAME_MAX, "%s", dent->d_name);
|
||||
snprintf(f->node.node.full_name, PATH_MAX, "%s/%s",
|
||||
parent->node.node.full_name, dent->d_name);
|
||||
f->parent = parent;
|
||||
fd = of_open_file(f->node.node.full_name);
|
||||
if (fd < 0) {
|
||||
DPAA_BUS_LOG(DEBUG, "Unable to open file node");
|
||||
free(f);
|
||||
return;
|
||||
}
|
||||
f->len = read(fd, f->buf, OF_FILE_BUF_MAX);
|
||||
close(fd);
|
||||
if (f->len < 0) {
|
||||
DPAA_BUS_LOG(DEBUG, "Unable to read file node");
|
||||
free(f);
|
||||
return;
|
||||
}
|
||||
list_add_tail(&f->node.list, &parent->files);
|
||||
}
|
||||
|
||||
static const struct dt_dir *
|
||||
node2dir(const struct device_node *n)
|
||||
{
|
||||
struct dt_node *dn = container_of((struct device_node *)n,
|
||||
struct dt_node, node);
|
||||
const struct dt_dir *d = container_of(dn, struct dt_dir, node);
|
||||
|
||||
assert(!dn->is_file);
|
||||
return d;
|
||||
}
|
||||
|
||||
/* process_dir() calls iterate_dir(), but the latter will also call the former
|
||||
* when recursing into sub-directories, so a predeclaration is needed.
|
||||
*/
|
||||
static int process_dir(const char *relative_path, struct dt_dir *dt);
|
||||
|
||||
static int
|
||||
iterate_dir(struct dirent **d, int num, struct dt_dir *dt)
|
||||
{
|
||||
int loop;
|
||||
/* Iterate the directory contents */
|
||||
for (loop = 0; loop < num; loop++) {
|
||||
struct dt_dir *subdir;
|
||||
int ret;
|
||||
/* Ignore dot files of all types (especially "..") */
|
||||
if (d[loop]->d_name[0] == '.')
|
||||
continue;
|
||||
switch (d[loop]->d_type) {
|
||||
case DT_REG:
|
||||
process_file(d[loop], dt);
|
||||
break;
|
||||
case DT_DIR:
|
||||
subdir = malloc(sizeof(*subdir));
|
||||
if (!subdir) {
|
||||
perror("malloc");
|
||||
return -ENOMEM;
|
||||
}
|
||||
snprintf(subdir->node.node.name, NAME_MAX, "%s",
|
||||
d[loop]->d_name);
|
||||
snprintf(subdir->node.node.full_name, PATH_MAX,
|
||||
"%s/%s", dt->node.node.full_name,
|
||||
d[loop]->d_name);
|
||||
subdir->parent = dt;
|
||||
ret = process_dir(subdir->node.node.full_name, subdir);
|
||||
if (ret)
|
||||
return ret;
|
||||
list_add_tail(&subdir->node.list, &dt->subdirs);
|
||||
break;
|
||||
default:
|
||||
DPAA_BUS_LOG(DEBUG, "Ignoring invalid dt entry %s/%s",
|
||||
dt->node.node.full_name, d[loop]->d_name);
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
process_dir(const char *relative_path, struct dt_dir *dt)
|
||||
{
|
||||
struct dirent **d;
|
||||
int ret, num;
|
||||
|
||||
dt->node.is_file = 0;
|
||||
INIT_LIST_HEAD(&dt->subdirs);
|
||||
INIT_LIST_HEAD(&dt->files);
|
||||
ret = of_open_dir(relative_path, &d);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
num = ret;
|
||||
ret = iterate_dir(d, num, dt);
|
||||
of_close_dir(d, num);
|
||||
return (ret < 0) ? ret : 0;
|
||||
}
|
||||
|
||||
static void
|
||||
linear_dir(struct dt_dir *d)
|
||||
{
|
||||
struct dt_file *f;
|
||||
struct dt_dir *dd;
|
||||
|
||||
d->compatible = NULL;
|
||||
d->status = NULL;
|
||||
d->lphandle = NULL;
|
||||
d->a_cells = NULL;
|
||||
d->s_cells = NULL;
|
||||
d->reg = NULL;
|
||||
list_for_each_entry(f, &d->files, node.list) {
|
||||
if (!strcmp(f->node.node.name, "compatible")) {
|
||||
if (d->compatible)
|
||||
DPAA_BUS_LOG(DEBUG, "Duplicate compatible in"
|
||||
" %s", d->node.node.full_name);
|
||||
d->compatible = f;
|
||||
} else if (!strcmp(f->node.node.name, "status")) {
|
||||
if (d->status)
|
||||
DPAA_BUS_LOG(DEBUG, "Duplicate status in %s",
|
||||
d->node.node.full_name);
|
||||
d->status = f;
|
||||
} else if (!strcmp(f->node.node.name, "linux,phandle")) {
|
||||
if (d->lphandle)
|
||||
DPAA_BUS_LOG(DEBUG, "Duplicate lphandle in %s",
|
||||
d->node.node.full_name);
|
||||
d->lphandle = f;
|
||||
} else if (!strcmp(f->node.node.name, "#address-cells")) {
|
||||
if (d->a_cells)
|
||||
DPAA_BUS_LOG(DEBUG, "Duplicate a_cells in %s",
|
||||
d->node.node.full_name);
|
||||
d->a_cells = f;
|
||||
} else if (!strcmp(f->node.node.name, "#size-cells")) {
|
||||
if (d->s_cells)
|
||||
DPAA_BUS_LOG(DEBUG, "Duplicate s_cells in %s",
|
||||
d->node.node.full_name);
|
||||
d->s_cells = f;
|
||||
} else if (!strcmp(f->node.node.name, "reg")) {
|
||||
if (d->reg)
|
||||
DPAA_BUS_LOG(DEBUG, "Duplicate reg in %s",
|
||||
d->node.node.full_name);
|
||||
d->reg = f;
|
||||
}
|
||||
}
|
||||
|
||||
list_for_each_entry(dd, &d->subdirs, node.list) {
|
||||
list_add_tail(&dd->linear, &linear);
|
||||
linear_dir(dd);
|
||||
}
|
||||
}
|
||||
|
||||
int
|
||||
of_init_path(const char *dt_path)
|
||||
{
|
||||
int ret;
|
||||
|
||||
base_dir = dt_path;
|
||||
|
||||
/* This needs to be singleton initialization */
|
||||
DPAA_BUS_HWWARN(alive, "Double-init of device-tree driver!");
|
||||
|
||||
/* Prepare root node (the remaining fields are set in process_dir()) */
|
||||
root_dir.node.node.name[0] = '\0';
|
||||
root_dir.node.node.full_name[0] = '\0';
|
||||
INIT_LIST_HEAD(&root_dir.node.list);
|
||||
root_dir.parent = NULL;
|
||||
|
||||
/* Kick things off... */
|
||||
ret = process_dir("", &root_dir);
|
||||
if (ret) {
|
||||
DPAA_BUS_LOG(ERR, "Unable to parse device tree");
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* Now make a flat, linear list of directories */
|
||||
linear_dir(&root_dir);
|
||||
alive = 1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void
|
||||
destroy_dir(struct dt_dir *d)
|
||||
{
|
||||
struct dt_file *f, *tmpf;
|
||||
struct dt_dir *dd, *tmpd;
|
||||
|
||||
list_for_each_entry_safe(f, tmpf, &d->files, node.list) {
|
||||
list_del(&f->node.list);
|
||||
free(f);
|
||||
}
|
||||
list_for_each_entry_safe(dd, tmpd, &d->subdirs, node.list) {
|
||||
destroy_dir(dd);
|
||||
list_del(&dd->node.list);
|
||||
free(dd);
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
of_finish(void)
|
||||
{
|
||||
DPAA_BUS_HWWARN(!alive, "Double-finish of device-tree driver!");
|
||||
|
||||
destroy_dir(&root_dir);
|
||||
INIT_LIST_HEAD(&linear);
|
||||
alive = 0;
|
||||
}
|
||||
|
||||
static const struct dt_dir *
|
||||
next_linear(const struct dt_dir *f)
|
||||
{
|
||||
if (f->linear.next == &linear)
|
||||
return NULL;
|
||||
return list_entry(f->linear.next, struct dt_dir, linear);
|
||||
}
|
||||
|
||||
static int
|
||||
check_compatible(const struct dt_file *f, const char *compatible)
|
||||
{
|
||||
const char *c = (char *)f->buf;
|
||||
unsigned int len, remains = f->len;
|
||||
|
||||
while (remains) {
|
||||
len = strlen(c);
|
||||
if (!strcmp(c, compatible))
|
||||
return 1;
|
||||
|
||||
if (remains < len + 1)
|
||||
break;
|
||||
|
||||
c += (len + 1);
|
||||
remains -= (len + 1);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
const struct device_node *
|
||||
of_find_compatible_node(const struct device_node *from,
|
||||
const char *type __always_unused,
|
||||
const char *compatible)
|
||||
{
|
||||
const struct dt_dir *d;
|
||||
|
||||
DPAA_BUS_HWWARN(!alive, "Device-tree driver not initialised!");
|
||||
|
||||
if (list_empty(&linear))
|
||||
return NULL;
|
||||
if (!from)
|
||||
d = list_entry(linear.next, struct dt_dir, linear);
|
||||
else
|
||||
d = node2dir(from);
|
||||
for (d = next_linear(d); d && (!d->compatible ||
|
||||
!check_compatible(d->compatible,
|
||||
compatible));
|
||||
d = next_linear(d))
|
||||
;
|
||||
if (d)
|
||||
return &d->node.node;
|
||||
return NULL;
|
||||
}
|
||||
|
||||
const void *
|
||||
of_get_property(const struct device_node *from, const char *name,
|
||||
size_t *lenp)
|
||||
{
|
||||
const struct dt_dir *d;
|
||||
const struct dt_file *f;
|
||||
|
||||
DPAA_BUS_HWWARN(!alive, "Device-tree driver not initialised!");
|
||||
|
||||
d = node2dir(from);
|
||||
list_for_each_entry(f, &d->files, node.list)
|
||||
if (!strcmp(f->node.node.name, name)) {
|
||||
if (lenp)
|
||||
*lenp = f->len;
|
||||
return f->buf;
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
bool
|
||||
of_device_is_available(const struct device_node *dev_node)
|
||||
{
|
||||
const struct dt_dir *d;
|
||||
|
||||
DPAA_BUS_HWWARN(!alive, "Device-tree driver not initialised!");
|
||||
d = node2dir(dev_node);
|
||||
if (!d->status)
|
||||
return true;
|
||||
if (!strcmp((char *)d->status->buf, "okay"))
|
||||
return true;
|
||||
if (!strcmp((char *)d->status->buf, "ok"))
|
||||
return true;
|
||||
return false;
|
||||
}
|
||||
|
||||
const struct device_node *
|
||||
of_find_node_by_phandle(phandle ph)
|
||||
{
|
||||
const struct dt_dir *d;
|
||||
|
||||
DPAA_BUS_HWWARN(!alive, "Device-tree driver not initialised!");
|
||||
list_for_each_entry(d, &linear, linear)
|
||||
if (d->lphandle && (d->lphandle->len == 4) &&
|
||||
!memcmp(d->lphandle->buf, &ph, 4))
|
||||
return &d->node.node;
|
||||
return NULL;
|
||||
}
|
||||
|
||||
const struct device_node *
|
||||
of_get_parent(const struct device_node *dev_node)
|
||||
{
|
||||
const struct dt_dir *d;
|
||||
|
||||
DPAA_BUS_HWWARN(!alive, "Device-tree driver not initialised!");
|
||||
|
||||
if (!dev_node)
|
||||
return NULL;
|
||||
d = node2dir(dev_node);
|
||||
if (!d->parent)
|
||||
return NULL;
|
||||
return &d->parent->node.node;
|
||||
}
|
||||
|
||||
const struct device_node *
|
||||
of_get_next_child(const struct device_node *dev_node,
|
||||
const struct device_node *prev)
|
||||
{
|
||||
const struct dt_dir *p, *c;
|
||||
|
||||
DPAA_BUS_HWWARN(!alive, "Device-tree driver not initialised!");
|
||||
|
||||
if (!dev_node)
|
||||
return NULL;
|
||||
p = node2dir(dev_node);
|
||||
if (prev) {
|
||||
c = node2dir(prev);
|
||||
DPAA_BUS_HWWARN((c->parent != p), "Parent/child mismatch");
|
||||
if (c->parent != p)
|
||||
return NULL;
|
||||
if (c->node.list.next == &p->subdirs)
|
||||
/* prev was the last child */
|
||||
return NULL;
|
||||
c = list_entry(c->node.list.next, struct dt_dir, node.list);
|
||||
return &c->node.node;
|
||||
}
|
||||
/* Return first child */
|
||||
if (list_empty(&p->subdirs))
|
||||
return NULL;
|
||||
c = list_entry(p->subdirs.next, struct dt_dir, node.list);
|
||||
return &c->node.node;
|
||||
}
|
||||
|
||||
uint32_t
|
||||
of_n_addr_cells(const struct device_node *dev_node)
|
||||
{
|
||||
const struct dt_dir *d;
|
||||
|
||||
DPAA_BUS_HWWARN(!alive, "Device-tree driver not initialised");
|
||||
if (!dev_node)
|
||||
return OF_DEFAULT_NA;
|
||||
d = node2dir(dev_node);
|
||||
while ((d = d->parent))
|
||||
if (d->a_cells) {
|
||||
unsigned char *buf =
|
||||
(unsigned char *)&d->a_cells->buf[0];
|
||||
assert(d->a_cells->len == 4);
|
||||
return ((uint32_t)buf[0] << 24) |
|
||||
((uint32_t)buf[1] << 16) |
|
||||
((uint32_t)buf[2] << 8) |
|
||||
(uint32_t)buf[3];
|
||||
}
|
||||
return OF_DEFAULT_NA;
|
||||
}
|
||||
|
||||
uint32_t
|
||||
of_n_size_cells(const struct device_node *dev_node)
|
||||
{
|
||||
const struct dt_dir *d;
|
||||
|
||||
DPAA_BUS_HWWARN(!alive, "Device-tree driver not initialised!");
|
||||
if (!dev_node)
|
||||
return OF_DEFAULT_NA;
|
||||
d = node2dir(dev_node);
|
||||
while ((d = d->parent))
|
||||
if (d->s_cells) {
|
||||
unsigned char *buf =
|
||||
(unsigned char *)&d->s_cells->buf[0];
|
||||
assert(d->s_cells->len == 4);
|
||||
return ((uint32_t)buf[0] << 24) |
|
||||
((uint32_t)buf[1] << 16) |
|
||||
((uint32_t)buf[2] << 8) |
|
||||
(uint32_t)buf[3];
|
||||
}
|
||||
return OF_DEFAULT_NS;
|
||||
}
|
||||
|
||||
const uint32_t *
|
||||
of_get_address(const struct device_node *dev_node, size_t idx,
|
||||
uint64_t *size, uint32_t *flags __rte_unused)
|
||||
{
|
||||
const struct dt_dir *d;
|
||||
const unsigned char *buf;
|
||||
uint32_t na = of_n_addr_cells(dev_node);
|
||||
uint32_t ns = of_n_size_cells(dev_node);
|
||||
|
||||
if (!dev_node)
|
||||
d = &root_dir;
|
||||
else
|
||||
d = node2dir(dev_node);
|
||||
if (!d->reg)
|
||||
return NULL;
|
||||
assert(d->reg->len % ((na + ns) * 4) == 0);
|
||||
assert(d->reg->len / ((na + ns) * 4) > (unsigned int) idx);
|
||||
buf = (const unsigned char *)&d->reg->buf[0];
|
||||
buf += (na + ns) * idx * 4;
|
||||
if (size)
|
||||
for (*size = 0; ns > 0; ns--, na++)
|
||||
*size = (*size << 32) +
|
||||
(((uint32_t)buf[4 * na] << 24) |
|
||||
((uint32_t)buf[4 * na + 1] << 16) |
|
||||
((uint32_t)buf[4 * na + 2] << 8) |
|
||||
(uint32_t)buf[4 * na + 3]);
|
||||
return (const uint32_t *)buf;
|
||||
}
|
||||
|
||||
uint64_t
|
||||
of_translate_address(const struct device_node *dev_node,
|
||||
const uint32_t *addr)
|
||||
{
|
||||
uint64_t phys_addr, tmp_addr;
|
||||
const struct device_node *parent;
|
||||
const uint32_t *ranges;
|
||||
size_t rlen;
|
||||
uint32_t na, pna;
|
||||
|
||||
DPAA_BUS_HWWARN(!alive, "Device-tree driver not initialised!");
|
||||
assert(dev_node != NULL);
|
||||
|
||||
na = of_n_addr_cells(dev_node);
|
||||
phys_addr = of_read_number(addr, na);
|
||||
|
||||
dev_node = of_get_parent(dev_node);
|
||||
if (!dev_node)
|
||||
return 0;
|
||||
else if (node2dir(dev_node) == &root_dir)
|
||||
return phys_addr;
|
||||
|
||||
do {
|
||||
pna = of_n_addr_cells(dev_node);
|
||||
parent = of_get_parent(dev_node);
|
||||
if (!parent)
|
||||
return 0;
|
||||
|
||||
ranges = of_get_property(dev_node, "ranges", &rlen);
|
||||
/* "ranges" property is missing. Translation breaks */
|
||||
if (!ranges)
|
||||
return 0;
|
||||
/* "ranges" property is empty. Do 1:1 translation */
|
||||
else if (rlen == 0)
|
||||
continue;
|
||||
else
|
||||
tmp_addr = of_read_number(ranges + na, pna);
|
||||
|
||||
na = pna;
|
||||
dev_node = parent;
|
||||
phys_addr += tmp_addr;
|
||||
} while (node2dir(parent) != &root_dir);
|
||||
|
||||
return phys_addr;
|
||||
}
|
||||
|
||||
bool
|
||||
of_device_is_compatible(const struct device_node *dev_node,
|
||||
const char *compatible)
|
||||
{
|
||||
const struct dt_dir *d;
|
||||
|
||||
DPAA_BUS_HWWARN(!alive, "Device-tree driver not initialised!");
|
||||
if (!dev_node)
|
||||
d = &root_dir;
|
||||
else
|
||||
d = node2dir(dev_node);
|
||||
if (d->compatible && check_compatible(d->compatible, compatible))
|
||||
return true;
|
||||
return false;
|
||||
}
|
190
drivers/bus/dpaa/include/of.h
Normal file
190
drivers/bus/dpaa/include/of.h
Normal file
@ -0,0 +1,190 @@
|
||||
/*-
|
||||
* This file is provided under a dual BSD/GPLv2 license. When using or
|
||||
* redistributing this file, you may do so under either license.
|
||||
*
|
||||
* BSD LICENSE
|
||||
*
|
||||
* Copyright 2010-2016 Freescale Semiconductor, Inc.
|
||||
* Copyright 2017 NXP.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
* modification, are permitted provided that the following conditions are met:
|
||||
* * Redistributions of source code must retain the above copyright
|
||||
* notice, this list of conditions and the following disclaimer.
|
||||
* * Redistributions in binary form must reproduce the above copyright
|
||||
* notice, this list of conditions and the following disclaimer in the
|
||||
* documentation and/or other materials provided with the distribution.
|
||||
* * Neither the name of the above-listed copyright holders nor the
|
||||
* names of any contributors may be used to endorse or promote products
|
||||
* derived from this software without specific prior written permission.
|
||||
*
|
||||
* GPL LICENSE SUMMARY
|
||||
*
|
||||
* ALTERNATIVELY, this software may be distributed under the terms of the
|
||||
* GNU General Public License ("GPL") as published by the Free Software
|
||||
* Foundation, either version 2 of that License or (at your option) any
|
||||
* later version.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||||
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE
|
||||
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
||||
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
||||
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
||||
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
||||
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
||||
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
||||
* POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
#ifndef __OF_H
|
||||
#define __OF_H
|
||||
|
||||
#include <compat.h>
|
||||
|
||||
#ifndef OF_INIT_DEFAULT_PATH
|
||||
#define OF_INIT_DEFAULT_PATH "/proc/device-tree"
|
||||
#endif
|
||||
|
||||
#define OF_DEFAULT_NA 1
|
||||
#define OF_DEFAULT_NS 1
|
||||
|
||||
#define OF_FILE_BUF_MAX 256
|
||||
|
||||
/**
|
||||
* Layout of Device Tree:
|
||||
* dt_dir
|
||||
* |- dt_dir
|
||||
* | |- dt_dir
|
||||
* | | |- dt_dir
|
||||
* | | | |- dt_file
|
||||
* | | | ``- dt_file
|
||||
* | | ``- dt_file
|
||||
* | `-dt_file`
|
||||
* ``- dt_file
|
||||
*
|
||||
* +------------------+
|
||||
* |dt_dir |
|
||||
* |+----------------+|
|
||||
* ||dt_node ||
|
||||
* ||+--------------+||
|
||||
* |||device_node |||
|
||||
* ||+--------------+||
|
||||
* || list_dt_nodes ||
|
||||
* |+----------------+|
|
||||
* | list of subdir |
|
||||
* | list of files |
|
||||
* +------------------+
|
||||
*/
|
||||
|
||||
/**
|
||||
* Device description on of a device node in device tree.
|
||||
*/
|
||||
struct device_node {
|
||||
char name[NAME_MAX];
|
||||
char full_name[PATH_MAX];
|
||||
};
|
||||
|
||||
/**
|
||||
* List of device nodes available in a device tree layout
|
||||
*/
|
||||
struct dt_node {
|
||||
struct device_node node; /**< Property of node */
|
||||
int is_file; /**< FALSE==dir, TRUE==file */
|
||||
struct list_head list; /**< Nodes within a parent subdir */
|
||||
};
|
||||
|
||||
/**
|
||||
* Types we use to represent directories and files
|
||||
*/
|
||||
struct dt_file;
|
||||
struct dt_dir {
|
||||
struct dt_node node;
|
||||
struct list_head subdirs;
|
||||
struct list_head files;
|
||||
struct list_head linear;
|
||||
struct dt_dir *parent;
|
||||
struct dt_file *compatible;
|
||||
struct dt_file *status;
|
||||
struct dt_file *lphandle;
|
||||
struct dt_file *a_cells;
|
||||
struct dt_file *s_cells;
|
||||
struct dt_file *reg;
|
||||
};
|
||||
|
||||
struct dt_file {
|
||||
struct dt_node node;
|
||||
struct dt_dir *parent;
|
||||
ssize_t len;
|
||||
uint64_t buf[OF_FILE_BUF_MAX >> 3];
|
||||
};
|
||||
|
||||
const struct device_node *of_find_compatible_node(
|
||||
const struct device_node *from,
|
||||
const char *type __always_unused,
|
||||
const char *compatible)
|
||||
__attribute__((nonnull(3)));
|
||||
|
||||
#define for_each_compatible_node(dev_node, type, compatible) \
|
||||
for (dev_node = of_find_compatible_node(NULL, type, compatible); \
|
||||
dev_node != NULL; \
|
||||
dev_node = of_find_compatible_node(dev_node, type, compatible))
|
||||
|
||||
const void *of_get_property(const struct device_node *from, const char *name,
|
||||
size_t *lenp) __attribute__((nonnull(2)));
|
||||
bool of_device_is_available(const struct device_node *dev_node);
|
||||
|
||||
const struct device_node *of_find_node_by_phandle(phandle ph);
|
||||
|
||||
const struct device_node *of_get_parent(const struct device_node *dev_node);
|
||||
|
||||
const struct device_node *of_get_next_child(const struct device_node *dev_node,
|
||||
const struct device_node *prev);
|
||||
|
||||
#define for_each_child_node(parent, child) \
|
||||
for (child = of_get_next_child(parent, NULL); child != NULL; \
|
||||
child = of_get_next_child(parent, child))
|
||||
|
||||
uint32_t of_n_addr_cells(const struct device_node *dev_node);
|
||||
uint32_t of_n_size_cells(const struct device_node *dev_node);
|
||||
|
||||
const uint32_t *of_get_address(const struct device_node *dev_node, size_t idx,
|
||||
uint64_t *size, uint32_t *flags);
|
||||
|
||||
uint64_t of_translate_address(const struct device_node *dev_node,
|
||||
const u32 *addr) __attribute__((nonnull));
|
||||
|
||||
bool of_device_is_compatible(const struct device_node *dev_node,
|
||||
const char *compatible);
|
||||
|
||||
/* of_init() must be called prior to initialisation or use of any driver
|
||||
* subsystem that is device-tree-dependent. Eg. Qman/Bman, config layers, etc.
|
||||
* The path should usually be "/proc/device-tree".
|
||||
*/
|
||||
int of_init_path(const char *dt_path);
|
||||
|
||||
/* of_finish() allows a controlled tear-down of the device-tree layer, eg. if a
|
||||
* full reload is desired without a process exit.
|
||||
*/
|
||||
void of_finish(void);
|
||||
|
||||
/* Use of this wrapper is recommended. */
|
||||
static inline int of_init(void)
|
||||
{
|
||||
return of_init_path(OF_INIT_DEFAULT_PATH);
|
||||
}
|
||||
|
||||
/* Read a numeric property according to its size and return it as a 64-bit
|
||||
* value.
|
||||
*/
|
||||
static inline uint64_t of_read_number(const __be32 *cell, int size)
|
||||
{
|
||||
uint64_t r = 0;
|
||||
|
||||
while (size--)
|
||||
r = (r << 32) | be32toh(*(cell++));
|
||||
return r;
|
||||
}
|
||||
|
||||
#endif /* __OF_H */
|
Loading…
x
Reference in New Issue
Block a user