blob: 5ed873c72ad1eb803d5ebcde83ba90ca0a6bb07a [file] [log] [blame]
David Gibsonfc14dad2005-06-08 17:18:34 +10001/*
2 * (C) Copyright David Gibson <dwg@au1.ibm.com>, IBM Corporation. 2005.
3 *
David Gibson63dc9c72007-09-18 11:44:04 +10004 *
David Gibsonfc14dad2005-06-08 17:18:34 +10005 * This program is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU General Public License as
7 * published by the Free Software Foundation; either version 2 of the
8 * License, or (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * General Public License for more details.
David Gibson63dc9c72007-09-18 11:44:04 +100014 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
18 * USA
David Gibsonfc14dad2005-06-08 17:18:34 +100019 */
20
Andre Przywara5e78dff2015-07-01 00:31:27 +010021#include <sys/stat.h>
22
David Gibsonfc14dad2005-06-08 17:18:34 +100023#include "dtc.h"
Jon Loeligere45e6fd2007-03-23 15:18:41 -050024#include "srcpos.h"
David Gibsonfc14dad2005-06-08 17:18:34 +100025
Jerry Van Baren4384b232007-04-04 22:04:33 -040026/*
27 * Command line options
28 */
29int quiet; /* Level of quietness */
30int reservenum; /* Number of memory reservation slots */
31int minsize; /* Minimum blob size */
Kumar Gala2b7dc8d2007-11-28 10:21:12 -060032int padsize; /* Additional padding to blob */
Tim Wang874f4052016-07-18 15:56:53 +080033int alignsize; /* Additional padding to blob accroding to the alignsize */
Rob Herring0016f8c2017-07-12 17:20:30 -050034int phandle_format = PHANDLE_EPAPR; /* Use linux,phandle or phandle properties */
Pantelis Antoniou20f29d82016-12-07 14:48:18 +020035int generate_symbols; /* enable symbols & fixup support */
36int generate_fixups; /* suppress generation of fixups on symbol support */
37int auto_label_aliases; /* auto generate labels -> aliases */
Jerry Van Baren4384b232007-04-04 22:04:33 -040038
Tim Wang874f4052016-07-18 15:56:53 +080039static int is_power_of_2(int x)
40{
41 return (x > 0) && ((x & (x - 1)) == 0);
42}
43
David Gibsonb2de5182008-02-29 16:51:28 +110044static void fill_fullpaths(struct node *tree, const char *prefix)
David Gibsonfc14dad2005-06-08 17:18:34 +100045{
46 struct node *child;
David Gibson92cb9a22007-12-04 14:26:15 +110047 const char *unit;
David Gibsonfc14dad2005-06-08 17:18:34 +100048
49 tree->fullpath = join_path(prefix, tree->name);
50
51 unit = strchr(tree->name, '@');
52 if (unit)
53 tree->basenamelen = unit - tree->name;
54 else
55 tree->basenamelen = strlen(tree->name);
56
57 for_each_child(tree, child)
58 fill_fullpaths(child, tree->fullpath);
59}
60
Mike Frysinger03449b82013-05-24 18:02:35 +100061/* Usage related data. */
Wang Long1e5ddb12014-09-11 15:16:37 +080062#define FDT_VERSION(version) _FDT_VERSION(version)
63#define _FDT_VERSION(version) #version
Mike Frysinger03449b82013-05-24 18:02:35 +100064static const char usage_synopsis[] = "dtc [options] <input file>";
Pantelis Antoniou20f29d82016-12-07 14:48:18 +020065static const char usage_short_opts[] = "qI:O:o:V:d:R:S:p:a:fb:i:H:sW:E:@Ahv";
Mike Frysinger03449b82013-05-24 18:02:35 +100066static struct option const usage_long_opts[] = {
67 {"quiet", no_argument, NULL, 'q'},
68 {"in-format", a_argument, NULL, 'I'},
69 {"out", a_argument, NULL, 'o'},
70 {"out-format", a_argument, NULL, 'O'},
71 {"out-version", a_argument, NULL, 'V'},
72 {"out-dependency", a_argument, NULL, 'd'},
73 {"reserve", a_argument, NULL, 'R'},
74 {"space", a_argument, NULL, 'S'},
75 {"pad", a_argument, NULL, 'p'},
Tim Wang874f4052016-07-18 15:56:53 +080076 {"align", a_argument, NULL, 'a'},
Mike Frysinger03449b82013-05-24 18:02:35 +100077 {"boot-cpu", a_argument, NULL, 'b'},
78 {"force", no_argument, NULL, 'f'},
79 {"include", a_argument, NULL, 'i'},
80 {"sort", no_argument, NULL, 's'},
81 {"phandle", a_argument, NULL, 'H'},
82 {"warning", a_argument, NULL, 'W'},
83 {"error", a_argument, NULL, 'E'},
Pantelis Antoniou20f29d82016-12-07 14:48:18 +020084 {"symbols", no_argument, NULL, '@'},
85 {"auto-alias", no_argument, NULL, 'A'},
Mike Frysinger03449b82013-05-24 18:02:35 +100086 {"help", no_argument, NULL, 'h'},
87 {"version", no_argument, NULL, 'v'},
88 {NULL, no_argument, NULL, 0x0},
89};
90static const char * const usage_opts_help[] = {
91 "\n\tQuiet: -q suppress warnings, -qq errors, -qqq all",
92 "\n\tInput formats are:\n"
93 "\t\tdts - device tree source text\n"
94 "\t\tdtb - device tree blob\n"
95 "\t\tfs - /proc/device-tree style directory",
96 "\n\tOutput file",
97 "\n\tOutput formats are:\n"
98 "\t\tdts - device tree source text\n"
99 "\t\tdtb - device tree blob\n"
100 "\t\tasm - assembler source",
Wang Long1e5ddb12014-09-11 15:16:37 +0800101 "\n\tBlob version to produce, defaults to "FDT_VERSION(DEFAULT_FDT_VERSION)" (for dtb and asm output)",
Mike Frysinger03449b82013-05-24 18:02:35 +1000102 "\n\tOutput dependency file",
Wang Long1e5ddb12014-09-11 15:16:37 +0800103 "\n\tMake space for <number> reserve map entries (for dtb and asm output)",
Mike Frysinger03449b82013-05-24 18:02:35 +1000104 "\n\tMake the blob at least <bytes> long (extra space)",
105 "\n\tAdd padding to the blob of <bytes> long (extra space)",
Tim Wang874f4052016-07-18 15:56:53 +0800106 "\n\tMake the blob align to the <bytes> (extra space)",
Mike Frysinger03449b82013-05-24 18:02:35 +1000107 "\n\tSet the physical boot cpu",
108 "\n\tTry to produce output even if the input tree has errors",
109 "\n\tAdd a path to search for include files",
110 "\n\tSort nodes and properties before outputting (useful for comparing trees)",
111 "\n\tValid phandle formats are:\n"
112 "\t\tlegacy - \"linux,phandle\" properties only\n"
113 "\t\tepapr - \"phandle\" properties only\n"
114 "\t\tboth - Both \"linux,phandle\" and \"phandle\" properties",
115 "\n\tEnable/disable warnings (prefix with \"no-\")",
116 "\n\tEnable/disable errors (prefix with \"no-\")",
Pantelis Antoniou20f29d82016-12-07 14:48:18 +0200117 "\n\tEnable generation of symbols",
118 "\n\tEnable auto-alias of labels",
Mike Frysinger03449b82013-05-24 18:02:35 +1000119 "\n\tPrint this help and exit",
120 "\n\tPrint version and exit",
121 NULL,
122};
David Gibsonfc14dad2005-06-08 17:18:34 +1000123
Andre Przywara5e78dff2015-07-01 00:31:27 +0100124static const char *guess_type_by_name(const char *fname, const char *fallback)
125{
126 const char *s;
127
128 s = strrchr(fname, '.');
129 if (s == NULL)
130 return fallback;
131 if (!strcasecmp(s, ".dts"))
132 return "dts";
133 if (!strcasecmp(s, ".dtb"))
134 return "dtb";
135 return fallback;
136}
137
138static const char *guess_input_format(const char *fname, const char *fallback)
139{
140 struct stat statbuf;
David Gibsonbad5b282017-03-06 12:08:53 +1100141 fdt32_t magic;
Andre Przywara5e78dff2015-07-01 00:31:27 +0100142 FILE *f;
143
144 if (stat(fname, &statbuf) != 0)
145 return fallback;
146
147 if (S_ISDIR(statbuf.st_mode))
148 return "fs";
149
150 if (!S_ISREG(statbuf.st_mode))
151 return fallback;
152
153 f = fopen(fname, "r");
154 if (f == NULL)
155 return fallback;
156 if (fread(&magic, 4, 1, f) != 1) {
157 fclose(f);
158 return fallback;
159 }
160 fclose(f);
161
David Gibsonbad5b282017-03-06 12:08:53 +1100162 if (fdt32_to_cpu(magic) == FDT_MAGIC)
Andre Przywara5e78dff2015-07-01 00:31:27 +0100163 return "dtb";
164
165 return guess_type_by_name(fname, fallback);
166}
167
David Gibsonfc14dad2005-06-08 17:18:34 +1000168int main(int argc, char *argv[])
169{
David Gibson00fbb862016-05-31 11:58:42 +1000170 struct dt_info *dti;
Andre Przywara5e78dff2015-07-01 00:31:27 +0100171 const char *inform = NULL;
Andre Przywaraf6dbc6c2015-07-01 00:31:28 +0100172 const char *outform = NULL;
David Gibson92cb9a22007-12-04 14:26:15 +1100173 const char *outname = "-";
Stephen Warren69df9f02012-01-12 11:31:00 -0700174 const char *depname = NULL;
David Gibson17625372013-10-28 21:06:53 +1100175 bool force = false, sort = false;
David Gibson92cb9a22007-12-04 14:26:15 +1100176 const char *arg;
David Gibsonfc14dad2005-06-08 17:18:34 +1000177 int opt;
David Gibsonfc14dad2005-06-08 17:18:34 +1000178 FILE *outf = NULL;
David Gibsonfb7c7ac2007-09-26 13:11:05 +1000179 int outversion = DEFAULT_FDT_VERSION;
David Gibson548767f2008-05-16 13:22:57 +1000180 long long cmdline_boot_cpuid = -1;
David Gibsonfc14dad2005-06-08 17:18:34 +1000181
Jerry Van Baren4384b232007-04-04 22:04:33 -0400182 quiet = 0;
183 reservenum = 0;
184 minsize = 0;
Kumar Gala2b7dc8d2007-11-28 10:21:12 -0600185 padsize = 0;
Tim Wang874f4052016-07-18 15:56:53 +0800186 alignsize = 0;
Jerry Van Barencd1da872007-03-18 16:49:24 -0400187
Mike Frysinger03449b82013-05-24 18:02:35 +1000188 while ((opt = util_getopt_long()) != EOF) {
David Gibsonfc14dad2005-06-08 17:18:34 +1000189 switch (opt) {
190 case 'I':
191 inform = optarg;
192 break;
193 case 'O':
194 outform = optarg;
195 break;
196 case 'o':
197 outname = optarg;
198 break;
199 case 'V':
200 outversion = strtol(optarg, NULL, 0);
201 break;
Stephen Warren69df9f02012-01-12 11:31:00 -0700202 case 'd':
203 depname = optarg;
204 break;
David Gibsonfc14dad2005-06-08 17:18:34 +1000205 case 'R':
206 reservenum = strtol(optarg, NULL, 0);
207 break;
Jerry Van Baren4384b232007-04-04 22:04:33 -0400208 case 'S':
209 minsize = strtol(optarg, NULL, 0);
210 break;
Kumar Gala2b7dc8d2007-11-28 10:21:12 -0600211 case 'p':
212 padsize = strtol(optarg, NULL, 0);
213 break;
Tim Wang874f4052016-07-18 15:56:53 +0800214 case 'a':
215 alignsize = strtol(optarg, NULL, 0);
216 if (!is_power_of_2(alignsize))
217 die("Invalid argument \"%d\" to -a option\n",
David Gibsonf72508e2017-02-27 10:38:38 +1100218 alignsize);
Tim Wang874f4052016-07-18 15:56:53 +0800219 break;
David Gibsonfc14dad2005-06-08 17:18:34 +1000220 case 'f':
David Gibson17625372013-10-28 21:06:53 +1100221 force = true;
David Gibsonfc14dad2005-06-08 17:18:34 +1000222 break;
Jerry Van Barencd1da872007-03-18 16:49:24 -0400223 case 'q':
224 quiet++;
225 break;
Michael Neuling38e8f8f2006-05-31 08:31:51 +1000226 case 'b':
David Gibson548767f2008-05-16 13:22:57 +1000227 cmdline_boot_cpuid = strtoll(optarg, NULL, 0);
Michael Neuling38e8f8f2006-05-31 08:31:51 +1000228 break;
Simon Glassde6b7622012-03-14 20:04:13 -0700229 case 'i':
230 srcfile_add_search_path(optarg);
231 break;
Jon Loeligera657ce82007-07-07 13:52:25 -0500232 case 'v':
Mike Frysinger31be4ce2013-04-10 14:29:09 -0400233 util_version();
David Gibsond75b33a2009-11-26 15:37:13 +1100234 case 'H':
235 if (streq(optarg, "legacy"))
236 phandle_format = PHANDLE_LEGACY;
237 else if (streq(optarg, "epapr"))
238 phandle_format = PHANDLE_EPAPR;
239 else if (streq(optarg, "both"))
240 phandle_format = PHANDLE_BOTH;
241 else
242 die("Invalid argument \"%s\" to -H option\n",
243 optarg);
244 break;
245
David Gibson37c0b6a2010-11-10 09:51:09 +1100246 case 's':
David Gibson17625372013-10-28 21:06:53 +1100247 sort = true;
David Gibson37c0b6a2010-11-10 09:51:09 +1100248 break;
249
David Gibsond5399192012-07-08 23:25:22 +1000250 case 'W':
251 parse_checks_option(true, false, optarg);
252 break;
253
254 case 'E':
255 parse_checks_option(false, true, optarg);
256 break;
257
Pantelis Antoniou20f29d82016-12-07 14:48:18 +0200258 case '@':
259 generate_symbols = 1;
260 break;
261 case 'A':
262 auto_label_aliases = 1;
263 break;
264
Jerry Van Barencd1da872007-03-18 16:49:24 -0400265 case 'h':
Mike Frysingerb9e80652013-05-24 18:04:43 +1000266 usage(NULL);
David Gibsonfc14dad2005-06-08 17:18:34 +1000267 default:
Mike Frysingerb9e80652013-05-24 18:04:43 +1000268 usage("unknown option");
David Gibsonfc14dad2005-06-08 17:18:34 +1000269 }
270 }
271
272 if (argc > (optind+1))
Mike Frysingerb9e80652013-05-24 18:04:43 +1000273 usage("missing files");
David Gibsonfc14dad2005-06-08 17:18:34 +1000274 else if (argc < (optind+1))
275 arg = "-";
276 else
277 arg = argv[optind];
278
Kumar Gala2b7dc8d2007-11-28 10:21:12 -0600279 /* minsize and padsize are mutually exclusive */
David Gibsonc2601542008-05-16 13:22:35 +1000280 if (minsize && padsize)
Kumar Gala2b7dc8d2007-11-28 10:21:12 -0600281 die("Can't set both -p and -S\n");
Kumar Gala2b7dc8d2007-11-28 10:21:12 -0600282
Stephen Warren69df9f02012-01-12 11:31:00 -0700283 if (depname) {
284 depfile = fopen(depname, "w");
285 if (!depfile)
286 die("Couldn't open dependency file %s: %s\n", depname,
287 strerror(errno));
288 fprintf(depfile, "%s:", outname);
289 }
290
Andre Przywara5e78dff2015-07-01 00:31:27 +0100291 if (inform == NULL)
292 inform = guess_input_format(arg, "dts");
Andre Przywaraf6dbc6c2015-07-01 00:31:28 +0100293 if (outform == NULL) {
294 outform = guess_type_by_name(outname, NULL);
295 if (outform == NULL) {
296 if (streq(inform, "dts"))
297 outform = "dtb";
298 else
299 outform = "dts";
300 }
301 }
David Gibsona742aad2008-05-16 13:22:09 +1000302 if (streq(inform, "dts"))
David Gibson00fbb862016-05-31 11:58:42 +1000303 dti = dt_from_source(arg);
David Gibsona742aad2008-05-16 13:22:09 +1000304 else if (streq(inform, "fs"))
David Gibson00fbb862016-05-31 11:58:42 +1000305 dti = dt_from_fs(arg);
David Gibsona742aad2008-05-16 13:22:09 +1000306 else if(streq(inform, "dtb"))
David Gibson00fbb862016-05-31 11:58:42 +1000307 dti = dt_from_blob(arg);
David Gibsona742aad2008-05-16 13:22:09 +1000308 else
David Gibsonfc14dad2005-06-08 17:18:34 +1000309 die("Unknown input format \"%s\"\n", inform);
David Gibsonfc14dad2005-06-08 17:18:34 +1000310
Ian Campbellacd1b532017-02-03 08:29:39 +0000311 dti->outname = outname;
312
Stephen Warren69df9f02012-01-12 11:31:00 -0700313 if (depfile) {
314 fputc('\n', depfile);
315 fclose(depfile);
316 }
317
David Gibson548767f2008-05-16 13:22:57 +1000318 if (cmdline_boot_cpuid != -1)
David Gibson00fbb862016-05-31 11:58:42 +1000319 dti->boot_cpuid_phys = cmdline_boot_cpuid;
David Gibson548767f2008-05-16 13:22:57 +1000320
David Gibson00fbb862016-05-31 11:58:42 +1000321 fill_fullpaths(dti->dt, "");
322 process_checks(force, dti);
David Gibsonfc14dad2005-06-08 17:18:34 +1000323
Pantelis Antoniou20f29d82016-12-07 14:48:18 +0200324 /* on a plugin, generate by default */
David Gibson00fbb862016-05-31 11:58:42 +1000325 if (dti->dtsflags & DTSF_PLUGIN) {
Pantelis Antoniou20f29d82016-12-07 14:48:18 +0200326 generate_fixups = 1;
327 }
328
329 if (auto_label_aliases)
David Gibson00fbb862016-05-31 11:58:42 +1000330 generate_label_tree(dti, "aliases", false);
Pantelis Antoniou20f29d82016-12-07 14:48:18 +0200331
332 if (generate_symbols)
David Gibson00fbb862016-05-31 11:58:42 +1000333 generate_label_tree(dti, "__symbols__", true);
Pantelis Antoniou20f29d82016-12-07 14:48:18 +0200334
335 if (generate_fixups) {
David Gibson00fbb862016-05-31 11:58:42 +1000336 generate_fixups_tree(dti, "__fixups__");
337 generate_local_fixups_tree(dti, "__local_fixups__");
Pantelis Antoniou20f29d82016-12-07 14:48:18 +0200338 }
339
David Gibson37c0b6a2010-11-10 09:51:09 +1100340 if (sort)
David Gibson00fbb862016-05-31 11:58:42 +1000341 sort_tree(dti);
David Gibson548767f2008-05-16 13:22:57 +1000342
David Gibsonfc14dad2005-06-08 17:18:34 +1000343 if (streq(outname, "-")) {
344 outf = stdout;
345 } else {
Andrei Errapart83e606a2014-06-19 21:12:27 +1000346 outf = fopen(outname, "wb");
David Gibsonfc14dad2005-06-08 17:18:34 +1000347 if (! outf)
348 die("Couldn't open output file %s: %s\n",
349 outname, strerror(errno));
350 }
351
352 if (streq(outform, "dts")) {
David Gibson00fbb862016-05-31 11:58:42 +1000353 dt_to_source(outf, dti);
David Gibsonfc14dad2005-06-08 17:18:34 +1000354 } else if (streq(outform, "dtb")) {
David Gibson00fbb862016-05-31 11:58:42 +1000355 dt_to_blob(outf, dti, outversion);
David Gibsonfc14dad2005-06-08 17:18:34 +1000356 } else if (streq(outform, "asm")) {
David Gibson00fbb862016-05-31 11:58:42 +1000357 dt_to_asm(outf, dti, outversion);
David Gibsonfc14dad2005-06-08 17:18:34 +1000358 } else if (streq(outform, "null")) {
359 /* do nothing */
360 } else {
361 die("Unknown output format \"%s\"\n", outform);
362 }
David Gibson63dc9c72007-09-18 11:44:04 +1000363
David Gibsonfc14dad2005-06-08 17:18:34 +1000364 exit(0);
365}