elf_test.py 7.7 KB
Newer Older
1
# SPDX-License-Identifier: GPL-2.0+
2 3 4 5 6 7
# Copyright (c) 2017 Google, Inc
# Written by Simon Glass <sjg@chromium.org>
#
# Test for the elf module

import os
8
import shutil
9
import sys
10
import tempfile
11 12
import unittest

13
import command
14
import elf
15
import test_util
16
import tools
17
import tout
18 19

binman_dir = os.path.dirname(os.path.realpath(sys.argv[0]))
20 21 22


class FakeEntry:
S
Simon Glass 已提交
23 24 25 26
    """A fake Entry object, usedfor testing

    This supports an entry with a given size.
    """
27 28
    def __init__(self, contents_size):
        self.contents_size = contents_size
S
Simon Glass 已提交
29
        self.data = tools.GetBytes(ord('a'), contents_size)
30 31 32 33

    def GetPath(self):
        return 'entry_path'

S
Simon Glass 已提交
34

35
class FakeSection:
S
Simon Glass 已提交
36 37 38 39 40 41
    """A fake Section object, used for testing

    This has the minimum feature set needed to support testing elf functions.
    A LookupSymbol() function is provided which returns a fake value for amu
    symbol requested.
    """
42 43 44 45
    def __init__(self, sym_value=1):
        self.sym_value = sym_value

    def GetPath(self):
46
        return 'section_path'
47 48

    def LookupSymbol(self, name, weak, msg):
S
Simon Glass 已提交
49
        """Fake implementation which returns the same value for all symbols"""
50
        return self.sym_value
51

S
Simon Glass 已提交
52

53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72
def BuildElfTestFiles(target_dir):
    """Build ELF files used for testing in binman

    This compiles and links the test files into the specified directory. It the
    Makefile and source files in the binman test/ directory.

    Args:
        target_dir: Directory to put the files into
    """
    if not os.path.exists(target_dir):
        os.mkdir(target_dir)
    testdir = os.path.join(binman_dir, 'test')

    # If binman is involved from the main U-Boot Makefile the -r and -R
    # flags are set in MAKEFLAGS. This prevents this Makefile from working
    # correctly. So drop any make flags here.
    if 'MAKEFLAGS' in os.environ:
        del os.environ['MAKEFLAGS']
    tools.Run('make', '-C', target_dir, '-f',
              os.path.join(testdir, 'Makefile'), 'SRC=%s/' % testdir,
73
              'bss_data', 'u_boot_ucode_ptr', 'u_boot_no_ucode_ptr')
74 75


76
class TestElf(unittest.TestCase):
77
    @classmethod
78 79
    def setUpClass(cls):
        cls._indir = tempfile.mkdtemp(prefix='elf.')
80
        tools.SetInputDirs(['.'])
81 82 83 84 85 86 87 88 89 90
        BuildElfTestFiles(cls._indir)

    @classmethod
    def tearDownClass(cls):
        if cls._indir:
            shutil.rmtree(cls._indir)

    @classmethod
    def ElfTestFile(cls, fname):
        return os.path.join(cls._indir, fname)
91

92
    def testAllSymbols(self):
S
Simon Glass 已提交
93
        """Test that we can obtain a symbol from the ELF file"""
94
        fname = self.ElfTestFile('u_boot_ucode_ptr')
95 96 97 98
        syms = elf.GetSymbols(fname, [])
        self.assertIn('.ucode', syms)

    def testRegexSymbols(self):
S
Simon Glass 已提交
99
        """Test that we can obtain from the ELF file by regular expression"""
100
        fname = self.ElfTestFile('u_boot_ucode_ptr')
101 102 103 104 105 106 107
        syms = elf.GetSymbols(fname, ['ucode'])
        self.assertIn('.ucode', syms)
        syms = elf.GetSymbols(fname, ['missing'])
        self.assertNotIn('.ucode', syms)
        syms = elf.GetSymbols(fname, ['missing', 'ucode'])
        self.assertIn('.ucode', syms)

108
    def testMissingFile(self):
S
Simon Glass 已提交
109
        """Test that a missing file is detected"""
110
        entry = FakeEntry(10)
111
        section = FakeSection()
112
        with self.assertRaises(ValueError) as e:
113
            syms = elf.LookupAndWriteSymbols('missing-file', entry, section)
114 115 116 117
        self.assertIn("Filename 'missing-file' not found in input path",
                      str(e.exception))

    def testOutsideFile(self):
S
Simon Glass 已提交
118
        """Test a symbol which extends outside the entry area is detected"""
119
        entry = FakeEntry(10)
120
        section = FakeSection()
