blob: 7c4a53d20e6dc5ad30fa192ef657904fc08c3254 [file] [log] [blame]
Rafael J. Wysockib31384f2014-11-04 01:28:56 +01001/*
2 * property.c - Unified device property interface.
3 *
4 * Copyright (C) 2014, Intel Corporation
5 * Authors: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
6 * Mika Westerberg <mika.westerberg@linux.intel.com>
7 *
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License version 2 as
10 * published by the Free Software Foundation.
11 */
12
Rafael J. Wysockib31384f2014-11-04 01:28:56 +010013#include <linux/acpi.h>
Rafael J. Wysocki16ba08d2015-04-03 16:05:11 +020014#include <linux/export.h>
15#include <linux/kernel.h>
Rafael J. Wysockib31384f2014-11-04 01:28:56 +010016#include <linux/of.h>
Suthikulpanit, Suravee05ca5562015-06-10 11:08:54 -050017#include <linux/of_address.h>
Mika Westerberg07bb80d2017-03-28 10:52:21 +030018#include <linux/of_graph.h>
Rafael J. Wysocki16ba08d2015-04-03 16:05:11 +020019#include <linux/property.h>
Jeremy Linton4c96b7d2015-08-12 17:06:26 -050020#include <linux/etherdevice.h>
21#include <linux/phy.h>
Rafael J. Wysocki16ba08d2015-04-03 16:05:11 +020022
Heikki Krogerusf4d05262016-03-29 14:52:23 +030023struct property_set {
Jarkko Nikula5ab894a2017-10-09 16:28:37 +030024 struct device *dev;
Heikki Krogerusf4d05262016-03-29 14:52:23 +030025 struct fwnode_handle fwnode;
Dmitry Torokhovbec84da82017-02-02 17:41:25 -080026 const struct property_entry *properties;
Heikki Krogerusf4d05262016-03-29 14:52:23 +030027};
28
Sakari Ailusdb3e50f2017-07-21 14:39:31 +030029static const struct fwnode_operations pset_fwnode_ops;
30
Sakari Ailus39e5aee2017-07-21 14:39:35 +030031static inline bool is_pset_node(const struct fwnode_handle *fwnode)
Rafael J. Wysocki16ba08d2015-04-03 16:05:11 +020032{
Sakari Ailusdb3e50f2017-07-21 14:39:31 +030033 return !IS_ERR_OR_NULL(fwnode) && fwnode->ops == &pset_fwnode_ops;
Rafael J. Wysocki16ba08d2015-04-03 16:05:11 +020034}
35
Sakari Ailus39e5aee2017-07-21 14:39:35 +030036#define to_pset_node(__fwnode) \
37 ({ \
38 typeof(__fwnode) __to_pset_node_fwnode = __fwnode; \
39 \
40 is_pset_node(__to_pset_node_fwnode) ? \
41 container_of(__to_pset_node_fwnode, \
42 struct property_set, fwnode) : \
43 NULL; \
44 })
Rafael J. Wysocki16ba08d2015-04-03 16:05:11 +020045
Sakari Ailus39e5aee2017-07-21 14:39:35 +030046static const struct property_entry *
47pset_prop_get(const struct property_set *pset, const char *name)
Rafael J. Wysocki16ba08d2015-04-03 16:05:11 +020048{
Dmitry Torokhovbec84da82017-02-02 17:41:25 -080049 const struct property_entry *prop;
Rafael J. Wysocki16ba08d2015-04-03 16:05:11 +020050
51 if (!pset || !pset->properties)
52 return NULL;
53
54 for (prop = pset->properties; prop->name; prop++)
55 if (!strcmp(name, prop->name))
56 return prop;
57
58 return NULL;
59}
60
Sakari Ailus39e5aee2017-07-21 14:39:35 +030061static const void *pset_prop_find(const struct property_set *pset,
Dmitry Torokhovbec84da82017-02-02 17:41:25 -080062 const char *propname, size_t length)
Rafael J. Wysocki16ba08d2015-04-03 16:05:11 +020063{
Dmitry Torokhovbec84da82017-02-02 17:41:25 -080064 const struct property_entry *prop;
65 const void *pointer;
Rafael J. Wysocki16ba08d2015-04-03 16:05:11 +020066
Andy Shevchenko318a19712015-11-30 17:11:31 +020067 prop = pset_prop_get(pset, propname);
Rafael J. Wysocki16ba08d2015-04-03 16:05:11 +020068 if (!prop)
Andy Shevchenko318a19712015-11-30 17:11:31 +020069 return ERR_PTR(-EINVAL);
Andy Shevchenko66586ba2015-11-30 17:11:32 +020070 if (prop->is_array)
71 pointer = prop->pointer.raw_data;
72 else
73 pointer = &prop->value.raw_data;
Andy Shevchenko318a19712015-11-30 17:11:31 +020074 if (!pointer)
75 return ERR_PTR(-ENODATA);
76 if (length > prop->length)
77 return ERR_PTR(-EOVERFLOW);
78 return pointer;
79}
Rafael J. Wysocki16ba08d2015-04-03 16:05:11 +020080
Sakari Ailus39e5aee2017-07-21 14:39:35 +030081static int pset_prop_read_u8_array(const struct property_set *pset,
Andy Shevchenko318a19712015-11-30 17:11:31 +020082 const char *propname,
83 u8 *values, size_t nval)
84{
Dmitry Torokhovbec84da82017-02-02 17:41:25 -080085 const void *pointer;
Andy Shevchenko318a19712015-11-30 17:11:31 +020086 size_t length = nval * sizeof(*values);
Rafael J. Wysocki16ba08d2015-04-03 16:05:11 +020087
Andy Shevchenko318a19712015-11-30 17:11:31 +020088 pointer = pset_prop_find(pset, propname, length);
89 if (IS_ERR(pointer))
90 return PTR_ERR(pointer);
Rafael J. Wysocki16ba08d2015-04-03 16:05:11 +020091
Andy Shevchenko318a19712015-11-30 17:11:31 +020092 memcpy(values, pointer, length);
93 return 0;
94}
Rafael J. Wysocki16ba08d2015-04-03 16:05:11 +020095
Sakari Ailus39e5aee2017-07-21 14:39:35 +030096static int pset_prop_read_u16_array(const struct property_set *pset,
Andy Shevchenko318a19712015-11-30 17:11:31 +020097 const char *propname,
98 u16 *values, size_t nval)
99{
Dmitry Torokhovbec84da82017-02-02 17:41:25 -0800100 const void *pointer;
Andy Shevchenko318a19712015-11-30 17:11:31 +0200101 size_t length = nval * sizeof(*values);
102
103 pointer = pset_prop_find(pset, propname, length);
104 if (IS_ERR(pointer))
105 return PTR_ERR(pointer);
106
107 memcpy(values, pointer, length);
108 return 0;
109}
110
Sakari Ailus39e5aee2017-07-21 14:39:35 +0300111static int pset_prop_read_u32_array(const struct property_set *pset,
Andy Shevchenko318a19712015-11-30 17:11:31 +0200112 const char *propname,
113 u32 *values, size_t nval)
114{
Dmitry Torokhovbec84da82017-02-02 17:41:25 -0800115 const void *pointer;
Andy Shevchenko318a19712015-11-30 17:11:31 +0200116 size_t length = nval * sizeof(*values);
117
118 pointer = pset_prop_find(pset, propname, length);
119 if (IS_ERR(pointer))
120 return PTR_ERR(pointer);
121
122 memcpy(values, pointer, length);
123 return 0;
124}
125
Sakari Ailus39e5aee2017-07-21 14:39:35 +0300126static int pset_prop_read_u64_array(const struct property_set *pset,
Andy Shevchenko318a19712015-11-30 17:11:31 +0200127 const char *propname,
128 u64 *values, size_t nval)
129{
Dmitry Torokhovbec84da82017-02-02 17:41:25 -0800130 const void *pointer;
Andy Shevchenko318a19712015-11-30 17:11:31 +0200131 size_t length = nval * sizeof(*values);
132
133 pointer = pset_prop_find(pset, propname, length);
134 if (IS_ERR(pointer))
135 return PTR_ERR(pointer);
136
137 memcpy(values, pointer, length);
138 return 0;
139}
140
Sakari Ailus39e5aee2017-07-21 14:39:35 +0300141static int pset_prop_count_elems_of_size(const struct property_set *pset,
Andy Shevchenko318a19712015-11-30 17:11:31 +0200142 const char *propname, size_t length)
143{
Dmitry Torokhovbec84da82017-02-02 17:41:25 -0800144 const struct property_entry *prop;
Andy Shevchenko318a19712015-11-30 17:11:31 +0200145
146 prop = pset_prop_get(pset, propname);
147 if (!prop)
Rafael J. Wysocki16ba08d2015-04-03 16:05:11 +0200148 return -EINVAL;
Andy Shevchenko318a19712015-11-30 17:11:31 +0200149
150 return prop->length / length;
151}
152
Sakari Ailus39e5aee2017-07-21 14:39:35 +0300153static int pset_prop_read_string_array(const struct property_set *pset,
Andy Shevchenko318a19712015-11-30 17:11:31 +0200154 const char *propname,
155 const char **strings, size_t nval)
156{
Sakari Ailus0f194992017-03-28 15:22:18 +0300157 const struct property_entry *prop;
Dmitry Torokhovbec84da82017-02-02 17:41:25 -0800158 const void *pointer;
Sakari Ailus0f194992017-03-28 15:22:18 +0300159 size_t array_len, length;
160
161 /* Find out the array length. */
162 prop = pset_prop_get(pset, propname);
163 if (!prop)
164 return -EINVAL;
165
166 if (!prop->is_array)
167 /* The array length for a non-array string property is 1. */
168 array_len = 1;
169 else
170 /* Find the length of an array. */
171 array_len = pset_prop_count_elems_of_size(pset, propname,
172 sizeof(const char *));
173
174 /* Return how many there are if strings is NULL. */
175 if (!strings)
176 return array_len;
177
178 array_len = min(nval, array_len);
179 length = array_len * sizeof(*strings);
Andy Shevchenko318a19712015-11-30 17:11:31 +0200180
181 pointer = pset_prop_find(pset, propname, length);
182 if (IS_ERR(pointer))
183 return PTR_ERR(pointer);
184
185 memcpy(strings, pointer, length);
Sakari Ailus0f194992017-03-28 15:22:18 +0300186
Sakari Ailusb0b027c2017-03-28 15:22:19 +0300187 return array_len;
Rafael J. Wysocki16ba08d2015-04-03 16:05:11 +0200188}
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100189
Sakari Ailuse44bb0c2017-03-28 10:52:24 +0300190struct fwnode_handle *dev_fwnode(struct device *dev)
Rafael J. Wysocki9017f252015-03-24 00:24:16 +0100191{
192 return IS_ENABLED(CONFIG_OF) && dev->of_node ?
193 &dev->of_node->fwnode : dev->fwnode;
194}
Sakari Ailuse44bb0c2017-03-28 10:52:24 +0300195EXPORT_SYMBOL_GPL(dev_fwnode);
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100196
Sakari Ailus37ba9832017-07-21 14:39:36 +0300197static bool pset_fwnode_property_present(const struct fwnode_handle *fwnode,
Sakari Ailus37081842017-06-06 12:37:37 +0300198 const char *propname)
199{
200 return !!pset_prop_get(to_pset_node(fwnode), propname);
201}
202
Sakari Ailus37ba9832017-07-21 14:39:36 +0300203static int pset_fwnode_read_int_array(const struct fwnode_handle *fwnode,
Sakari Ailus37081842017-06-06 12:37:37 +0300204 const char *propname,
205 unsigned int elem_size, void *val,
206 size_t nval)
207{
Sakari Ailus37ba9832017-07-21 14:39:36 +0300208 const struct property_set *node = to_pset_node(fwnode);
Sakari Ailus37081842017-06-06 12:37:37 +0300209
210 if (!val)
211 return pset_prop_count_elems_of_size(node, propname, elem_size);
212
213 switch (elem_size) {
214 case sizeof(u8):
215 return pset_prop_read_u8_array(node, propname, val, nval);
216 case sizeof(u16):
217 return pset_prop_read_u16_array(node, propname, val, nval);
218 case sizeof(u32):
219 return pset_prop_read_u32_array(node, propname, val, nval);
220 case sizeof(u64):
221 return pset_prop_read_u64_array(node, propname, val, nval);
222 }
223
224 return -ENXIO;
225}
226
Sakari Ailus37ba9832017-07-21 14:39:36 +0300227static int
228pset_fwnode_property_read_string_array(const struct fwnode_handle *fwnode,
229 const char *propname,
230 const char **val, size_t nval)
Sakari Ailus37081842017-06-06 12:37:37 +0300231{
232 return pset_prop_read_string_array(to_pset_node(fwnode), propname,
233 val, nval);
234}
235
236static const struct fwnode_operations pset_fwnode_ops = {
237 .property_present = pset_fwnode_property_present,
238 .property_read_int_array = pset_fwnode_read_int_array,
239 .property_read_string_array = pset_fwnode_property_read_string_array,
240};
241
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100242/**
243 * device_property_present - check if a property of a device is present
244 * @dev: Device whose property is being checked
245 * @propname: Name of the property
246 *
247 * Check if property @propname is present in the device firmware description.
248 */
249bool device_property_present(struct device *dev, const char *propname)
250{
Rafael J. Wysocki9017f252015-03-24 00:24:16 +0100251 return fwnode_property_present(dev_fwnode(dev), propname);
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100252}
253EXPORT_SYMBOL_GPL(device_property_present);
254
Andy Shevchenko362c0b32015-11-30 17:11:36 +0200255/**
256 * fwnode_property_present - check if a property of a firmware node is present
257 * @fwnode: Firmware node whose property to check
258 * @propname: Name of the property
259 */
Sakari Ailus37ba9832017-07-21 14:39:36 +0300260bool fwnode_property_present(const struct fwnode_handle *fwnode,
261 const char *propname)
Andy Shevchenko362c0b32015-11-30 17:11:36 +0200262{
263 bool ret;
264
Sakari Ailuse8158b482017-07-11 18:20:20 +0300265 ret = fwnode_call_bool_op(fwnode, property_present, propname);
Heikki Krogerus0d67e0f2016-03-10 13:03:18 +0200266 if (ret == false && !IS_ERR_OR_NULL(fwnode) &&
267 !IS_ERR_OR_NULL(fwnode->secondary))
Sakari Ailuse8158b482017-07-11 18:20:20 +0300268 ret = fwnode_call_bool_op(fwnode->secondary, property_present,
Sakari Ailus37081842017-06-06 12:37:37 +0300269 propname);
Andy Shevchenko362c0b32015-11-30 17:11:36 +0200270 return ret;
271}
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100272EXPORT_SYMBOL_GPL(fwnode_property_present);
273
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100274/**
275 * device_property_read_u8_array - return a u8 array property of a device
276 * @dev: Device to get the property of
277 * @propname: Name of the property
Adrian Hunter5c0acf32015-03-17 09:58:58 +0200278 * @val: The values are stored here or %NULL to return the number of values
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100279 * @nval: Size of the @val array
280 *
281 * Function reads an array of u8 properties with @propname from the device
282 * firmware description and stores them to @val if found.
283 *
Adrian Hunter5c0acf32015-03-17 09:58:58 +0200284 * Return: number of values if @val was %NULL,
285 * %0 if the property was found (success),
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100286 * %-EINVAL if given arguments are not valid,
287 * %-ENODATA if the property does not have a value,
288 * %-EPROTO if the property is not an array of numbers,
289 * %-EOVERFLOW if the size of the property is not as expected.
Guenter Roeck4fa7508e2015-08-26 20:27:04 -0700290 * %-ENXIO if no suitable firmware interface is present.
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100291 */
292int device_property_read_u8_array(struct device *dev, const char *propname,
293 u8 *val, size_t nval)
294{
Rafael J. Wysocki9017f252015-03-24 00:24:16 +0100295 return fwnode_property_read_u8_array(dev_fwnode(dev), propname, val, nval);
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100296}
297EXPORT_SYMBOL_GPL(device_property_read_u8_array);
298
299/**
300 * device_property_read_u16_array - return a u16 array property of a device
301 * @dev: Device to get the property of
302 * @propname: Name of the property
Adrian Hunter5c0acf32015-03-17 09:58:58 +0200303 * @val: The values are stored here or %NULL to return the number of values
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100304 * @nval: Size of the @val array
305 *
306 * Function reads an array of u16 properties with @propname from the device
307 * firmware description and stores them to @val if found.
308 *
Adrian Hunter5c0acf32015-03-17 09:58:58 +0200309 * Return: number of values if @val was %NULL,
310 * %0 if the property was found (success),
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100311 * %-EINVAL if given arguments are not valid,
312 * %-ENODATA if the property does not have a value,
313 * %-EPROTO if the property is not an array of numbers,
314 * %-EOVERFLOW if the size of the property is not as expected.
Guenter Roeck4fa7508e2015-08-26 20:27:04 -0700315 * %-ENXIO if no suitable firmware interface is present.
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100316 */
317int device_property_read_u16_array(struct device *dev, const char *propname,
318 u16 *val, size_t nval)
319{
Rafael J. Wysocki9017f252015-03-24 00:24:16 +0100320 return fwnode_property_read_u16_array(dev_fwnode(dev), propname, val, nval);
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100321}
322EXPORT_SYMBOL_GPL(device_property_read_u16_array);
323
324/**
325 * device_property_read_u32_array - return a u32 array property of a device
326 * @dev: Device to get the property of
327 * @propname: Name of the property
Adrian Hunter5c0acf32015-03-17 09:58:58 +0200328 * @val: The values are stored here or %NULL to return the number of values
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100329 * @nval: Size of the @val array
330 *
331 * Function reads an array of u32 properties with @propname from the device
332 * firmware description and stores them to @val if found.
333 *
Adrian Hunter5c0acf32015-03-17 09:58:58 +0200334 * Return: number of values if @val was %NULL,
335 * %0 if the property was found (success),
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100336 * %-EINVAL if given arguments are not valid,
337 * %-ENODATA if the property does not have a value,
338 * %-EPROTO if the property is not an array of numbers,
339 * %-EOVERFLOW if the size of the property is not as expected.
Guenter Roeck4fa7508e2015-08-26 20:27:04 -0700340 * %-ENXIO if no suitable firmware interface is present.
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100341 */
342int device_property_read_u32_array(struct device *dev, const char *propname,
343 u32 *val, size_t nval)
344{
Rafael J. Wysocki9017f252015-03-24 00:24:16 +0100345 return fwnode_property_read_u32_array(dev_fwnode(dev), propname, val, nval);
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100346}
347EXPORT_SYMBOL_GPL(device_property_read_u32_array);
348
349/**
350 * device_property_read_u64_array - return a u64 array property of a device
351 * @dev: Device to get the property of
352 * @propname: Name of the property
Adrian Hunter5c0acf32015-03-17 09:58:58 +0200353 * @val: The values are stored here or %NULL to return the number of values
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100354 * @nval: Size of the @val array
355 *
356 * Function reads an array of u64 properties with @propname from the device
357 * firmware description and stores them to @val if found.
358 *
Adrian Hunter5c0acf32015-03-17 09:58:58 +0200359 * Return: number of values if @val was %NULL,
360 * %0 if the property was found (success),
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100361 * %-EINVAL if given arguments are not valid,
362 * %-ENODATA if the property does not have a value,
363 * %-EPROTO if the property is not an array of numbers,
364 * %-EOVERFLOW if the size of the property is not as expected.
Guenter Roeck4fa7508e2015-08-26 20:27:04 -0700365 * %-ENXIO if no suitable firmware interface is present.
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100366 */
367int device_property_read_u64_array(struct device *dev, const char *propname,
368 u64 *val, size_t nval)
369{
Rafael J. Wysocki9017f252015-03-24 00:24:16 +0100370 return fwnode_property_read_u64_array(dev_fwnode(dev), propname, val, nval);
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100371}
372EXPORT_SYMBOL_GPL(device_property_read_u64_array);
373
374/**
375 * device_property_read_string_array - return a string array property of device
376 * @dev: Device to get the property of
377 * @propname: Name of the property
Adrian Hunter5c0acf32015-03-17 09:58:58 +0200378 * @val: The values are stored here or %NULL to return the number of values
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100379 * @nval: Size of the @val array
380 *
381 * Function reads an array of string properties with @propname from the device
382 * firmware description and stores them to @val if found.
383 *
Sakari Ailusb0b027c2017-03-28 15:22:19 +0300384 * Return: number of values read on success if @val is non-NULL,
385 * number of values available on success if @val is NULL,
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100386 * %-EINVAL if given arguments are not valid,
387 * %-ENODATA if the property does not have a value,
388 * %-EPROTO or %-EILSEQ if the property is not an array of strings,
389 * %-EOVERFLOW if the size of the property is not as expected.
Guenter Roeck4fa7508e2015-08-26 20:27:04 -0700390 * %-ENXIO if no suitable firmware interface is present.
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100391 */
392int device_property_read_string_array(struct device *dev, const char *propname,
393 const char **val, size_t nval)
394{
Rafael J. Wysocki9017f252015-03-24 00:24:16 +0100395 return fwnode_property_read_string_array(dev_fwnode(dev), propname, val, nval);
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100396}
397EXPORT_SYMBOL_GPL(device_property_read_string_array);
398
399/**
400 * device_property_read_string - return a string property of a device
401 * @dev: Device to get the property of
402 * @propname: Name of the property
403 * @val: The value is stored here
404 *
405 * Function reads property @propname from the device firmware description and
406 * stores the value into @val if found. The value is checked to be a string.
407 *
408 * Return: %0 if the property was found (success),
409 * %-EINVAL if given arguments are not valid,
410 * %-ENODATA if the property does not have a value,
411 * %-EPROTO or %-EILSEQ if the property type is not a string.
Guenter Roeck4fa7508e2015-08-26 20:27:04 -0700412 * %-ENXIO if no suitable firmware interface is present.
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100413 */
414int device_property_read_string(struct device *dev, const char *propname,
415 const char **val)
416{
Rafael J. Wysocki9017f252015-03-24 00:24:16 +0100417 return fwnode_property_read_string(dev_fwnode(dev), propname, val);
Rafael J. Wysockib31384f2014-11-04 01:28:56 +0100418}
419EXPORT_SYMBOL_GPL(device_property_read_string);
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100420
Mika Westerberg3f5c8d32015-09-14 17:37:35 +0300421/**
422 * device_property_match_string - find a string in an array and return index
423 * @dev: Device to get the property of
424 * @propname: Name of the property holding the array
425 * @string: String to look for
426 *
427 * Find a given string in a string array and if it is found return the
428 * index back.
429 *
430 * Return: %0 if the property was found (success),
431 * %-EINVAL if given arguments are not valid,
432 * %-ENODATA if the property does not have a value,
433 * %-EPROTO if the property is not an array of strings,
434 * %-ENXIO if no suitable firmware interface is present.
435 */
436int device_property_match_string(struct device *dev, const char *propname,
437 const char *string)
438{
439 return fwnode_property_match_string(dev_fwnode(dev), propname, string);
440}
441EXPORT_SYMBOL_GPL(device_property_match_string);
442
Sakari Ailus37ba9832017-07-21 14:39:36 +0300443static int fwnode_property_read_int_array(const struct fwnode_handle *fwnode,
Sakari Ailus37081842017-06-06 12:37:37 +0300444 const char *propname,
445 unsigned int elem_size, void *val,
446 size_t nval)
447{
448 int ret;
Rafael J. Wysocki9017f252015-03-24 00:24:16 +0100449
Sakari Ailus37081842017-06-06 12:37:37 +0300450 ret = fwnode_call_int_op(fwnode, property_read_int_array, propname,
451 elem_size, val, nval);
452 if (ret == -EINVAL && !IS_ERR_OR_NULL(fwnode) &&
453 !IS_ERR_OR_NULL(fwnode->secondary))
454 ret = fwnode_call_int_op(
455 fwnode->secondary, property_read_int_array, propname,
456 elem_size, val, nval);
Andy Shevchenko318a19712015-11-30 17:11:31 +0200457
Sakari Ailus37081842017-06-06 12:37:37 +0300458 return ret;
459}
Andy Shevchenko362c0b32015-11-30 17:11:36 +0200460
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100461/**
462 * fwnode_property_read_u8_array - return a u8 array property of firmware node
463 * @fwnode: Firmware node to get the property of
464 * @propname: Name of the property
Adrian Hunter5c0acf32015-03-17 09:58:58 +0200465 * @val: The values are stored here or %NULL to return the number of values
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100466 * @nval: Size of the @val array
467 *
468 * Read an array of u8 properties with @propname from @fwnode and stores them to
469 * @val if found.
470 *
Adrian Hunter5c0acf32015-03-17 09:58:58 +0200471 * Return: number of values if @val was %NULL,
472 * %0 if the property was found (success),
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100473 * %-EINVAL if given arguments are not valid,
474 * %-ENODATA if the property does not have a value,
475 * %-EPROTO if the property is not an array of numbers,
476 * %-EOVERFLOW if the size of the property is not as expected,
477 * %-ENXIO if no suitable firmware interface is present.
478 */
Sakari Ailus37ba9832017-07-21 14:39:36 +0300479int fwnode_property_read_u8_array(const struct fwnode_handle *fwnode,
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100480 const char *propname, u8 *val, size_t nval)
481{
Sakari Ailus37081842017-06-06 12:37:37 +0300482 return fwnode_property_read_int_array(fwnode, propname, sizeof(u8),
483 val, nval);
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100484}
485EXPORT_SYMBOL_GPL(fwnode_property_read_u8_array);
486
487/**
488 * fwnode_property_read_u16_array - return a u16 array property of firmware node
489 * @fwnode: Firmware node to get the property of
490 * @propname: Name of the property
Adrian Hunter5c0acf32015-03-17 09:58:58 +0200491 * @val: The values are stored here or %NULL to return the number of values
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100492 * @nval: Size of the @val array
493 *
494 * Read an array of u16 properties with @propname from @fwnode and store them to
495 * @val if found.
496 *
Adrian Hunter5c0acf32015-03-17 09:58:58 +0200497 * Return: number of values if @val was %NULL,
498 * %0 if the property was found (success),
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100499 * %-EINVAL if given arguments are not valid,
500 * %-ENODATA if the property does not have a value,
501 * %-EPROTO if the property is not an array of numbers,
502 * %-EOVERFLOW if the size of the property is not as expected,
503 * %-ENXIO if no suitable firmware interface is present.
504 */
Sakari Ailus37ba9832017-07-21 14:39:36 +0300505int fwnode_property_read_u16_array(const struct fwnode_handle *fwnode,
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100506 const char *propname, u16 *val, size_t nval)
507{
Sakari Ailus37081842017-06-06 12:37:37 +0300508 return fwnode_property_read_int_array(fwnode, propname, sizeof(u16),
509 val, nval);
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100510}
511EXPORT_SYMBOL_GPL(fwnode_property_read_u16_array);
512
513/**
514 * fwnode_property_read_u32_array - return a u32 array property of firmware node
515 * @fwnode: Firmware node to get the property of
516 * @propname: Name of the property
Adrian Hunter5c0acf32015-03-17 09:58:58 +0200517 * @val: The values are stored here or %NULL to return the number of values
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100518 * @nval: Size of the @val array
519 *
520 * Read an array of u32 properties with @propname from @fwnode store them to
521 * @val if found.
522 *
Adrian Hunter5c0acf32015-03-17 09:58:58 +0200523 * Return: number of values if @val was %NULL,
524 * %0 if the property was found (success),
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100525 * %-EINVAL if given arguments are not valid,
526 * %-ENODATA if the property does not have a value,
527 * %-EPROTO if the property is not an array of numbers,
528 * %-EOVERFLOW if the size of the property is not as expected,
529 * %-ENXIO if no suitable firmware interface is present.
530 */
Sakari Ailus37ba9832017-07-21 14:39:36 +0300531int fwnode_property_read_u32_array(const struct fwnode_handle *fwnode,
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100532 const char *propname, u32 *val, size_t nval)
533{
Sakari Ailus37081842017-06-06 12:37:37 +0300534 return fwnode_property_read_int_array(fwnode, propname, sizeof(u32),
535 val, nval);
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100536}
537EXPORT_SYMBOL_GPL(fwnode_property_read_u32_array);
538
539/**
540 * fwnode_property_read_u64_array - return a u64 array property firmware node
541 * @fwnode: Firmware node to get the property of
542 * @propname: Name of the property
Adrian Hunter5c0acf32015-03-17 09:58:58 +0200543 * @val: The values are stored here or %NULL to return the number of values
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100544 * @nval: Size of the @val array
545 *
546 * Read an array of u64 properties with @propname from @fwnode and store them to
547 * @val if found.
548 *
Adrian Hunter5c0acf32015-03-17 09:58:58 +0200549 * Return: number of values if @val was %NULL,
550 * %0 if the property was found (success),
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100551 * %-EINVAL if given arguments are not valid,
552 * %-ENODATA if the property does not have a value,
553 * %-EPROTO if the property is not an array of numbers,
554 * %-EOVERFLOW if the size of the property is not as expected,
555 * %-ENXIO if no suitable firmware interface is present.
556 */
Sakari Ailus37ba9832017-07-21 14:39:36 +0300557int fwnode_property_read_u64_array(const struct fwnode_handle *fwnode,
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100558 const char *propname, u64 *val, size_t nval)
559{
Sakari Ailus37081842017-06-06 12:37:37 +0300560 return fwnode_property_read_int_array(fwnode, propname, sizeof(u64),
561 val, nval);
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100562}
563EXPORT_SYMBOL_GPL(fwnode_property_read_u64_array);
564
565/**
566 * fwnode_property_read_string_array - return string array property of a node
567 * @fwnode: Firmware node to get the property of
568 * @propname: Name of the property
Adrian Hunter5c0acf32015-03-17 09:58:58 +0200569 * @val: The values are stored here or %NULL to return the number of values
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100570 * @nval: Size of the @val array
571 *
572 * Read an string list property @propname from the given firmware node and store
573 * them to @val if found.
574 *
Sakari Ailusb0b027c2017-03-28 15:22:19 +0300575 * Return: number of values read on success if @val is non-NULL,
576 * number of values available on success if @val is NULL,
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100577 * %-EINVAL if given arguments are not valid,
578 * %-ENODATA if the property does not have a value,
Sakari Ailus026b8212017-03-28 15:22:17 +0300579 * %-EPROTO or %-EILSEQ if the property is not an array of strings,
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100580 * %-EOVERFLOW if the size of the property is not as expected,
581 * %-ENXIO if no suitable firmware interface is present.
582 */
Sakari Ailus37ba9832017-07-21 14:39:36 +0300583int fwnode_property_read_string_array(const struct fwnode_handle *fwnode,
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100584 const char *propname, const char **val,
585 size_t nval)
586{
Andy Shevchenko362c0b32015-11-30 17:11:36 +0200587 int ret;
588
Sakari Ailus37081842017-06-06 12:37:37 +0300589 ret = fwnode_call_int_op(fwnode, property_read_string_array, propname,
590 val, nval);
Heikki Krogerus0d67e0f2016-03-10 13:03:18 +0200591 if (ret == -EINVAL && !IS_ERR_OR_NULL(fwnode) &&
592 !IS_ERR_OR_NULL(fwnode->secondary))
Sakari Ailus37081842017-06-06 12:37:37 +0300593 ret = fwnode_call_int_op(fwnode->secondary,
594 property_read_string_array, propname,
595 val, nval);
Andy Shevchenko362c0b32015-11-30 17:11:36 +0200596 return ret;
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100597}
598EXPORT_SYMBOL_GPL(fwnode_property_read_string_array);
599
600/**
601 * fwnode_property_read_string - return a string property of a firmware node
602 * @fwnode: Firmware node to get the property of
603 * @propname: Name of the property
604 * @val: The value is stored here
605 *
606 * Read property @propname from the given firmware node and store the value into
607 * @val if found. The value is checked to be a string.
608 *
609 * Return: %0 if the property was found (success),
610 * %-EINVAL if given arguments are not valid,
611 * %-ENODATA if the property does not have a value,
612 * %-EPROTO or %-EILSEQ if the property is not a string,
613 * %-ENXIO if no suitable firmware interface is present.
614 */
Sakari Ailus37ba9832017-07-21 14:39:36 +0300615int fwnode_property_read_string(const struct fwnode_handle *fwnode,
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100616 const char *propname, const char **val)
617{
Sakari Ailuse4817472017-03-28 15:26:22 +0300618 int ret = fwnode_property_read_string_array(fwnode, propname, val, 1);
Andy Shevchenko362c0b32015-11-30 17:11:36 +0200619
Sakari Ailusb0b027c2017-03-28 15:22:19 +0300620 return ret < 0 ? ret : 0;
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +0100621}
622EXPORT_SYMBOL_GPL(fwnode_property_read_string);
623
624/**
Mika Westerberg3f5c8d32015-09-14 17:37:35 +0300625 * fwnode_property_match_string - find a string in an array and return index
626 * @fwnode: Firmware node to get the property of
627 * @propname: Name of the property holding the array
628 * @string: String to look for
629 *
630 * Find a given string in a string array and if it is found return the
631 * index back.
632 *
633 * Return: %0 if the property was found (success),
634 * %-EINVAL if given arguments are not valid,
635 * %-ENODATA if the property does not have a value,
636 * %-EPROTO if the property is not an array of strings,
637 * %-ENXIO if no suitable firmware interface is present.
638 */
Sakari Ailus37ba9832017-07-21 14:39:36 +0300639int fwnode_property_match_string(const struct fwnode_handle *fwnode,
Mika Westerberg3f5c8d32015-09-14 17:37:35 +0300640 const char *propname, const char *string)
641{
642 const char **values;
Andy Shevchenkoa7c1d0a2016-03-17 14:22:17 -0700643 int nval, ret;
Mika Westerberg3f5c8d32015-09-14 17:37:35 +0300644
645 nval = fwnode_property_read_string_array(fwnode, propname, NULL, 0);
646 if (nval < 0)
647 return nval;
648
Andy Shevchenkof6740c12015-12-29 13:07:50 +0200649 if (nval == 0)
650 return -ENODATA;
651
Mika Westerberg3f5c8d32015-09-14 17:37:35 +0300652 values = kcalloc(nval, sizeof(*values), GFP_KERNEL);
653 if (!values)
654 return -ENOMEM;
655
656 ret = fwnode_property_read_string_array(fwnode, propname, values, nval);
657 if (ret < 0)
658 goto out;
659
Andy Shevchenkoa7c1d0a2016-03-17 14:22:17 -0700660 ret = match_string(values, nval, string);
661 if (ret < 0)
662 ret = -ENODATA;
Mika Westerberg3f5c8d32015-09-14 17:37:35 +0300663out:
664 kfree(values);
665 return ret;
666}
667EXPORT_SYMBOL_GPL(fwnode_property_match_string);
668
Sakari Ailus3e3119d2017-07-21 15:11:49 +0300669/**
670 * fwnode_property_get_reference_args() - Find a reference with arguments
671 * @fwnode: Firmware node where to look for the reference
672 * @prop: The name of the property
673 * @nargs_prop: The name of the property telling the number of
674 * arguments in the referred node. NULL if @nargs is known,
675 * otherwise @nargs is ignored. Only relevant on OF.
676 * @nargs: Number of arguments. Ignored if @nargs_prop is non-NULL.
677 * @index: Index of the reference, from zero onwards.
678 * @args: Result structure with reference and integer arguments.
679 *
680 * Obtain a reference based on a named property in an fwnode, with
681 * integer arguments.
682 *
683 * Caller is responsible to call fwnode_handle_put() on the returned
684 * args->fwnode pointer.
685 *
Sakari Ailusc343bc22017-09-26 12:08:27 +0300686 * Returns: %0 on success
687 * %-ENOENT when the index is out of bounds, the index has an empty
688 * reference or the property was not found
689 * %-EINVAL on parse error
Sakari Ailus3e3119d2017-07-21 15:11:49 +0300690 */
691int fwnode_property_get_reference_args(const struct fwnode_handle *fwnode,
692 const char *prop, const char *nargs_prop,
693 unsigned int nargs, unsigned int index,
694 struct fwnode_reference_args *args)
695{
696 return fwnode_call_int_op(fwnode, get_reference_args, prop, nargs_prop,
697 nargs, index, args);
698}
699EXPORT_SYMBOL_GPL(fwnode_property_get_reference_args);
700
Dmitry Torokhov2d479e12017-02-02 17:41:27 -0800701static int property_copy_string_array(struct property_entry *dst,
702 const struct property_entry *src)
Mika Westerberg13141e12015-11-30 17:11:37 +0200703{
Dmitry Torokhov2d479e12017-02-02 17:41:27 -0800704 char **d;
705 size_t nval = src->length / sizeof(*d);
706 int i;
Mika Westerberg13141e12015-11-30 17:11:37 +0200707
Dmitry Torokhov2d479e12017-02-02 17:41:27 -0800708 d = kcalloc(nval, sizeof(*d), GFP_KERNEL);
709 if (!d)
710 return -ENOMEM;
Mika Westerberg13141e12015-11-30 17:11:37 +0200711
Dmitry Torokhov2d479e12017-02-02 17:41:27 -0800712 for (i = 0; i < nval; i++) {
713 d[i] = kstrdup(src->pointer.str[i], GFP_KERNEL);
714 if (!d[i] && src->pointer.str[i]) {
715 while (--i >= 0)
716 kfree(d[i]);
717 kfree(d);
718 return -ENOMEM;
Mika Westerberg13141e12015-11-30 17:11:37 +0200719 }
Mika Westerberg13141e12015-11-30 17:11:37 +0200720 }
721
Dmitry Torokhov2d479e12017-02-02 17:41:27 -0800722 dst->pointer.raw_data = d;
723 return 0;
Mika Westerberg13141e12015-11-30 17:11:37 +0200724}
725
Dmitry Torokhov2d479e12017-02-02 17:41:27 -0800726static int property_entry_copy_data(struct property_entry *dst,
727 const struct property_entry *src)
Mika Westerberg13141e12015-11-30 17:11:37 +0200728{
Dmitry Torokhov2d479e12017-02-02 17:41:27 -0800729 int error;
Mika Westerberg13141e12015-11-30 17:11:37 +0200730
731 dst->name = kstrdup(src->name, GFP_KERNEL);
732 if (!dst->name)
733 return -ENOMEM;
734
735 if (src->is_array) {
Dmitry Torokhov2d479e12017-02-02 17:41:27 -0800736 if (!src->length) {
737 error = -ENODATA;
738 goto out_free_name;
739 }
Andy Shevchenkof6740c12015-12-29 13:07:50 +0200740
Mika Westerberg13141e12015-11-30 17:11:37 +0200741 if (src->is_string) {
Dmitry Torokhov2d479e12017-02-02 17:41:27 -0800742 error = property_copy_string_array(dst, src);
743 if (error)
744 goto out_free_name;
Mika Westerberg13141e12015-11-30 17:11:37 +0200745 } else {
746 dst->pointer.raw_data = kmemdup(src->pointer.raw_data,
747 src->length, GFP_KERNEL);
Dmitry Torokhov2d479e12017-02-02 17:41:27 -0800748 if (!dst->pointer.raw_data) {
749 error = -ENOMEM;
750 goto out_free_name;
751 }
Mika Westerberg13141e12015-11-30 17:11:37 +0200752 }
753 } else if (src->is_string) {
754 dst->value.str = kstrdup(src->value.str, GFP_KERNEL);
Dmitry Torokhov2d479e12017-02-02 17:41:27 -0800755 if (!dst->value.str && src->value.str) {
756 error = -ENOMEM;
757 goto out_free_name;
758 }
Mika Westerberg13141e12015-11-30 17:11:37 +0200759 } else {
760 dst->value.raw_data = src->value.raw_data;
761 }
762
763 dst->length = src->length;
764 dst->is_array = src->is_array;
765 dst->is_string = src->is_string;
766
767 return 0;
Dmitry Torokhov2d479e12017-02-02 17:41:27 -0800768
769out_free_name:
770 kfree(dst->name);
771 return error;
772}
773
774static void property_entry_free_data(const struct property_entry *p)
775{
776 size_t i, nval;
777
778 if (p->is_array) {
779 if (p->is_string && p->pointer.str) {
780 nval = p->length / sizeof(const char *);
781 for (i = 0; i < nval; i++)
782 kfree(p->pointer.str[i]);
783 }
784 kfree(p->pointer.raw_data);
785 } else if (p->is_string) {
786 kfree(p->value.str);
787 }
788 kfree(p->name);
789}
790
791/**
792 * property_entries_dup - duplicate array of properties
793 * @properties: array of properties to copy
794 *
795 * This function creates a deep copy of the given NULL-terminated array
796 * of property entries.
797 */
798struct property_entry *
799property_entries_dup(const struct property_entry *properties)
800{
801 struct property_entry *p;
802 int i, n = 0;
803
804 while (properties[n].name)
805 n++;
806
807 p = kcalloc(n + 1, sizeof(*p), GFP_KERNEL);
808 if (!p)
809 return ERR_PTR(-ENOMEM);
810
811 for (i = 0; i < n; i++) {
812 int ret = property_entry_copy_data(&p[i], &properties[i]);
813 if (ret) {
814 while (--i >= 0)
815 property_entry_free_data(&p[i]);
816 kfree(p);
817 return ERR_PTR(ret);
818 }
819 }
820
821 return p;
822}
823EXPORT_SYMBOL_GPL(property_entries_dup);
824
825/**
826 * property_entries_free - free previously allocated array of properties
827 * @properties: array of properties to destroy
828 *
829 * This function frees given NULL-terminated array of property entries,
830 * along with their data.
831 */
832void property_entries_free(const struct property_entry *properties)
833{
834 const struct property_entry *p;
835
836 for (p = properties; p->name; p++)
837 property_entry_free_data(p);
838
839 kfree(properties);
840}
841EXPORT_SYMBOL_GPL(property_entries_free);
842
843/**
844 * pset_free_set - releases memory allocated for copied property set
845 * @pset: Property set to release
846 *
847 * Function takes previously copied property set and releases all the
848 * memory allocated to it.
849 */
850static void pset_free_set(struct property_set *pset)
851{
852 if (!pset)
853 return;
854
855 property_entries_free(pset->properties);
856 kfree(pset);
Mika Westerberg13141e12015-11-30 17:11:37 +0200857}
858
859/**
860 * pset_copy_set - copies property set
861 * @pset: Property set to copy
862 *
863 * This function takes a deep copy of the given property set and returns
864 * pointer to the copy. Call device_free_property_set() to free resources
865 * allocated in this function.
866 *
867 * Return: Pointer to the new property set or error pointer.
868 */
869static struct property_set *pset_copy_set(const struct property_set *pset)
870{
Dmitry Torokhov2d479e12017-02-02 17:41:27 -0800871 struct property_entry *properties;
Mika Westerberg13141e12015-11-30 17:11:37 +0200872 struct property_set *p;
Mika Westerberg13141e12015-11-30 17:11:37 +0200873
874 p = kzalloc(sizeof(*p), GFP_KERNEL);
875 if (!p)
876 return ERR_PTR(-ENOMEM);
877
Dmitry Torokhov2d479e12017-02-02 17:41:27 -0800878 properties = property_entries_dup(pset->properties);
879 if (IS_ERR(properties)) {
Mika Westerberg13141e12015-11-30 17:11:37 +0200880 kfree(p);
Dmitry Torokhov2d479e12017-02-02 17:41:27 -0800881 return ERR_CAST(properties);
Mika Westerberg13141e12015-11-30 17:11:37 +0200882 }
883
Dmitry Torokhov2d479e12017-02-02 17:41:27 -0800884 p->properties = properties;
Mika Westerberg13141e12015-11-30 17:11:37 +0200885 return p;
886}
887
888/**
Heikki Krogerusf4d05262016-03-29 14:52:23 +0300889 * device_remove_properties - Remove properties from a device object.
Mika Westerberg13141e12015-11-30 17:11:37 +0200890 * @dev: Device whose properties to remove.
891 *
892 * The function removes properties previously associated to the device
Heikki Krogerusf4d05262016-03-29 14:52:23 +0300893 * secondary firmware node with device_add_properties(). Memory allocated
Mika Westerberg13141e12015-11-30 17:11:37 +0200894 * to the properties will also be released.
895 */
Heikki Krogerusf4d05262016-03-29 14:52:23 +0300896void device_remove_properties(struct device *dev)
Mika Westerberg13141e12015-11-30 17:11:37 +0200897{
898 struct fwnode_handle *fwnode;
Jarkko Nikula5ab894a2017-10-09 16:28:37 +0300899 struct property_set *pset;
Mika Westerberg13141e12015-11-30 17:11:37 +0200900
901 fwnode = dev_fwnode(dev);
902 if (!fwnode)
903 return;
904 /*
905 * Pick either primary or secondary node depending which one holds
906 * the pset. If there is no real firmware node (ACPI/DT) primary
907 * will hold the pset.
908 */
Jarkko Nikula5ab894a2017-10-09 16:28:37 +0300909 pset = to_pset_node(fwnode);
910 if (pset) {
Heikki Krogerus0d67e0f2016-03-10 13:03:18 +0200911 set_primary_fwnode(dev, NULL);
Heikki Krogerus0d67e0f2016-03-10 13:03:18 +0200912 } else {
Jarkko Nikula5ab894a2017-10-09 16:28:37 +0300913 pset = to_pset_node(fwnode->secondary);
914 if (pset && dev == pset->dev)
Heikki Krogerus0d67e0f2016-03-10 13:03:18 +0200915 set_secondary_fwnode(dev, NULL);
Heikki Krogerus0d67e0f2016-03-10 13:03:18 +0200916 }
Jarkko Nikula5ab894a2017-10-09 16:28:37 +0300917 if (pset && dev == pset->dev)
918 pset_free_set(pset);
Mika Westerberg13141e12015-11-30 17:11:37 +0200919}
Heikki Krogerusf4d05262016-03-29 14:52:23 +0300920EXPORT_SYMBOL_GPL(device_remove_properties);
Mika Westerberg13141e12015-11-30 17:11:37 +0200921
922/**
Heikki Krogerusf4d05262016-03-29 14:52:23 +0300923 * device_add_properties - Add a collection of properties to a device object.
Mika Westerberg13141e12015-11-30 17:11:37 +0200924 * @dev: Device to add properties to.
Heikki Krogerusf4d05262016-03-29 14:52:23 +0300925 * @properties: Collection of properties to add.
Mika Westerberg13141e12015-11-30 17:11:37 +0200926 *
Heikki Krogerusf4d05262016-03-29 14:52:23 +0300927 * Associate a collection of device properties represented by @properties with
928 * @dev as its secondary firmware node. The function takes a copy of
929 * @properties.
Mika Westerberg13141e12015-11-30 17:11:37 +0200930 */
Dmitry Torokhovbec84da82017-02-02 17:41:25 -0800931int device_add_properties(struct device *dev,
932 const struct property_entry *properties)
Mika Westerberg13141e12015-11-30 17:11:37 +0200933{
Heikki Krogerusf4d05262016-03-29 14:52:23 +0300934 struct property_set *p, pset;
Mika Westerberg13141e12015-11-30 17:11:37 +0200935
Heikki Krogerusf4d05262016-03-29 14:52:23 +0300936 if (!properties)
Mika Westerberg13141e12015-11-30 17:11:37 +0200937 return -EINVAL;
938
Heikki Krogerusf4d05262016-03-29 14:52:23 +0300939 pset.properties = properties;
940
941 p = pset_copy_set(&pset);
Mika Westerberg13141e12015-11-30 17:11:37 +0200942 if (IS_ERR(p))
943 return PTR_ERR(p);
944
Sakari Ailus37081842017-06-06 12:37:37 +0300945 p->fwnode.ops = &pset_fwnode_ops;
Mika Westerberg13141e12015-11-30 17:11:37 +0200946 set_secondary_fwnode(dev, &p->fwnode);
Jarkko Nikula5ab894a2017-10-09 16:28:37 +0300947 p->dev = dev;
Mika Westerberg13141e12015-11-30 17:11:37 +0200948 return 0;
949}
Heikki Krogerusf4d05262016-03-29 14:52:23 +0300950EXPORT_SYMBOL_GPL(device_add_properties);
Mika Westerberg13141e12015-11-30 17:11:37 +0200951
952/**
Sakari Ailus23387252017-03-28 10:52:26 +0300953 * fwnode_get_next_parent - Iterate to the node's parent
954 * @fwnode: Firmware whose parent is retrieved
955 *
956 * This is like fwnode_get_parent() except that it drops the refcount
957 * on the passed node, making it suitable for iterating through a
958 * node's parents.
959 *
960 * Returns a node pointer with refcount incremented, use
961 * fwnode_handle_node() on it when done.
962 */
963struct fwnode_handle *fwnode_get_next_parent(struct fwnode_handle *fwnode)
964{
965 struct fwnode_handle *parent = fwnode_get_parent(fwnode);
966
967 fwnode_handle_put(fwnode);
968
969 return parent;
970}
971EXPORT_SYMBOL_GPL(fwnode_get_next_parent);
972
973/**
Mika Westerbergafaf26f2017-03-28 10:52:17 +0300974 * fwnode_get_parent - Return parent firwmare node
975 * @fwnode: Firmware whose parent is retrieved
976 *
977 * Return parent firmware node of the given node if possible or %NULL if no
978 * parent was available.
979 */
Sakari Ailus37ba9832017-07-21 14:39:36 +0300980struct fwnode_handle *fwnode_get_parent(const struct fwnode_handle *fwnode)
Mika Westerbergafaf26f2017-03-28 10:52:17 +0300981{
Sakari Ailus37081842017-06-06 12:37:37 +0300982 return fwnode_call_ptr_op(fwnode, get_parent);
Mika Westerbergafaf26f2017-03-28 10:52:17 +0300983}
984EXPORT_SYMBOL_GPL(fwnode_get_parent);
985
986/**
Mika Westerberg34055192017-03-28 10:52:18 +0300987 * fwnode_get_next_child_node - Return the next child node handle for a node
988 * @fwnode: Firmware node to find the next child node for.
989 * @child: Handle to one of the node's child nodes or a %NULL handle.
990 */
Sakari Ailus37ba9832017-07-21 14:39:36 +0300991struct fwnode_handle *
992fwnode_get_next_child_node(const struct fwnode_handle *fwnode,
993 struct fwnode_handle *child)
Mika Westerberg34055192017-03-28 10:52:18 +0300994{
Sakari Ailus37081842017-06-06 12:37:37 +0300995 return fwnode_call_ptr_op(fwnode, get_next_child_node, child);
Mika Westerberg34055192017-03-28 10:52:18 +0300996}
997EXPORT_SYMBOL_GPL(fwnode_get_next_child_node);
998
999/**
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +01001000 * device_get_next_child_node - Return the next child node handle for a device
1001 * @dev: Device to find the next child node for.
1002 * @child: Handle to one of the device's child nodes or a null handle.
1003 */
1004struct fwnode_handle *device_get_next_child_node(struct device *dev,
1005 struct fwnode_handle *child)
1006{
Mika Westerberg34055192017-03-28 10:52:18 +03001007 struct acpi_device *adev = ACPI_COMPANION(dev);
1008 struct fwnode_handle *fwnode = NULL;
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +01001009
Mika Westerberg34055192017-03-28 10:52:18 +03001010 if (dev->of_node)
1011 fwnode = &dev->of_node->fwnode;
1012 else if (adev)
1013 fwnode = acpi_fwnode_handle(adev);
1014
1015 return fwnode_get_next_child_node(fwnode, child);
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +01001016}
1017EXPORT_SYMBOL_GPL(device_get_next_child_node);
1018
1019/**
Mika Westerberg21ea73f2017-03-28 10:52:19 +03001020 * fwnode_get_named_child_node - Return first matching named child node handle
1021 * @fwnode: Firmware node to find the named child node for.
Adam Thomson613e9722016-06-21 18:50:20 +01001022 * @childname: String to match child node name against.
1023 */
Sakari Ailus37ba9832017-07-21 14:39:36 +03001024struct fwnode_handle *
1025fwnode_get_named_child_node(const struct fwnode_handle *fwnode,
1026 const char *childname)
Adam Thomson613e9722016-06-21 18:50:20 +01001027{
Sakari Ailus37081842017-06-06 12:37:37 +03001028 return fwnode_call_ptr_op(fwnode, get_named_child_node, childname);
Adam Thomson613e9722016-06-21 18:50:20 +01001029}
Mika Westerberg21ea73f2017-03-28 10:52:19 +03001030EXPORT_SYMBOL_GPL(fwnode_get_named_child_node);
1031
1032/**
1033 * device_get_named_child_node - Return first matching named child node handle
1034 * @dev: Device to find the named child node for.
1035 * @childname: String to match child node name against.
1036 */
1037struct fwnode_handle *device_get_named_child_node(struct device *dev,
1038 const char *childname)
1039{
1040 return fwnode_get_named_child_node(dev_fwnode(dev), childname);
1041}
Adam Thomson613e9722016-06-21 18:50:20 +01001042EXPORT_SYMBOL_GPL(device_get_named_child_node);
1043
1044/**
Sakari Ailuse7887c22017-03-28 10:52:22 +03001045 * fwnode_handle_get - Obtain a reference to a device node
1046 * @fwnode: Pointer to the device node to obtain the reference to.
Sakari Ailuscf89a312017-09-19 12:39:11 +03001047 *
1048 * Returns the fwnode handle.
Sakari Ailuse7887c22017-03-28 10:52:22 +03001049 */
Sakari Ailuscf89a312017-09-19 12:39:11 +03001050struct fwnode_handle *fwnode_handle_get(struct fwnode_handle *fwnode)
Sakari Ailuse7887c22017-03-28 10:52:22 +03001051{
Sakari Ailuscf89a312017-09-19 12:39:11 +03001052 if (!fwnode_has_op(fwnode, get))
1053 return fwnode;
1054
1055 return fwnode_call_ptr_op(fwnode, get);
Sakari Ailuse7887c22017-03-28 10:52:22 +03001056}
1057EXPORT_SYMBOL_GPL(fwnode_handle_get);
1058
1059/**
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +01001060 * fwnode_handle_put - Drop reference to a device node
1061 * @fwnode: Pointer to the device node to drop the reference to.
1062 *
1063 * This has to be used when terminating device_for_each_child_node() iteration
1064 * with break or return to prevent stale device node references from being left
1065 * behind.
1066 */
1067void fwnode_handle_put(struct fwnode_handle *fwnode)
1068{
Sakari Ailus37081842017-06-06 12:37:37 +03001069 fwnode_call_void_op(fwnode, put);
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +01001070}
1071EXPORT_SYMBOL_GPL(fwnode_handle_put);
1072
1073/**
Sakari Ailus2294b3a2017-06-06 12:37:39 +03001074 * fwnode_device_is_available - check if a device is available for use
1075 * @fwnode: Pointer to the fwnode of the device.
1076 */
Sakari Ailus37ba9832017-07-21 14:39:36 +03001077bool fwnode_device_is_available(const struct fwnode_handle *fwnode)
Sakari Ailus2294b3a2017-06-06 12:37:39 +03001078{
Sakari Ailuse8158b482017-07-11 18:20:20 +03001079 return fwnode_call_bool_op(fwnode, device_is_available);
Sakari Ailus2294b3a2017-06-06 12:37:39 +03001080}
1081EXPORT_SYMBOL_GPL(fwnode_device_is_available);
1082
1083/**
Rafael J. Wysocki8a0662d2014-11-04 14:03:59 +01001084 * device_get_child_node_count - return the number of child nodes for device
1085 * @dev: Device to cound the child nodes for
1086 */
1087unsigned int device_get_child_node_count(struct device *dev)
1088{
1089 struct fwnode_handle *child;
1090 unsigned int count = 0;
1091
1092 device_for_each_child_node(dev, child)
1093 count++;
1094
1095 return count;
1096}
1097EXPORT_SYMBOL_GPL(device_get_child_node_count);
Suthikulpanit, Suravee05ca5562015-06-10 11:08:54 -05001098
Suthikulpanit, Suraveee5e55862015-10-28 15:50:49 -07001099bool device_dma_supported(struct device *dev)
1100{
1101 /* For DT, this is always supported.
1102 * For ACPI, this depends on CCA, which
1103 * is determined by the acpi_dma_supported().
1104 */
1105 if (IS_ENABLED(CONFIG_OF) && dev->of_node)
1106 return true;
1107
1108 return acpi_dma_supported(ACPI_COMPANION(dev));
1109}
1110EXPORT_SYMBOL_GPL(device_dma_supported);
1111
1112enum dev_dma_attr device_get_dma_attr(struct device *dev)
1113{
1114 enum dev_dma_attr attr = DEV_DMA_NOT_SUPPORTED;
1115
1116 if (IS_ENABLED(CONFIG_OF) && dev->of_node) {
1117 if (of_dma_is_coherent(dev->of_node))
1118 attr = DEV_DMA_COHERENT;
1119 else
1120 attr = DEV_DMA_NON_COHERENT;
1121 } else
1122 attr = acpi_get_dma_attr(ACPI_COMPANION(dev));
1123
1124 return attr;
1125}
1126EXPORT_SYMBOL_GPL(device_get_dma_attr);
1127
Jeremy Linton4c96b7d2015-08-12 17:06:26 -05001128/**
Marcin Wojtasb28f2632018-01-18 13:31:39 +01001129 * fwnode_get_phy_mode - Get phy mode for given firmware node
1130 * @fwnode: Pointer to the given node
Jeremy Linton4c96b7d2015-08-12 17:06:26 -05001131 *
1132 * The function gets phy interface string from property 'phy-mode' or
1133 * 'phy-connection-type', and return its index in phy_modes table, or errno in
1134 * error case.
1135 */
Marcin Wojtasb28f2632018-01-18 13:31:39 +01001136int fwnode_get_phy_mode(struct fwnode_handle *fwnode)
Jeremy Linton4c96b7d2015-08-12 17:06:26 -05001137{
1138 const char *pm;
1139 int err, i;
1140
Marcin Wojtasb28f2632018-01-18 13:31:39 +01001141 err = fwnode_property_read_string(fwnode, "phy-mode", &pm);
Jeremy Linton4c96b7d2015-08-12 17:06:26 -05001142 if (err < 0)
Marcin Wojtasb28f2632018-01-18 13:31:39 +01001143 err = fwnode_property_read_string(fwnode,
Jeremy Linton4c96b7d2015-08-12 17:06:26 -05001144 "phy-connection-type", &pm);
1145 if (err < 0)
1146 return err;
1147
1148 for (i = 0; i < PHY_INTERFACE_MODE_MAX; i++)
1149 if (!strcasecmp(pm, phy_modes(i)))
1150 return i;
1151
1152 return -ENODEV;
1153}
Marcin Wojtasb28f2632018-01-18 13:31:39 +01001154EXPORT_SYMBOL_GPL(fwnode_get_phy_mode);
1155
1156/**
1157 * device_get_phy_mode - Get phy mode for given device
1158 * @dev: Pointer to the given device
1159 *
1160 * The function gets phy interface string from property 'phy-mode' or
1161 * 'phy-connection-type', and return its index in phy_modes table, or errno in
1162 * error case.
1163 */
1164int device_get_phy_mode(struct device *dev)
1165{
1166 return fwnode_get_phy_mode(dev_fwnode(dev));
1167}
Jeremy Linton4c96b7d2015-08-12 17:06:26 -05001168EXPORT_SYMBOL_GPL(device_get_phy_mode);
1169
Marcin Wojtasbabe2db2018-01-18 13:31:38 +01001170static void *fwnode_get_mac_addr(struct fwnode_handle *fwnode,
Jeremy Linton4c96b7d2015-08-12 17:06:26 -05001171 const char *name, char *addr,
1172 int alen)
1173{
Marcin Wojtasbabe2db2018-01-18 13:31:38 +01001174 int ret = fwnode_property_read_u8_array(fwnode, name, addr, alen);
Jeremy Linton4c96b7d2015-08-12 17:06:26 -05001175
Jeremy Linton2f710a32015-08-19 11:46:42 -05001176 if (ret == 0 && alen == ETH_ALEN && is_valid_ether_addr(addr))
Jeremy Linton4c96b7d2015-08-12 17:06:26 -05001177 return addr;
1178 return NULL;
1179}
1180
1181/**
Marcin Wojtasbabe2db2018-01-18 13:31:38 +01001182 * fwnode_get_mac_address - Get the MAC from the firmware node
1183 * @fwnode: Pointer to the firmware node
Jeremy Linton2f710a32015-08-19 11:46:42 -05001184 * @addr: Address of buffer to store the MAC in
1185 * @alen: Length of the buffer pointed to by addr, should be ETH_ALEN
1186 *
1187 * Search the firmware node for the best MAC address to use. 'mac-address' is
Jeremy Linton4c96b7d2015-08-12 17:06:26 -05001188 * checked first, because that is supposed to contain to "most recent" MAC
1189 * address. If that isn't set, then 'local-mac-address' is checked next,
1190 * because that is the default address. If that isn't set, then the obsolete
1191 * 'address' is checked, just in case we're using an old device tree.
1192 *
1193 * Note that the 'address' property is supposed to contain a virtual address of
1194 * the register set, but some DTS files have redefined that property to be the
1195 * MAC address.
1196 *
1197 * All-zero MAC addresses are rejected, because those could be properties that
Jeremy Linton2f710a32015-08-19 11:46:42 -05001198 * exist in the firmware tables, but were not updated by the firmware. For
1199 * example, the DTS could define 'mac-address' and 'local-mac-address', with
1200 * zero MAC addresses. Some older U-Boots only initialized 'local-mac-address'.
1201 * In this case, the real MAC is in 'local-mac-address', and 'mac-address'
1202 * exists but is all zeros.
Jeremy Linton4c96b7d2015-08-12 17:06:26 -05001203*/
Marcin Wojtasbabe2db2018-01-18 13:31:38 +01001204void *fwnode_get_mac_address(struct fwnode_handle *fwnode, char *addr, int alen)
Jeremy Linton4c96b7d2015-08-12 17:06:26 -05001205{
Julien Grall5b902d62015-09-03 23:59:50 +01001206 char *res;
Jeremy Linton4c96b7d2015-08-12 17:06:26 -05001207
Marcin Wojtasbabe2db2018-01-18 13:31:38 +01001208 res = fwnode_get_mac_addr(fwnode, "mac-address", addr, alen);
Julien Grall5b902d62015-09-03 23:59:50 +01001209 if (res)
1210 return res;
1211
Marcin Wojtasbabe2db2018-01-18 13:31:38 +01001212 res = fwnode_get_mac_addr(fwnode, "local-mac-address", addr, alen);
Julien Grall5b902d62015-09-03 23:59:50 +01001213 if (res)
1214 return res;
Jeremy Linton4c96b7d2015-08-12 17:06:26 -05001215
Marcin Wojtasbabe2db2018-01-18 13:31:38 +01001216 return fwnode_get_mac_addr(fwnode, "address", addr, alen);
1217}
1218EXPORT_SYMBOL(fwnode_get_mac_address);
1219
1220/**
1221 * device_get_mac_address - Get the MAC for a given device
1222 * @dev: Pointer to the device
1223 * @addr: Address of buffer to store the MAC in
1224 * @alen: Length of the buffer pointed to by addr, should be ETH_ALEN
1225 */
1226void *device_get_mac_address(struct device *dev, char *addr, int alen)
1227{
1228 return fwnode_get_mac_address(dev_fwnode(dev), addr, alen);
Jeremy Linton4c96b7d2015-08-12 17:06:26 -05001229}
1230EXPORT_SYMBOL(device_get_mac_address);
Mika Westerberg07bb80d2017-03-28 10:52:21 +03001231
1232/**
1233 * device_graph_get_next_endpoint - Get next endpoint firmware node
1234 * @fwnode: Pointer to the parent firmware node
1235 * @prev: Previous endpoint node or %NULL to get the first
1236 *
1237 * Returns an endpoint firmware node pointer or %NULL if no more endpoints
1238 * are available.
1239 */
1240struct fwnode_handle *
Sakari Ailus37ba9832017-07-21 14:39:36 +03001241fwnode_graph_get_next_endpoint(const struct fwnode_handle *fwnode,
Mika Westerberg07bb80d2017-03-28 10:52:21 +03001242 struct fwnode_handle *prev)
1243{
Sakari Ailus3b27d002017-06-06 12:37:38 +03001244 return fwnode_call_ptr_op(fwnode, graph_get_next_endpoint, prev);
Mika Westerberg07bb80d2017-03-28 10:52:21 +03001245}
1246EXPORT_SYMBOL_GPL(fwnode_graph_get_next_endpoint);
1247
1248/**
Kieran Bingham6a71d8d2017-06-06 12:37:41 +03001249 * fwnode_graph_get_port_parent - Return the device fwnode of a port endpoint
1250 * @endpoint: Endpoint firmware node of the port
1251 *
1252 * Return: the firmware node of the device the @endpoint belongs to.
1253 */
1254struct fwnode_handle *
Sakari Ailus37ba9832017-07-21 14:39:36 +03001255fwnode_graph_get_port_parent(const struct fwnode_handle *endpoint)
Kieran Bingham6a71d8d2017-06-06 12:37:41 +03001256{
1257 struct fwnode_handle *port, *parent;
1258
1259 port = fwnode_get_parent(endpoint);
1260 parent = fwnode_call_ptr_op(port, graph_get_port_parent);
1261
1262 fwnode_handle_put(port);
1263
1264 return parent;
1265}
1266EXPORT_SYMBOL_GPL(fwnode_graph_get_port_parent);
1267
1268/**
Mika Westerberg07bb80d2017-03-28 10:52:21 +03001269 * fwnode_graph_get_remote_port_parent - Return fwnode of a remote device
1270 * @fwnode: Endpoint firmware node pointing to the remote endpoint
1271 *
1272 * Extracts firmware node of a remote device the @fwnode points to.
1273 */
1274struct fwnode_handle *
Sakari Ailus37ba9832017-07-21 14:39:36 +03001275fwnode_graph_get_remote_port_parent(const struct fwnode_handle *fwnode)
Mika Westerberg07bb80d2017-03-28 10:52:21 +03001276{
Kieran Bingham6a71d8d2017-06-06 12:37:41 +03001277 struct fwnode_handle *endpoint, *parent;
Mika Westerberg07bb80d2017-03-28 10:52:21 +03001278
Kieran Bingham6a71d8d2017-06-06 12:37:41 +03001279 endpoint = fwnode_graph_get_remote_endpoint(fwnode);
1280 parent = fwnode_graph_get_port_parent(endpoint);
Mika Westerberg07bb80d2017-03-28 10:52:21 +03001281
Kieran Bingham6a71d8d2017-06-06 12:37:41 +03001282 fwnode_handle_put(endpoint);
Mika Westerberg07bb80d2017-03-28 10:52:21 +03001283
1284 return parent;
1285}
1286EXPORT_SYMBOL_GPL(fwnode_graph_get_remote_port_parent);
1287
1288/**
1289 * fwnode_graph_get_remote_port - Return fwnode of a remote port
1290 * @fwnode: Endpoint firmware node pointing to the remote endpoint
1291 *
1292 * Extracts firmware node of a remote port the @fwnode points to.
1293 */
Sakari Ailus37ba9832017-07-21 14:39:36 +03001294struct fwnode_handle *
1295fwnode_graph_get_remote_port(const struct fwnode_handle *fwnode)
Mika Westerberg07bb80d2017-03-28 10:52:21 +03001296{
Sakari Ailus3b27d002017-06-06 12:37:38 +03001297 return fwnode_get_next_parent(fwnode_graph_get_remote_endpoint(fwnode));
Mika Westerberg07bb80d2017-03-28 10:52:21 +03001298}
1299EXPORT_SYMBOL_GPL(fwnode_graph_get_remote_port);
1300
1301/**
1302 * fwnode_graph_get_remote_endpoint - Return fwnode of a remote endpoint
1303 * @fwnode: Endpoint firmware node pointing to the remote endpoint
1304 *
1305 * Extracts firmware node of a remote endpoint the @fwnode points to.
1306 */
1307struct fwnode_handle *
Sakari Ailus37ba9832017-07-21 14:39:36 +03001308fwnode_graph_get_remote_endpoint(const struct fwnode_handle *fwnode)
Mika Westerberg07bb80d2017-03-28 10:52:21 +03001309{
Sakari Ailus3b27d002017-06-06 12:37:38 +03001310 return fwnode_call_ptr_op(fwnode, graph_get_remote_endpoint);
Mika Westerberg07bb80d2017-03-28 10:52:21 +03001311}
1312EXPORT_SYMBOL_GPL(fwnode_graph_get_remote_endpoint);
Sakari Ailus2bd54522017-03-28 10:52:25 +03001313
1314/**
Sakari Ailus125ee6b2017-06-06 12:37:40 +03001315 * fwnode_graph_get_remote_node - get remote parent node for given port/endpoint
1316 * @fwnode: pointer to parent fwnode_handle containing graph port/endpoint
1317 * @port_id: identifier of the parent port node
1318 * @endpoint_id: identifier of the endpoint node
1319 *
1320 * Return: Remote fwnode handle associated with remote endpoint node linked
1321 * to @node. Use fwnode_node_put() on it when done.
1322 */
Sakari Ailus37ba9832017-07-21 14:39:36 +03001323struct fwnode_handle *
1324fwnode_graph_get_remote_node(const struct fwnode_handle *fwnode, u32 port_id,
1325 u32 endpoint_id)
Sakari Ailus125ee6b2017-06-06 12:37:40 +03001326{
1327 struct fwnode_handle *endpoint = NULL;
1328
1329 while ((endpoint = fwnode_graph_get_next_endpoint(fwnode, endpoint))) {
1330 struct fwnode_endpoint fwnode_ep;
1331 struct fwnode_handle *remote;
1332 int ret;
1333
1334 ret = fwnode_graph_parse_endpoint(endpoint, &fwnode_ep);
1335 if (ret < 0)
1336 continue;
1337
1338 if (fwnode_ep.port != port_id || fwnode_ep.id != endpoint_id)
1339 continue;
1340
1341 remote = fwnode_graph_get_remote_port_parent(endpoint);
1342 if (!remote)
1343 return NULL;
1344
1345 return fwnode_device_is_available(remote) ? remote : NULL;
1346 }
1347
1348 return NULL;
1349}
1350EXPORT_SYMBOL_GPL(fwnode_graph_get_remote_node);
1351
1352/**
Sakari Ailus2bd54522017-03-28 10:52:25 +03001353 * fwnode_graph_parse_endpoint - parse common endpoint node properties
1354 * @fwnode: pointer to endpoint fwnode_handle
1355 * @endpoint: pointer to the fwnode endpoint data structure
1356 *
1357 * Parse @fwnode representing a graph endpoint node and store the
1358 * information in @endpoint. The caller must hold a reference to
1359 * @fwnode.
1360 */
Sakari Ailus37ba9832017-07-21 14:39:36 +03001361int fwnode_graph_parse_endpoint(const struct fwnode_handle *fwnode,
Sakari Ailus2bd54522017-03-28 10:52:25 +03001362 struct fwnode_endpoint *endpoint)
1363{
Sakari Ailus2bd54522017-03-28 10:52:25 +03001364 memset(endpoint, 0, sizeof(*endpoint));
1365
Sakari Ailus3b27d002017-06-06 12:37:38 +03001366 return fwnode_call_int_op(fwnode, graph_parse_endpoint, endpoint);
Sakari Ailus2bd54522017-03-28 10:52:25 +03001367}
1368EXPORT_SYMBOL(fwnode_graph_parse_endpoint);