blob: 403ca2b4123adb72220b17c12bdbdefd22c94e56 [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 Glassf58558a2019-07-08 13:18:34 -060013import command
Simon Glassb50e5612017-11-13 18:54:54 -070014import elf
Simon Glassc3f94542018-07-06 10:27:34 -060015import test_util
Simon Glasse0e62752018-10-01 21:12:41 -060016import tools
Simon Glass9f297b02019-07-20 12:23:36 -060017import 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
48 def LookupSymbol(self, name, weak, msg):
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 Glassb2b0df82018-07-17 13:25:26 -060052
Simon Glass53e22bf2019-08-24 07:22:53 -060053def BuildElfTestFiles(target_dir):
54 """Build ELF files used for testing in binman
55
56 This compiles and links the test files into the specified directory. It the
57 Makefile and source files in the binman test/ directory.
58
59 Args:
60 target_dir: Directory to put the files into
61 """
62 if not os.path.exists(target_dir):
63 os.mkdir(target_dir)
64 testdir = os.path.join(binman_dir, 'test')
65
66 # If binman is involved from the main U-Boot Makefile the -r and -R
67 # flags are set in MAKEFLAGS. This prevents this Makefile from working
68 # correctly. So drop any make flags here.
69 if 'MAKEFLAGS' in os.environ:
70 del os.environ['MAKEFLAGS']
71 tools.Run('make', '-C', target_dir, '-f',
72 os.path.join(testdir, 'Makefile'), 'SRC=%s/' % testdir,
73 'bss_data', 'u_boot_ucode_ptr')
74
75
Simon Glassb50e5612017-11-13 18:54:54 -070076class TestElf(unittest.TestCase):
Simon Glasse0e62752018-10-01 21:12:41 -060077 @classmethod
Simon Glassf514d8f2019-08-24 07:22:54 -060078 def setUpClass(cls):
79 cls._indir = tempfile.mkdtemp(prefix='elf.')
Simon Glasse0e62752018-10-01 21:12:41 -060080 tools.SetInputDirs(['.'])
Simon Glassf514d8f2019-08-24 07:22:54 -060081 BuildElfTestFiles(cls._indir)
82
83 @classmethod
84 def tearDownClass(cls):
85 if cls._indir:
86 shutil.rmtree(cls._indir)
87
88 @classmethod
89 def ElfTestFile(cls, fname):
90 return os.path.join(cls._indir, fname)
Simon Glasse0e62752018-10-01 21:12:41 -060091
Simon Glassb50e5612017-11-13 18:54:54 -070092 def testAllSymbols(self):
Simon Glassb2b0df82018-07-17 13:25:26 -060093 """Test that we can obtain a symbol from the ELF file"""
Simon Glassf514d8f2019-08-24 07:22:54 -060094 fname = self.ElfTestFile('u_boot_ucode_ptr')
Simon Glassb50e5612017-11-13 18:54:54 -070095 syms = elf.GetSymbols(fname, [])
96 self.assertIn('.ucode', syms)
97
98 def testRegexSymbols(self):
Simon Glassb2b0df82018-07-17 13:25:26 -060099 """Test that we can obtain from the ELF file by regular expression"""
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, ['ucode'])
102 self.assertIn('.ucode', syms)
103 syms = elf.GetSymbols(fname, ['missing'])
104 self.assertNotIn('.ucode', syms)
105 syms = elf.GetSymbols(fname, ['missing', 'ucode'])
106 self.assertIn('.ucode', syms)
107
Simon Glass19790632017-11-13 18:55:01 -0700108 def testMissingFile(self):
Simon Glassb2b0df82018-07-17 13:25:26 -0600109 """Test that a missing file is detected"""
Simon Glass19790632017-11-13 18:55:01 -0700110 entry = FakeEntry(10)
Simon Glassf55382b2018-06-01 09:38:13 -0600111 section = FakeSection()
Simon Glass19790632017-11-13 18:55:01 -0700112 with self.assertRaises(ValueError) as e:
Simon Glassf55382b2018-06-01 09:38:13 -0600113 syms = elf.LookupAndWriteSymbols('missing-file', entry, section)
Simon Glass19790632017-11-13 18:55:01 -0700114 self.assertIn("Filename 'missing-file' not found in input path",
115 str(e.exception))
116
117 def testOutsideFile(self):
Simon Glassb2b0df82018-07-17 13:25:26 -0600118 """Test a symbol which extends outside the entry area is detected"""
Simon Glass19790632017-11-13 18:55:01 -0700119 entry = FakeEntry(10)
Simon Glassf55382b2018-06-01 09:38:13 -0600120 section = FakeSection()
Simon Glass19790632017-11-13 18:55:01 -0700121 elf_fname = os.path.join(binman_dir, 'test', 'u_boot_binman_syms')
122 with self.assertRaises(ValueError) as e:
Simon Glassf55382b2018-06-01 09:38:13 -0600123 syms = elf.LookupAndWriteSymbols(elf_fname, entry, section)
Simon Glass19790632017-11-13 18:55:01 -0700124 self.assertIn('entry_path has offset 4 (size 8) but the contents size '
125 'is a', str(e.exception))
126
127 def testMissingImageStart(self):
Simon Glassb2b0df82018-07-17 13:25:26 -0600128 """Test that we detect a missing __image_copy_start symbol
129
130 This is needed to mark the start of the image. Without it we cannot
131 locate the offset of a binman symbol within the image.
132 """
Simon Glass19790632017-11-13 18:55:01 -0700133 entry = FakeEntry(10)
Simon Glassf55382b2018-06-01 09:38:13 -0600134 section = FakeSection()
Simon Glass19790632017-11-13 18:55:01 -0700135 elf_fname = os.path.join(binman_dir, 'test', 'u_boot_binman_syms_bad')
Simon Glassf55382b2018-06-01 09:38:13 -0600136 self.assertEqual(elf.LookupAndWriteSymbols(elf_fname, entry, section),
Simon Glass19790632017-11-13 18:55:01 -0700137 None)
138
139 def testBadSymbolSize(self):
Simon Glassb2b0df82018-07-17 13:25:26 -0600140 """Test that an attempt to use an 8-bit symbol are detected
141
142 Only 32 and 64 bits are supported, since we need to store an offset
143 into the image.
144 """
Simon Glass19790632017-11-13 18:55:01 -0700145 entry = FakeEntry(10)
Simon Glassf55382b2018-06-01 09:38:13 -0600146 section = FakeSection()
Simon Glass19790632017-11-13 18:55:01 -0700147 elf_fname = os.path.join(binman_dir, 'test', 'u_boot_binman_syms_size')
148 with self.assertRaises(ValueError) as e:
Simon Glassf55382b2018-06-01 09:38:13 -0600149 syms = elf.LookupAndWriteSymbols(elf_fname, entry, section)
Simon Glass19790632017-11-13 18:55:01 -0700150 self.assertIn('has size 1: only 4 and 8 are supported',
151 str(e.exception))
152
153 def testNoValue(self):
Simon Glassb2b0df82018-07-17 13:25:26 -0600154 """Test the case where we have no value for the symbol
155
156 This should produce -1 values for all thress symbols, taking up the
157 first 16 bytes of the image.
158 """
Simon Glass19790632017-11-13 18:55:01 -0700159 entry = FakeEntry(20)
Simon Glassf55382b2018-06-01 09:38:13 -0600160 section = FakeSection(sym_value=None)
Simon Glass19790632017-11-13 18:55:01 -0700161 elf_fname = os.path.join(binman_dir, 'test', 'u_boot_binman_syms')
Simon Glassf55382b2018-06-01 09:38:13 -0600162 syms = elf.LookupAndWriteSymbols(elf_fname, entry, section)
Simon Glassc6c10e72019-05-17 22:00:46 -0600163 self.assertEqual(tools.GetBytes(255, 16) + tools.GetBytes(ord('a'), 4),
164 entry.data)
Simon Glass19790632017-11-13 18:55:01 -0700165
166 def testDebug(self):
Simon Glassb2b0df82018-07-17 13:25:26 -0600167 """Check that enabling debug in the elf module produced debug output"""
Simon Glass9f297b02019-07-20 12:23:36 -0600168 try:
169 tout.Init(tout.DEBUG)
170 entry = FakeEntry(20)
171 section = FakeSection()
172 elf_fname = os.path.join(binman_dir, 'test', 'u_boot_binman_syms')
173 with test_util.capture_sys_output() as (stdout, stderr):
174 syms = elf.LookupAndWriteSymbols(elf_fname, entry, section)
175 self.assertTrue(len(stdout.getvalue()) > 0)
176 finally:
177 tout.Init(tout.WARNING)
Simon Glass19790632017-11-13 18:55:01 -0700178
Simon Glassf58558a2019-07-08 13:18:34 -0600179 def testMakeElf(self):
180 """Test for the MakeElf function"""
181 outdir = tempfile.mkdtemp(prefix='elf.')
182 expected_text = b'1234'
183 expected_data = b'wxyz'
184 elf_fname = os.path.join(outdir, 'elf')
Simon Glass9d44a7e2019-08-24 07:22:45 -0600185 bin_fname = os.path.join(outdir, 'bin')
Simon Glassf58558a2019-07-08 13:18:34 -0600186
187 # Make an Elf file and then convert it to a fkat binary file. This
188 # should produce the original data.
189 elf.MakeElf(elf_fname, expected_text, expected_data)
190 stdout = command.Output('objcopy', '-O', 'binary', elf_fname, bin_fname)
191 with open(bin_fname, 'rb') as fd:
192 data = fd.read()
193 self.assertEqual(expected_text + expected_data, data)
194 shutil.rmtree(outdir)
195
Simon Glassd8d40742019-07-08 13:18:35 -0600196 def testDecodeElf(self):
197 """Test for the MakeElf function"""
198 if not elf.ELF_TOOLS:
199 self.skipTest('Python elftools not available')
200 outdir = tempfile.mkdtemp(prefix='elf.')
201 expected_text = b'1234'
202 expected_data = b'wxyz'
203 elf_fname = os.path.join(outdir, 'elf')
204 elf.MakeElf(elf_fname, expected_text, expected_data)
205 data = tools.ReadFile(elf_fname)
206
207 load = 0xfef20000
208 entry = load + 2
209 expected = expected_text + expected_data
210 self.assertEqual(elf.ElfInfo(expected, load, entry, len(expected)),
211 elf.DecodeElf(data, 0))
212 self.assertEqual(elf.ElfInfo(b'\0\0' + expected[2:],
213 load, entry, len(expected)),
214 elf.DecodeElf(data, load + 2))
Simon Glassf514d8f2019-08-24 07:22:54 -0600215 shutil.rmtree(outdir)
Simon Glassd8d40742019-07-08 13:18:35 -0600216
Simon Glass19790632017-11-13 18:55:01 -0700217
Simon Glassb50e5612017-11-13 18:54:54 -0700218if __name__ == '__main__':
219 unittest.main()