blob: 02bc108374925165676e933719f7bf6ba6e62c50 [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001# SPDX-License-Identifier: GPL-2.0+
Simon Glassb50e5612017-11-13 18:54:54 -07002# Copyright (c) 2017 Google, Inc
3# Written by Simon Glass <sjg@chromium.org>
4#
Simon Glassb50e5612017-11-13 18:54:54 -07005# Test for the elf module
6
7import os
Simon Glassf58558a2019-07-08 13:18:34 -06008import shutil
Simon Glass056f0ef2021-11-03 21:09:16 -06009import struct
Simon Glassb50e5612017-11-13 18:54:54 -070010import sys
Simon Glassf58558a2019-07-08 13:18:34 -060011import tempfile
Simon Glassb50e5612017-11-13 18:54:54 -070012import unittest
13
Simon Glass16287932020-04-17 18:09:03 -060014from binman import elf
Simon Glassbf776672020-04-17 18:09:04 -060015from patman import command
16from patman import test_util
17from patman import tools
18from patman import tout
Simon Glassb50e5612017-11-13 18:54:54 -070019
20binman_dir = os.path.dirname(os.path.realpath(sys.argv[0]))
Simon Glass19790632017-11-13 18:55:01 -070021
Simon Glass19790632017-11-13 18:55:01 -070022
23class FakeEntry:
Simon Glassb2b0df82018-07-17 13:25:26 -060024 """A fake Entry object, usedfor testing
25
26 This supports an entry with a given size.
27 """
Simon Glass19790632017-11-13 18:55:01 -070028 def __init__(self, contents_size):
29 self.contents_size = contents_size
Simon Glassc1aa66e2022-01-29 14:14:04 -070030 self.data = tools.get_bytes(ord('a'), contents_size)
Simon Glass19790632017-11-13 18:55:01 -070031
32 def GetPath(self):
33 return 'entry_path'
34
Simon Glassb2b0df82018-07-17 13:25:26 -060035
Simon Glassf55382b2018-06-01 09:38:13 -060036class FakeSection:
Simon Glassb2b0df82018-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 Glass19790632017-11-13 18:55:01 -070043 def __init__(self, sym_value=1):
44 self.sym_value = sym_value
45
46 def GetPath(self):
Simon Glassf55382b2018-06-01 09:38:13 -060047 return 'section_path'
Simon Glass19790632017-11-13 18:55:01 -070048
Simon Glass870a9ea2021-01-06 21:35:15 -070049 def LookupImageSymbol(self, name, weak, msg, base_addr):
Simon Glassb2b0df82018-07-17 13:25:26 -060050 """Fake implementation which returns the same value for all symbols"""
Simon Glass19790632017-11-13 18:55:01 -070051 return self.sym_value
Simon Glassb50e5612017-11-13 18:54:54 -070052
Simon Glass870a9ea2021-01-06 21:35:15 -070053 def GetImage(self):
54 return self
Simon Glassb2b0df82018-07-17 13:25:26 -060055
Simon Glass53e22bf2019-08-24 07:22:53 -060056def BuildElfTestFiles(target_dir):
57 """Build ELF files used for testing in binman
58
Simon Glass4d38dd72022-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 Glass53e22bf2019-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 Glass2fb2cd72021-11-03 21:09:15 -060074 try:
Simon Glassc1aa66e2022-01-29 14:14:04 -070075 tools.run('make', '-C', target_dir, '-f',
Simon Glass2fb2cd72021-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 Glass53e22bf2019-08-24 07:22:53 -060080
81
Simon Glassb50e5612017-11-13 18:54:54 -070082class TestElf(unittest.TestCase):
Simon Glasse0e62752018-10-01 21:12:41 -060083 @classmethod
Simon Glassf514d8f2019-08-24 07:22:54 -060084 def setUpClass(cls):
85 cls._indir = tempfile.mkdtemp(prefix='elf.')
Simon Glassc1aa66e2022-01-29 14:14:04 -070086 tools.set_input_dirs(['.'])
Simon Glassf514d8f2019-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 Glasse0e62752018-10-01 21:12:41 -060097
Simon Glassb50e5612017-11-13 18:54:54 -070098 def testAllSymbols(self):
Simon Glassb2b0df82018-07-17 13:25:26 -060099 """Test that we can obtain a symbol from the ELF file"""
Simon Glassf514d8f2019-08-24 07:22:54 -0600100 fname = self.ElfTestFile('u_boot_ucode_ptr')
Simon Glassb50e5612017-11-13 18:54:54 -0700101 syms = elf.GetSymbols(fname, [])
Simon Glass1e18a692022-01-09 20:13:37 -0700102 self.assertIn('_dt_ucode_base_size', syms)
Simon Glassb50e5612017-11-13 18:54:54 -0700103
104 def testRegexSymbols(self):
Simon Glassb2b0df82018-07-17 13:25:26 -0600105 """Test that we can obtain from the ELF file by regular expression"""
Simon Glassf514d8f2019-08-24 07:22:54 -0600106 fname = self.ElfTestFile('u_boot_ucode_ptr')
Simon Glassb50e5612017-11-13 18:54:54 -0700107 syms = elf.GetSymbols(fname, ['ucode'])
Simon Glass1e18a692022-01-09 20:13:37 -0700108 self.assertIn('_dt_ucode_base_size', syms)
Simon Glassb50e5612017-11-13 18:54:54 -0700109 syms = elf.GetSymbols(fname, ['missing'])
Simon Glass1e18a692022-01-09 20:13:37 -0700110 self.assertNotIn('_dt_ucode_base_size', syms)
Simon Glassb50e5612017-11-13 18:54:54 -0700111 syms = elf.GetSymbols(fname, ['missing', 'ucode'])
Simon Glass1e18a692022-01-09 20:13:37 -0700112 self.assertIn('_dt_ucode_base_size', syms)
Simon Glassb50e5612017-11-13 18:54:54 -0700113
Simon Glass19790632017-11-13 18:55:01 -0700114 def testMissingFile(self):
Simon Glassb2b0df82018-07-17 13:25:26 -0600115 """Test that a missing file is detected"""
Simon Glass19790632017-11-13 18:55:01 -0700116 entry = FakeEntry(10)
Simon Glassf55382b2018-06-01 09:38:13 -0600117 section = FakeSection()
Simon Glass19790632017-11-13 18:55:01 -0700118 with self.assertRaises(ValueError) as e:
Simon Glass8d2ef3e2022-02-11 13:23:21 -0700119 elf.LookupAndWriteSymbols('missing-file', entry, section)
Simon Glass19790632017-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 Glassb2b0df82018-07-17 13:25:26 -0600124 """Test a symbol which extends outside the entry area is detected"""
Simon Glass19790632017-11-13 18:55:01 -0700125 entry = FakeEntry(10)
Simon Glassf55382b2018-06-01 09:38:13 -0600126 section = FakeSection()
Simon Glass1542c8b2019-08-24 07:22:56 -0600127 elf_fname = self.ElfTestFile('u_boot_binman_syms')
Simon Glass19790632017-11-13 18:55:01 -0700128 with self.assertRaises(ValueError) as e:
Simon Glass8d2ef3e2022-02-11 13:23:21 -0700129 elf.LookupAndWriteSymbols(elf_fname, entry, section)
Simon Glass19790632017-11-13 18:55:01 -0700130 self.assertIn('entry_path has offset 4 (size 8) but the contents size '
131 'is a', str(e.exception))
132
133 def testMissingImageStart(self):
Simon Glassb2b0df82018-07-17 13:25:26 -0600134 """Test that we detect a missing __image_copy_start symbol
135
136 This is needed to mark the start of the image. Without it we cannot
137 locate the offset of a binman symbol within the image.
138 """
Simon Glass19790632017-11-13 18:55:01 -0700139 entry = FakeEntry(10)
Simon Glassf55382b2018-06-01 09:38:13 -0600140 section = FakeSection()
Simon Glass8dc60f92019-08-24 07:22:58 -0600141 elf_fname = self.ElfTestFile('u_boot_binman_syms_bad')
Simon Glass8d2ef3e2022-02-11 13:23:21 -0700142 elf.LookupAndWriteSymbols(elf_fname, entry, section)
Simon Glass19790632017-11-13 18:55:01 -0700143
144 def testBadSymbolSize(self):
Simon Glassb2b0df82018-07-17 13:25:26 -0600145 """Test that an attempt to use an 8-bit symbol are detected
146
147 Only 32 and 64 bits are supported, since we need to store an offset
148 into the image.
149 """
Simon Glass19790632017-11-13 18:55:01 -0700150 entry = FakeEntry(10)
Simon Glassf55382b2018-06-01 09:38:13 -0600151 section = FakeSection()
Simon Glasse9d2ee32019-08-24 07:22:57 -0600152 elf_fname =self.ElfTestFile('u_boot_binman_syms_size')
Simon Glass19790632017-11-13 18:55:01 -0700153 with self.assertRaises(ValueError) as e:
Simon Glass8d2ef3e2022-02-11 13:23:21 -0700154 elf.LookupAndWriteSymbols(elf_fname, entry, section)
Simon Glass19790632017-11-13 18:55:01 -0700155 self.assertIn('has size 1: only 4 and 8 are supported',
156 str(e.exception))
157
158 def testNoValue(self):
Simon Glassb2b0df82018-07-17 13:25:26 -0600159 """Test the case where we have no value for the symbol
160
161 This should produce -1 values for all thress symbols, taking up the
162 first 16 bytes of the image.
163 """
Simon Glassb87064c2019-08-24 07:23:05 -0600164 entry = FakeEntry(24)
Simon Glassf55382b2018-06-01 09:38:13 -0600165 section = FakeSection(sym_value=None)
Simon Glass1542c8b2019-08-24 07:22:56 -0600166 elf_fname = self.ElfTestFile('u_boot_binman_syms')
Simon Glass8d2ef3e2022-02-11 13:23:21 -0700167 elf.LookupAndWriteSymbols(elf_fname, entry, section)
Simon Glassc1aa66e2022-01-29 14:14:04 -0700168 self.assertEqual(tools.get_bytes(255, 20) + tools.get_bytes(ord('a'), 4),
Simon Glassc6c10e72019-05-17 22:00:46 -0600169 entry.data)
Simon Glass19790632017-11-13 18:55:01 -0700170
171 def testDebug(self):
Simon Glassb2b0df82018-07-17 13:25:26 -0600172 """Check that enabling debug in the elf module produced debug output"""
Simon Glass9f297b02019-07-20 12:23:36 -0600173 try:
Simon Glassf3385a52022-01-29 14:14:15 -0700174 tout.init(tout.DEBUG)
Simon Glass9f297b02019-07-20 12:23:36 -0600175 entry = FakeEntry(20)
176 section = FakeSection()
Simon Glass1542c8b2019-08-24 07:22:56 -0600177 elf_fname = self.ElfTestFile('u_boot_binman_syms')
Simon Glass9f297b02019-07-20 12:23:36 -0600178 with test_util.capture_sys_output() as (stdout, stderr):
Simon Glass8d2ef3e2022-02-11 13:23:21 -0700179 elf.LookupAndWriteSymbols(elf_fname, entry, section)
Simon Glass9f297b02019-07-20 12:23:36 -0600180 self.assertTrue(len(stdout.getvalue()) > 0)
181 finally:
Simon Glassf3385a52022-01-29 14:14:15 -0700182 tout.init(tout.WARNING)
Simon Glass19790632017-11-13 18:55:01 -0700183
Simon Glassf58558a2019-07-08 13:18:34 -0600184 def testMakeElf(self):
185 """Test for the MakeElf function"""
186 outdir = tempfile.mkdtemp(prefix='elf.')
187 expected_text = b'1234'
188 expected_data = b'wxyz'
189 elf_fname = os.path.join(outdir, 'elf')
Simon Glass9d44a7e2019-08-24 07:22:45 -0600190 bin_fname = os.path.join(outdir, 'bin')
Simon Glassf58558a2019-07-08 13:18:34 -0600191
192 # Make an Elf file and then convert it to a fkat binary file. This
193 # should produce the original data.
194 elf.MakeElf(elf_fname, expected_text, expected_data)
Simon Glassc1aa66e2022-01-29 14:14:04 -0700195 objcopy, args = tools.get_target_compile_tool('objcopy')
Alper Nebi Yasak1e4687a2020-09-06 14:46:05 +0300196 args += ['-O', 'binary', elf_fname, bin_fname]
Simon Glassd9800692022-01-29 14:14:05 -0700197 stdout = command.output(objcopy, *args)
Simon Glassf58558a2019-07-08 13:18:34 -0600198 with open(bin_fname, 'rb') as fd:
199 data = fd.read()
200 self.assertEqual(expected_text + expected_data, data)
201 shutil.rmtree(outdir)
202
Simon Glassd8d40742019-07-08 13:18:35 -0600203 def testDecodeElf(self):
204 """Test for the MakeElf function"""
205 if not elf.ELF_TOOLS:
206 self.skipTest('Python elftools not available')
207 outdir = tempfile.mkdtemp(prefix='elf.')
208 expected_text = b'1234'
209 expected_data = b'wxyz'
210 elf_fname = os.path.join(outdir, 'elf')
211 elf.MakeElf(elf_fname, expected_text, expected_data)
Simon Glassc1aa66e2022-01-29 14:14:04 -0700212 data = tools.read_file(elf_fname)
Simon Glassd8d40742019-07-08 13:18:35 -0600213
214 load = 0xfef20000
215 entry = load + 2
216 expected = expected_text + expected_data
217 self.assertEqual(elf.ElfInfo(expected, load, entry, len(expected)),
218 elf.DecodeElf(data, 0))
219 self.assertEqual(elf.ElfInfo(b'\0\0' + expected[2:],
220 load, entry, len(expected)),
221 elf.DecodeElf(data, load + 2))
Simon Glassf514d8f2019-08-24 07:22:54 -0600222 shutil.rmtree(outdir)
Simon Glassd8d40742019-07-08 13:18:35 -0600223
Simon Glass056f0ef2021-11-03 21:09:16 -0600224 def testEmbedData(self):
225 """Test for the GetSymbolFileOffset() function"""
226 if not elf.ELF_TOOLS:
227 self.skipTest('Python elftools not available')
228
229 fname = self.ElfTestFile('embed_data')
230 offset = elf.GetSymbolFileOffset(fname, ['embed_start', 'embed_end'])
231 start = offset['embed_start'].offset
232 end = offset['embed_end'].offset
Simon Glassc1aa66e2022-01-29 14:14:04 -0700233 data = tools.read_file(fname)
Simon Glass056f0ef2021-11-03 21:09:16 -0600234 embed_data = data[start:end]
235 expect = struct.pack('<III', 0x1234, 0x5678, 0)
236 self.assertEqual(expect, embed_data)
237
238 def testEmbedFail(self):
239 """Test calling GetSymbolFileOffset() without elftools"""
240 try:
241 old_val = elf.ELF_TOOLS
242 elf.ELF_TOOLS = False
243 fname = self.ElfTestFile('embed_data')
244 with self.assertRaises(ValueError) as e:
245 elf.GetSymbolFileOffset(fname, ['embed_start', 'embed_end'])
Simon Glass17b4ffc2022-03-05 20:18:57 -0700246 self.assertIn("Python: No module named 'elftools'",
Simon Glass056f0ef2021-11-03 21:09:16 -0600247 str(e.exception))
248 finally:
249 elf.ELF_TOOLS = old_val
250
251 def testEmbedDataNoSym(self):
252 """Test for GetSymbolFileOffset() getting no symbols"""
253 if not elf.ELF_TOOLS:
254 self.skipTest('Python elftools not available')
255
256 fname = self.ElfTestFile('embed_data')
257 offset = elf.GetSymbolFileOffset(fname, ['missing_sym'])
258 self.assertEqual({}, offset)
259
Simon Glass17b4ffc2022-03-05 20:18:57 -0700260 def test_read_loadable_segments(self):
261 """Test for read_loadable_segments()"""
Simon Glass4d38dd72022-02-08 11:49:55 -0700262 if not elf.ELF_TOOLS:
263 self.skipTest('Python elftools not available')
264 fname = self.ElfTestFile('embed_data')
Simon Glass17b4ffc2022-03-05 20:18:57 -0700265 segments, entry = elf.read_loadable_segments(tools.read_file(fname))
Simon Glass4d38dd72022-02-08 11:49:55 -0700266
267 def test_read_segments_fail(self):
Simon Glass17b4ffc2022-03-05 20:18:57 -0700268 """Test for read_loadable_segments() without elftools"""
Simon Glass4d38dd72022-02-08 11:49:55 -0700269 try:
270 old_val = elf.ELF_TOOLS
271 elf.ELF_TOOLS = False
272 fname = self.ElfTestFile('embed_data')
273 with self.assertRaises(ValueError) as e:
Simon Glass17b4ffc2022-03-05 20:18:57 -0700274 elf.read_loadable_segments(tools.read_file(fname))
275 self.assertIn("Python: No module named 'elftools'",
Simon Glass4d38dd72022-02-08 11:49:55 -0700276 str(e.exception))
277 finally:
278 elf.ELF_TOOLS = old_val
279
280 def test_read_segments_bad_data(self):
Simon Glass17b4ffc2022-03-05 20:18:57 -0700281 """Test for read_loadable_segments() with an invalid ELF file"""
Simon Glass4d38dd72022-02-08 11:49:55 -0700282 fname = self.ElfTestFile('embed_data')
283 with self.assertRaises(ValueError) as e:
Simon Glass17b4ffc2022-03-05 20:18:57 -0700284 elf.read_loadable_segments(tools.get_bytes(100, 100))
Simon Glass4d38dd72022-02-08 11:49:55 -0700285 self.assertIn('Magic number does not match', str(e.exception))
286
Simon Glass40def8a2022-03-18 19:19:49 -0600287 def test_get_file_offset(self):
288 """Test GetFileOffset() gives the correct file offset for a symbol"""
289 fname = self.ElfTestFile('embed_data')
290 syms = elf.GetSymbols(fname, ['embed'])
291 addr = syms['embed'].address
292 offset = elf.GetFileOffset(fname, addr)
293 data = tools.read_file(fname)
294
295 # Just use the first 4 bytes and assume it is little endian
296 embed_data = data[offset:offset + 4]
297 embed_value = struct.unpack('<I', embed_data)[0]
298 self.assertEqual(0x1234, embed_value)
299
300 def test_get_file_offset_fail(self):
301 """Test calling GetFileOffset() without elftools"""
302 try:
303 old_val = elf.ELF_TOOLS
304 elf.ELF_TOOLS = False
305 fname = self.ElfTestFile('embed_data')
306 with self.assertRaises(ValueError) as e:
307 elf.GetFileOffset(fname, 0)
308 self.assertIn("Python: No module named 'elftools'",
309 str(e.exception))
310 finally:
311 elf.ELF_TOOLS = old_val
312
313 def test_get_symbol_from_address(self):
314 """Test GetSymbolFromAddress()"""
315 fname = self.ElfTestFile('elf_sections')
316 sym_name = 'calculate'
317 syms = elf.GetSymbols(fname, [sym_name])
318 addr = syms[sym_name].address
319 sym = elf.GetSymbolFromAddress(fname, addr)
320 self.assertEqual(sym_name, sym)
321
322 def test_get_symbol_from_address_fail(self):
323 """Test calling GetSymbolFromAddress() without elftools"""
324 try:
325 old_val = elf.ELF_TOOLS
326 elf.ELF_TOOLS = False
327 fname = self.ElfTestFile('embed_data')
328 with self.assertRaises(ValueError) as e:
329 elf.GetSymbolFromAddress(fname, 0x1000)
330 self.assertIn("Python: No module named 'elftools'",
331 str(e.exception))
332 finally:
333 elf.ELF_TOOLS = old_val
334
Simon Glass19790632017-11-13 18:55:01 -0700335
Simon Glassb50e5612017-11-13 18:54:54 -0700336if __name__ == '__main__':
337 unittest.main()