121 122
        elf_fname = os.path.join(binman_dir, 'test', 'u_boot_binman_syms')
        with self.assertRaises(ValueError) as e:
123
            syms = elf.LookupAndWriteSymbols(elf_fname, entry, section)
124 125 126 127
        self.assertIn('entry_path has offset 4 (size 8) but the contents size '
                      'is a', str(e.exception))

    def testMissingImageStart(self):
S
Simon Glass 已提交
128 129 130 131 132
        """Test that we detect a missing __image_copy_start symbol

        This is needed to mark the start of the image. Without it we cannot
        locate the offset of a binman symbol within the image.
        """
133
        entry = FakeEntry(10)
134
        section = FakeSection()
135
        elf_fname = os.path.join(binman_dir, 'test', 'u_boot_binman_syms_bad')
136
        self.assertEqual(elf.LookupAndWriteSymbols(elf_fname, entry, section),
137 138 139
                         None)

    def testBadSymbolSize(self):
S
Simon Glass 已提交
140 141 142 143 144
        """Test that an attempt to use an 8-bit symbol are detected

        Only 32 and 64 bits are supported, since we need to store an offset
        into the image.
        """
145
        entry = FakeEntry(10)
146
        section = FakeSection()
147 148
        elf_fname = os.path.join(binman_dir, 'test', 'u_boot_binman_syms_size')
        with self.assertRaises(ValueError) as e:
149
            syms = elf.LookupAndWriteSymbols(elf_fname, entry, section)
150 151 152 153
        self.assertIn('has size 1: only 4 and 8 are supported',
                      str(e.exception))

    def testNoValue(self):
S
Simon Glass 已提交
154 155 156 157 158
        """Test the case where we have no value for the symbol

        This should produce -1 values for all thress symbols, taking up the
        first 16 bytes of the image.
        """
159
        entry = FakeEntry(20)
160
        section = FakeSection(sym_value=None)
161
        elf_fname = os.path.join(binman_dir, 'test', 'u_boot_binman_syms')
162
        syms = elf.LookupAndWriteSymbols(elf_fname, entry, section)
S
Simon Glass 已提交
163 164
        self.assertEqual(tools.GetBytes(255, 16) + tools.GetBytes(ord('a'), 4),
                                                                  entry.data)
165 166

    def testDebug(self):
S
Simon Glass 已提交
167
        """Check that enabling debug in the elf module produced debug output"""
168 169 170 171 172 173 174 175 176 177
        try:
            tout.Init(tout.DEBUG)
            entry = FakeEntry(20)
            section = FakeSection()
            elf_fname = os.path.join(binman_dir, 'test', 'u_boot_binman_syms')
            with test_util.capture_sys_output() as (stdout, stderr):
                syms = elf.LookupAndWriteSymbols(elf_fname, entry, section)
            self.assertTrue(len(stdout.getvalue()) > 0)
        finally:
            tout.Init(tout.WARNING)
178

179 180 181 182 183 184
    def testMakeElf(self):
        """Test for the MakeElf function"""
        outdir = tempfile.mkdtemp(prefix='elf.')
        expected_text = b'1234'
        expected_data = b'wxyz'
        elf_fname = os.path.join(outdir, 'elf')
S
Simon Glass 已提交
185
        bin_fname = os.path.join(outdir, 'bin')
186 187 188 189 190 191 192 193 194 195

        # Make an Elf file and then convert it to a fkat binary file. This
        # should produce the original data.
        elf.MakeElf(elf_fname, expected_text, expected_data)
        stdout = command.Output('objcopy', '-O', 'binary', elf_fname, bin_fname)
        with open(bin_fname, 'rb') as fd:
            data = fd.read()
        self.assertEqual(expected_text + expected_data, data)
        shutil.rmtree(outdir)

196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214
    def testDecodeElf(self):
        """Test for the MakeElf function"""
        if not elf.ELF_TOOLS:
            self.skipTest('Python elftools not available')
        outdir = tempfile.mkdtemp(prefix='elf.')
        expected_text = b'1234'
        expected_data = b'wxyz'
        elf_fname = os.path.join(outdir, 'elf')
        elf.MakeElf(elf_fname, expected_text, expected_data)
        data = tools.ReadFile(elf_fname)

        load = 0xfef20000
        entry = load + 2
        expected = expected_text + expected_data
        self.assertEqual(elf.ElfInfo(expected, load, entry, len(expected)),
                         elf.DecodeElf(data, 0))
        self.assertEqual(elf.ElfInfo(b'\0\0' + expected[2:],
                                     load, entry, len(expected)),
                         elf.DecodeElf(data, load + 2))
215
        shutil.rmtree(outdir)
216

217

218 219
if __name__ == '__main__':
    unittest.main()