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