437 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			437 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
/*
 | 
						|
 * Copyright (c) 2006-2022, RT-Thread Development Team
 | 
						|
 *
 | 
						|
 * SPDX-License-Identifier: Apache-2.0
 | 
						|
 *
 | 
						|
 * Change Logs:
 | 
						|
 * Date           Author       Notes
 | 
						|
 * 2022-08-25     GuEe-GUI     first version
 | 
						|
 */
 | 
						|
 | 
						|
#include <rtthread.h>
 | 
						|
 | 
						|
#include <drivers/ofw.h>
 | 
						|
#include <drivers/ofw_io.h>
 | 
						|
#include <drivers/ofw_fdt.h>
 | 
						|
 | 
						|
#define DBG_TAG "rtdm.ofw"
 | 
						|
#define DBG_LVL DBG_INFO
 | 
						|
#include <rtdbg.h>
 | 
						|
 | 
						|
#include "ofw_internal.h"
 | 
						|
 | 
						|
static int ofw_bus_addr_cells(struct rt_ofw_node *np)
 | 
						|
{
 | 
						|
    int res = OFW_ROOT_NODE_ADDR_CELLS_DEFAULT;
 | 
						|
 | 
						|
    for (rt_uint32_t cells; np; np = np->parent)
 | 
						|
    {
 | 
						|
        if (!rt_ofw_prop_read_u32(np, "#address-cells", &cells))
 | 
						|
        {
 | 
						|
            res = cells;
 | 
						|
            break;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    return res;
 | 
						|
}
 | 
						|
 | 
						|
static int ofw_bus_size_cells(struct rt_ofw_node *np)
 | 
						|
{
 | 
						|
    int res = OFW_ROOT_NODE_SIZE_CELLS_DEFAULT;
 | 
						|
 | 
						|
    for (rt_uint32_t cells; np; np = np->parent)
 | 
						|
    {
 | 
						|
        if (!rt_ofw_prop_read_u32(np, "#size-cells", &cells))
 | 
						|
        {
 | 
						|
            res = cells;
 | 
						|
            break;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    return res;
 | 
						|
}
 | 
						|
 | 
						|
int rt_ofw_bus_addr_cells(struct rt_ofw_node *np)
 | 
						|
{
 | 
						|
    return np ? ofw_bus_addr_cells(np) : -RT_EINVAL;
 | 
						|
}
 | 
						|
 | 
						|
int rt_ofw_bus_size_cells(struct rt_ofw_node *np)
 | 
						|
{
 | 
						|
    return np ? ofw_bus_size_cells(np) : -RT_EINVAL;
 | 
						|
}
 | 
						|
 | 
						|
int rt_ofw_io_addr_cells(struct rt_ofw_node *np)
 | 
						|
{
 | 
						|
    return np ? ofw_bus_addr_cells(np->parent ? np->parent : np) : -RT_EINVAL;
 | 
						|
}
 | 
						|
 | 
						|
int rt_ofw_io_size_cells(struct rt_ofw_node *np)
 | 
						|
{
 | 
						|
    return np ? ofw_bus_size_cells(np->parent ? np->parent : np) : -RT_EINVAL;
 | 
						|
}
 | 
						|
 | 
						|
int rt_ofw_get_address_count(struct rt_ofw_node *np)
 | 
						|
{
 | 
						|
    int count;
 | 
						|
 | 
						|
    if (np)
 | 
						|
    {
 | 
						|
        rt_ssize_t len;
 | 
						|
 | 
						|
        count = 0;
 | 
						|
 | 
						|
        if (rt_ofw_get_prop(np, "reg", &len))
 | 
						|
        {
 | 
						|
            count = len / (sizeof(fdt32_t) * (rt_ofw_io_addr_cells(np) + rt_ofw_io_size_cells(np)));
 | 
						|
        }
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        count = -RT_EINVAL;
 | 
						|
    }
 | 
						|
 | 
						|
    return count;
 | 
						|
}
 | 
						|
 | 
						|
static rt_err_t ofw_get_address(struct rt_ofw_node *np, int index, rt_uint64_t *out_address, rt_uint64_t *out_size)
 | 
						|
{
 | 
						|
    rt_ssize_t len;
 | 
						|
    rt_err_t err = RT_EOK;
 | 
						|
    int addr_cells = rt_ofw_io_addr_cells(np);
 | 
						|
    int size_cells = rt_ofw_io_size_cells(np);
 | 
						|
    int skip_cells = (addr_cells + size_cells) * index;
 | 
						|
    const fdt32_t *cell = rt_ofw_prop_read_raw(np, "reg", &len);
 | 
						|
 | 
						|
    if (cell && skip_cells < (len / sizeof(*cell)))
 | 
						|
    {
 | 
						|
        cell += skip_cells;
 | 
						|
        *out_address = rt_fdt_next_cell(&cell, addr_cells);
 | 
						|
        *out_address = rt_ofw_translate_address(np, RT_NULL, *out_address);
 | 
						|
        *out_size = rt_fdt_read_number(cell, size_cells);
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        err = -RT_EINVAL;
 | 
						|
    }
 | 
						|
 | 
						|
    return err;
 | 
						|
}
 | 
						|
 | 
						|
rt_err_t rt_ofw_get_address(struct rt_ofw_node *np, int index, rt_uint64_t *out_address, rt_uint64_t *out_size)
 | 
						|
{
 | 
						|
    rt_err_t err;
 | 
						|
 | 
						|
    if (np && index >= 0 && (out_address || out_size))
 | 
						|
    {
 | 
						|
        rt_uint64_t address, size;
 | 
						|
 | 
						|
        err = ofw_get_address(np, index, &address, &size);
 | 
						|
 | 
						|
        if (!err)
 | 
						|
        {
 | 
						|
            if (out_address)
 | 
						|
            {
 | 
						|
                *out_address = address;
 | 
						|
            }
 | 
						|
            if (out_size)
 | 
						|
            {
 | 
						|
                *out_size = size;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        err = -RT_EINVAL;
 | 
						|
    }
 | 
						|
 | 
						|
    return err;
 | 
						|
}
 | 
						|
 | 
						|
static rt_err_t ofw_get_address_by_name(struct rt_ofw_node *np, const char *name,
 | 
						|
        rt_uint64_t *out_address, rt_uint64_t *out_size)
 | 
						|
 | 
						|
{
 | 
						|
    int index = 0;
 | 
						|
    rt_err_t err = -RT_EEMPTY;
 | 
						|
    const char *reg_name;
 | 
						|
    struct rt_ofw_prop *prop;
 | 
						|
 | 
						|
    rt_ofw_foreach_prop_string(np, "reg-names", prop, reg_name)
 | 
						|
    {
 | 
						|
        if (!rt_strcmp(name, reg_name))
 | 
						|
        {
 | 
						|
            err = rt_ofw_get_address(np, index, out_address, out_size);
 | 
						|
 | 
						|
            break;
 | 
						|
        }
 | 
						|
 | 
						|
        ++index;
 | 
						|
    }
 | 
						|
 | 
						|
    return err;
 | 
						|
}
 | 
						|
 | 
						|
rt_err_t rt_ofw_get_address_by_name(struct rt_ofw_node *np, const char *name,
 | 
						|
        rt_uint64_t *out_address, rt_uint64_t *out_size)
 | 
						|
{
 | 
						|
    rt_err_t err;
 | 
						|
 | 
						|
    if (np && name && (out_address || out_size))
 | 
						|
    {
 | 
						|
        rt_uint64_t address, size;
 | 
						|
 | 
						|
        err = ofw_get_address_by_name(np, name, &address, &size);
 | 
						|
 | 
						|
        if (!err)
 | 
						|
        {
 | 
						|
            if (out_address)
 | 
						|
            {
 | 
						|
                *out_address = address;
 | 
						|
            }
 | 
						|
            if (out_size)
 | 
						|
            {
 | 
						|
                *out_size = size;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        err = -RT_EINVAL;
 | 
						|
    }
 | 
						|
 | 
						|
    return err;
 | 
						|
}
 | 
						|
 | 
						|
int rt_ofw_get_address_array(struct rt_ofw_node *np, int nr, rt_uint64_t *out_regs)
 | 
						|
{
 | 
						|
    int count;
 | 
						|
 | 
						|
    if (np && nr > 0 && out_regs)
 | 
						|
    {
 | 
						|
        rt_ssize_t len;
 | 
						|
        int max_nr;
 | 
						|
        int addr_cells = rt_ofw_io_addr_cells(np);
 | 
						|
        int size_cells = rt_ofw_io_size_cells(np);
 | 
						|
        const fdt32_t *cell = rt_ofw_prop_read_raw(np, "reg", &len);
 | 
						|
 | 
						|
        max_nr = len / (sizeof(*cell) * (addr_cells + size_cells));
 | 
						|
 | 
						|
        if (nr > max_nr)
 | 
						|
        {
 | 
						|
            nr = max_nr;
 | 
						|
        }
 | 
						|
 | 
						|
        count = nr;
 | 
						|
 | 
						|
        while (nr --> 0)
 | 
						|
        {
 | 
						|
            *out_regs = rt_fdt_next_cell(&cell, addr_cells);
 | 
						|
            *out_regs = rt_ofw_translate_address(np, RT_NULL, *out_regs);
 | 
						|
            ++out_regs;
 | 
						|
 | 
						|
            *out_regs = rt_fdt_next_cell(&cell, size_cells);
 | 
						|
            ++out_regs;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        count = -RT_EINVAL;
 | 
						|
    }
 | 
						|
 | 
						|
    return count;
 | 
						|
}
 | 
						|
 | 
						|
static struct bus_ranges *ofw_bus_ranges(struct rt_ofw_node *np, struct rt_ofw_prop *prop)
 | 
						|
{
 | 
						|
    const fdt32_t *cell;
 | 
						|
    struct bus_ranges *ranges = RT_NULL;
 | 
						|
    int child_address_cells, child_size_cells, parent_address_cells, groups;
 | 
						|
    rt_uint64_t *child_addr, *parent_addr, *child_size;
 | 
						|
 | 
						|
    /*
 | 
						|
     * Address Translation Example:
 | 
						|
     *
 | 
						|
     *  / {
 | 
						|
     *      #address-cells = <1>;
 | 
						|
     *      #size-cells = <1>;
 | 
						|
     *
 | 
						|
     *      soc {
 | 
						|
     *          compatible = "simple-bus";
 | 
						|
     *          #address-cells = <1>;
 | 
						|
     *          #size-cells = <1>;
 | 
						|
     *          ranges = <0x0 0xe0000000 0x00100000>;
 | 
						|
     *
 | 
						|
     *          serial@4600 {
 | 
						|
     *              device_type = "serial";
 | 
						|
     *              reg = <0x4600 0x100>;
 | 
						|
     *              clock-frequency = <0>;
 | 
						|
     *          };
 | 
						|
     *      };
 | 
						|
     *  }
 | 
						|
     *
 | 
						|
     * The soc node specifies a ranges property of <0x0 0xe0000000 0x00100000>;
 | 
						|
     * This property value specifies that for a 1024 KB range of address space, a
 | 
						|
     * child node addressed at physical 0x0 maps to a parent address of physical
 | 
						|
     * 0xe0000000. With this mapping, the serial device node can be addressed by a
 | 
						|
     * load or store at address 0xe0004600, an offset of 0x4600 (specified in reg)
 | 
						|
     * plus the 0xe0000000 mapping specified in ranges:
 | 
						|
     *
 | 
						|
     *      bus-address = parent-bus-address + (reg-address - child-bus-address)
 | 
						|
     */
 | 
						|
 | 
						|
    do {
 | 
						|
        child_address_cells = rt_ofw_bus_addr_cells(np);
 | 
						|
        child_size_cells = rt_ofw_bus_size_cells(np);
 | 
						|
        parent_address_cells = rt_ofw_io_addr_cells(np);
 | 
						|
 | 
						|
        if (child_address_cells < 0 || child_size_cells < 0 || parent_address_cells < 0)
 | 
						|
        {
 | 
						|
            LOG_D("%s read address/size cells fail: child[%d, %d] parent[%d]",
 | 
						|
                    np->full_name, child_address_cells, child_size_cells, parent_address_cells);
 | 
						|
 | 
						|
            break;
 | 
						|
        }
 | 
						|
 | 
						|
        groups = prop->length / sizeof(*cell);
 | 
						|
        groups /= child_address_cells + child_size_cells + parent_address_cells;
 | 
						|
 | 
						|
        ranges = rt_malloc(sizeof(*ranges) + sizeof(rt_uint64_t) * 3 * groups);
 | 
						|
 | 
						|
        if (!ranges)
 | 
						|
        {
 | 
						|
            break;
 | 
						|
        }
 | 
						|
 | 
						|
        ranges->nr = groups;
 | 
						|
        ranges->child_addr = (void *)ranges + sizeof(*ranges);
 | 
						|
        ranges->parent_addr = &ranges->child_addr[groups];
 | 
						|
        ranges->child_size = &ranges->parent_addr[groups];
 | 
						|
 | 
						|
        cell = prop->value;
 | 
						|
 | 
						|
        child_addr = ranges->child_addr;
 | 
						|
        parent_addr = ranges->parent_addr;
 | 
						|
        child_size = ranges->child_size;
 | 
						|
 | 
						|
        while (groups --> 0)
 | 
						|
        {
 | 
						|
            *child_addr++ = rt_fdt_next_cell(&cell, child_address_cells);
 | 
						|
            *parent_addr++ = rt_fdt_next_cell(&cell, parent_address_cells);
 | 
						|
            *child_size++ = rt_fdt_next_cell(&cell, child_size_cells);
 | 
						|
        }
 | 
						|
 | 
						|
        rt_ofw_data(np) = ranges;
 | 
						|
    } while (0);
 | 
						|
 | 
						|
    return ranges;
 | 
						|
}
 | 
						|
 | 
						|
rt_uint64_t rt_ofw_translate_address(struct rt_ofw_node *np, const char *range_type, rt_uint64_t address)
 | 
						|
{
 | 
						|
    rt_uint64_t cpu_addr = address;
 | 
						|
 | 
						|
    if (!range_type)
 | 
						|
    {
 | 
						|
        range_type = "ranges";
 | 
						|
    }
 | 
						|
 | 
						|
    rt_ofw_foreach_parent_node(np)
 | 
						|
    {
 | 
						|
        rt_ssize_t len;
 | 
						|
        struct rt_ofw_prop *prop;
 | 
						|
        struct bus_ranges *ranges;
 | 
						|
 | 
						|
        prop = rt_ofw_get_prop(np, range_type, &len);
 | 
						|
 | 
						|
        if (!prop || !len)
 | 
						|
        {
 | 
						|
            continue;
 | 
						|
        }
 | 
						|
 | 
						|
        ranges = rt_ofw_data(np);
 | 
						|
 | 
						|
        if (!ranges)
 | 
						|
        {
 | 
						|
            ranges = ofw_bus_ranges(np, prop);
 | 
						|
        }
 | 
						|
 | 
						|
        if (ranges)
 | 
						|
        {
 | 
						|
            for (int i = 0; i < ranges->nr; ++i)
 | 
						|
            {
 | 
						|
                rt_uint64_t child_addr = ranges->child_addr[i];
 | 
						|
                rt_uint64_t child_size = ranges->child_size[i];
 | 
						|
 | 
						|
                if (address >= child_addr && address < child_addr + child_size)
 | 
						|
                {
 | 
						|
                    cpu_addr = address + (ranges->parent_addr[i] - child_addr);
 | 
						|
 | 
						|
                    break;
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
        else
 | 
						|
        {
 | 
						|
            cpu_addr = ~0ULL;
 | 
						|
        }
 | 
						|
 | 
						|
        rt_ofw_node_put(np);
 | 
						|
 | 
						|
        break;
 | 
						|
    }
 | 
						|
 | 
						|
    return cpu_addr;
 | 
						|
}
 | 
						|
 | 
						|
#ifdef ARCH_CPU_64BIT
 | 
						|
#define ofw_address_cpu_cast(np, address) (void *)(address)
 | 
						|
#else
 | 
						|
#define ofw_address_cpu_cast(np, address)                       \
 | 
						|
({                                                              \
 | 
						|
    if (((address) >> 32))                                      \
 | 
						|
    {                                                           \
 | 
						|
        LOG_W("%s find 64 bits address = %x%x",                 \
 | 
						|
                rt_ofw_node_full_name(np),                      \
 | 
						|
                ofw_static_cast(rt_ubase_t, (address) >> 32),   \
 | 
						|
                ofw_static_cast(rt_ubase_t, (address)));        \
 | 
						|
    }                                                           \
 | 
						|
    (void *)ofw_static_cast(rt_ubase_t, (address));             \
 | 
						|
})
 | 
						|
#endif
 | 
						|
 | 
						|
void *rt_ofw_iomap(struct rt_ofw_node *np, int index)
 | 
						|
{
 | 
						|
    void *iomem = RT_NULL;
 | 
						|
 | 
						|
    if (np)
 | 
						|
    {
 | 
						|
        rt_uint64_t regs[2];
 | 
						|
 | 
						|
        if (!ofw_get_address(np, index, ®s[0], ®s[1]))
 | 
						|
        {
 | 
						|
            iomem = rt_ioremap(ofw_address_cpu_cast(np, regs[0]), (size_t)regs[1]);
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    return iomem;
 | 
						|
}
 | 
						|
 | 
						|
void *rt_ofw_iomap_by_name(struct rt_ofw_node *np, const char *name)
 | 
						|
{
 | 
						|
    void *iomem = RT_NULL;
 | 
						|
 | 
						|
    if (np)
 | 
						|
    {
 | 
						|
        rt_uint64_t regs[2];
 | 
						|
 | 
						|
        if (!ofw_get_address_by_name(np, name, ®s[0], ®s[1]))
 | 
						|
        {
 | 
						|
            iomem = rt_ioremap(ofw_address_cpu_cast(np, regs[0]), (size_t)regs[1]);
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    return iomem;
 | 
						|
}
 |