xref: /dpdk/drivers/common/dpaax/dpaa_of.c (revision 8c83f28cc8a4dc7ee0fab676051dd483695d4140)
1*8c83f28cSHemant Agrawal /* SPDX-License-Identifier: (BSD-3-Clause OR GPL-2.0)
2*8c83f28cSHemant Agrawal  *
3*8c83f28cSHemant Agrawal  * Copyright 2010-2016 Freescale Semiconductor Inc.
4*8c83f28cSHemant Agrawal  * Copyright 2017 NXP
5*8c83f28cSHemant Agrawal  *
6*8c83f28cSHemant Agrawal  */
7*8c83f28cSHemant Agrawal 
8*8c83f28cSHemant Agrawal #include <dpaa_of.h>
9*8c83f28cSHemant Agrawal #include <assert.h>
10*8c83f28cSHemant Agrawal #include <rte_string_fns.h>
11*8c83f28cSHemant Agrawal #include <dpaax_logs.h>
12*8c83f28cSHemant Agrawal 
13*8c83f28cSHemant Agrawal static int alive;
14*8c83f28cSHemant Agrawal static struct dt_dir root_dir;
15*8c83f28cSHemant Agrawal static const char *base_dir;
16*8c83f28cSHemant Agrawal static COMPAT_LIST_HEAD(linear);
17*8c83f28cSHemant Agrawal 
18*8c83f28cSHemant Agrawal static int
of_open_dir(const char * relative_path,struct dirent *** d)19*8c83f28cSHemant Agrawal of_open_dir(const char *relative_path, struct dirent ***d)
20*8c83f28cSHemant Agrawal {
21*8c83f28cSHemant Agrawal 	int ret;
22*8c83f28cSHemant Agrawal 	char full_path[PATH_MAX];
23*8c83f28cSHemant Agrawal 
24*8c83f28cSHemant Agrawal 	snprintf(full_path, PATH_MAX, "%s/%s", base_dir, relative_path);
25*8c83f28cSHemant Agrawal 	ret = scandir(full_path, d, 0, versionsort);
26*8c83f28cSHemant Agrawal 	if (ret < 0)
27*8c83f28cSHemant Agrawal 		DPAAX_LOG(ERR, "Failed to open directory %s",
28*8c83f28cSHemant Agrawal 			     full_path);
29*8c83f28cSHemant Agrawal 	return ret;
30*8c83f28cSHemant Agrawal }
31*8c83f28cSHemant Agrawal 
32*8c83f28cSHemant Agrawal static void
of_close_dir(struct dirent ** d,int num)33*8c83f28cSHemant Agrawal of_close_dir(struct dirent **d, int num)
34*8c83f28cSHemant Agrawal {
35*8c83f28cSHemant Agrawal 	while (num--)
36*8c83f28cSHemant Agrawal 		free(d[num]);
37*8c83f28cSHemant Agrawal 	free(d);
38*8c83f28cSHemant Agrawal }
39*8c83f28cSHemant Agrawal 
40*8c83f28cSHemant Agrawal static int
of_open_file(const char * relative_path)41*8c83f28cSHemant Agrawal of_open_file(const char *relative_path)
42*8c83f28cSHemant Agrawal {
43*8c83f28cSHemant Agrawal 	int ret;
44*8c83f28cSHemant Agrawal 	char full_path[PATH_MAX];
45*8c83f28cSHemant Agrawal 
46*8c83f28cSHemant Agrawal 	snprintf(full_path, PATH_MAX, "%s/%s", base_dir, relative_path);
47*8c83f28cSHemant Agrawal 	ret = open(full_path, O_RDONLY);
48*8c83f28cSHemant Agrawal 	if (ret < 0)
49*8c83f28cSHemant Agrawal 		DPAAX_LOG(ERR, "Failed to open directory %s",
50*8c83f28cSHemant Agrawal 			     full_path);
51*8c83f28cSHemant Agrawal 	return ret;
52*8c83f28cSHemant Agrawal }
53*8c83f28cSHemant Agrawal 
54*8c83f28cSHemant Agrawal static void
process_file(struct dirent * dent,struct dt_dir * parent)55*8c83f28cSHemant Agrawal process_file(struct dirent *dent, struct dt_dir *parent)
56*8c83f28cSHemant Agrawal {
57*8c83f28cSHemant Agrawal 	int fd;
58*8c83f28cSHemant Agrawal 	struct dt_file *f = malloc(sizeof(*f));
59*8c83f28cSHemant Agrawal 
60*8c83f28cSHemant Agrawal 	if (!f) {
61*8c83f28cSHemant Agrawal 		DPAAX_LOG(DEBUG, "Unable to allocate memory for file node");
62*8c83f28cSHemant Agrawal 		return;
63*8c83f28cSHemant Agrawal 	}
64*8c83f28cSHemant Agrawal 	f->node.is_file = 1;
65*8c83f28cSHemant Agrawal 	strlcpy(f->node.node.name, dent->d_name, NAME_MAX);
66*8c83f28cSHemant Agrawal 	snprintf(f->node.node.full_name, PATH_MAX, "%s/%s",
67*8c83f28cSHemant Agrawal 		 parent->node.node.full_name, dent->d_name);
68*8c83f28cSHemant Agrawal 	f->parent = parent;
69*8c83f28cSHemant Agrawal 	fd = of_open_file(f->node.node.full_name);
70*8c83f28cSHemant Agrawal 	if (fd < 0) {
71*8c83f28cSHemant Agrawal 		DPAAX_LOG(DEBUG, "Unable to open file node");
72*8c83f28cSHemant Agrawal 		free(f);
73*8c83f28cSHemant Agrawal 		return;
74*8c83f28cSHemant Agrawal 	}
75*8c83f28cSHemant Agrawal 	f->len = read(fd, f->buf, OF_FILE_BUF_MAX);
76*8c83f28cSHemant Agrawal 	close(fd);
77*8c83f28cSHemant Agrawal 	if (f->len < 0) {
78*8c83f28cSHemant Agrawal 		DPAAX_LOG(DEBUG, "Unable to read file node");
79*8c83f28cSHemant Agrawal 		free(f);
80*8c83f28cSHemant Agrawal 		return;
81*8c83f28cSHemant Agrawal 	}
82*8c83f28cSHemant Agrawal 	list_add_tail(&f->node.list, &parent->files);
83*8c83f28cSHemant Agrawal }
84*8c83f28cSHemant Agrawal 
85*8c83f28cSHemant Agrawal static const struct dt_dir *
node2dir(const struct device_node * n)86*8c83f28cSHemant Agrawal node2dir(const struct device_node *n)
87*8c83f28cSHemant Agrawal {
88*8c83f28cSHemant Agrawal 	struct dt_node *dn = container_of((struct device_node *)n,
89*8c83f28cSHemant Agrawal 					  struct dt_node, node);
90*8c83f28cSHemant Agrawal 	const struct dt_dir *d = container_of(dn, struct dt_dir, node);
91*8c83f28cSHemant Agrawal 
92*8c83f28cSHemant Agrawal 	assert(!dn->is_file);
93*8c83f28cSHemant Agrawal 	return d;
94*8c83f28cSHemant Agrawal }
95*8c83f28cSHemant Agrawal 
96*8c83f28cSHemant Agrawal /* process_dir() calls iterate_dir(), but the latter will also call the former
97*8c83f28cSHemant Agrawal  * when recursing into sub-directories, so a predeclaration is needed.
98*8c83f28cSHemant Agrawal  */
99*8c83f28cSHemant Agrawal static int process_dir(const char *relative_path, struct dt_dir *dt);
100*8c83f28cSHemant Agrawal 
101*8c83f28cSHemant Agrawal static int
iterate_dir(struct dirent ** d,int num,struct dt_dir * dt)102*8c83f28cSHemant Agrawal iterate_dir(struct dirent **d, int num, struct dt_dir *dt)
103*8c83f28cSHemant Agrawal {
104*8c83f28cSHemant Agrawal 	int loop;
105*8c83f28cSHemant Agrawal 	/* Iterate the directory contents */
106*8c83f28cSHemant Agrawal 	for (loop = 0; loop < num; loop++) {
107*8c83f28cSHemant Agrawal 		struct dt_dir *subdir;
108*8c83f28cSHemant Agrawal 		int ret;
109*8c83f28cSHemant Agrawal 		/* Ignore dot files of all types (especially "..") */
110*8c83f28cSHemant Agrawal 		if (d[loop]->d_name[0] == '.')
111*8c83f28cSHemant Agrawal 			continue;
112*8c83f28cSHemant Agrawal 		switch (d[loop]->d_type) {
113*8c83f28cSHemant Agrawal 		case DT_REG:
114*8c83f28cSHemant Agrawal 			process_file(d[loop], dt);
115*8c83f28cSHemant Agrawal 			break;
116*8c83f28cSHemant Agrawal 		case DT_DIR:
117*8c83f28cSHemant Agrawal 			subdir = malloc(sizeof(*subdir));
118*8c83f28cSHemant Agrawal 			if (!subdir) {
119*8c83f28cSHemant Agrawal 				perror("malloc");
120*8c83f28cSHemant Agrawal 				return -ENOMEM;
121*8c83f28cSHemant Agrawal 			}
122*8c83f28cSHemant Agrawal 			strlcpy(subdir->node.node.name, d[loop]->d_name,
123*8c83f28cSHemant Agrawal 				NAME_MAX);
124*8c83f28cSHemant Agrawal 			snprintf(subdir->node.node.full_name, PATH_MAX,
125*8c83f28cSHemant Agrawal 				 "%s/%s", dt->node.node.full_name,
126*8c83f28cSHemant Agrawal 				 d[loop]->d_name);
127*8c83f28cSHemant Agrawal 			subdir->parent = dt;
128*8c83f28cSHemant Agrawal 			ret = process_dir(subdir->node.node.full_name, subdir);
129*8c83f28cSHemant Agrawal 			if (ret)
130*8c83f28cSHemant Agrawal 				return ret;
131*8c83f28cSHemant Agrawal 			list_add_tail(&subdir->node.list, &dt->subdirs);
132*8c83f28cSHemant Agrawal 			break;
133*8c83f28cSHemant Agrawal 		default:
134*8c83f28cSHemant Agrawal 			DPAAX_LOG(DEBUG, "Ignoring invalid dt entry %s/%s",
135*8c83f28cSHemant Agrawal 				     dt->node.node.full_name, d[loop]->d_name);
136*8c83f28cSHemant Agrawal 		}
137*8c83f28cSHemant Agrawal 	}
138*8c83f28cSHemant Agrawal 	return 0;
139*8c83f28cSHemant Agrawal }
140*8c83f28cSHemant Agrawal 
141*8c83f28cSHemant Agrawal static int
process_dir(const char * relative_path,struct dt_dir * dt)142*8c83f28cSHemant Agrawal process_dir(const char *relative_path, struct dt_dir *dt)
143*8c83f28cSHemant Agrawal {
144*8c83f28cSHemant Agrawal 	struct dirent **d;
145*8c83f28cSHemant Agrawal 	int ret, num;
146*8c83f28cSHemant Agrawal 
147*8c83f28cSHemant Agrawal 	dt->node.is_file = 0;
148*8c83f28cSHemant Agrawal 	INIT_LIST_HEAD(&dt->subdirs);
149*8c83f28cSHemant Agrawal 	INIT_LIST_HEAD(&dt->files);
150*8c83f28cSHemant Agrawal 	ret = of_open_dir(relative_path, &d);
151*8c83f28cSHemant Agrawal 	if (ret < 0)
152*8c83f28cSHemant Agrawal 		return ret;
153*8c83f28cSHemant Agrawal 	num = ret;
154*8c83f28cSHemant Agrawal 	ret = iterate_dir(d, num, dt);
155*8c83f28cSHemant Agrawal 	of_close_dir(d, num);
156*8c83f28cSHemant Agrawal 	return (ret < 0) ? ret : 0;
157*8c83f28cSHemant Agrawal }
158*8c83f28cSHemant Agrawal 
159*8c83f28cSHemant Agrawal static void
linear_dir(struct dt_dir * d)160*8c83f28cSHemant Agrawal linear_dir(struct dt_dir *d)
161*8c83f28cSHemant Agrawal {
162*8c83f28cSHemant Agrawal 	struct dt_file *f;
163*8c83f28cSHemant Agrawal 	struct dt_dir *dd;
164*8c83f28cSHemant Agrawal 
165*8c83f28cSHemant Agrawal 	d->compatible = NULL;
166*8c83f28cSHemant Agrawal 	d->status = NULL;
167*8c83f28cSHemant Agrawal 	d->lphandle = NULL;
168*8c83f28cSHemant Agrawal 	d->a_cells = NULL;
169*8c83f28cSHemant Agrawal 	d->s_cells = NULL;
170*8c83f28cSHemant Agrawal 	d->reg = NULL;
171*8c83f28cSHemant Agrawal 	list_for_each_entry(f, &d->files, node.list) {
172*8c83f28cSHemant Agrawal 		if (!strcmp(f->node.node.name, "compatible")) {
173*8c83f28cSHemant Agrawal 			if (d->compatible)
174*8c83f28cSHemant Agrawal 				DPAAX_LOG(DEBUG, "Duplicate compatible in"
175*8c83f28cSHemant Agrawal 					     " %s", d->node.node.full_name);
176*8c83f28cSHemant Agrawal 			d->compatible = f;
177*8c83f28cSHemant Agrawal 		} else if (!strcmp(f->node.node.name, "status")) {
178*8c83f28cSHemant Agrawal 			if (d->status)
179*8c83f28cSHemant Agrawal 				DPAAX_LOG(DEBUG, "Duplicate status in %s",
180*8c83f28cSHemant Agrawal 					     d->node.node.full_name);
181*8c83f28cSHemant Agrawal 			d->status = f;
182*8c83f28cSHemant Agrawal 		} else if (!strcmp(f->node.node.name, "linux,phandle")) {
183*8c83f28cSHemant Agrawal 			if (d->lphandle)
184*8c83f28cSHemant Agrawal 				DPAAX_LOG(DEBUG, "Duplicate lphandle in %s",
185*8c83f28cSHemant Agrawal 					     d->node.node.full_name);
186*8c83f28cSHemant Agrawal 			d->lphandle = f;
187*8c83f28cSHemant Agrawal 		} else if (!strcmp(f->node.node.name, "phandle")) {
188*8c83f28cSHemant Agrawal 			if (d->lphandle)
189*8c83f28cSHemant Agrawal 				DPAAX_LOG(DEBUG, "Duplicate lphandle in %s",
190*8c83f28cSHemant Agrawal 					     d->node.node.full_name);
191*8c83f28cSHemant Agrawal 			d->lphandle = f;
192*8c83f28cSHemant Agrawal 		} else if (!strcmp(f->node.node.name, "#address-cells")) {
193*8c83f28cSHemant Agrawal 			if (d->a_cells)
194*8c83f28cSHemant Agrawal 				DPAAX_LOG(DEBUG, "Duplicate a_cells in %s",
195*8c83f28cSHemant Agrawal 					     d->node.node.full_name);
196*8c83f28cSHemant Agrawal 			d->a_cells = f;
197*8c83f28cSHemant Agrawal 		} else if (!strcmp(f->node.node.name, "#size-cells")) {
198*8c83f28cSHemant Agrawal 			if (d->s_cells)
199*8c83f28cSHemant Agrawal 				DPAAX_LOG(DEBUG, "Duplicate s_cells in %s",
200*8c83f28cSHemant Agrawal 					     d->node.node.full_name);
201*8c83f28cSHemant Agrawal 			d->s_cells = f;
202*8c83f28cSHemant Agrawal 		} else if (!strcmp(f->node.node.name, "reg")) {
203*8c83f28cSHemant Agrawal 			if (d->reg)
204*8c83f28cSHemant Agrawal 				DPAAX_LOG(DEBUG, "Duplicate reg in %s",
205*8c83f28cSHemant Agrawal 					     d->node.node.full_name);
206*8c83f28cSHemant Agrawal 			d->reg = f;
207*8c83f28cSHemant Agrawal 		}
208*8c83f28cSHemant Agrawal 	}
209*8c83f28cSHemant Agrawal 
210*8c83f28cSHemant Agrawal 	list_for_each_entry(dd, &d->subdirs, node.list) {
211*8c83f28cSHemant Agrawal 		list_add_tail(&dd->linear, &linear);
212*8c83f28cSHemant Agrawal 		linear_dir(dd);
213*8c83f28cSHemant Agrawal 	}
214*8c83f28cSHemant Agrawal }
215*8c83f28cSHemant Agrawal 
216*8c83f28cSHemant Agrawal int
of_init_path(const char * dt_path)217*8c83f28cSHemant Agrawal of_init_path(const char *dt_path)
218*8c83f28cSHemant Agrawal {
219*8c83f28cSHemant Agrawal 	int ret;
220*8c83f28cSHemant Agrawal 
221*8c83f28cSHemant Agrawal 	base_dir = dt_path;
222*8c83f28cSHemant Agrawal 
223*8c83f28cSHemant Agrawal 	/* This needs to be singleton initialization */
224*8c83f28cSHemant Agrawal 	DPAAX_HWWARN(alive, "Double-init of device-tree driver!");
225*8c83f28cSHemant Agrawal 
226*8c83f28cSHemant Agrawal 	/* Prepare root node (the remaining fields are set in process_dir()) */
227*8c83f28cSHemant Agrawal 	root_dir.node.node.name[0] = '\0';
228*8c83f28cSHemant Agrawal 	root_dir.node.node.full_name[0] = '\0';
229*8c83f28cSHemant Agrawal 	INIT_LIST_HEAD(&root_dir.node.list);
230*8c83f28cSHemant Agrawal 	root_dir.parent = NULL;
231*8c83f28cSHemant Agrawal 
232*8c83f28cSHemant Agrawal 	/* Kick things off... */
233*8c83f28cSHemant Agrawal 	ret = process_dir("", &root_dir);
234*8c83f28cSHemant Agrawal 	if (ret) {
235*8c83f28cSHemant Agrawal 		DPAAX_LOG(ERR, "Unable to parse device tree");
236*8c83f28cSHemant Agrawal 		return ret;
237*8c83f28cSHemant Agrawal 	}
238*8c83f28cSHemant Agrawal 
239*8c83f28cSHemant Agrawal 	/* Now make a flat, linear list of directories */
240*8c83f28cSHemant Agrawal 	linear_dir(&root_dir);
241*8c83f28cSHemant Agrawal 	alive = 1;
242*8c83f28cSHemant Agrawal 	return 0;
243*8c83f28cSHemant Agrawal }
244*8c83f28cSHemant Agrawal 
245*8c83f28cSHemant Agrawal static void
destroy_dir(struct dt_dir * d)246*8c83f28cSHemant Agrawal destroy_dir(struct dt_dir *d)
247*8c83f28cSHemant Agrawal {
248*8c83f28cSHemant Agrawal 	struct dt_file *f, *tmpf;
249*8c83f28cSHemant Agrawal 	struct dt_dir *dd, *tmpd;
250*8c83f28cSHemant Agrawal 
251*8c83f28cSHemant Agrawal 	list_for_each_entry_safe(f, tmpf, &d->files, node.list) {
252*8c83f28cSHemant Agrawal 		list_del(&f->node.list);
253*8c83f28cSHemant Agrawal 		free(f);
254*8c83f28cSHemant Agrawal 	}
255*8c83f28cSHemant Agrawal 	list_for_each_entry_safe(dd, tmpd, &d->subdirs, node.list) {
256*8c83f28cSHemant Agrawal 		destroy_dir(dd);
257*8c83f28cSHemant Agrawal 		list_del(&dd->node.list);
258*8c83f28cSHemant Agrawal 		free(dd);
259*8c83f28cSHemant Agrawal 	}
260*8c83f28cSHemant Agrawal }
261*8c83f28cSHemant Agrawal 
262*8c83f28cSHemant Agrawal void
of_finish(void)263*8c83f28cSHemant Agrawal of_finish(void)
264*8c83f28cSHemant Agrawal {
265*8c83f28cSHemant Agrawal 	DPAAX_HWWARN(!alive, "Double-finish of device-tree driver!");
266*8c83f28cSHemant Agrawal 
267*8c83f28cSHemant Agrawal 	destroy_dir(&root_dir);
268*8c83f28cSHemant Agrawal 	INIT_LIST_HEAD(&linear);
269*8c83f28cSHemant Agrawal 	alive = 0;
270*8c83f28cSHemant Agrawal }
271*8c83f28cSHemant Agrawal 
272*8c83f28cSHemant Agrawal static const struct dt_dir *
next_linear(const struct dt_dir * f)273*8c83f28cSHemant Agrawal next_linear(const struct dt_dir *f)
274*8c83f28cSHemant Agrawal {
275*8c83f28cSHemant Agrawal 	if (f->linear.next == &linear)
276*8c83f28cSHemant Agrawal 		return NULL;
277*8c83f28cSHemant Agrawal 	return list_entry(f->linear.next, struct dt_dir, linear);
278*8c83f28cSHemant Agrawal }
279*8c83f28cSHemant Agrawal 
280*8c83f28cSHemant Agrawal static int
check_compatible(const struct dt_file * f,const char * compatible)281*8c83f28cSHemant Agrawal check_compatible(const struct dt_file *f, const char *compatible)
282*8c83f28cSHemant Agrawal {
283*8c83f28cSHemant Agrawal 	const char *c = (char *)f->buf;
284*8c83f28cSHemant Agrawal 	unsigned int len, remains = f->len;
285*8c83f28cSHemant Agrawal 
286*8c83f28cSHemant Agrawal 	while (remains) {
287*8c83f28cSHemant Agrawal 		len = strlen(c);
288*8c83f28cSHemant Agrawal 		if (!strcmp(c, compatible))
289*8c83f28cSHemant Agrawal 			return 1;
290*8c83f28cSHemant Agrawal 
291*8c83f28cSHemant Agrawal 		if (remains < len + 1)
292*8c83f28cSHemant Agrawal 			break;
293*8c83f28cSHemant Agrawal 
294*8c83f28cSHemant Agrawal 		c += (len + 1);
295*8c83f28cSHemant Agrawal 		remains -= (len + 1);
296*8c83f28cSHemant Agrawal 	}
297*8c83f28cSHemant Agrawal 	return 0;
298*8c83f28cSHemant Agrawal }
299*8c83f28cSHemant Agrawal 
300*8c83f28cSHemant Agrawal const struct device_node *
of_find_compatible_node(const struct device_node * from,const char * type __rte_unused,const char * compatible)301*8c83f28cSHemant Agrawal of_find_compatible_node(const struct device_node *from,
302*8c83f28cSHemant Agrawal 			const char *type __rte_unused,
303*8c83f28cSHemant Agrawal 			const char *compatible)
304*8c83f28cSHemant Agrawal {
305*8c83f28cSHemant Agrawal 	const struct dt_dir *d;
306*8c83f28cSHemant Agrawal 
307*8c83f28cSHemant Agrawal 	DPAAX_HWWARN(!alive, "Device-tree driver not initialised!");
308*8c83f28cSHemant Agrawal 
309*8c83f28cSHemant Agrawal 	if (list_empty(&linear))
310*8c83f28cSHemant Agrawal 		return NULL;
311*8c83f28cSHemant Agrawal 	if (!from)
312*8c83f28cSHemant Agrawal 		d = list_entry(linear.next, struct dt_dir, linear);
313*8c83f28cSHemant Agrawal 	else
314*8c83f28cSHemant Agrawal 		d = node2dir(from);
315*8c83f28cSHemant Agrawal 	for (d = next_linear(d); d && (!d->compatible ||
316*8c83f28cSHemant Agrawal 				       !check_compatible(d->compatible,
317*8c83f28cSHemant Agrawal 				       compatible));
318*8c83f28cSHemant Agrawal 			d = next_linear(d))
319*8c83f28cSHemant Agrawal 		;
320*8c83f28cSHemant Agrawal 	if (d)
321*8c83f28cSHemant Agrawal 		return &d->node.node;
322*8c83f28cSHemant Agrawal 	return NULL;
323*8c83f28cSHemant Agrawal }
324*8c83f28cSHemant Agrawal 
325*8c83f28cSHemant Agrawal const void *
of_get_property(const struct device_node * from,const char * name,size_t * lenp)326*8c83f28cSHemant Agrawal of_get_property(const struct device_node *from, const char *name,
327*8c83f28cSHemant Agrawal 		size_t *lenp)
328*8c83f28cSHemant Agrawal {
329*8c83f28cSHemant Agrawal 	const struct dt_dir *d;
330*8c83f28cSHemant Agrawal 	const struct dt_file *f;
331*8c83f28cSHemant Agrawal 
332*8c83f28cSHemant Agrawal 	DPAAX_HWWARN(!alive, "Device-tree driver not initialised!");
333*8c83f28cSHemant Agrawal 
334*8c83f28cSHemant Agrawal 	d = node2dir(from);
335*8c83f28cSHemant Agrawal 	list_for_each_entry(f, &d->files, node.list)
336*8c83f28cSHemant Agrawal 		if (!strcmp(f->node.node.name, name)) {
337*8c83f28cSHemant Agrawal 			if (lenp)
338*8c83f28cSHemant Agrawal 				*lenp = f->len;
339*8c83f28cSHemant Agrawal 			return f->buf;
340*8c83f28cSHemant Agrawal 		}
341*8c83f28cSHemant Agrawal 	return NULL;
342*8c83f28cSHemant Agrawal }
343*8c83f28cSHemant Agrawal 
344*8c83f28cSHemant Agrawal bool
of_device_is_available(const struct device_node * dev_node)345*8c83f28cSHemant Agrawal of_device_is_available(const struct device_node *dev_node)
346*8c83f28cSHemant Agrawal {
347*8c83f28cSHemant Agrawal 	const struct dt_dir *d;
348*8c83f28cSHemant Agrawal 
349*8c83f28cSHemant Agrawal 	DPAAX_HWWARN(!alive, "Device-tree driver not initialised!");
350*8c83f28cSHemant Agrawal 	d = node2dir(dev_node);
351*8c83f28cSHemant Agrawal 	if (!d->status)
352*8c83f28cSHemant Agrawal 		return true;
353*8c83f28cSHemant Agrawal 	if (!strcmp((char *)d->status->buf, "okay"))
354*8c83f28cSHemant Agrawal 		return true;
355*8c83f28cSHemant Agrawal 	if (!strcmp((char *)d->status->buf, "ok"))
356*8c83f28cSHemant Agrawal 		return true;
357*8c83f28cSHemant Agrawal 	return false;
358*8c83f28cSHemant Agrawal }
359*8c83f28cSHemant Agrawal 
360*8c83f28cSHemant Agrawal const struct device_node *
of_find_node_by_phandle(uint64_t ph)361*8c83f28cSHemant Agrawal of_find_node_by_phandle(uint64_t ph)
362*8c83f28cSHemant Agrawal {
363*8c83f28cSHemant Agrawal 	const struct dt_dir *d;
364*8c83f28cSHemant Agrawal 
365*8c83f28cSHemant Agrawal 	DPAAX_HWWARN(!alive, "Device-tree driver not initialised!");
366*8c83f28cSHemant Agrawal 	list_for_each_entry(d, &linear, linear)
367*8c83f28cSHemant Agrawal 		if (d->lphandle && (d->lphandle->len == 4) &&
368*8c83f28cSHemant Agrawal 		    !memcmp(d->lphandle->buf, &ph, 4))
369*8c83f28cSHemant Agrawal 			return &d->node.node;
370*8c83f28cSHemant Agrawal 	return NULL;
371*8c83f28cSHemant Agrawal }
372*8c83f28cSHemant Agrawal 
373*8c83f28cSHemant Agrawal const struct device_node *
of_get_parent(const struct device_node * dev_node)374*8c83f28cSHemant Agrawal of_get_parent(const struct device_node *dev_node)
375*8c83f28cSHemant Agrawal {
376*8c83f28cSHemant Agrawal 	const struct dt_dir *d;
377*8c83f28cSHemant Agrawal 
378*8c83f28cSHemant Agrawal 	DPAAX_HWWARN(!alive, "Device-tree driver not initialised!");
379*8c83f28cSHemant Agrawal 
380*8c83f28cSHemant Agrawal 	if (!dev_node)
381*8c83f28cSHemant Agrawal 		return NULL;
382*8c83f28cSHemant Agrawal 	d = node2dir(dev_node);
383*8c83f28cSHemant Agrawal 	if (!d->parent)
384*8c83f28cSHemant Agrawal 		return NULL;
385*8c83f28cSHemant Agrawal 	return &d->parent->node.node;
386*8c83f28cSHemant Agrawal }
387*8c83f28cSHemant Agrawal 
388*8c83f28cSHemant Agrawal const struct device_node *
of_get_next_child(const struct device_node * dev_node,const struct device_node * prev)389*8c83f28cSHemant Agrawal of_get_next_child(const struct device_node *dev_node,
390*8c83f28cSHemant Agrawal 		  const struct device_node *prev)
391*8c83f28cSHemant Agrawal {
392*8c83f28cSHemant Agrawal 	const struct dt_dir *p, *c;
393*8c83f28cSHemant Agrawal 
394*8c83f28cSHemant Agrawal 	DPAAX_HWWARN(!alive, "Device-tree driver not initialised!");
395*8c83f28cSHemant Agrawal 
396*8c83f28cSHemant Agrawal 	if (!dev_node)
397*8c83f28cSHemant Agrawal 		return NULL;
398*8c83f28cSHemant Agrawal 	p = node2dir(dev_node);
399*8c83f28cSHemant Agrawal 	if (prev) {
400*8c83f28cSHemant Agrawal 		c = node2dir(prev);
401*8c83f28cSHemant Agrawal 		DPAAX_HWWARN((c->parent != p), "Parent/child mismatch");
402*8c83f28cSHemant Agrawal 		if (c->parent != p)
403*8c83f28cSHemant Agrawal 			return NULL;
404*8c83f28cSHemant Agrawal 		if (c->node.list.next == &p->subdirs)
405*8c83f28cSHemant Agrawal 			/* prev was the last child */
406*8c83f28cSHemant Agrawal 			return NULL;
407*8c83f28cSHemant Agrawal 		c = list_entry(c->node.list.next, struct dt_dir, node.list);
408*8c83f28cSHemant Agrawal 		return &c->node.node;
409*8c83f28cSHemant Agrawal 	}
410*8c83f28cSHemant Agrawal 	/* Return first child */
411*8c83f28cSHemant Agrawal 	if (list_empty(&p->subdirs))
412*8c83f28cSHemant Agrawal 		return NULL;
413*8c83f28cSHemant Agrawal 	c = list_entry(p->subdirs.next, struct dt_dir, node.list);
414*8c83f28cSHemant Agrawal 	return &c->node.node;
415*8c83f28cSHemant Agrawal }
416*8c83f28cSHemant Agrawal 
417*8c83f28cSHemant Agrawal uint32_t
of_n_addr_cells(const struct device_node * dev_node)418*8c83f28cSHemant Agrawal of_n_addr_cells(const struct device_node *dev_node)
419*8c83f28cSHemant Agrawal {
420*8c83f28cSHemant Agrawal 	const struct dt_dir *d;
421*8c83f28cSHemant Agrawal 
422*8c83f28cSHemant Agrawal 	DPAAX_HWWARN(!alive, "Device-tree driver not initialised");
423*8c83f28cSHemant Agrawal 	if (!dev_node)
424*8c83f28cSHemant Agrawal 		return OF_DEFAULT_NA;
425*8c83f28cSHemant Agrawal 	d = node2dir(dev_node);
426*8c83f28cSHemant Agrawal 	while ((d = d->parent))
427*8c83f28cSHemant Agrawal 		if (d->a_cells) {
428*8c83f28cSHemant Agrawal 			unsigned char *buf =
429*8c83f28cSHemant Agrawal 				(unsigned char *)&d->a_cells->buf[0];
430*8c83f28cSHemant Agrawal 			assert(d->a_cells->len == 4);
431*8c83f28cSHemant Agrawal 			return ((uint32_t)buf[0] << 24) |
432*8c83f28cSHemant Agrawal 				((uint32_t)buf[1] << 16) |
433*8c83f28cSHemant Agrawal 				((uint32_t)buf[2] << 8) |
434*8c83f28cSHemant Agrawal 				(uint32_t)buf[3];
435*8c83f28cSHemant Agrawal 		}
436*8c83f28cSHemant Agrawal 	return OF_DEFAULT_NA;
437*8c83f28cSHemant Agrawal }
438*8c83f28cSHemant Agrawal 
439*8c83f28cSHemant Agrawal uint32_t
of_n_size_cells(const struct device_node * dev_node)440*8c83f28cSHemant Agrawal of_n_size_cells(const struct device_node *dev_node)
441*8c83f28cSHemant Agrawal {
442*8c83f28cSHemant Agrawal 	const struct dt_dir *d;
443*8c83f28cSHemant Agrawal 
444*8c83f28cSHemant Agrawal 	DPAAX_HWWARN(!alive, "Device-tree driver not initialised!");
445*8c83f28cSHemant Agrawal 	if (!dev_node)
446*8c83f28cSHemant Agrawal 		return OF_DEFAULT_NA;
447*8c83f28cSHemant Agrawal 	d = node2dir(dev_node);
448*8c83f28cSHemant Agrawal 	while ((d = d->parent))
449*8c83f28cSHemant Agrawal 		if (d->s_cells) {
450*8c83f28cSHemant Agrawal 			unsigned char *buf =
451*8c83f28cSHemant Agrawal 				(unsigned char *)&d->s_cells->buf[0];
452*8c83f28cSHemant Agrawal 			assert(d->s_cells->len == 4);
453*8c83f28cSHemant Agrawal 			return ((uint32_t)buf[0] << 24) |
454*8c83f28cSHemant Agrawal 				((uint32_t)buf[1] << 16) |
455*8c83f28cSHemant Agrawal 				((uint32_t)buf[2] << 8) |
456*8c83f28cSHemant Agrawal 				(uint32_t)buf[3];
457*8c83f28cSHemant Agrawal 		}
458*8c83f28cSHemant Agrawal 	return OF_DEFAULT_NS;
459*8c83f28cSHemant Agrawal }
460*8c83f28cSHemant Agrawal 
461*8c83f28cSHemant Agrawal const uint32_t *
of_get_address(const struct device_node * dev_node,size_t idx,uint64_t * size,uint32_t * flags __rte_unused)462*8c83f28cSHemant Agrawal of_get_address(const struct device_node *dev_node, size_t idx,
463*8c83f28cSHemant Agrawal 	       uint64_t *size, uint32_t *flags __rte_unused)
464*8c83f28cSHemant Agrawal {
465*8c83f28cSHemant Agrawal 	const struct dt_dir *d;
466*8c83f28cSHemant Agrawal 	const unsigned char *buf;
467*8c83f28cSHemant Agrawal 	uint32_t na = of_n_addr_cells(dev_node);
468*8c83f28cSHemant Agrawal 	uint32_t ns = of_n_size_cells(dev_node);
469*8c83f28cSHemant Agrawal 
470*8c83f28cSHemant Agrawal 	if (!dev_node)
471*8c83f28cSHemant Agrawal 		d = &root_dir;
472*8c83f28cSHemant Agrawal 	else
473*8c83f28cSHemant Agrawal 		d = node2dir(dev_node);
474*8c83f28cSHemant Agrawal 	if (!d->reg)
475*8c83f28cSHemant Agrawal 		return NULL;
476*8c83f28cSHemant Agrawal 	assert(d->reg->len % ((na + ns) * 4) == 0);
477*8c83f28cSHemant Agrawal 	assert(d->reg->len / ((na + ns) * 4) > (unsigned int) idx);
478*8c83f28cSHemant Agrawal 	buf = (const unsigned char *)&d->reg->buf[0];
479*8c83f28cSHemant Agrawal 	buf += (na + ns) * idx * 4;
480*8c83f28cSHemant Agrawal 	if (size)
481*8c83f28cSHemant Agrawal 		for (*size = 0; ns > 0; ns--, na++)
482*8c83f28cSHemant Agrawal 			*size = (*size << 32) +
483*8c83f28cSHemant Agrawal 				(((uint32_t)buf[4 * na] << 24) |
484*8c83f28cSHemant Agrawal 				((uint32_t)buf[4 * na + 1] << 16) |
485*8c83f28cSHemant Agrawal 				((uint32_t)buf[4 * na + 2] << 8) |
486*8c83f28cSHemant Agrawal 				(uint32_t)buf[4 * na + 3]);
487*8c83f28cSHemant Agrawal 	return (const uint32_t *)buf;
488*8c83f28cSHemant Agrawal }
489*8c83f28cSHemant Agrawal 
490*8c83f28cSHemant Agrawal uint64_t
of_translate_address(const struct device_node * dev_node,const uint32_t * addr)491*8c83f28cSHemant Agrawal of_translate_address(const struct device_node *dev_node,
492*8c83f28cSHemant Agrawal 		     const uint32_t *addr)
493*8c83f28cSHemant Agrawal {
494*8c83f28cSHemant Agrawal 	uint64_t phys_addr, tmp_addr;
495*8c83f28cSHemant Agrawal 	const struct device_node *parent;
496*8c83f28cSHemant Agrawal 	const uint32_t *ranges;
497*8c83f28cSHemant Agrawal 	size_t rlen;
498*8c83f28cSHemant Agrawal 	uint32_t na, pna;
499*8c83f28cSHemant Agrawal 
500*8c83f28cSHemant Agrawal 	DPAAX_HWWARN(!alive, "Device-tree driver not initialised!");
501*8c83f28cSHemant Agrawal 	assert(dev_node != NULL);
502*8c83f28cSHemant Agrawal 
503*8c83f28cSHemant Agrawal 	na = of_n_addr_cells(dev_node);
504*8c83f28cSHemant Agrawal 	phys_addr = of_read_number(addr, na);
505*8c83f28cSHemant Agrawal 
506*8c83f28cSHemant Agrawal 	dev_node = of_get_parent(dev_node);
507*8c83f28cSHemant Agrawal 	if (!dev_node)
508*8c83f28cSHemant Agrawal 		return 0;
509*8c83f28cSHemant Agrawal 	else if (node2dir(dev_node) == &root_dir)
510*8c83f28cSHemant Agrawal 		return phys_addr;
511*8c83f28cSHemant Agrawal 
512*8c83f28cSHemant Agrawal 	do {
513*8c83f28cSHemant Agrawal 		pna = of_n_addr_cells(dev_node);
514*8c83f28cSHemant Agrawal 		parent = of_get_parent(dev_node);
515*8c83f28cSHemant Agrawal 		if (!parent)
516*8c83f28cSHemant Agrawal 			return 0;
517*8c83f28cSHemant Agrawal 
518*8c83f28cSHemant Agrawal 		ranges = of_get_property(dev_node, "ranges", &rlen);
519*8c83f28cSHemant Agrawal 		/* "ranges" property is missing. Translation breaks */
520*8c83f28cSHemant Agrawal 		if (!ranges)
521*8c83f28cSHemant Agrawal 			return 0;
522*8c83f28cSHemant Agrawal 		/* "ranges" property is empty. Do 1:1 translation */
523*8c83f28cSHemant Agrawal 		else if (rlen == 0)
524*8c83f28cSHemant Agrawal 			continue;
525*8c83f28cSHemant Agrawal 		else
526*8c83f28cSHemant Agrawal 			tmp_addr = of_read_number(ranges + na, pna);
527*8c83f28cSHemant Agrawal 
528*8c83f28cSHemant Agrawal 		na = pna;
529*8c83f28cSHemant Agrawal 		dev_node = parent;
530*8c83f28cSHemant Agrawal 		phys_addr += tmp_addr;
531*8c83f28cSHemant Agrawal 	} while (node2dir(parent) != &root_dir);
532*8c83f28cSHemant Agrawal 
533*8c83f28cSHemant Agrawal 	return phys_addr;
534*8c83f28cSHemant Agrawal }
535*8c83f28cSHemant Agrawal 
536*8c83f28cSHemant Agrawal bool
of_device_is_compatible(const struct device_node * dev_node,const char * compatible)537*8c83f28cSHemant Agrawal of_device_is_compatible(const struct device_node *dev_node,
538*8c83f28cSHemant Agrawal 			const char *compatible)
539*8c83f28cSHemant Agrawal {
540*8c83f28cSHemant Agrawal 	const struct dt_dir *d;
541*8c83f28cSHemant Agrawal 
542*8c83f28cSHemant Agrawal 	DPAAX_HWWARN(!alive, "Device-tree driver not initialised!");
543*8c83f28cSHemant Agrawal 	if (!dev_node)
544*8c83f28cSHemant Agrawal 		d = &root_dir;
545*8c83f28cSHemant Agrawal 	else
546*8c83f28cSHemant Agrawal 		d = node2dir(dev_node);
547*8c83f28cSHemant Agrawal 	if (d->compatible && check_compatible(d->compatible, compatible))
548*8c83f28cSHemant Agrawal 		return true;
549*8c83f28cSHemant Agrawal 	return false;
550*8c83f28cSHemant Agrawal }
551*8c83f28cSHemant Agrawal 
of_get_mac_addr(const struct device_node * np,const char * name)552*8c83f28cSHemant Agrawal static const void *of_get_mac_addr(const struct device_node *np,
553*8c83f28cSHemant Agrawal 		const char *name)
554*8c83f28cSHemant Agrawal {
555*8c83f28cSHemant Agrawal 	return of_get_property(np, name, NULL);
556*8c83f28cSHemant Agrawal }
557*8c83f28cSHemant Agrawal 
558*8c83f28cSHemant Agrawal /**
559*8c83f28cSHemant Agrawal  * Search the device tree for the best MAC address to use.  'mac-address' is
560*8c83f28cSHemant Agrawal  * checked first, because that is supposed to contain to "most recent" MAC
561*8c83f28cSHemant Agrawal  * address. If that isn't set, then 'local-mac-address' is checked next,
562*8c83f28cSHemant Agrawal  * because that is the default address.  If that isn't set, then the obsolete
563*8c83f28cSHemant Agrawal  * 'address' is checked, just in case we're using an old device tree.
564*8c83f28cSHemant Agrawal  *
565*8c83f28cSHemant Agrawal  * Note that the 'address' property is supposed to contain a virtual address of
566*8c83f28cSHemant Agrawal  * the register set, but some DTS files have redefined that property to be the
567*8c83f28cSHemant Agrawal  * MAC address.
568*8c83f28cSHemant Agrawal  *
569*8c83f28cSHemant Agrawal  * All-zero MAC addresses are rejected, because those could be properties that
570*8c83f28cSHemant Agrawal  * exist in the device tree, but were not set by U-Boot.  For example, the
571*8c83f28cSHemant Agrawal  * DTS could define 'mac-address' and 'local-mac-address', with zero MAC
572*8c83f28cSHemant Agrawal  * addresses.  Some older U-Boots only initialized 'local-mac-address'.  In
573*8c83f28cSHemant Agrawal  * this case, the real MAC is in 'local-mac-address', and 'mac-address' exists
574*8c83f28cSHemant Agrawal  * but is all zeros.
575*8c83f28cSHemant Agrawal  */
of_get_mac_address(const struct device_node * np)576*8c83f28cSHemant Agrawal const void *of_get_mac_address(const struct device_node *np)
577*8c83f28cSHemant Agrawal {
578*8c83f28cSHemant Agrawal 	const void *addr;
579*8c83f28cSHemant Agrawal 
580*8c83f28cSHemant Agrawal 	addr = of_get_mac_addr(np, "mac-address");
581*8c83f28cSHemant Agrawal 	if (addr)
582*8c83f28cSHemant Agrawal 		return addr;
583*8c83f28cSHemant Agrawal 
584*8c83f28cSHemant Agrawal 	addr = of_get_mac_addr(np, "local-mac-address");
585*8c83f28cSHemant Agrawal 	if (addr)
586*8c83f28cSHemant Agrawal 		return addr;
587*8c83f28cSHemant Agrawal 
588*8c83f28cSHemant Agrawal 	return of_get_mac_addr(np, "address");
589*8c83f28cSHemant Agrawal }
590