xref: /aosp_15_r20/external/dtc/fdtput.c (revision cd60bc56d4bea3af4ec04523e4d71c2b272c8aff)
1*cd60bc56SAndroid Build Coastguard Worker // SPDX-License-Identifier: GPL-2.0-or-later
2*cd60bc56SAndroid Build Coastguard Worker /*
3*cd60bc56SAndroid Build Coastguard Worker  * Copyright (c) 2011 The Chromium OS Authors. All rights reserved.
4*cd60bc56SAndroid Build Coastguard Worker  */
5*cd60bc56SAndroid Build Coastguard Worker 
6*cd60bc56SAndroid Build Coastguard Worker #include <assert.h>
7*cd60bc56SAndroid Build Coastguard Worker #include <ctype.h>
8*cd60bc56SAndroid Build Coastguard Worker #include <getopt.h>
9*cd60bc56SAndroid Build Coastguard Worker #include <stdio.h>
10*cd60bc56SAndroid Build Coastguard Worker #include <stdlib.h>
11*cd60bc56SAndroid Build Coastguard Worker #include <string.h>
12*cd60bc56SAndroid Build Coastguard Worker 
13*cd60bc56SAndroid Build Coastguard Worker #include <libfdt.h>
14*cd60bc56SAndroid Build Coastguard Worker 
15*cd60bc56SAndroid Build Coastguard Worker #include "util.h"
16*cd60bc56SAndroid Build Coastguard Worker 
17*cd60bc56SAndroid Build Coastguard Worker /* These are the operations we support */
18*cd60bc56SAndroid Build Coastguard Worker enum oper_type {
19*cd60bc56SAndroid Build Coastguard Worker 	OPER_WRITE_PROP,		/* Write a property in a node */
20*cd60bc56SAndroid Build Coastguard Worker 	OPER_CREATE_NODE,		/* Create a new node */
21*cd60bc56SAndroid Build Coastguard Worker 	OPER_REMOVE_NODE,		/* Delete a node */
22*cd60bc56SAndroid Build Coastguard Worker 	OPER_DELETE_PROP,		/* Delete a property in a node */
23*cd60bc56SAndroid Build Coastguard Worker };
24*cd60bc56SAndroid Build Coastguard Worker 
25*cd60bc56SAndroid Build Coastguard Worker struct display_info {
26*cd60bc56SAndroid Build Coastguard Worker 	enum oper_type oper;	/* operation to perform */
27*cd60bc56SAndroid Build Coastguard Worker 	int type;		/* data type (s/i/u/x or 0 for default) */
28*cd60bc56SAndroid Build Coastguard Worker 	int size;		/* data size (1/2/4) */
29*cd60bc56SAndroid Build Coastguard Worker 	int verbose;		/* verbose output */
30*cd60bc56SAndroid Build Coastguard Worker 	int auto_path;		/* automatically create all path components */
31*cd60bc56SAndroid Build Coastguard Worker };
32*cd60bc56SAndroid Build Coastguard Worker 
33*cd60bc56SAndroid Build Coastguard Worker 
34*cd60bc56SAndroid Build Coastguard Worker /**
35*cd60bc56SAndroid Build Coastguard Worker  * Report an error with a particular node.
36*cd60bc56SAndroid Build Coastguard Worker  *
37*cd60bc56SAndroid Build Coastguard Worker  * @param name		Node name to report error on
38*cd60bc56SAndroid Build Coastguard Worker  * @param namelen	Length of node name, or -1 to use entire string
39*cd60bc56SAndroid Build Coastguard Worker  * @param err		Error number to report (-FDT_ERR_...)
40*cd60bc56SAndroid Build Coastguard Worker  */
report_error(const char * name,int namelen,int err)41*cd60bc56SAndroid Build Coastguard Worker static void report_error(const char *name, int namelen, int err)
42*cd60bc56SAndroid Build Coastguard Worker {
43*cd60bc56SAndroid Build Coastguard Worker 	if (namelen == -1)
44*cd60bc56SAndroid Build Coastguard Worker 		namelen = strlen(name);
45*cd60bc56SAndroid Build Coastguard Worker 	fprintf(stderr, "Error at '%1.*s': %s\n", namelen, name,
46*cd60bc56SAndroid Build Coastguard Worker 		fdt_strerror(err));
47*cd60bc56SAndroid Build Coastguard Worker }
48*cd60bc56SAndroid Build Coastguard Worker 
49*cd60bc56SAndroid Build Coastguard Worker /**
50*cd60bc56SAndroid Build Coastguard Worker  * Encode a series of arguments in a property value.
51*cd60bc56SAndroid Build Coastguard Worker  *
52*cd60bc56SAndroid Build Coastguard Worker  * @param disp		Display information / options
53*cd60bc56SAndroid Build Coastguard Worker  * @param arg		List of arguments from command line
54*cd60bc56SAndroid Build Coastguard Worker  * @param arg_count	Number of arguments (may be 0)
55*cd60bc56SAndroid Build Coastguard Worker  * @param valuep	Returns buffer containing value
56*cd60bc56SAndroid Build Coastguard Worker  * @param value_len	Returns length of value encoded
57*cd60bc56SAndroid Build Coastguard Worker  */
encode_value(struct display_info * disp,char ** arg,int arg_count,char ** valuep,int * value_len)58*cd60bc56SAndroid Build Coastguard Worker static int encode_value(struct display_info *disp, char **arg, int arg_count,
59*cd60bc56SAndroid Build Coastguard Worker 			char **valuep, int *value_len)
60*cd60bc56SAndroid Build Coastguard Worker {
61*cd60bc56SAndroid Build Coastguard Worker 	char *value = NULL;	/* holding area for value */
62*cd60bc56SAndroid Build Coastguard Worker 	int value_size = 0;	/* size of holding area */
63*cd60bc56SAndroid Build Coastguard Worker 	char *ptr;		/* pointer to current value position */
64*cd60bc56SAndroid Build Coastguard Worker 	int len;		/* length of this cell/string/byte */
65*cd60bc56SAndroid Build Coastguard Worker 	int ival;
66*cd60bc56SAndroid Build Coastguard Worker 	int upto;	/* the number of bytes we have written to buf */
67*cd60bc56SAndroid Build Coastguard Worker 	char fmt[3];
68*cd60bc56SAndroid Build Coastguard Worker 
69*cd60bc56SAndroid Build Coastguard Worker 	upto = 0;
70*cd60bc56SAndroid Build Coastguard Worker 
71*cd60bc56SAndroid Build Coastguard Worker 	if (disp->verbose)
72*cd60bc56SAndroid Build Coastguard Worker 		fprintf(stderr, "Decoding value:\n");
73*cd60bc56SAndroid Build Coastguard Worker 
74*cd60bc56SAndroid Build Coastguard Worker 	fmt[0] = '%';
75*cd60bc56SAndroid Build Coastguard Worker 	fmt[1] = disp->type ? disp->type : 'd';
76*cd60bc56SAndroid Build Coastguard Worker 	fmt[2] = '\0';
77*cd60bc56SAndroid Build Coastguard Worker 	for (; arg_count > 0; arg++, arg_count--, upto += len) {
78*cd60bc56SAndroid Build Coastguard Worker 		/* assume integer unless told otherwise */
79*cd60bc56SAndroid Build Coastguard Worker 		if (disp->type == 's')
80*cd60bc56SAndroid Build Coastguard Worker 			len = strlen(*arg) + 1;
81*cd60bc56SAndroid Build Coastguard Worker 		else
82*cd60bc56SAndroid Build Coastguard Worker 			len = disp->size == -1 ? 4 : disp->size;
83*cd60bc56SAndroid Build Coastguard Worker 
84*cd60bc56SAndroid Build Coastguard Worker 		/* enlarge our value buffer by a suitable margin if needed */
85*cd60bc56SAndroid Build Coastguard Worker 		if (upto + len > value_size) {
86*cd60bc56SAndroid Build Coastguard Worker 			value_size = (upto + len) + 500;
87*cd60bc56SAndroid Build Coastguard Worker 			value = xrealloc(value, value_size);
88*cd60bc56SAndroid Build Coastguard Worker 		}
89*cd60bc56SAndroid Build Coastguard Worker 
90*cd60bc56SAndroid Build Coastguard Worker 		ptr = value + upto;
91*cd60bc56SAndroid Build Coastguard Worker 		if (disp->type == 's') {
92*cd60bc56SAndroid Build Coastguard Worker 			memcpy(ptr, *arg, len);
93*cd60bc56SAndroid Build Coastguard Worker 			if (disp->verbose)
94*cd60bc56SAndroid Build Coastguard Worker 				fprintf(stderr, "\tstring: '%s'\n", ptr);
95*cd60bc56SAndroid Build Coastguard Worker 		} else {
96*cd60bc56SAndroid Build Coastguard Worker 			fdt32_t *iptr = (fdt32_t *)ptr;
97*cd60bc56SAndroid Build Coastguard Worker 			sscanf(*arg, fmt, &ival);
98*cd60bc56SAndroid Build Coastguard Worker 			if (len == 4)
99*cd60bc56SAndroid Build Coastguard Worker 				*iptr = cpu_to_fdt32(ival);
100*cd60bc56SAndroid Build Coastguard Worker 			else
101*cd60bc56SAndroid Build Coastguard Worker 				*ptr = (uint8_t)ival;
102*cd60bc56SAndroid Build Coastguard Worker 			if (disp->verbose) {
103*cd60bc56SAndroid Build Coastguard Worker 				fprintf(stderr, "\t%s: %d\n",
104*cd60bc56SAndroid Build Coastguard Worker 					disp->size == 1 ? "byte" :
105*cd60bc56SAndroid Build Coastguard Worker 					disp->size == 2 ? "short" : "int",
106*cd60bc56SAndroid Build Coastguard Worker 					ival);
107*cd60bc56SAndroid Build Coastguard Worker 			}
108*cd60bc56SAndroid Build Coastguard Worker 		}
109*cd60bc56SAndroid Build Coastguard Worker 	}
110*cd60bc56SAndroid Build Coastguard Worker 	*value_len = upto;
111*cd60bc56SAndroid Build Coastguard Worker 	*valuep = value;
112*cd60bc56SAndroid Build Coastguard Worker 	if (disp->verbose)
113*cd60bc56SAndroid Build Coastguard Worker 		fprintf(stderr, "Value size %d\n", upto);
114*cd60bc56SAndroid Build Coastguard Worker 	return 0;
115*cd60bc56SAndroid Build Coastguard Worker }
116*cd60bc56SAndroid Build Coastguard Worker 
117*cd60bc56SAndroid Build Coastguard Worker #define ALIGN(x)		(((x) + (FDT_TAGSIZE) - 1) & ~((FDT_TAGSIZE) - 1))
118*cd60bc56SAndroid Build Coastguard Worker 
realloc_fdt(char * fdt,int delta)119*cd60bc56SAndroid Build Coastguard Worker static char *realloc_fdt(char *fdt, int delta)
120*cd60bc56SAndroid Build Coastguard Worker {
121*cd60bc56SAndroid Build Coastguard Worker 	int new_sz = fdt_totalsize(fdt) + delta;
122*cd60bc56SAndroid Build Coastguard Worker 	fdt = xrealloc(fdt, new_sz);
123*cd60bc56SAndroid Build Coastguard Worker 	fdt_open_into(fdt, fdt, new_sz);
124*cd60bc56SAndroid Build Coastguard Worker 	return fdt;
125*cd60bc56SAndroid Build Coastguard Worker }
126*cd60bc56SAndroid Build Coastguard Worker 
realloc_node(char * fdt,const char * name)127*cd60bc56SAndroid Build Coastguard Worker static char *realloc_node(char *fdt, const char *name)
128*cd60bc56SAndroid Build Coastguard Worker {
129*cd60bc56SAndroid Build Coastguard Worker 	int delta;
130*cd60bc56SAndroid Build Coastguard Worker 	/* FDT_BEGIN_NODE, node name in off_struct and FDT_END_NODE */
131*cd60bc56SAndroid Build Coastguard Worker 	delta = sizeof(struct fdt_node_header) + ALIGN(strlen(name) + 1)
132*cd60bc56SAndroid Build Coastguard Worker 			+ FDT_TAGSIZE;
133*cd60bc56SAndroid Build Coastguard Worker 	return realloc_fdt(fdt, delta);
134*cd60bc56SAndroid Build Coastguard Worker }
135*cd60bc56SAndroid Build Coastguard Worker 
realloc_property(char * fdt,int nodeoffset,const char * name,int newlen)136*cd60bc56SAndroid Build Coastguard Worker static char *realloc_property(char *fdt, int nodeoffset,
137*cd60bc56SAndroid Build Coastguard Worker 		const char *name, int newlen)
138*cd60bc56SAndroid Build Coastguard Worker {
139*cd60bc56SAndroid Build Coastguard Worker 	int delta = 0;
140*cd60bc56SAndroid Build Coastguard Worker 	int oldlen = 0;
141*cd60bc56SAndroid Build Coastguard Worker 
142*cd60bc56SAndroid Build Coastguard Worker 	if (!fdt_get_property(fdt, nodeoffset, name, &oldlen))
143*cd60bc56SAndroid Build Coastguard Worker 		/* strings + property header */
144*cd60bc56SAndroid Build Coastguard Worker 		delta = sizeof(struct fdt_property) + strlen(name) + 1;
145*cd60bc56SAndroid Build Coastguard Worker 
146*cd60bc56SAndroid Build Coastguard Worker 	if (newlen > oldlen)
147*cd60bc56SAndroid Build Coastguard Worker 		/* actual value in off_struct */
148*cd60bc56SAndroid Build Coastguard Worker 		delta += ALIGN(newlen) - ALIGN(oldlen);
149*cd60bc56SAndroid Build Coastguard Worker 
150*cd60bc56SAndroid Build Coastguard Worker 	return realloc_fdt(fdt, delta);
151*cd60bc56SAndroid Build Coastguard Worker }
152*cd60bc56SAndroid Build Coastguard Worker 
store_key_value(char ** blob,const char * node_name,const char * property,const char * buf,int len)153*cd60bc56SAndroid Build Coastguard Worker static int store_key_value(char **blob, const char *node_name,
154*cd60bc56SAndroid Build Coastguard Worker 		const char *property, const char *buf, int len)
155*cd60bc56SAndroid Build Coastguard Worker {
156*cd60bc56SAndroid Build Coastguard Worker 	int node;
157*cd60bc56SAndroid Build Coastguard Worker 	int err;
158*cd60bc56SAndroid Build Coastguard Worker 
159*cd60bc56SAndroid Build Coastguard Worker 	node = fdt_path_offset(*blob, node_name);
160*cd60bc56SAndroid Build Coastguard Worker 	if (node < 0) {
161*cd60bc56SAndroid Build Coastguard Worker 		report_error(node_name, -1, node);
162*cd60bc56SAndroid Build Coastguard Worker 		return -1;
163*cd60bc56SAndroid Build Coastguard Worker 	}
164*cd60bc56SAndroid Build Coastguard Worker 
165*cd60bc56SAndroid Build Coastguard Worker 	err = fdt_setprop(*blob, node, property, buf, len);
166*cd60bc56SAndroid Build Coastguard Worker 	if (err == -FDT_ERR_NOSPACE) {
167*cd60bc56SAndroid Build Coastguard Worker 		*blob = realloc_property(*blob, node, property, len);
168*cd60bc56SAndroid Build Coastguard Worker 		err = fdt_setprop(*blob, node, property, buf, len);
169*cd60bc56SAndroid Build Coastguard Worker 	}
170*cd60bc56SAndroid Build Coastguard Worker 	if (err) {
171*cd60bc56SAndroid Build Coastguard Worker 		report_error(property, -1, err);
172*cd60bc56SAndroid Build Coastguard Worker 		return -1;
173*cd60bc56SAndroid Build Coastguard Worker 	}
174*cd60bc56SAndroid Build Coastguard Worker 	return 0;
175*cd60bc56SAndroid Build Coastguard Worker }
176*cd60bc56SAndroid Build Coastguard Worker 
177*cd60bc56SAndroid Build Coastguard Worker /**
178*cd60bc56SAndroid Build Coastguard Worker  * Create paths as needed for all components of a path
179*cd60bc56SAndroid Build Coastguard Worker  *
180*cd60bc56SAndroid Build Coastguard Worker  * Any components of the path that do not exist are created. Errors are
181*cd60bc56SAndroid Build Coastguard Worker  * reported.
182*cd60bc56SAndroid Build Coastguard Worker  *
183*cd60bc56SAndroid Build Coastguard Worker  * @param blob		FDT blob to write into
184*cd60bc56SAndroid Build Coastguard Worker  * @param in_path	Path to process
185*cd60bc56SAndroid Build Coastguard Worker  * @return 0 if ok, -1 on error
186*cd60bc56SAndroid Build Coastguard Worker  */
create_paths(char ** blob,const char * in_path)187*cd60bc56SAndroid Build Coastguard Worker static int create_paths(char **blob, const char *in_path)
188*cd60bc56SAndroid Build Coastguard Worker {
189*cd60bc56SAndroid Build Coastguard Worker 	const char *path = in_path;
190*cd60bc56SAndroid Build Coastguard Worker 	const char *sep;
191*cd60bc56SAndroid Build Coastguard Worker 	int node, offset = 0;
192*cd60bc56SAndroid Build Coastguard Worker 
193*cd60bc56SAndroid Build Coastguard Worker 	/* skip leading '/' */
194*cd60bc56SAndroid Build Coastguard Worker 	while (*path == '/')
195*cd60bc56SAndroid Build Coastguard Worker 		path++;
196*cd60bc56SAndroid Build Coastguard Worker 
197*cd60bc56SAndroid Build Coastguard Worker 	for (sep = path; *sep; path = sep + 1, offset = node) {
198*cd60bc56SAndroid Build Coastguard Worker 		/* equivalent to strchrnul(), but it requires _GNU_SOURCE */
199*cd60bc56SAndroid Build Coastguard Worker 		sep = strchr(path, '/');
200*cd60bc56SAndroid Build Coastguard Worker 		if (!sep)
201*cd60bc56SAndroid Build Coastguard Worker 			sep = path + strlen(path);
202*cd60bc56SAndroid Build Coastguard Worker 
203*cd60bc56SAndroid Build Coastguard Worker 		node = fdt_subnode_offset_namelen(*blob, offset, path,
204*cd60bc56SAndroid Build Coastguard Worker 				sep - path);
205*cd60bc56SAndroid Build Coastguard Worker 		if (node == -FDT_ERR_NOTFOUND) {
206*cd60bc56SAndroid Build Coastguard Worker 			*blob = realloc_node(*blob, path);
207*cd60bc56SAndroid Build Coastguard Worker 			node = fdt_add_subnode_namelen(*blob, offset, path,
208*cd60bc56SAndroid Build Coastguard Worker 						       sep - path);
209*cd60bc56SAndroid Build Coastguard Worker 		}
210*cd60bc56SAndroid Build Coastguard Worker 		if (node < 0) {
211*cd60bc56SAndroid Build Coastguard Worker 			report_error(path, sep - path, node);
212*cd60bc56SAndroid Build Coastguard Worker 			return -1;
213*cd60bc56SAndroid Build Coastguard Worker 		}
214*cd60bc56SAndroid Build Coastguard Worker 	}
215*cd60bc56SAndroid Build Coastguard Worker 
216*cd60bc56SAndroid Build Coastguard Worker 	return 0;
217*cd60bc56SAndroid Build Coastguard Worker }
218*cd60bc56SAndroid Build Coastguard Worker 
219*cd60bc56SAndroid Build Coastguard Worker /**
220*cd60bc56SAndroid Build Coastguard Worker  * Create a new node in the fdt.
221*cd60bc56SAndroid Build Coastguard Worker  *
222*cd60bc56SAndroid Build Coastguard Worker  * This will overwrite the node_name string. Any error is reported.
223*cd60bc56SAndroid Build Coastguard Worker  *
224*cd60bc56SAndroid Build Coastguard Worker  * TODO: Perhaps create fdt_path_offset_namelen() so we don't need to do this.
225*cd60bc56SAndroid Build Coastguard Worker  *
226*cd60bc56SAndroid Build Coastguard Worker  * @param blob		FDT blob to write into
227*cd60bc56SAndroid Build Coastguard Worker  * @param node_name	Name of node to create
228*cd60bc56SAndroid Build Coastguard Worker  * @return new node offset if found, or -1 on failure
229*cd60bc56SAndroid Build Coastguard Worker  */
create_node(char ** blob,const char * node_name)230*cd60bc56SAndroid Build Coastguard Worker static int create_node(char **blob, const char *node_name)
231*cd60bc56SAndroid Build Coastguard Worker {
232*cd60bc56SAndroid Build Coastguard Worker 	int node = 0;
233*cd60bc56SAndroid Build Coastguard Worker 	char *p;
234*cd60bc56SAndroid Build Coastguard Worker 
235*cd60bc56SAndroid Build Coastguard Worker 	p = strrchr(node_name, '/');
236*cd60bc56SAndroid Build Coastguard Worker 	if (!p) {
237*cd60bc56SAndroid Build Coastguard Worker 		report_error(node_name, -1, -FDT_ERR_BADPATH);
238*cd60bc56SAndroid Build Coastguard Worker 		return -1;
239*cd60bc56SAndroid Build Coastguard Worker 	}
240*cd60bc56SAndroid Build Coastguard Worker 	*p = '\0';
241*cd60bc56SAndroid Build Coastguard Worker 
242*cd60bc56SAndroid Build Coastguard Worker 	*blob = realloc_node(*blob, p + 1);
243*cd60bc56SAndroid Build Coastguard Worker 
244*cd60bc56SAndroid Build Coastguard Worker 	if (p > node_name) {
245*cd60bc56SAndroid Build Coastguard Worker 		node = fdt_path_offset(*blob, node_name);
246*cd60bc56SAndroid Build Coastguard Worker 		if (node < 0) {
247*cd60bc56SAndroid Build Coastguard Worker 			report_error(node_name, -1, node);
248*cd60bc56SAndroid Build Coastguard Worker 			return -1;
249*cd60bc56SAndroid Build Coastguard Worker 		}
250*cd60bc56SAndroid Build Coastguard Worker 	}
251*cd60bc56SAndroid Build Coastguard Worker 
252*cd60bc56SAndroid Build Coastguard Worker 	node = fdt_add_subnode(*blob, node, p + 1);
253*cd60bc56SAndroid Build Coastguard Worker 	if (node < 0) {
254*cd60bc56SAndroid Build Coastguard Worker 		report_error(p + 1, -1, node);
255*cd60bc56SAndroid Build Coastguard Worker 		return -1;
256*cd60bc56SAndroid Build Coastguard Worker 	}
257*cd60bc56SAndroid Build Coastguard Worker 
258*cd60bc56SAndroid Build Coastguard Worker 	return 0;
259*cd60bc56SAndroid Build Coastguard Worker }
260*cd60bc56SAndroid Build Coastguard Worker 
261*cd60bc56SAndroid Build Coastguard Worker /**
262*cd60bc56SAndroid Build Coastguard Worker  * Delete a property of a node in the fdt.
263*cd60bc56SAndroid Build Coastguard Worker  *
264*cd60bc56SAndroid Build Coastguard Worker  * @param blob		FDT blob to write into
265*cd60bc56SAndroid Build Coastguard Worker  * @param node_name	Path to node containing the property to delete
266*cd60bc56SAndroid Build Coastguard Worker  * @param prop_name	Name of property to delete
267*cd60bc56SAndroid Build Coastguard Worker  * @return 0 on success, or -1 on failure
268*cd60bc56SAndroid Build Coastguard Worker  */
delete_prop(char * blob,const char * node_name,const char * prop_name)269*cd60bc56SAndroid Build Coastguard Worker static int delete_prop(char *blob, const char *node_name, const char *prop_name)
270*cd60bc56SAndroid Build Coastguard Worker {
271*cd60bc56SAndroid Build Coastguard Worker 	int node = 0;
272*cd60bc56SAndroid Build Coastguard Worker 
273*cd60bc56SAndroid Build Coastguard Worker 	node = fdt_path_offset(blob, node_name);
274*cd60bc56SAndroid Build Coastguard Worker 	if (node < 0) {
275*cd60bc56SAndroid Build Coastguard Worker 		report_error(node_name, -1, node);
276*cd60bc56SAndroid Build Coastguard Worker 		return -1;
277*cd60bc56SAndroid Build Coastguard Worker 	}
278*cd60bc56SAndroid Build Coastguard Worker 
279*cd60bc56SAndroid Build Coastguard Worker 	node = fdt_delprop(blob, node, prop_name);
280*cd60bc56SAndroid Build Coastguard Worker 	if (node < 0) {
281*cd60bc56SAndroid Build Coastguard Worker 		report_error(node_name, -1, node);
282*cd60bc56SAndroid Build Coastguard Worker 		return -1;
283*cd60bc56SAndroid Build Coastguard Worker 	}
284*cd60bc56SAndroid Build Coastguard Worker 
285*cd60bc56SAndroid Build Coastguard Worker 	return 0;
286*cd60bc56SAndroid Build Coastguard Worker }
287*cd60bc56SAndroid Build Coastguard Worker 
288*cd60bc56SAndroid Build Coastguard Worker /**
289*cd60bc56SAndroid Build Coastguard Worker  * Delete a node in the fdt.
290*cd60bc56SAndroid Build Coastguard Worker  *
291*cd60bc56SAndroid Build Coastguard Worker  * @param blob		FDT blob to write into
292*cd60bc56SAndroid Build Coastguard Worker  * @param node_name	Name of node to delete
293*cd60bc56SAndroid Build Coastguard Worker  * @return 0 on success, or -1 on failure
294*cd60bc56SAndroid Build Coastguard Worker  */
delete_node(char * blob,const char * node_name)295*cd60bc56SAndroid Build Coastguard Worker static int delete_node(char *blob, const char *node_name)
296*cd60bc56SAndroid Build Coastguard Worker {
297*cd60bc56SAndroid Build Coastguard Worker 	int node = 0;
298*cd60bc56SAndroid Build Coastguard Worker 
299*cd60bc56SAndroid Build Coastguard Worker 	node = fdt_path_offset(blob, node_name);
300*cd60bc56SAndroid Build Coastguard Worker 	if (node < 0) {
301*cd60bc56SAndroid Build Coastguard Worker 		report_error(node_name, -1, node);
302*cd60bc56SAndroid Build Coastguard Worker 		return -1;
303*cd60bc56SAndroid Build Coastguard Worker 	}
304*cd60bc56SAndroid Build Coastguard Worker 
305*cd60bc56SAndroid Build Coastguard Worker 	node = fdt_del_node(blob, node);
306*cd60bc56SAndroid Build Coastguard Worker 	if (node < 0) {
307*cd60bc56SAndroid Build Coastguard Worker 		report_error(node_name, -1, node);
308*cd60bc56SAndroid Build Coastguard Worker 		return -1;
309*cd60bc56SAndroid Build Coastguard Worker 	}
310*cd60bc56SAndroid Build Coastguard Worker 
311*cd60bc56SAndroid Build Coastguard Worker 	return 0;
312*cd60bc56SAndroid Build Coastguard Worker }
313*cd60bc56SAndroid Build Coastguard Worker 
do_fdtput(struct display_info * disp,const char * filename,char ** arg,int arg_count)314*cd60bc56SAndroid Build Coastguard Worker static int do_fdtput(struct display_info *disp, const char *filename,
315*cd60bc56SAndroid Build Coastguard Worker 		    char **arg, int arg_count)
316*cd60bc56SAndroid Build Coastguard Worker {
317*cd60bc56SAndroid Build Coastguard Worker 	char *value = NULL;
318*cd60bc56SAndroid Build Coastguard Worker 	char *blob;
319*cd60bc56SAndroid Build Coastguard Worker 	char *node;
320*cd60bc56SAndroid Build Coastguard Worker 	int len, ret = 0;
321*cd60bc56SAndroid Build Coastguard Worker 
322*cd60bc56SAndroid Build Coastguard Worker 	blob = utilfdt_read(filename, NULL);
323*cd60bc56SAndroid Build Coastguard Worker 	if (!blob)
324*cd60bc56SAndroid Build Coastguard Worker 		return -1;
325*cd60bc56SAndroid Build Coastguard Worker 
326*cd60bc56SAndroid Build Coastguard Worker 	switch (disp->oper) {
327*cd60bc56SAndroid Build Coastguard Worker 	case OPER_WRITE_PROP:
328*cd60bc56SAndroid Build Coastguard Worker 		/*
329*cd60bc56SAndroid Build Coastguard Worker 		 * Convert the arguments into a single binary value, then
330*cd60bc56SAndroid Build Coastguard Worker 		 * store them into the property.
331*cd60bc56SAndroid Build Coastguard Worker 		 */
332*cd60bc56SAndroid Build Coastguard Worker 		assert(arg_count >= 2);
333*cd60bc56SAndroid Build Coastguard Worker 		if (disp->auto_path && create_paths(&blob, *arg))
334*cd60bc56SAndroid Build Coastguard Worker 			return -1;
335*cd60bc56SAndroid Build Coastguard Worker 		if (encode_value(disp, arg + 2, arg_count - 2, &value, &len) ||
336*cd60bc56SAndroid Build Coastguard Worker 			store_key_value(&blob, *arg, arg[1], value, len))
337*cd60bc56SAndroid Build Coastguard Worker 			ret = -1;
338*cd60bc56SAndroid Build Coastguard Worker 		break;
339*cd60bc56SAndroid Build Coastguard Worker 	case OPER_CREATE_NODE:
340*cd60bc56SAndroid Build Coastguard Worker 		for (; ret >= 0 && arg_count--; arg++) {
341*cd60bc56SAndroid Build Coastguard Worker 			if (disp->auto_path)
342*cd60bc56SAndroid Build Coastguard Worker 				ret = create_paths(&blob, *arg);
343*cd60bc56SAndroid Build Coastguard Worker 			else
344*cd60bc56SAndroid Build Coastguard Worker 				ret = create_node(&blob, *arg);
345*cd60bc56SAndroid Build Coastguard Worker 		}
346*cd60bc56SAndroid Build Coastguard Worker 		break;
347*cd60bc56SAndroid Build Coastguard Worker 	case OPER_REMOVE_NODE:
348*cd60bc56SAndroid Build Coastguard Worker 		for (; ret >= 0 && arg_count--; arg++)
349*cd60bc56SAndroid Build Coastguard Worker 			ret = delete_node(blob, *arg);
350*cd60bc56SAndroid Build Coastguard Worker 		break;
351*cd60bc56SAndroid Build Coastguard Worker 	case OPER_DELETE_PROP:
352*cd60bc56SAndroid Build Coastguard Worker 		node = *arg;
353*cd60bc56SAndroid Build Coastguard Worker 		for (arg++; ret >= 0 && arg_count-- > 1; arg++)
354*cd60bc56SAndroid Build Coastguard Worker 			ret = delete_prop(blob, node, *arg);
355*cd60bc56SAndroid Build Coastguard Worker 		break;
356*cd60bc56SAndroid Build Coastguard Worker 	}
357*cd60bc56SAndroid Build Coastguard Worker 	if (ret >= 0) {
358*cd60bc56SAndroid Build Coastguard Worker 		fdt_pack(blob);
359*cd60bc56SAndroid Build Coastguard Worker 		ret = utilfdt_write(filename, blob);
360*cd60bc56SAndroid Build Coastguard Worker 	}
361*cd60bc56SAndroid Build Coastguard Worker 
362*cd60bc56SAndroid Build Coastguard Worker 	free(blob);
363*cd60bc56SAndroid Build Coastguard Worker 
364*cd60bc56SAndroid Build Coastguard Worker 	if (value) {
365*cd60bc56SAndroid Build Coastguard Worker 		free(value);
366*cd60bc56SAndroid Build Coastguard Worker 	}
367*cd60bc56SAndroid Build Coastguard Worker 
368*cd60bc56SAndroid Build Coastguard Worker 	return ret;
369*cd60bc56SAndroid Build Coastguard Worker }
370*cd60bc56SAndroid Build Coastguard Worker 
371*cd60bc56SAndroid Build Coastguard Worker /* Usage related data. */
372*cd60bc56SAndroid Build Coastguard Worker static const char usage_synopsis[] =
373*cd60bc56SAndroid Build Coastguard Worker 	"write a property value to a device tree\n"
374*cd60bc56SAndroid Build Coastguard Worker 	"	fdtput <options> <dt file> <node> <property> [<value>...]\n"
375*cd60bc56SAndroid Build Coastguard Worker 	"	fdtput -c <options> <dt file> [<node>...]\n"
376*cd60bc56SAndroid Build Coastguard Worker 	"	fdtput -r <options> <dt file> [<node>...]\n"
377*cd60bc56SAndroid Build Coastguard Worker 	"	fdtput -d <options> <dt file> <node> [<property>...]\n"
378*cd60bc56SAndroid Build Coastguard Worker 	"\n"
379*cd60bc56SAndroid Build Coastguard Worker 	"The command line arguments are joined together into a single value.\n"
380*cd60bc56SAndroid Build Coastguard Worker 	USAGE_TYPE_MSG;
381*cd60bc56SAndroid Build Coastguard Worker static const char usage_short_opts[] = "crdpt:v" USAGE_COMMON_SHORT_OPTS;
382*cd60bc56SAndroid Build Coastguard Worker static struct option const usage_long_opts[] = {
383*cd60bc56SAndroid Build Coastguard Worker 	{"create",           no_argument, NULL, 'c'},
384*cd60bc56SAndroid Build Coastguard Worker 	{"remove",	     no_argument, NULL, 'r'},
385*cd60bc56SAndroid Build Coastguard Worker 	{"delete",	     no_argument, NULL, 'd'},
386*cd60bc56SAndroid Build Coastguard Worker 	{"auto-path",        no_argument, NULL, 'p'},
387*cd60bc56SAndroid Build Coastguard Worker 	{"type",              a_argument, NULL, 't'},
388*cd60bc56SAndroid Build Coastguard Worker 	{"verbose",          no_argument, NULL, 'v'},
389*cd60bc56SAndroid Build Coastguard Worker 	USAGE_COMMON_LONG_OPTS,
390*cd60bc56SAndroid Build Coastguard Worker };
391*cd60bc56SAndroid Build Coastguard Worker static const char * const usage_opts_help[] = {
392*cd60bc56SAndroid Build Coastguard Worker 	"Create nodes if they don't already exist",
393*cd60bc56SAndroid Build Coastguard Worker 	"Delete nodes (and any subnodes) if they already exist",
394*cd60bc56SAndroid Build Coastguard Worker 	"Delete properties if they already exist",
395*cd60bc56SAndroid Build Coastguard Worker 	"Automatically create nodes as needed for the node path",
396*cd60bc56SAndroid Build Coastguard Worker 	"Type of data",
397*cd60bc56SAndroid Build Coastguard Worker 	"Display each value decoded from command line",
398*cd60bc56SAndroid Build Coastguard Worker 	USAGE_COMMON_OPTS_HELP
399*cd60bc56SAndroid Build Coastguard Worker };
400*cd60bc56SAndroid Build Coastguard Worker 
main(int argc,char * argv[])401*cd60bc56SAndroid Build Coastguard Worker int main(int argc, char *argv[])
402*cd60bc56SAndroid Build Coastguard Worker {
403*cd60bc56SAndroid Build Coastguard Worker 	int opt;
404*cd60bc56SAndroid Build Coastguard Worker 	struct display_info disp;
405*cd60bc56SAndroid Build Coastguard Worker 	char *filename = NULL;
406*cd60bc56SAndroid Build Coastguard Worker 
407*cd60bc56SAndroid Build Coastguard Worker 	memset(&disp, '\0', sizeof(disp));
408*cd60bc56SAndroid Build Coastguard Worker 	disp.size = -1;
409*cd60bc56SAndroid Build Coastguard Worker 	disp.oper = OPER_WRITE_PROP;
410*cd60bc56SAndroid Build Coastguard Worker 	while ((opt = util_getopt_long()) != EOF) {
411*cd60bc56SAndroid Build Coastguard Worker 		/*
412*cd60bc56SAndroid Build Coastguard Worker 		 * TODO: add options to:
413*cd60bc56SAndroid Build Coastguard Worker 		 * - rename node
414*cd60bc56SAndroid Build Coastguard Worker 		 * - pack fdt before writing
415*cd60bc56SAndroid Build Coastguard Worker 		 * - set amount of free space when writing
416*cd60bc56SAndroid Build Coastguard Worker 		 */
417*cd60bc56SAndroid Build Coastguard Worker 		switch (opt) {
418*cd60bc56SAndroid Build Coastguard Worker 		case_USAGE_COMMON_FLAGS
419*cd60bc56SAndroid Build Coastguard Worker 
420*cd60bc56SAndroid Build Coastguard Worker 		case 'c':
421*cd60bc56SAndroid Build Coastguard Worker 			disp.oper = OPER_CREATE_NODE;
422*cd60bc56SAndroid Build Coastguard Worker 			break;
423*cd60bc56SAndroid Build Coastguard Worker 		case 'r':
424*cd60bc56SAndroid Build Coastguard Worker 			disp.oper = OPER_REMOVE_NODE;
425*cd60bc56SAndroid Build Coastguard Worker 			break;
426*cd60bc56SAndroid Build Coastguard Worker 		case 'd':
427*cd60bc56SAndroid Build Coastguard Worker 			disp.oper = OPER_DELETE_PROP;
428*cd60bc56SAndroid Build Coastguard Worker 			break;
429*cd60bc56SAndroid Build Coastguard Worker 		case 'p':
430*cd60bc56SAndroid Build Coastguard Worker 			disp.auto_path = 1;
431*cd60bc56SAndroid Build Coastguard Worker 			break;
432*cd60bc56SAndroid Build Coastguard Worker 		case 't':
433*cd60bc56SAndroid Build Coastguard Worker 			if (utilfdt_decode_type(optarg, &disp.type,
434*cd60bc56SAndroid Build Coastguard Worker 					&disp.size))
435*cd60bc56SAndroid Build Coastguard Worker 				usage("Invalid type string");
436*cd60bc56SAndroid Build Coastguard Worker 			if (disp.type == 'r')
437*cd60bc56SAndroid Build Coastguard Worker 				usage("Unsupported raw data type");
438*cd60bc56SAndroid Build Coastguard Worker 			break;
439*cd60bc56SAndroid Build Coastguard Worker 
440*cd60bc56SAndroid Build Coastguard Worker 		case 'v':
441*cd60bc56SAndroid Build Coastguard Worker 			disp.verbose = 1;
442*cd60bc56SAndroid Build Coastguard Worker 			break;
443*cd60bc56SAndroid Build Coastguard Worker 		}
444*cd60bc56SAndroid Build Coastguard Worker 	}
445*cd60bc56SAndroid Build Coastguard Worker 
446*cd60bc56SAndroid Build Coastguard Worker 	if (optind < argc)
447*cd60bc56SAndroid Build Coastguard Worker 		filename = argv[optind++];
448*cd60bc56SAndroid Build Coastguard Worker 	if (!filename)
449*cd60bc56SAndroid Build Coastguard Worker 		usage("missing filename");
450*cd60bc56SAndroid Build Coastguard Worker 
451*cd60bc56SAndroid Build Coastguard Worker 	argv += optind;
452*cd60bc56SAndroid Build Coastguard Worker 	argc -= optind;
453*cd60bc56SAndroid Build Coastguard Worker 
454*cd60bc56SAndroid Build Coastguard Worker 	if (disp.oper == OPER_WRITE_PROP) {
455*cd60bc56SAndroid Build Coastguard Worker 		if (argc < 1)
456*cd60bc56SAndroid Build Coastguard Worker 			usage("missing node");
457*cd60bc56SAndroid Build Coastguard Worker 		if (argc < 2)
458*cd60bc56SAndroid Build Coastguard Worker 			usage("missing property");
459*cd60bc56SAndroid Build Coastguard Worker 	}
460*cd60bc56SAndroid Build Coastguard Worker 
461*cd60bc56SAndroid Build Coastguard Worker 	if (disp.oper == OPER_DELETE_PROP)
462*cd60bc56SAndroid Build Coastguard Worker 		if (argc < 1)
463*cd60bc56SAndroid Build Coastguard Worker 			usage("missing node");
464*cd60bc56SAndroid Build Coastguard Worker 
465*cd60bc56SAndroid Build Coastguard Worker 	if (do_fdtput(&disp, filename, argv, argc))
466*cd60bc56SAndroid Build Coastguard Worker 		return 1;
467*cd60bc56SAndroid Build Coastguard Worker 	return 0;
468*cd60bc56SAndroid Build Coastguard Worker }
469