blob: cc95b424b338d101fe0efeef4b13c9649b320606 [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001# SPDX-License-Identifier: GPL-2.0+
Simon Glass24ad3652017-11-13 18:54:54 -07002# Copyright (c) 2017 Google, Inc
3# Written by Simon Glass <sjg@chromium.org>
4#
Simon Glass24ad3652017-11-13 18:54:54 -07005# Test for the elf module
6
7import os
Simon Glass4f379ea2019-07-08 13:18:34 -06008import shutil
Simon Glassa4e259e2021-11-03 21:09:16 -06009import struct
Simon Glass24ad3652017-11-13 18:54:54 -070010import sys
Simon Glass4f379ea2019-07-08 13:18:34 -060011import tempfile
Simon Glass24ad3652017-11-13 18:54:54 -070012import unittest
13
Simon Glassc585dd42020-04-17 18:09:03 -060014from binman import elf
Simon Glass131444f2023-02-23 18:18:04 -070015from u_boot_pylib import command
16from u_boot_pylib import test_util
17from u_boot_pylib import tools
18from u_boot_pylib import tout
Simon Glass24ad3652017-11-13 18:54:54 -070019
20binman_dir = os.path.dirname(os.path.realpath(sys.argv[0]))
Simon Glass4ca8e042017-11-13 18:55:01 -070021
Simon Glass4ca8e042017-11-13 18:55:01 -070022
23class FakeEntry:
Simon Glass4114f972018-07-17 13:25:26 -060024 """A fake Entry object, usedfor testing
25
26 This supports an entry with a given size.
27 """
Simon Glass4ca8e042017-11-13 18:55:01 -070028 def __init__(self, contents_size):
29 self.contents_size = contents_size
Simon Glass80025522022-01-29 14:14:04 -070030 self.data = tools.get_bytes(ord('a'), contents_size)
Simon Glass4ca8e042017-11-13 18:55:01 -070031
32 def GetPath(self):
33 return 'entry_path'
34
Simon Glass4114f972018-07-17 13:25:26 -060035
Simon Glass8a6f56e2018-06-01 09:38:13 -060036class FakeSection:
Simon Glass4114f972018-07-17 13:25:26 -060037 """A fake Section object, used for testing
38
39 This has the minimum feature set needed to support testing elf functions.
40 A LookupSymbol() function is provided which returns a fake value for amu
41 symbol requested.
42 """
Simon Glass4ca8e042017-11-13 18:55:01 -070043 def __init__(self, sym_value=1):
44 self.sym_value = sym_value
45
46 def GetPath(self):
Simon Glass8a6f56e2018-06-01 09:38:13 -060047 return 'section_path'
Simon Glass4ca8e042017-11-13 18:55:01 -070048
Simon Glassecbe4732021-01-06 21:35:15 -070049 def LookupImageSymbol(self, name, weak, msg, base_addr):
Simon Glass4114f972018-07-17 13:25:26 -060050 """Fake implementation which returns the same value for all symbols"""
Simon Glass4ca8e042017-11-13 18:55:01 -070051 return self.sym_value
Simon Glass24ad3652017-11-13 18:54:54 -070052
Simon Glassecbe4732021-01-06 21:35:15 -070053 def GetImage(self):
54 return self
Simon Glass4114f972018-07-17 13:25:26 -060055
Simon Glassf6290892019-08-24 07:22:53 -060056def BuildElfTestFiles(target_dir):
57 """Build ELF files used for testing in binman
58
Simon Glass571adc82022-02-08 11:49:55 -070059 This compiles and links the test files into the specified directory. It uses
60 the Makefile and source files in the binman test/ directory.
Simon Glassf6290892019-08-24 07:22:53 -060061
62 Args:
63 target_dir: Directory to put the files into
64 """
65 if not os.path.exists(target_dir):
66 os.mkdir(target_dir)
67 testdir = os.path.join(binman_dir, 'test')
68
69 # If binman is involved from the main U-Boot Makefile the -r and -R
70 # flags are set in MAKEFLAGS. This prevents this Makefile from working
71 # correctly. So drop any make flags here.
72 if 'MAKEFLAGS' in os.environ:
73 del os.environ['MAKEFLAGS']
Simon Glass271fd8f2021-11-03 21:09:15 -060074 try:
Simon Glass80025522022-01-29 14:14:04 -070075 tools.run('make', '-C', target_dir, '-f',
Simon Glass271fd8f2021-11-03 21:09:15 -060076 os.path.join(testdir, 'Makefile'), 'SRC=%s/' % testdir)
77 except ValueError as e:
78 # The test system seems to suppress this in a strange way
79 print(e)
Simon Glassf6290892019-08-24 07:22:53 -060080
81
Simon Glass24ad3652017-11-13 18:54:54 -070082class TestElf(unittest.TestCase):
Simon Glass752e7552018-10-01 21:12:41 -060083 @classmethod
Simon Glass4affd4b2019-08-24 07:22:54 -060084 def setUpClass(cls):
85 cls._indir = tempfile.mkdtemp(prefix='elf.')
Simon Glass80025522022-01-29 14:14:04 -070086 tools.set_input_dirs(['.'])
Simon Glass4affd4b2019-08-24 07:22:54 -060087 BuildElfTestFiles(cls._indir)
88
89 @classmethod
90 def tearDownClass(cls):
91 if cls._indir:
92 shutil.rmtree(cls._indir)
93
94 @classmethod
95 def ElfTestFile(cls, fname):
96 return os.path.join(cls._indir, fname)
Simon Glass752e7552018-10-01 21:12:41 -060097
Simon Glass24ad3652017-11-13 18:54:54 -070098 def testAllSymbols(self):
Simon Glass4114f972018-07-17 13:25:26 -060099 """Test that we can obtain a symbol from the ELF file"""
Simon Glass4affd4b2019-08-24 07:22:54 -0600100 fname = self.ElfTestFile('u_boot_ucode_ptr')
Simon Glass24ad3652017-11-13 18:54:54 -0700101 syms = elf.GetSymbols(fname, [])
Simon Glass315400e2022-01-09 20:13:37 -0700102 self.assertIn('_dt_ucode_base_size', syms)
Simon Glass24ad3652017-11-13 18:54:54 -0700103
104 def testRegexSymbols(self):
Simon Glass4114f972018-07-17 13:25:26 -0600105 """Test that we can obtain from the ELF file by regular expression"""
Simon Glass4affd4b2019-08-24 07:22:54 -0600106 fname = self.ElfTestFile('u_boot_ucode_ptr')
Simon Glass24ad3652017-11-13 18:54:54 -0700107 syms = elf.GetSymbols(fname, ['ucode'])
Simon Glass315400e2022-01-09 20:13:37 -0700108 self.assertIn('_dt_ucode_base_size', syms)
Simon Glass24ad3652017-11-13 18:54:54 -0700109 syms = elf.GetSymbols(fname, ['missing'])
Simon Glass315400e2022-01-09 20:13:37 -0700110 self.assertNotIn('_dt_ucode_base_size', syms)
Simon Glass24ad3652017-11-13 18:54:54 -0700111 syms = elf.GetSymbols(fname, ['missing', 'ucode'])
Simon Glass315400e2022-01-09 20:13:37 -0700112 self.assertIn('_dt_ucode_base_size', syms)
Simon Glass24ad3652017-11-13 18:54:54 -0700113
Simon Glass4ca8e042017-11-13 18:55:01 -0700114 def testMissingFile(self):
Simon Glass4114f972018-07-17 13:25:26 -0600115 """Test that a missing file is detected"""
Simon Glass4ca8e042017-11-13 18:55:01 -0700116 entry = FakeEntry(10)
Simon Glass8a6f56e2018-06-01 09:38:13 -0600117 section = FakeSection()
Simon Glass4ca8e042017-11-13 18:55:01 -0700118 with self.assertRaises(ValueError) as e:
Simon Glass2a0fa982022-02-11 13:23:21 -0700119 elf.LookupAndWriteSymbols('missing-file', entry, section)
Simon Glass4ca8e042017-11-13 18:55:01 -0700120 self.assertIn("Filename 'missing-file' not found in input path",
121 str(e.exception))
122
123 def testOutsideFile(self):
Simon Glass4114f972018-07-17 13:25:26 -0600124 """Test a symbol which extends outside the entry area is detected"""
Stefan Herbrechtsmeier732742e2022-08-19 16:25:18 +0200125 if not elf.ELF_TOOLS:
126 self.skipTest('Python elftools not available')
Simon Glass4ca8e042017-11-13 18:55:01 -0700127 entry = FakeEntry(10)
Simon Glass8a6f56e2018-06-01 09:38:13 -0600128 section = FakeSection()
Simon Glass5d0c0262019-08-24 07:22:56 -0600129 elf_fname = self.ElfTestFile('u_boot_binman_syms')
Simon Glass4ca8e042017-11-13 18:55:01 -0700130 with self.assertRaises(ValueError) as e:
Simon Glass2a0fa982022-02-11 13:23:21 -0700131 elf.LookupAndWriteSymbols(elf_fname, entry, section)
Alper Nebi Yasak9634dc92022-06-18 15:13:11 +0300132 self.assertIn('entry_path has offset 8 (size 8) but the contents size '
Simon Glass4ca8e042017-11-13 18:55:01 -0700133 'is a', str(e.exception))
134
135 def testMissingImageStart(self):
Simon Glass4114f972018-07-17 13:25:26 -0600136 """Test that we detect a missing __image_copy_start symbol
137
138 This is needed to mark the start of the image. Without it we cannot
139 locate the offset of a binman symbol within the image.
140 """
Simon Glass4ca8e042017-11-13 18:55:01 -0700141 entry = FakeEntry(10)
Simon Glass8a6f56e2018-06-01 09:38:13 -0600142 section = FakeSection()
Simon Glass46ea6912019-08-24 07:22:58 -0600143 elf_fname = self.ElfTestFile('u_boot_binman_syms_bad')
Simon Glassea1d77d2023-07-18 07:23:56 -0600144 count = elf.LookupAndWriteSymbols(elf_fname, entry, section)
145 self.assertEqual(0, count)
Simon Glass4ca8e042017-11-13 18:55:01 -0700146
147 def testBadSymbolSize(self):
Simon Glass4114f972018-07-17 13:25:26 -0600148 """Test that an attempt to use an 8-bit symbol are detected
149
150 Only 32 and 64 bits are supported, since we need to store an offset
151 into the image.
152 """
Stefan Herbrechtsmeier732742e2022-08-19 16:25:18 +0200153 if not elf.ELF_TOOLS:
154 self.skipTest('Python elftools not available')
Simon Glass4ca8e042017-11-13 18:55:01 -0700155 entry = FakeEntry(10)
Simon Glass8a6f56e2018-06-01 09:38:13 -0600156 section = FakeSection()
Simon Glassb8deb122019-08-24 07:22:57 -0600157 elf_fname =self.ElfTestFile('u_boot_binman_syms_size')
Simon Glass4ca8e042017-11-13 18:55:01 -0700158 with self.assertRaises(ValueError) as e:
Simon Glass2a0fa982022-02-11 13:23:21 -0700159 elf.LookupAndWriteSymbols(elf_fname, entry, section)
Simon Glass4ca8e042017-11-13 18:55:01 -0700160 self.assertIn('has size 1: only 4 and 8 are supported',
161 str(e.exception))
162
163 def testNoValue(self):
Simon Glass4114f972018-07-17 13:25:26 -0600164 """Test the case where we have no value for the symbol
165
Simon Glassea1d77d2023-07-18 07:23:56 -0600166 This should produce -1 values for all three symbols, taking up the
Simon Glass4114f972018-07-17 13:25:26 -0600167 first 16 bytes of the image.
168 """
Stefan Herbrechtsmeier732742e2022-08-19 16:25:18 +0200169 if not elf.ELF_TOOLS:
170 self.skipTest('Python elftools not available')
Alper Nebi Yasak9634dc92022-06-18 15:13:11 +0300171 entry = FakeEntry(28)
Simon Glass8a6f56e2018-06-01 09:38:13 -0600172 section = FakeSection(sym_value=None)
Simon Glass5d0c0262019-08-24 07:22:56 -0600173 elf_fname = self.ElfTestFile('u_boot_binman_syms')
Simon Glassea1d77d2023-07-18 07:23:56 -0600174 count = elf.LookupAndWriteSymbols(elf_fname, entry, section)
175 self.assertEqual(5, count)
Alper Nebi Yasak9634dc92022-06-18 15:13:11 +0300176 expected = (struct.pack('<L', elf.BINMAN_SYM_MAGIC_VALUE) +
177 tools.get_bytes(255, 20) +
178 tools.get_bytes(ord('a'), 4))
179 self.assertEqual(expected, entry.data)
Simon Glass4ca8e042017-11-13 18:55:01 -0700180
181 def testDebug(self):
Simon Glass4114f972018-07-17 13:25:26 -0600182 """Check that enabling debug in the elf module produced debug output"""
Stefan Herbrechtsmeier732742e2022-08-19 16:25:18 +0200183 if not elf.ELF_TOOLS:
184 self.skipTest('Python elftools not available')
Simon Glassb6dff4c2019-07-20 12:23:36 -0600185 try:
Simon Glass011f1b32022-01-29 14:14:15 -0700186 tout.init(tout.DEBUG)
Alper Nebi Yasak9634dc92022-06-18 15:13:11 +0300187 entry = FakeEntry(24)
Simon Glassb6dff4c2019-07-20 12:23:36 -0600188 section = FakeSection()
Simon Glass5d0c0262019-08-24 07:22:56 -0600189 elf_fname = self.ElfTestFile('u_boot_binman_syms')
Simon Glassb6dff4c2019-07-20 12:23:36 -0600190 with test_util.capture_sys_output() as (stdout, stderr):
Simon Glass2a0fa982022-02-11 13:23:21 -0700191 elf.LookupAndWriteSymbols(elf_fname, entry, section)
Simon Glassb6dff4c2019-07-20 12:23:36 -0600192 self.assertTrue(len(stdout.getvalue()) > 0)
193 finally:
Simon Glass011f1b32022-01-29 14:14:15 -0700194 tout.init(tout.WARNING)
Simon Glass4ca8e042017-11-13 18:55:01 -0700195
Simon Glass4f379ea2019-07-08 13:18:34 -0600196 def testMakeElf(self):
197 """Test for the MakeElf function"""
198 outdir = tempfile.mkdtemp(prefix='elf.')
199 expected_text = b'1234'
200 expected_data = b'wxyz'
201 elf_fname = os.path.join(outdir, 'elf')
Simon Glassd349ada2019-08-24 07:22:45 -0600202 bin_fname = os.path.join(outdir, 'bin')
Simon Glass4f379ea2019-07-08 13:18:34 -0600203
204 # Make an Elf file and then convert it to a fkat binary file. This
205 # should produce the original data.
206 elf.MakeElf(elf_fname, expected_text, expected_data)
Simon Glass80025522022-01-29 14:14:04 -0700207 objcopy, args = tools.get_target_compile_tool('objcopy')
Alper Nebi Yasak5cd321d2020-09-06 14:46:05 +0300208 args += ['-O', 'binary', elf_fname, bin_fname]
Simon Glass840be732022-01-29 14:14:05 -0700209 stdout = command.output(objcopy, *args)
Simon Glass4f379ea2019-07-08 13:18:34 -0600210 with open(bin_fname, 'rb') as fd:
211 data = fd.read()
212 self.assertEqual(expected_text + expected_data, data)
213 shutil.rmtree(outdir)
214
Simon Glass567b6822019-07-08 13:18:35 -0600215 def testDecodeElf(self):
216 """Test for the MakeElf function"""
217 if not elf.ELF_TOOLS:
218 self.skipTest('Python elftools not available')
219 outdir = tempfile.mkdtemp(prefix='elf.')
220 expected_text = b'1234'
221 expected_data = b'wxyz'
222 elf_fname = os.path.join(outdir, 'elf')
223 elf.MakeElf(elf_fname, expected_text, expected_data)
Simon Glass80025522022-01-29 14:14:04 -0700224 data = tools.read_file(elf_fname)
Simon Glass567b6822019-07-08 13:18:35 -0600225
226 load = 0xfef20000
227 entry = load + 2
228 expected = expected_text + expected_data
229 self.assertEqual(elf.ElfInfo(expected, load, entry, len(expected)),
230 elf.DecodeElf(data, 0))
231 self.assertEqual(elf.ElfInfo(b'\0\0' + expected[2:],
232 load, entry, len(expected)),
233 elf.DecodeElf(data, load + 2))
Simon Glass4affd4b2019-08-24 07:22:54 -0600234 shutil.rmtree(outdir)
Simon Glass567b6822019-07-08 13:18:35 -0600235
Simon Glassa4e259e2021-11-03 21:09:16 -0600236 def testEmbedData(self):
237 """Test for the GetSymbolFileOffset() function"""
238 if not elf.ELF_TOOLS:
239 self.skipTest('Python elftools not available')
240
241 fname = self.ElfTestFile('embed_data')
242 offset = elf.GetSymbolFileOffset(fname, ['embed_start', 'embed_end'])
243 start = offset['embed_start'].offset
244 end = offset['embed_end'].offset
Simon Glass80025522022-01-29 14:14:04 -0700245 data = tools.read_file(fname)
Simon Glassa4e259e2021-11-03 21:09:16 -0600246 embed_data = data[start:end]
Simon Glass6490d4b2023-01-23 11:29:41 -0700247 expect = struct.pack('<IIIII', 2, 3, 0x1234, 0x5678, 0)
Simon Glassa4e259e2021-11-03 21:09:16 -0600248 self.assertEqual(expect, embed_data)
249
250 def testEmbedFail(self):
251 """Test calling GetSymbolFileOffset() without elftools"""
252 try:
253 old_val = elf.ELF_TOOLS
254 elf.ELF_TOOLS = False
255 fname = self.ElfTestFile('embed_data')
256 with self.assertRaises(ValueError) as e:
257 elf.GetSymbolFileOffset(fname, ['embed_start', 'embed_end'])
Simon Glassacc03752022-03-05 20:18:57 -0700258 self.assertIn("Python: No module named 'elftools'",
Simon Glassa4e259e2021-11-03 21:09:16 -0600259 str(e.exception))
260 finally:
261 elf.ELF_TOOLS = old_val
262
263 def testEmbedDataNoSym(self):
264 """Test for GetSymbolFileOffset() getting no symbols"""
265 if not elf.ELF_TOOLS:
266 self.skipTest('Python elftools not available')
267
268 fname = self.ElfTestFile('embed_data')
269 offset = elf.GetSymbolFileOffset(fname, ['missing_sym'])
270 self.assertEqual({}, offset)
271
Simon Glassacc03752022-03-05 20:18:57 -0700272 def test_read_loadable_segments(self):
273 """Test for read_loadable_segments()"""
Simon Glass571adc82022-02-08 11:49:55 -0700274 if not elf.ELF_TOOLS:
275 self.skipTest('Python elftools not available')
276 fname = self.ElfTestFile('embed_data')
Simon Glassacc03752022-03-05 20:18:57 -0700277 segments, entry = elf.read_loadable_segments(tools.read_file(fname))
Simon Glass571adc82022-02-08 11:49:55 -0700278
279 def test_read_segments_fail(self):
Simon Glassacc03752022-03-05 20:18:57 -0700280 """Test for read_loadable_segments() without elftools"""
Simon Glass571adc82022-02-08 11:49:55 -0700281 try:
282 old_val = elf.ELF_TOOLS
283 elf.ELF_TOOLS = False
284 fname = self.ElfTestFile('embed_data')
285 with self.assertRaises(ValueError) as e:
Simon Glassacc03752022-03-05 20:18:57 -0700286 elf.read_loadable_segments(tools.read_file(fname))
287 self.assertIn("Python: No module named 'elftools'",
Simon Glass571adc82022-02-08 11:49:55 -0700288 str(e.exception))
289 finally:
290 elf.ELF_TOOLS = old_val
291
292 def test_read_segments_bad_data(self):
Simon Glassacc03752022-03-05 20:18:57 -0700293 """Test for read_loadable_segments() with an invalid ELF file"""
Stefan Herbrechtsmeier732742e2022-08-19 16:25:18 +0200294 if not elf.ELF_TOOLS:
295 self.skipTest('Python elftools not available')
Simon Glass571adc82022-02-08 11:49:55 -0700296 fname = self.ElfTestFile('embed_data')
297 with self.assertRaises(ValueError) as e:
Simon Glassacc03752022-03-05 20:18:57 -0700298 elf.read_loadable_segments(tools.get_bytes(100, 100))
Simon Glass571adc82022-02-08 11:49:55 -0700299 self.assertIn('Magic number does not match', str(e.exception))
300
Simon Glassea64c022022-03-18 19:19:49 -0600301 def test_get_file_offset(self):
302 """Test GetFileOffset() gives the correct file offset for a symbol"""
Stefan Herbrechtsmeier732742e2022-08-19 16:25:18 +0200303 if not elf.ELF_TOOLS:
304 self.skipTest('Python elftools not available')
Simon Glassea64c022022-03-18 19:19:49 -0600305 fname = self.ElfTestFile('embed_data')
306 syms = elf.GetSymbols(fname, ['embed'])
307 addr = syms['embed'].address
308 offset = elf.GetFileOffset(fname, addr)
309 data = tools.read_file(fname)
310
311 # Just use the first 4 bytes and assume it is little endian
312 embed_data = data[offset:offset + 4]
313 embed_value = struct.unpack('<I', embed_data)[0]
314 self.assertEqual(0x1234, embed_value)
315
316 def test_get_file_offset_fail(self):
317 """Test calling GetFileOffset() without elftools"""
318 try:
319 old_val = elf.ELF_TOOLS
320 elf.ELF_TOOLS = False
321 fname = self.ElfTestFile('embed_data')
322 with self.assertRaises(ValueError) as e:
323 elf.GetFileOffset(fname, 0)
324 self.assertIn("Python: No module named 'elftools'",
325 str(e.exception))
326 finally:
327 elf.ELF_TOOLS = old_val
328
329 def test_get_symbol_from_address(self):
330 """Test GetSymbolFromAddress()"""
Stefan Herbrechtsmeier732742e2022-08-19 16:25:18 +0200331 if not elf.ELF_TOOLS:
332 self.skipTest('Python elftools not available')
Simon Glassea64c022022-03-18 19:19:49 -0600333 fname = self.ElfTestFile('elf_sections')
334 sym_name = 'calculate'
335 syms = elf.GetSymbols(fname, [sym_name])
336 addr = syms[sym_name].address
337 sym = elf.GetSymbolFromAddress(fname, addr)
338 self.assertEqual(sym_name, sym)
339
340 def test_get_symbol_from_address_fail(self):
341 """Test calling GetSymbolFromAddress() without elftools"""
342 try:
343 old_val = elf.ELF_TOOLS
344 elf.ELF_TOOLS = False
345 fname = self.ElfTestFile('embed_data')
346 with self.assertRaises(ValueError) as e:
347 elf.GetSymbolFromAddress(fname, 0x1000)
348 self.assertIn("Python: No module named 'elftools'",
349 str(e.exception))
350 finally:
351 elf.ELF_TOOLS = old_val
352
Simon Glass6e657f62023-01-07 14:07:13 -0700353 def test_is_valid(self):
354 """Test is_valid()"""
355 self.assertEqual(False, elf.is_valid(b''))
356 self.assertEqual(False, elf.is_valid(b'1234'))
357
358 fname = self.ElfTestFile('elf_sections')
359 data = tools.read_file(fname)
360 self.assertEqual(True, elf.is_valid(data))
361 self.assertEqual(False, elf.is_valid(data[4:]))
362
Simon Glass6490d4b2023-01-23 11:29:41 -0700363 def test_get_symbol_offset(self):
364 fname = self.ElfTestFile('embed_data')
365 syms = elf.GetSymbols(fname, ['embed_start', 'embed'])
366 expected = syms['embed'].address - syms['embed_start'].address
367 val = elf.GetSymbolOffset(fname, 'embed', 'embed_start')
368 self.assertEqual(expected, val)
369
370 with self.assertRaises(KeyError) as e:
371 elf.GetSymbolOffset(fname, 'embed')
372 self.assertIn('__image_copy_start', str(e.exception))
373
Simon Glass83b8bfe2023-07-18 07:24:01 -0600374 def test_get_symbol_address(self):
375 fname = self.ElfTestFile('embed_data')
376 addr = elf.GetSymbolAddress(fname, 'region_size')
377 self.assertEqual(0, addr)
378
Simon Glass4ca8e042017-11-13 18:55:01 -0700379
Simon Glass24ad3652017-11-13 18:54:54 -0700380if __name__ == '__main__':
381 unittest.main()