blob: bcccd78c0a131eabe586e9da9bec83e213c7d4b3 [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 Glassb50e5612017-11-13 18:54:54 -07009import sys
Simon Glassf58558a2019-07-08 13:18:34 -060010import tempfile
Simon Glassb50e5612017-11-13 18:54:54 -070011import unittest
12
Simon Glass16287932020-04-17 18:09:03 -060013from binman import elf
Simon Glassbf776672020-04-17 18:09:04 -060014from patman import command
15from patman import test_util
16from patman import tools
17from patman import tout
Simon Glassb50e5612017-11-13 18:54:54 -070018
19binman_dir = os.path.dirname(os.path.realpath(sys.argv[0]))
Simon Glass19790632017-11-13 18:55:01 -070020
Simon Glass19790632017-11-13 18:55:01 -070021
22class FakeEntry:
Simon Glassb2b0df82018-07-17 13:25:26 -060023 """A fake Entry object, usedfor testing
24
25 This supports an entry with a given size.
26 """
Simon Glass19790632017-11-13 18:55:01 -070027 def __init__(self, contents_size):
28 self.contents_size = contents_size
Simon Glassc6c10e72019-05-17 22:00:46 -060029 self.data = tools.GetBytes(ord('a'), contents_size)
Simon Glass19790632017-11-13 18:55:01 -070030
31 def GetPath(self):
32 return 'entry_path'
33
Simon Glassb2b0df82018-07-17 13:25:26 -060034
Simon Glassf55382b2018-06-01 09:38:13 -060035class FakeSection:
Simon Glassb2b0df82018-07-17 13:25:26 -060036 """A fake Section object, used for testing
37
38 This has the minimum feature set needed to support testing elf functions.
39 A LookupSymbol() function is provided which returns a fake value for amu
40 symbol requested.
41 """
Simon Glass19790632017-11-13 18:55:01 -070042 def __init__(self, sym_value=1):
43 self.sym_value = sym_value
44
45 def GetPath(self):
Simon Glassf55382b2018-06-01 09:38:13 -060046 return 'section_path'
Simon Glass19790632017-11-13 18:55:01 -070047
Simon Glass870a9ea2021-01-06 21:35:15 -070048 def LookupImageSymbol(self, name, weak, msg, base_addr):
Simon Glassb2b0df82018-07-17 13:25:26 -060049 """Fake implementation which returns the same value for all symbols"""
Simon Glass19790632017-11-13 18:55:01 -070050 return self.sym_value
Simon Glassb50e5612017-11-13 18:54:54 -070051
Simon Glass870a9ea2021-01-06 21:35:15 -070052 def GetImage(self):
53 return self
Simon Glassb2b0df82018-07-17 13:25:26 -060054
Simon Glass53e22bf2019-08-24 07:22:53 -060055def BuildElfTestFiles(target_dir):
56 """Build ELF files used for testing in binman
57
58 This compiles and links the test files into the specified directory. It the
59 Makefile and source files in the binman test/ directory.
60
61 Args:
62 target_dir: Directory to put the files into
63 """
64 if not os.path.exists(target_dir):
65 os.mkdir(target_dir)
66 testdir = os.path.join(binman_dir, 'test')
67
68 # If binman is involved from the main U-Boot Makefile the -r and -R
69 # flags are set in MAKEFLAGS. This prevents this Makefile from working
70 # correctly. So drop any make flags here.
71 if 'MAKEFLAGS' in os.environ:
72 del os.environ['MAKEFLAGS']
Simon Glass2fb2cd72021-11-03 21:09:15 -060073 try:
74 tools.Run('make', '-C', target_dir, '-f',
75 os.path.join(testdir, 'Makefile'), 'SRC=%s/' % testdir)
76 except ValueError as e:
77 # The test system seems to suppress this in a strange way
78 print(e)
Simon Glass53e22bf2019-08-24 07:22:53 -060079
80
Simon Glassb50e5612017-11-13 18:54:54 -070081class TestElf(unittest.TestCase):
Simon Glasse0e62752018-10-01 21:12:41 -060082 @classmethod
Simon Glassf514d8f2019-08-24 07:22:54 -060083 def setUpClass(cls):
84 cls._indir = tempfile.mkdtemp(prefix='elf.')
Simon Glasse0e62752018-10-01 21:12:41 -060085 tools.SetInputDirs(['.'])
Simon Glassf514d8f2019-08-24 07:22:54 -060086 BuildElfTestFiles(cls._indir)
87
88 @classmethod
89 def tearDownClass(cls):
90 if cls._indir:
91 shutil.rmtree(cls._indir)
92
93 @classmethod
94 def ElfTestFile(cls, fname):
95 return os.path.join(cls._indir, fname)
Simon Glasse0e62752018-10-01 21:12:41 -060096
Simon Glassb50e5612017-11-13 18:54:54 -070097 def testAllSymbols(self):
Simon Glassb2b0df82018-07-17 13:25:26 -060098 """Test that we can obtain a symbol from the ELF file"""
Simon Glassf514d8f2019-08-24 07:22:54 -060099 fname = self.ElfTestFile('u_boot_ucode_ptr')
Simon Glassb50e5612017-11-13 18:54:54 -0700100 syms = elf.GetSymbols(fname, [])
101 self.assertIn('.ucode', syms)
102
103 def testRegexSymbols(self):
Simon Glassb2b0df82018-07-17 13:25:26 -0600104 """Test that we can obtain from the ELF file by regular expression"""
Simon Glassf514d8f2019-08-24 07:22:54 -0600105 fname = self.ElfTestFile('u_boot_ucode_ptr')
Simon Glassb50e5612017-11-13 18:54:54 -0700106 syms = elf.GetSymbols(fname, ['ucode'])
107 self.assertIn('.ucode', syms)
108 syms = elf.GetSymbols(fname, ['missing'])
109 self.assertNotIn('.ucode', syms)
110 syms = elf.GetSymbols(fname, ['missing', 'ucode'])
111 self.assertIn('.ucode', syms)
112
Simon Glass19790632017-11-13 18:55:01 -0700113 def testMissingFile(self):
Simon Glassb2b0df82018-07-17 13:25:26 -0600114 """Test that a missing file is detected"""
Simon Glass19790632017-11-13 18:55:01 -0700115 entry = FakeEntry(10)
Simon Glassf55382b2018-06-01 09:38:13 -0600116 section = FakeSection()
Simon Glass19790632017-11-13 18:55:01 -0700117 with self.assertRaises(ValueError) as e:
Simon Glassf55382b2018-06-01 09:38:13 -0600118 syms = elf.LookupAndWriteSymbols('missing-file', entry, section)
Simon Glass19790632017-11-13 18:55:01 -0700119 self.assertIn("Filename 'missing-file' not found in input path",
120 str(e.exception))
121
122 def testOutsideFile(self):
Simon Glassb2b0df82018-07-17 13:25:26 -0600123 """Test a symbol which extends outside the entry area is detected"""
Simon Glass19790632017-11-13 18:55:01 -0700124 entry = FakeEntry(10)
Simon Glassf55382b2018-06-01 09:38:13 -0600125 section = FakeSection()
Simon Glass1542c8b2019-08-24 07:22:56 -0600126 elf_fname = self.ElfTestFile('u_boot_binman_syms')
Simon Glass19790632017-11-13 18:55:01 -0700127 with self.assertRaises(ValueError) as e:
Simon Glassf55382b2018-06-01 09:38:13 -0600128 syms = elf.LookupAndWriteSymbols(elf_fname, entry, section)
Simon Glass19790632017-11-13 18:55:01 -0700129 self.assertIn('entry_path has offset 4 (size 8) but the contents size '
130 'is a', str(e.exception))
131
132 def testMissingImageStart(self):
Simon Glassb2b0df82018-07-17 13:25:26 -0600133 """Test that we detect a missing __image_copy_start symbol
134
135 This is needed to mark the start of the image. Without it we cannot
136 locate the offset of a binman symbol within the image.
137 """
Simon Glass19790632017-11-13 18:55:01 -0700138 entry = FakeEntry(10)
Simon Glassf55382b2018-06-01 09:38:13 -0600139 section = FakeSection()
Simon Glass8dc60f92019-08-24 07:22:58 -0600140 elf_fname = self.ElfTestFile('u_boot_binman_syms_bad')
Simon Glassf55382b2018-06-01 09:38:13 -0600141 self.assertEqual(elf.LookupAndWriteSymbols(elf_fname, entry, section),
Simon Glass19790632017-11-13 18:55:01 -0700142 None)
143
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 Glassf55382b2018-06-01 09:38:13 -0600154 syms = 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 Glassf55382b2018-06-01 09:38:13 -0600167 syms = elf.LookupAndWriteSymbols(elf_fname, entry, section)
Simon Glassb87064c2019-08-24 07:23:05 -0600168 self.assertEqual(tools.GetBytes(255, 20) + tools.GetBytes(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:
174 tout.Init(tout.DEBUG)
175 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):
179 syms = elf.LookupAndWriteSymbols(elf_fname, entry, section)
180 self.assertTrue(len(stdout.getvalue()) > 0)
181 finally:
182 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)
Alper Nebi Yasak1e4687a2020-09-06 14:46:05 +0300195 objcopy, args = tools.GetTargetCompileTool('objcopy')
196 args += ['-O', 'binary', elf_fname, bin_fname]
197 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)
212 data = tools.ReadFile(elf_fname)
213
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 Glass19790632017-11-13 18:55:01 -0700224
Simon Glassb50e5612017-11-13 18:54:54 -0700225if __name__ == '__main__':
226 unittest.main()