mirror of
https://github.com/AsahiLinux/u-boot
synced 2024-11-10 15:14:43 +00:00
dm: core: add ofnode and dev function to iterate on node property
Add functions to iterate on all property with livetree - dev_read_first_prop - dev_read_next_prop - dev_read_prop_by_prop and - ofnode_get_first_property - ofnode_get_next_property - ofnode_get_property_by_prop And helper: dev_for_each_property For example: struct ofprop property; dev_for_each_property(property, config) { value = dev_read_prop_by_prop(&property, &propname, &len); or: for (res = ofnode_get_first_property(node, &property); !res; res = ofnode_get_next_property(&property)) { value = ofnode_get_property_by_prop(&property, &propname, &len); .... } Signed-off-by: Patrick Delaunay <patrick.delaunay@st.com> Reviewed-by: Simon Glass <sjg@chromium.org>
This commit is contained in:
parent
719cab6d2e
commit
ce891fcada
8 changed files with 317 additions and 2 deletions
|
@ -171,6 +171,38 @@ const void *of_get_property(const struct device_node *np, const char *name,
|
|||
return pp ? pp->value : NULL;
|
||||
}
|
||||
|
||||
const struct property *of_get_first_property(const struct device_node *np)
|
||||
{
|
||||
if (!np)
|
||||
return NULL;
|
||||
|
||||
return np->properties;
|
||||
}
|
||||
|
||||
const struct property *of_get_next_property(const struct device_node *np,
|
||||
const struct property *property)
|
||||
{
|
||||
if (!np)
|
||||
return NULL;
|
||||
|
||||
return property->next;
|
||||
}
|
||||
|
||||
const void *of_get_property_by_prop(const struct device_node *np,
|
||||
const struct property *property,
|
||||
const char **name,
|
||||
int *lenp)
|
||||
{
|
||||
if (!np || !property)
|
||||
return NULL;
|
||||
if (name)
|
||||
*name = property->name;
|
||||
if (lenp)
|
||||
*lenp = property->length;
|
||||
|
||||
return property->value;
|
||||
}
|
||||
|
||||
static const char *of_prop_next_string(struct property *prop, const char *cur)
|
||||
{
|
||||
const void *curv = cur;
|
||||
|
|
|
@ -571,6 +571,54 @@ const void *ofnode_get_property(ofnode node, const char *propname, int *lenp)
|
|||
propname, lenp);
|
||||
}
|
||||
|
||||
int ofnode_get_first_property(ofnode node, struct ofprop *prop)
|
||||
{
|
||||
prop->node = node;
|
||||
|
||||
if (ofnode_is_np(node)) {
|
||||
prop->prop = of_get_first_property(ofnode_to_np(prop->node));
|
||||
if (!prop->prop)
|
||||
return -FDT_ERR_NOTFOUND;
|
||||
} else {
|
||||
prop->offset =
|
||||
fdt_first_property_offset(gd->fdt_blob,
|
||||
ofnode_to_offset(prop->node));
|
||||
if (prop->offset < 0)
|
||||
return prop->offset;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int ofnode_get_next_property(struct ofprop *prop)
|
||||
{
|
||||
if (ofnode_is_np(prop->node)) {
|
||||
prop->prop = of_get_next_property(ofnode_to_np(prop->node),
|
||||
prop->prop);
|
||||
if (!prop->prop)
|
||||
return -FDT_ERR_NOTFOUND;
|
||||
} else {
|
||||
prop->offset = fdt_next_property_offset(gd->fdt_blob,
|
||||
prop->offset);
|
||||
if (prop->offset < 0)
|
||||
return prop->offset;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
const void *ofnode_get_property_by_prop(const struct ofprop *prop,
|
||||
const char **propname, int *lenp)
|
||||
{
|
||||
if (ofnode_is_np(prop->node))
|
||||
return of_get_property_by_prop(ofnode_to_np(prop->node),
|
||||
prop->prop, propname, lenp);
|
||||
else
|
||||
return fdt_getprop_by_offset(gd->fdt_blob,
|
||||
prop->offset,
|
||||
propname, lenp);
|
||||
}
|
||||
|
||||
bool ofnode_is_available(ofnode node)
|
||||
{
|
||||
if (ofnode_is_np(node))
|
||||
|
|
|
@ -255,6 +255,22 @@ const void *dev_read_prop(const struct udevice *dev, const char *propname,
|
|||
return ofnode_get_property(dev_ofnode(dev), propname, lenp);
|
||||
}
|
||||
|
||||
int dev_read_first_prop(const struct udevice *dev, struct ofprop *prop)
|
||||
{
|
||||
return ofnode_get_first_property(dev_ofnode(dev), prop);
|
||||
}
|
||||
|
||||
int dev_read_next_prop(struct ofprop *prop)
|
||||
{
|
||||
return ofnode_get_next_property(prop);
|
||||
}
|
||||
|
||||
const void *dev_read_prop_by_prop(struct ofprop *prop,
|
||||
const char **propname, int *lenp)
|
||||
{
|
||||
return ofnode_get_property_by_prop(prop, propname, lenp);
|
||||
}
|
||||
|
||||
int dev_read_alias_seq(const struct udevice *dev, int *devnump)
|
||||
{
|
||||
ofnode node = dev_ofnode(dev);
|
||||
|
|
|
@ -103,6 +103,46 @@ struct property *of_find_property(const struct device_node *np,
|
|||
const void *of_get_property(const struct device_node *np, const char *name,
|
||||
int *lenp);
|
||||
|
||||
/**
|
||||
* of_get_first_property()- get to the pointer of the first property
|
||||
*
|
||||
* Get pointer to the first property of the node, it is used to iterate
|
||||
* and read all the property with of_get_next_property_by_prop().
|
||||
*
|
||||
* @np: Pointer to device node
|
||||
* @return pointer to property or NULL if not found
|
||||
*/
|
||||
const struct property *of_get_first_property(const struct device_node *np);
|
||||
|
||||
/**
|
||||
* of_get_next_property() - get to the pointer of the next property
|
||||
*
|
||||
* Get pointer to the next property of the node, it is used to iterate
|
||||
* and read all the property with of_get_property_by_prop().
|
||||
*
|
||||
* @np: Pointer to device node
|
||||
* @property: pointer of the current property
|
||||
* @return pointer to next property or NULL if not found
|
||||
*/
|
||||
const struct property *of_get_next_property(const struct device_node *np,
|
||||
const struct property *property);
|
||||
|
||||
/**
|
||||
* of_get_property_by_prop() - get a property value of a node property
|
||||
*
|
||||
* Get value for the property identified by node and property pointer.
|
||||
*
|
||||
* @node: node to read
|
||||
* @property: pointer of the property to read
|
||||
* @propname: place to property name on success
|
||||
* @lenp: place to put length on success
|
||||
* @return pointer to property value or NULL if error
|
||||
*/
|
||||
const void *of_get_property_by_prop(const struct device_node *np,
|
||||
const struct property *property,
|
||||
const char **name,
|
||||
int *lenp);
|
||||
|
||||
/**
|
||||
* of_device_is_compatible() - Check if the node matches given constraints
|
||||
* @device: pointer to node
|
||||
|
|
|
@ -58,6 +58,31 @@ struct ofnode_phandle_args {
|
|||
uint32_t args[OF_MAX_PHANDLE_ARGS];
|
||||
};
|
||||
|
||||
/**
|
||||
* ofprop - reference to a property of a device tree node
|
||||
*
|
||||
* This struct hold the reference on one property of one node,
|
||||
* using struct ofnode and an offset within the flat device tree or either
|
||||
* a pointer to a struct property in the live device tree.
|
||||
*
|
||||
* Thus we can reference arguments in both the live tree and the flat tree.
|
||||
*
|
||||
* The property reference can also hold a null reference. This corresponds to
|
||||
* a struct property NULL pointer or an offset of -1.
|
||||
*
|
||||
* @node: Pointer to device node
|
||||
* @offset: Pointer into flat device tree, used for flat tree.
|
||||
* @prop: Pointer to property, used for live treee.
|
||||
*/
|
||||
|
||||
struct ofprop {
|
||||
ofnode node;
|
||||
union {
|
||||
int offset;
|
||||
const struct property *prop;
|
||||
};
|
||||
};
|
||||
|
||||
/**
|
||||
* _ofnode_to_np() - convert an ofnode to a live DT node pointer
|
||||
*
|
||||
|
@ -595,7 +620,7 @@ int ofnode_decode_display_timing(ofnode node, int index,
|
|||
struct display_timing *config);
|
||||
|
||||
/**
|
||||
* ofnode_get_property()- - get a pointer to the value of a node property
|
||||
* ofnode_get_property() - get a pointer to the value of a node property
|
||||
*
|
||||
* @node: node to read
|
||||
* @propname: property to read
|
||||
|
@ -604,6 +629,42 @@ int ofnode_decode_display_timing(ofnode node, int index,
|
|||
*/
|
||||
const void *ofnode_get_property(ofnode node, const char *propname, int *lenp);
|
||||
|
||||
/**
|
||||
* ofnode_get_first_property()- get the reference of the first property
|
||||
*
|
||||
* Get reference to the first property of the node, it is used to iterate
|
||||
* and read all the property with ofnode_get_property_by_prop().
|
||||
*
|
||||
* @node: node to read
|
||||
* @prop: place to put argument reference
|
||||
* @return 0 if OK, -ve on error. -FDT_ERR_NOTFOUND if not found
|
||||
*/
|
||||
int ofnode_get_first_property(ofnode node, struct ofprop *prop);
|
||||
|
||||
/**
|
||||
* ofnode_get_next_property() - get the reference of the next property
|
||||
*
|
||||
* Get reference to the next property of the node, it is used to iterate
|
||||
* and read all the property with ofnode_get_property_by_prop().
|
||||
*
|
||||
* @prop: reference of current argument and place to put reference of next one
|
||||
* @return 0 if OK, -ve on error. -FDT_ERR_NOTFOUND if not found
|
||||
*/
|
||||
int ofnode_get_next_property(struct ofprop *prop);
|
||||
|
||||
/**
|
||||
* ofnode_get_property_by_prop() - get a pointer to the value of a property
|
||||
*
|
||||
* Get value for the property identified by the provided reference.
|
||||
*
|
||||
* @prop: reference on property
|
||||
* @propname: If non-NULL, place to property name on success,
|
||||
* @lenp: If non-NULL, place to put length on success
|
||||
* @return 0 if OK, -ve on error. -FDT_ERR_NOTFOUND if not found
|
||||
*/
|
||||
const void *ofnode_get_property_by_prop(const struct ofprop *prop,
|
||||
const char **propname, int *lenp);
|
||||
|
||||
/**
|
||||
* ofnode_is_available() - check if a node is marked available
|
||||
*
|
||||
|
|
|
@ -494,6 +494,42 @@ int dev_read_phandle(const struct udevice *dev);
|
|||
const void *dev_read_prop(const struct udevice *dev, const char *propname,
|
||||
int *lenp);
|
||||
|
||||
/**
|
||||
* dev_read_first_prop()- get the reference of the first property
|
||||
*
|
||||
* Get reference to the first property of the node, it is used to iterate
|
||||
* and read all the property with dev_read_prop_by_prop().
|
||||
*
|
||||
* @dev: device to check
|
||||
* @prop: place to put argument reference
|
||||
* @return 0 if OK, -ve on error. -FDT_ERR_NOTFOUND if not found
|
||||
*/
|
||||
int dev_read_first_prop(const struct udevice *dev, struct ofprop *prop);
|
||||
|
||||
/**
|
||||
* ofnode_get_next_property() - get the reference of the next property
|
||||
*
|
||||
* Get reference to the next property of the node, it is used to iterate
|
||||
* and read all the property with dev_read_prop_by_prop().
|
||||
*
|
||||
* @prop: reference of current argument and place to put reference of next one
|
||||
* @return 0 if OK, -ve on error. -FDT_ERR_NOTFOUND if not found
|
||||
*/
|
||||
int dev_read_next_prop(struct ofprop *prop);
|
||||
|
||||
/**
|
||||
* dev_read_prop_by_prop() - get a pointer to the value of a property
|
||||
*
|
||||
* Get value for the property identified by the provided reference.
|
||||
*
|
||||
* @prop: reference on property
|
||||
* @propname: If non-NULL, place to property name on success,
|
||||
* @lenp: If non-NULL, place to put length on success
|
||||
* @return 0 if OK, -ve on error. -FDT_ERR_NOTFOUND if not found
|
||||
*/
|
||||
const void *dev_read_prop_by_prop(struct ofprop *prop,
|
||||
const char **propname, int *lenp);
|
||||
|
||||
/**
|
||||
* dev_read_alias_seq() - Get the alias sequence number of a node
|
||||
*
|
||||
|
@ -860,6 +896,23 @@ static inline const void *dev_read_prop(const struct udevice *dev,
|
|||
return ofnode_get_property(dev_ofnode(dev), propname, lenp);
|
||||
}
|
||||
|
||||
static inline int dev_read_first_prop(const struct udevice *dev, struct ofprop *prop)
|
||||
{
|
||||
return ofnode_get_first_property(dev_ofnode(dev), prop);
|
||||
}
|
||||
|
||||
static inline int dev_read_next_prop(struct ofprop *prop)
|
||||
{
|
||||
return ofnode_get_next_property(prop);
|
||||
}
|
||||
|
||||
static inline const void *dev_read_prop_by_prop(struct ofprop *prop,
|
||||
const char **propname,
|
||||
int *lenp)
|
||||
{
|
||||
return ofnode_get_property_by_prop(prop, propname, lenp);
|
||||
}
|
||||
|
||||
static inline int dev_read_alias_seq(const struct udevice *dev, int *devnump)
|
||||
{
|
||||
return fdtdec_get_alias_seq(gd->fdt_blob, dev->uclass->uc_drv->name,
|
||||
|
@ -941,4 +994,18 @@ static inline int dev_read_alias_highest_id(const char *stem)
|
|||
ofnode_valid(subnode); \
|
||||
subnode = ofnode_next_subnode(subnode))
|
||||
|
||||
/**
|
||||
* dev_for_each_property() - Helper function to iterate through property
|
||||
*
|
||||
* This creates a for() loop which works through the property in a device's
|
||||
* device-tree node.
|
||||
*
|
||||
* @prop: struct ofprop holding the current property
|
||||
* @dev: device to use for interation (struct udevice *)
|
||||
*/
|
||||
#define dev_for_each_property(prop, dev) \
|
||||
for (int ret_prop = dev_read_first_prop(dev, &prop); \
|
||||
!ret_prop; \
|
||||
ret_prop = dev_read_next_prop(&prop))
|
||||
|
||||
#endif
|
||||
|
|
|
@ -31,8 +31,9 @@ obj-y += irq.o
|
|||
obj-$(CONFIG_LED) += led.o
|
||||
obj-$(CONFIG_DM_MAILBOX) += mailbox.o
|
||||
obj-$(CONFIG_DM_MMC) += mmc.o
|
||||
obj-y += ofnode.o
|
||||
obj-y += fdtdec.o
|
||||
obj-y += ofnode.o
|
||||
obj-y += ofread.o
|
||||
obj-$(CONFIG_OSD) += osd.o
|
||||
obj-$(CONFIG_DM_VIDEO) += panel.o
|
||||
obj-$(CONFIG_DM_PCI) += pci.o
|
||||
|
|
50
test/dm/ofread.c
Normal file
50
test/dm/ofread.c
Normal file
|
@ -0,0 +1,50 @@
|
|||
// SPDX-License-Identifier: GPL-2.0+
|
||||
|
||||
#include <common.h>
|
||||
#include <dm.h>
|
||||
#include <dm/test.h>
|
||||
#include <test/ut.h>
|
||||
|
||||
static int dm_test_ofnode_get_property_by_prop(struct unit_test_state *uts)
|
||||
{
|
||||
ofnode node;
|
||||
struct ofprop prop;
|
||||
const void *value;
|
||||
const char *propname;
|
||||
int res, len, count = 0;
|
||||
|
||||
node = ofnode_path("/cros-ec/flash");
|
||||
for (res = ofnode_get_first_property(node, &prop);
|
||||
!res;
|
||||
res = ofnode_get_next_property(&prop)) {
|
||||
value = ofnode_get_property_by_prop(&prop, &propname, &len);
|
||||
ut_assertnonnull(value);
|
||||
switch (count) {
|
||||
case 0:
|
||||
ut_asserteq_str("image-pos", propname);
|
||||
ut_asserteq(4, len);
|
||||
break;
|
||||
case 1:
|
||||
ut_asserteq_str("size", propname);
|
||||
ut_asserteq(4, len);
|
||||
break;
|
||||
case 2:
|
||||
ut_asserteq_str("erase-value", propname);
|
||||
ut_asserteq(4, len);
|
||||
break;
|
||||
case 3:
|
||||
/* only for platdata */
|
||||
ut_asserteq_str("name", propname);
|
||||
ut_asserteq(6, len);
|
||||
ut_asserteq_str("flash", value);
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
count++;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
DM_TEST(dm_test_ofnode_get_property_by_prop,
|
||||
DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);
|
Loading…
Reference in a new issue