blob: 4b3ce23fb4c4e6726bc0a948e9bc339d268767bf [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001# SPDX-License-Identifier: GPL-2.0+
Simon Glass2574ef62016-11-25 20:15:51 -07002# Copyright (c) 2016 Google, Inc
3# Written by Simon Glass <sjg@chromium.org>
4#
Simon Glass2574ef62016-11-25 20:15:51 -07005# Creates binary images from input files controlled by a description
6#
7
8from collections import OrderedDict
Simon Glass220ff5f2020-08-05 13:27:46 -06009import glob
Simon Glass2574ef62016-11-25 20:15:51 -070010import os
Simon Glassc1dc2f82020-08-29 11:36:14 -060011import pkg_resources
Simon Glassa820af72020-09-06 10:39:09 -060012import re
Simon Glassc1dc2f82020-08-29 11:36:14 -060013
Simon Glass2574ef62016-11-25 20:15:51 -070014import sys
Simon Glassa997ea52020-04-17 18:09:04 -060015from patman import tools
Simon Glass2574ef62016-11-25 20:15:51 -070016
Simon Glassc585dd42020-04-17 18:09:03 -060017from binman import cbfs_util
18from binman import elf
Simon Glassa997ea52020-04-17 18:09:04 -060019from patman import command
20from patman import tout
Simon Glass2574ef62016-11-25 20:15:51 -070021
22# List of images we plan to create
23# Make this global so that it can be referenced from tests
24images = OrderedDict()
25
Simon Glassa820af72020-09-06 10:39:09 -060026# Help text for each type of missing blob, dict:
27# key: Value of the entry's 'missing-msg' or entry name
28# value: Text for the help
29missing_blob_help = {}
30
Simon Glass55ab0b62021-03-18 20:25:06 +130031def _ReadImageDesc(binman_node, use_expanded):
Simon Glass2574ef62016-11-25 20:15:51 -070032 """Read the image descriptions from the /binman node
33
34 This normally produces a single Image object called 'image'. But if
35 multiple images are present, they will all be returned.
36
37 Args:
38 binman_node: Node object of the /binman node
Simon Glass55ab0b62021-03-18 20:25:06 +130039 use_expanded: True if the FDT will be updated with the entry information
Simon Glass2574ef62016-11-25 20:15:51 -070040 Returns:
41 OrderedDict of Image objects, each of which describes an image
42 """
43 images = OrderedDict()
44 if 'multiple-images' in binman_node.props:
45 for node in binman_node.subnodes:
Simon Glass55ab0b62021-03-18 20:25:06 +130046 images[node.name] = Image(node.name, node,
47 use_expanded=use_expanded)
Simon Glass2574ef62016-11-25 20:15:51 -070048 else:
Simon Glass55ab0b62021-03-18 20:25:06 +130049 images['image'] = Image('image', binman_node, use_expanded=use_expanded)
Simon Glass2574ef62016-11-25 20:15:51 -070050 return images
51
Simon Glass22c92ca2017-05-27 07:38:29 -060052def _FindBinmanNode(dtb):
Simon Glass2574ef62016-11-25 20:15:51 -070053 """Find the 'binman' node in the device tree
54
55 Args:
Simon Glass22c92ca2017-05-27 07:38:29 -060056 dtb: Fdt object to scan
Simon Glass2574ef62016-11-25 20:15:51 -070057 Returns:
58 Node object of /binman node, or None if not found
59 """
Simon Glass22c92ca2017-05-27 07:38:29 -060060 for node in dtb.GetRoot().subnodes:
Simon Glass2574ef62016-11-25 20:15:51 -070061 if node.name == 'binman':
62 return node
63 return None
64
Simon Glassa820af72020-09-06 10:39:09 -060065def _ReadMissingBlobHelp():
66 """Read the missing-blob-help file
67
68 This file containins help messages explaining what to do when external blobs
69 are missing.
70
71 Returns:
72 Dict:
73 key: Message tag (str)
74 value: Message text (str)
75 """
76
77 def _FinishTag(tag, msg, result):
78 if tag:
79 result[tag] = msg.rstrip()
80 tag = None
81 msg = ''
82 return tag, msg
83
84 my_data = pkg_resources.resource_string(__name__, 'missing-blob-help')
85 re_tag = re.compile('^([-a-z0-9]+):$')
86 result = {}
87 tag = None
88 msg = ''
89 for line in my_data.decode('utf-8').splitlines():
90 if not line.startswith('#'):
91 m_tag = re_tag.match(line)
92 if m_tag:
93 _, msg = _FinishTag(tag, msg, result)
94 tag = m_tag.group(1)
95 elif tag:
96 msg += line + '\n'
97 _FinishTag(tag, msg, result)
98 return result
99
100def _ShowBlobHelp(path, text):
101 tout.Warning('\n%s:' % path)
102 for line in text.splitlines():
103 tout.Warning(' %s' % line)
104
105def _ShowHelpForMissingBlobs(missing_list):
106 """Show help for each missing blob to help the user take action
107
108 Args:
109 missing_list: List of Entry objects to show help for
110 """
111 global missing_blob_help
112
113 if not missing_blob_help:
114 missing_blob_help = _ReadMissingBlobHelp()
115
116 for entry in missing_list:
117 tags = entry.GetHelpTags()
118
119 # Show the first match help message
120 for tag in tags:
121 if tag in missing_blob_help:
122 _ShowBlobHelp(entry._node.path, missing_blob_help[tag])
123 break
124
Simon Glass220ff5f2020-08-05 13:27:46 -0600125def GetEntryModules(include_testing=True):
126 """Get a set of entry class implementations
127
128 Returns:
129 Set of paths to entry class filenames
130 """
Simon Glassc1dc2f82020-08-29 11:36:14 -0600131 glob_list = pkg_resources.resource_listdir(__name__, 'etype')
132 glob_list = [fname for fname in glob_list if fname.endswith('.py')]
Simon Glass220ff5f2020-08-05 13:27:46 -0600133 return set([os.path.splitext(os.path.basename(item))[0]
134 for item in glob_list
135 if include_testing or '_testing' not in item])
136
Simon Glass29aa7362018-09-14 04:57:19 -0600137def WriteEntryDocs(modules, test_missing=None):
138 """Write out documentation for all entries
Simon Glass92307732018-07-06 10:27:40 -0600139
140 Args:
Simon Glass29aa7362018-09-14 04:57:19 -0600141 modules: List of Module objects to get docs for
142 test_missing: Used for testing only, to force an entry's documeentation
143 to show as missing even if it is present. Should be set to None in
144 normal use.
Simon Glass92307732018-07-06 10:27:40 -0600145 """
Simon Glassc585dd42020-04-17 18:09:03 -0600146 from binman.entry import Entry
Simon Glass969616c2018-07-17 13:25:36 -0600147 Entry.WriteDocs(modules, test_missing)
148
Simon Glassb2fd11d2019-07-08 14:25:48 -0600149
150def ListEntries(image_fname, entry_paths):
151 """List the entries in an image
152
153 This decodes the supplied image and displays a table of entries from that
154 image, preceded by a header.
155
156 Args:
157 image_fname: Image filename to process
158 entry_paths: List of wildcarded paths (e.g. ['*dtb*', 'u-boot*',
159 'section/u-boot'])
160 """
161 image = Image.FromFile(image_fname)
162
163 entries, lines, widths = image.GetListEntries(entry_paths)
164
165 num_columns = len(widths)
166 for linenum, line in enumerate(lines):
167 if linenum == 1:
168 # Print header line
169 print('-' * (sum(widths) + num_columns * 2))
170 out = ''
171 for i, item in enumerate(line):
172 width = -widths[i]
173 if item.startswith('>'):
174 width = -width
175 item = item[1:]
176 txt = '%*s ' % (width, item)
177 out += txt
178 print(out.rstrip())
179
Simon Glass4c613bf2019-07-08 14:25:50 -0600180
181def ReadEntry(image_fname, entry_path, decomp=True):
182 """Extract an entry from an image
183
184 This extracts the data from a particular entry in an image
185
186 Args:
187 image_fname: Image filename to process
188 entry_path: Path to entry to extract
189 decomp: True to return uncompressed data, if the data is compress
190 False to return the raw data
191
192 Returns:
193 data extracted from the entry
194 """
Simon Glassb9ba4e02019-08-24 07:22:44 -0600195 global Image
Simon Glass90cd6f02020-08-05 13:27:47 -0600196 from binman.image import Image
Simon Glassb9ba4e02019-08-24 07:22:44 -0600197
Simon Glass4c613bf2019-07-08 14:25:50 -0600198 image = Image.FromFile(image_fname)
199 entry = image.FindEntryPath(entry_path)
200 return entry.ReadData(decomp)
201
202
Simon Glass637958f2021-11-23 21:09:50 -0700203def ShowAltFormats(image):
204 """Show alternative formats available for entries in the image
205
206 This shows a list of formats available.
207
208 Args:
209 image (Image): Image to check
210 """
211 alt_formats = {}
212 image.CheckAltFormats(alt_formats)
213 print('%-10s %-20s %s' % ('Flag (-F)', 'Entry type', 'Description'))
214 for name, val in alt_formats.items():
215 entry, helptext = val
216 print('%-10s %-20s %s' % (name, entry.etype, helptext))
217
218
Simon Glass980a2842019-07-08 14:25:52 -0600219def ExtractEntries(image_fname, output_fname, outdir, entry_paths,
Simon Glass637958f2021-11-23 21:09:50 -0700220 decomp=True, alt_format=None):
Simon Glass980a2842019-07-08 14:25:52 -0600221 """Extract the data from one or more entries and write it to files
222
223 Args:
224 image_fname: Image filename to process
225 output_fname: Single output filename to use if extracting one file, None
226 otherwise
227 outdir: Output directory to use (for any number of files), else None
228 entry_paths: List of entry paths to extract
Simon Glassd48f94e2019-07-20 12:24:12 -0600229 decomp: True to decompress the entry data
Simon Glass980a2842019-07-08 14:25:52 -0600230
231 Returns:
232 List of EntryInfo records that were written
233 """
234 image = Image.FromFile(image_fname)
235
Simon Glass637958f2021-11-23 21:09:50 -0700236 if alt_format == 'list':
237 ShowAltFormats(image)
238 return
239
Simon Glass980a2842019-07-08 14:25:52 -0600240 # Output an entry to a single file, as a special case
241 if output_fname:
242 if not entry_paths:
Simon Glassa772d3f2019-07-20 12:24:14 -0600243 raise ValueError('Must specify an entry path to write with -f')
Simon Glass980a2842019-07-08 14:25:52 -0600244 if len(entry_paths) != 1:
Simon Glassa772d3f2019-07-20 12:24:14 -0600245 raise ValueError('Must specify exactly one entry path to write with -f')
Simon Glass980a2842019-07-08 14:25:52 -0600246 entry = image.FindEntryPath(entry_paths[0])
Simon Glass637958f2021-11-23 21:09:50 -0700247 data = entry.ReadData(decomp, alt_format)
Simon Glass980a2842019-07-08 14:25:52 -0600248 tools.WriteFile(output_fname, data)
249 tout.Notice("Wrote %#x bytes to file '%s'" % (len(data), output_fname))
250 return
251
252 # Otherwise we will output to a path given by the entry path of each entry.
253 # This means that entries will appear in subdirectories if they are part of
254 # a sub-section.
255 einfos = image.GetListEntries(entry_paths)[0]
256 tout.Notice('%d entries match and will be written' % len(einfos))
257 for einfo in einfos:
258 entry = einfo.entry
Simon Glass637958f2021-11-23 21:09:50 -0700259 data = entry.ReadData(decomp, alt_format)
Simon Glass980a2842019-07-08 14:25:52 -0600260 path = entry.GetPath()[1:]
261 fname = os.path.join(outdir, path)
262
263 # If this entry has children, create a directory for it and put its
264 # data in a file called 'root' in that directory
265 if entry.GetEntries():
Simon Glass4ef93d92021-03-18 20:24:51 +1300266 if fname and not os.path.exists(fname):
Simon Glass980a2842019-07-08 14:25:52 -0600267 os.makedirs(fname)
268 fname = os.path.join(fname, 'root')
Simon Glass08349452021-01-06 21:35:13 -0700269 tout.Notice("Write entry '%s' size %x to '%s'" %
270 (entry.GetPath(), len(data), fname))
Simon Glass980a2842019-07-08 14:25:52 -0600271 tools.WriteFile(fname, data)
272 return einfos
273
274
Simon Glass274bd0e2019-07-20 12:24:13 -0600275def BeforeReplace(image, allow_resize):
276 """Handle getting an image ready for replacing entries in it
277
278 Args:
279 image: Image to prepare
280 """
281 state.PrepareFromLoadedData(image)
282 image.LoadData()
283
284 # If repacking, drop the old offset/size values except for the original
285 # ones, so we are only left with the constraints.
286 if allow_resize:
287 image.ResetForPack()
288
289
290def ReplaceOneEntry(image, entry, data, do_compress, allow_resize):
291 """Handle replacing a single entry an an image
292
293 Args:
294 image: Image to update
295 entry: Entry to write
296 data: Data to replace with
297 do_compress: True to compress the data if needed, False if data is
298 already compressed so should be used as is
299 allow_resize: True to allow entries to change size (this does a re-pack
300 of the entries), False to raise an exception
301 """
302 if not entry.WriteData(data, do_compress):
303 if not image.allow_repack:
304 entry.Raise('Entry data size does not match, but allow-repack is not present for this image')
305 if not allow_resize:
306 entry.Raise('Entry data size does not match, but resize is disabled')
307
308
309def AfterReplace(image, allow_resize, write_map):
310 """Handle write out an image after replacing entries in it
311
312 Args:
313 image: Image to write
314 allow_resize: True to allow entries to change size (this does a re-pack
315 of the entries), False to raise an exception
316 write_map: True to write a map file
317 """
318 tout.Info('Processing image')
319 ProcessImage(image, update_fdt=True, write_map=write_map,
320 get_contents=False, allow_resize=allow_resize)
321
322
323def WriteEntryToImage(image, entry, data, do_compress=True, allow_resize=True,
324 write_map=False):
325 BeforeReplace(image, allow_resize)
326 tout.Info('Writing data to %s' % entry.GetPath())
327 ReplaceOneEntry(image, entry, data, do_compress, allow_resize)
328 AfterReplace(image, allow_resize=allow_resize, write_map=write_map)
329
330
Simon Glassd48f94e2019-07-20 12:24:12 -0600331def WriteEntry(image_fname, entry_path, data, do_compress=True,
332 allow_resize=True, write_map=False):
Simon Glass3971c952019-07-20 12:24:11 -0600333 """Replace an entry in an image
334
335 This replaces the data in a particular entry in an image. This size of the
336 new data must match the size of the old data unless allow_resize is True.
337
338 Args:
339 image_fname: Image filename to process
340 entry_path: Path to entry to extract
341 data: Data to replace with
Simon Glassd48f94e2019-07-20 12:24:12 -0600342 do_compress: True to compress the data if needed, False if data is
Simon Glass3971c952019-07-20 12:24:11 -0600343 already compressed so should be used as is
344 allow_resize: True to allow entries to change size (this does a re-pack
345 of the entries), False to raise an exception
Simon Glassd48f94e2019-07-20 12:24:12 -0600346 write_map: True to write a map file
Simon Glass3971c952019-07-20 12:24:11 -0600347
348 Returns:
349 Image object that was updated
350 """
Simon Glass274bd0e2019-07-20 12:24:13 -0600351 tout.Info("Write entry '%s', file '%s'" % (entry_path, image_fname))
Simon Glass3971c952019-07-20 12:24:11 -0600352 image = Image.FromFile(image_fname)
353 entry = image.FindEntryPath(entry_path)
Simon Glass274bd0e2019-07-20 12:24:13 -0600354 WriteEntryToImage(image, entry, data, do_compress=do_compress,
355 allow_resize=allow_resize, write_map=write_map)
Simon Glass3971c952019-07-20 12:24:11 -0600356
Simon Glass3971c952019-07-20 12:24:11 -0600357 return image
358
Simon Glass30033c22019-07-20 12:24:15 -0600359
360def ReplaceEntries(image_fname, input_fname, indir, entry_paths,
361 do_compress=True, allow_resize=True, write_map=False):
362 """Replace the data from one or more entries from input files
363
364 Args:
365 image_fname: Image filename to process
Jan Kiszka8ea44432021-11-11 08:13:30 +0100366 input_fname: Single input filename to use if replacing one file, None
Simon Glass30033c22019-07-20 12:24:15 -0600367 otherwise
368 indir: Input directory to use (for any number of files), else None
Jan Kiszka8ea44432021-11-11 08:13:30 +0100369 entry_paths: List of entry paths to replace
Simon Glass30033c22019-07-20 12:24:15 -0600370 do_compress: True if the input data is uncompressed and may need to be
371 compressed if the entry requires it, False if the data is already
372 compressed.
373 write_map: True to write a map file
374
375 Returns:
376 List of EntryInfo records that were written
377 """
Jan Kiszkaafc8f292021-11-11 08:14:18 +0100378 image_fname = os.path.abspath(image_fname)
Simon Glass30033c22019-07-20 12:24:15 -0600379 image = Image.FromFile(image_fname)
380
381 # Replace an entry from a single file, as a special case
382 if input_fname:
383 if not entry_paths:
384 raise ValueError('Must specify an entry path to read with -f')
385 if len(entry_paths) != 1:
386 raise ValueError('Must specify exactly one entry path to write with -f')
387 entry = image.FindEntryPath(entry_paths[0])
388 data = tools.ReadFile(input_fname)
389 tout.Notice("Read %#x bytes from file '%s'" % (len(data), input_fname))
390 WriteEntryToImage(image, entry, data, do_compress=do_compress,
391 allow_resize=allow_resize, write_map=write_map)
392 return
393
394 # Otherwise we will input from a path given by the entry path of each entry.
395 # This means that files must appear in subdirectories if they are part of
396 # a sub-section.
397 einfos = image.GetListEntries(entry_paths)[0]
398 tout.Notice("Replacing %d matching entries in image '%s'" %
399 (len(einfos), image_fname))
400
401 BeforeReplace(image, allow_resize)
402
403 for einfo in einfos:
404 entry = einfo.entry
405 if entry.GetEntries():
406 tout.Info("Skipping section entry '%s'" % entry.GetPath())
407 continue
408
409 path = entry.GetPath()[1:]
410 fname = os.path.join(indir, path)
411
412 if os.path.exists(fname):
413 tout.Notice("Write entry '%s' from file '%s'" %
414 (entry.GetPath(), fname))
415 data = tools.ReadFile(fname)
416 ReplaceOneEntry(image, entry, data, do_compress, allow_resize)
417 else:
418 tout.Warning("Skipping entry '%s' from missing file '%s'" %
419 (entry.GetPath(), fname))
420
421 AfterReplace(image, allow_resize=allow_resize, write_map=write_map)
422 return image
423
424
Simon Glass55ab0b62021-03-18 20:25:06 +1300425def PrepareImagesAndDtbs(dtb_fname, select_images, update_fdt, use_expanded):
Simon Glassd3151ff2019-07-20 12:23:27 -0600426 """Prepare the images to be processed and select the device tree
427
428 This function:
429 - reads in the device tree
430 - finds and scans the binman node to create all entries
431 - selects which images to build
432 - Updates the device tress with placeholder properties for offset,
433 image-pos, etc.
434
435 Args:
436 dtb_fname: Filename of the device tree file to use (.dts or .dtb)
437 selected_images: List of images to output, or None for all
438 update_fdt: True to update the FDT wth entry offsets, etc.
Simon Glass55ab0b62021-03-18 20:25:06 +1300439 use_expanded: True to use expanded versions of entries, if available.
440 So if 'u-boot' is called for, we use 'u-boot-expanded' instead. This
441 is needed if update_fdt is True (although tests may disable it)
Simon Glass31ee50f2020-09-01 05:13:55 -0600442
443 Returns:
444 OrderedDict of images:
445 key: Image name (str)
446 value: Image object
Simon Glassd3151ff2019-07-20 12:23:27 -0600447 """
448 # Import these here in case libfdt.py is not available, in which case
449 # the above help option still works.
Simon Glassc585dd42020-04-17 18:09:03 -0600450 from dtoc import fdt
451 from dtoc import fdt_util
Simon Glassd3151ff2019-07-20 12:23:27 -0600452 global images
453
454 # Get the device tree ready by compiling it and copying the compiled
455 # output into a file in our output directly. Then scan it for use
456 # in binman.
457 dtb_fname = fdt_util.EnsureCompiled(dtb_fname)
458 fname = tools.GetOutputFilename('u-boot.dtb.out')
459 tools.WriteFile(fname, tools.ReadFile(dtb_fname))
460 dtb = fdt.FdtScan(fname)
461
462 node = _FindBinmanNode(dtb)
463 if not node:
464 raise ValueError("Device tree '%s' does not have a 'binman' "
465 "node" % dtb_fname)
466
Simon Glass55ab0b62021-03-18 20:25:06 +1300467 images = _ReadImageDesc(node, use_expanded)
Simon Glassd3151ff2019-07-20 12:23:27 -0600468
469 if select_images:
470 skip = []
471 new_images = OrderedDict()
472 for name, image in images.items():
473 if name in select_images:
474 new_images[name] = image
475 else:
476 skip.append(name)
477 images = new_images
478 tout.Notice('Skipping images: %s' % ', '.join(skip))
479
480 state.Prepare(images, dtb)
481
482 # Prepare the device tree by making sure that any missing
483 # properties are added (e.g. 'pos' and 'size'). The values of these
484 # may not be correct yet, but we add placeholders so that the
485 # size of the device tree is correct. Later, in
486 # SetCalculatedProperties() we will insert the correct values
487 # without changing the device-tree size, thus ensuring that our
488 # entry offsets remain the same.
489 for image in images.values():
490 image.ExpandEntries()
491 if update_fdt:
Simon Glassacd6c6e2020-10-26 17:40:17 -0600492 image.AddMissingProperties(True)
Simon Glassd3151ff2019-07-20 12:23:27 -0600493 image.ProcessFdt(dtb)
494
Simon Glass5a300602019-07-20 12:23:29 -0600495 for dtb_item in state.GetAllFdts():
Simon Glassd3151ff2019-07-20 12:23:27 -0600496 dtb_item.Sync(auto_resize=True)
497 dtb_item.Pack()
498 dtb_item.Flush()
499 return images
500
501
Simon Glassf8a54bc2019-07-20 12:23:56 -0600502def ProcessImage(image, update_fdt, write_map, get_contents=True,
Heiko Thiery6d451362022-01-06 11:49:41 +0100503 allow_resize=True, allow_missing=False,
504 allow_fake_blobs=False):
Simon Glassb766c5e52019-07-20 12:23:24 -0600505 """Perform all steps for this image, including checking and # writing it.
506
507 This means that errors found with a later image will be reported after
508 earlier images are already completed and written, but that does not seem
509 important.
510
511 Args:
512 image: Image to process
513 update_fdt: True to update the FDT wth entry offsets, etc.
514 write_map: True to write a map file
Simon Glass072959a2019-07-20 12:23:50 -0600515 get_contents: True to get the image contents from files, etc., False if
516 the contents is already present
Simon Glassf8a54bc2019-07-20 12:23:56 -0600517 allow_resize: True to allow entries to change size (this does a re-pack
518 of the entries), False to raise an exception
Simon Glass5d94cc62020-07-09 18:39:38 -0600519 allow_missing: Allow blob_ext objects to be missing
Heiko Thiery6d451362022-01-06 11:49:41 +0100520 allow_fake_blobs: Allow blob_ext objects to be faked with dummy files
Simon Glassa003cd32020-07-09 18:39:40 -0600521
522 Returns:
Heiko Thiery6d451362022-01-06 11:49:41 +0100523 True if one or more external blobs are missing or faked,
524 False if all are present
Simon Glassb766c5e52019-07-20 12:23:24 -0600525 """
Simon Glass072959a2019-07-20 12:23:50 -0600526 if get_contents:
Simon Glass5d94cc62020-07-09 18:39:38 -0600527 image.SetAllowMissing(allow_missing)
Heiko Thiery6d451362022-01-06 11:49:41 +0100528 image.SetAllowFakeBlob(allow_fake_blobs)
Simon Glass072959a2019-07-20 12:23:50 -0600529 image.GetEntryContents()
Simon Glassb766c5e52019-07-20 12:23:24 -0600530 image.GetEntryOffsets()
531
532 # We need to pack the entries to figure out where everything
533 # should be placed. This sets the offset/size of each entry.
534 # However, after packing we call ProcessEntryContents() which
535 # may result in an entry changing size. In that case we need to
536 # do another pass. Since the device tree often contains the
537 # final offset/size information we try to make space for this in
538 # AddMissingProperties() above. However, if the device is
539 # compressed we cannot know this compressed size in advance,
540 # since changing an offset from 0x100 to 0x104 (for example) can
541 # alter the compressed size of the device tree. So we need a
542 # third pass for this.
Simon Glass37fdd142019-07-20 12:24:06 -0600543 passes = 5
Simon Glassb766c5e52019-07-20 12:23:24 -0600544 for pack_pass in range(passes):
545 try:
546 image.PackEntries()
Simon Glassb766c5e52019-07-20 12:23:24 -0600547 except Exception as e:
548 if write_map:
549 fname = image.WriteMap()
550 print("Wrote map file '%s' to show errors" % fname)
551 raise
552 image.SetImagePos()
553 if update_fdt:
554 image.SetCalculatedProperties()
Simon Glass5a300602019-07-20 12:23:29 -0600555 for dtb_item in state.GetAllFdts():
Simon Glassb766c5e52019-07-20 12:23:24 -0600556 dtb_item.Sync()
Simon Glassf8a54bc2019-07-20 12:23:56 -0600557 dtb_item.Flush()
Simon Glasse5943412019-08-24 07:23:12 -0600558 image.WriteSymbols()
Simon Glassb766c5e52019-07-20 12:23:24 -0600559 sizes_ok = image.ProcessEntryContents()
560 if sizes_ok:
561 break
562 image.ResetForPack()
Simon Glass6bf9b472019-08-24 07:23:13 -0600563 tout.Info('Pack completed after %d pass(es)' % (pack_pass + 1))
Simon Glassb766c5e52019-07-20 12:23:24 -0600564 if not sizes_ok:
Simon Glass9d8ee322019-07-20 12:23:58 -0600565 image.Raise('Entries changed size after packing (tried %s passes)' %
Simon Glassb766c5e52019-07-20 12:23:24 -0600566 passes)
567
Simon Glassb766c5e52019-07-20 12:23:24 -0600568 image.BuildImage()
569 if write_map:
570 image.WriteMap()
Simon Glassa003cd32020-07-09 18:39:40 -0600571 missing_list = []
572 image.CheckMissing(missing_list)
573 if missing_list:
574 tout.Warning("Image '%s' is missing external blobs and is non-functional: %s" %
575 (image.name, ' '.join([e.name for e in missing_list])))
Simon Glassa820af72020-09-06 10:39:09 -0600576 _ShowHelpForMissingBlobs(missing_list)
Heiko Thiery6d451362022-01-06 11:49:41 +0100577 faked_list = []
578 image.CheckFakedBlobs(faked_list)
579 if faked_list:
580 tout.Warning("Image '%s:%s' has faked external blobs and is non-functional: %s" %
581 (image.name, image.image_name,
582 ' '.join([e.GetDefaultFilename() for e in faked_list])))
583 return bool(missing_list) or bool(faked_list)
Simon Glassb766c5e52019-07-20 12:23:24 -0600584
585
Simon Glassf46732a2019-07-08 14:25:29 -0600586def Binman(args):
Simon Glass2574ef62016-11-25 20:15:51 -0700587 """The main control code for binman
588
589 This assumes that help and test options have already been dealt with. It
590 deals with the core task of building images.
591
592 Args:
Simon Glassf46732a2019-07-08 14:25:29 -0600593 args: Command line arguments Namespace object
Simon Glass2574ef62016-11-25 20:15:51 -0700594 """
Simon Glassb9ba4e02019-08-24 07:22:44 -0600595 global Image
596 global state
597
Simon Glassf46732a2019-07-08 14:25:29 -0600598 if args.full_help:
Paul Barker25ecd972021-09-08 12:38:01 +0100599 tools.PrintFullHelp(
600 os.path.join(os.path.dirname(os.path.realpath(sys.argv[0])), 'README.rst')
601 )
Simon Glass2574ef62016-11-25 20:15:51 -0700602 return 0
603
Simon Glassb9ba4e02019-08-24 07:22:44 -0600604 # Put these here so that we can import this module without libfdt
Simon Glass90cd6f02020-08-05 13:27:47 -0600605 from binman.image import Image
Simon Glassc585dd42020-04-17 18:09:03 -0600606 from binman import state
Simon Glassb9ba4e02019-08-24 07:22:44 -0600607
Simon Glassdf08cbb2019-09-15 18:10:36 -0600608 if args.cmd in ['ls', 'extract', 'replace']:
Simon Glass9b7f5002019-07-20 12:23:53 -0600609 try:
Simon Glassdf08cbb2019-09-15 18:10:36 -0600610 tout.Init(args.verbosity)
Simon Glass9b7f5002019-07-20 12:23:53 -0600611 tools.PrepareOutputDir(None)
Simon Glassdf08cbb2019-09-15 18:10:36 -0600612 if args.cmd == 'ls':
613 ListEntries(args.image, args.paths)
Simon Glassb2fd11d2019-07-08 14:25:48 -0600614
Simon Glassdf08cbb2019-09-15 18:10:36 -0600615 if args.cmd == 'extract':
616 ExtractEntries(args.image, args.filename, args.outdir, args.paths,
Simon Glass637958f2021-11-23 21:09:50 -0700617 not args.uncompressed, args.format)
Simon Glass980a2842019-07-08 14:25:52 -0600618
Simon Glassdf08cbb2019-09-15 18:10:36 -0600619 if args.cmd == 'replace':
620 ReplaceEntries(args.image, args.filename, args.indir, args.paths,
621 do_compress=not args.compressed,
622 allow_resize=not args.fix_size, write_map=args.map)
623 except:
624 raise
Simon Glass30033c22019-07-20 12:24:15 -0600625 finally:
626 tools.FinaliseOutputDir()
627 return 0
628
Simon Glassadfb8492021-11-03 21:09:18 -0600629 elf_params = None
630 if args.update_fdt_in_elf:
631 elf_params = args.update_fdt_in_elf.split(',')
632 if len(elf_params) != 4:
633 raise ValueError('Invalid args %s to --update-fdt-in-elf: expected infile,outfile,begin_sym,end_sym' %
634 elf_params)
635
Simon Glass2574ef62016-11-25 20:15:51 -0700636 # Try to figure out which device tree contains our image description
Simon Glassf46732a2019-07-08 14:25:29 -0600637 if args.dt:
638 dtb_fname = args.dt
Simon Glass2574ef62016-11-25 20:15:51 -0700639 else:
Simon Glassf46732a2019-07-08 14:25:29 -0600640 board = args.board
Simon Glass2574ef62016-11-25 20:15:51 -0700641 if not board:
642 raise ValueError('Must provide a board to process (use -b <board>)')
Simon Glassf46732a2019-07-08 14:25:29 -0600643 board_pathname = os.path.join(args.build_dir, board)
Simon Glass2574ef62016-11-25 20:15:51 -0700644 dtb_fname = os.path.join(board_pathname, 'u-boot.dtb')
Simon Glassf46732a2019-07-08 14:25:29 -0600645 if not args.indir:
646 args.indir = ['.']
647 args.indir.append(board_pathname)
Simon Glass2574ef62016-11-25 20:15:51 -0700648
649 try:
Simon Glassf46732a2019-07-08 14:25:29 -0600650 tout.Init(args.verbosity)
651 elf.debug = args.debug
652 cbfs_util.VERBOSE = args.verbosity > 2
653 state.use_fake_dtb = args.fake_dtb
Simon Glass55ab0b62021-03-18 20:25:06 +1300654
655 # Normally we replace the 'u-boot' etype with 'u-boot-expanded', etc.
656 # When running tests this can be disabled using this flag. When not
657 # updating the FDT in image, it is not needed by binman, but we use it
658 # for consistency, so that the images look the same to U-Boot at
659 # runtime.
660 use_expanded = not args.no_expanded
Simon Glass2574ef62016-11-25 20:15:51 -0700661 try:
Simon Glassf46732a2019-07-08 14:25:29 -0600662 tools.SetInputDirs(args.indir)
663 tools.PrepareOutputDir(args.outdir, args.preserve)
664 tools.SetToolPaths(args.toolpath)
665 state.SetEntryArgs(args.entry_arg)
Simon Glass76f496d2021-07-06 10:36:37 -0600666 state.SetThreads(args.threads)
Simon Glass92307732018-07-06 10:27:40 -0600667
Simon Glassd3151ff2019-07-20 12:23:27 -0600668 images = PrepareImagesAndDtbs(dtb_fname, args.image,
Simon Glass55ab0b62021-03-18 20:25:06 +1300669 args.update_fdt, use_expanded)
Heiko Thiery6d451362022-01-06 11:49:41 +0100670
Simon Glass76f496d2021-07-06 10:36:37 -0600671 if args.test_section_timeout:
672 # Set the first image to timeout, used in testThreadTimeout()
673 images[list(images.keys())[0]].test_section_timeout = True
Heiko Thiery6d451362022-01-06 11:49:41 +0100674 invalid = False
Simon Glass2574ef62016-11-25 20:15:51 -0700675 for image in images.values():
Heiko Thiery6d451362022-01-06 11:49:41 +0100676 invalid |= ProcessImage(image, args.update_fdt, args.map,
677 allow_missing=args.allow_missing,
678 allow_fake_blobs=args.fake_ext_blobs)
Simon Glassbdb40312018-09-14 04:57:20 -0600679
680 # Write the updated FDTs to our output files
Simon Glass5a300602019-07-20 12:23:29 -0600681 for dtb_item in state.GetAllFdts():
Simon Glassbdb40312018-09-14 04:57:20 -0600682 tools.WriteFile(dtb_item._fname, dtb_item.GetContents())
683
Simon Glassadfb8492021-11-03 21:09:18 -0600684 if elf_params:
685 data = state.GetFdtForEtype('u-boot-dtb').GetContents()
686 elf.UpdateFile(*elf_params, data)
687
Heiko Thiery6d451362022-01-06 11:49:41 +0100688 if invalid:
Simon Glassa820af72020-09-06 10:39:09 -0600689 tout.Warning("\nSome images are invalid")
Simon Glass748a1d42021-07-06 10:36:41 -0600690
691 # Use this to debug the time take to pack the image
692 #state.TimingShow()
Simon Glass2574ef62016-11-25 20:15:51 -0700693 finally:
694 tools.FinaliseOutputDir()
695 finally:
696 tout.Uninit()
697
698 return 0