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

import os
import sys
import unittest

import elf
12
import test_util
13
import tools
14 15

binman_dir = os.path.dirname(os.path.realpath(sys.argv[0]))
16 17 18


class FakeEntry:
S
Simon Glass 已提交
19 20 21 22
    """A fake Entry object, usedfor testing

    This supports an entry with a given size.
    """
23 24 25 26 27 28 29
    def __init__(self, contents_size):
        self.contents_size = contents_size
        self.data = 'a' * contents_size

    def GetPath(self):
        return 'entry_path'

S
Simon Glass 已提交
30

31
class FakeSection:
S
Simon Glass 已提交
32 33 34 35 36 37
    """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.
    """
38 39 40 41
    def __init__(self, sym_value=1):
        self.sym_value = sym_value

    def GetPath(self):
42
        return 'section_path'
43 44

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

S
Simon Glass 已提交
48

49
class TestElf(unittest.TestCase):
50 51 52 53
    @classmethod
    def setUpClass(self):
        tools.SetInputDirs(['.'])

54
    def testAllSymbols(self):
S
Simon Glass 已提交
55
        """Test that we can obtain a symbol from the ELF file"""
56
        fname = os.path.join(binman_dir, 'test', 'u_boot_ucode_ptr')
57 58 59 60
        syms = elf.GetSymbols(fname, [])
        self.assertIn('.ucode', syms)

    def testRegexSymbols(self):
S
Simon Glass 已提交
61
        """Test that we can obtain from the ELF file by regular expression"""
62
        fname = os.path.join(binman_dir, 'test', 'u_boot_ucode_ptr')
63 64 65 66 67 68 69
        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)

70
    def testMissingFile(self):
S
Simon Glass 已提交
71
        """Test that a missing file is detected"""
72
        entry = FakeEntry(10)
73
        section = FakeSection()
74
        with self.assertRaises(ValueError) as e:
75
            syms = elf.LookupAndWriteSymbols('missing-file', entry, section)
76 77 78 79
        self.assertIn("Filename 'missing-file' not found in input path",
                      str(e.exception))

    def testOutsideFile(self):
S
Simon Glass 已提交
80
        """Test a symbol which extends outside the entry area is detected"""
81
        entry = FakeEntry(10)
82
        section = FakeSection()
83 84
        elf_fname = os.path.join(binman_dir, 'test', 'u_boot_binman_syms')
        with self.assertRaises(ValueError) as e:
85
            syms = elf.LookupAndWriteSymbols(elf_fname, entry, section)
86 87 88 89
        self.assertIn('entry_path has offset 4 (size 8) but the contents size '
                      'is a', str(e.exception))

    def testMissingImageStart(self):
S
Simon Glass 已提交
90 91 92 93 94
        """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.
        """
95
        entry = FakeEntry(10)
96
        section = FakeSection()
97
        elf_fname = os.path.join(binman_dir, 'test', 'u_boot_binman_syms_bad')
98
        self.assertEqual(elf.LookupAndWriteSymbols(elf_fname, entry, section),
99 100 101
                         None)

    def testBadSymbolSize(self):
S
Simon Glass 已提交
102 103 104 105 106
        """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.
        """
107
        entry = FakeEntry(10)
108
        section = FakeSection()
109 110
        elf_fname = os.path.join(binman_dir, 'test', 'u_boot_binman_syms_size')
        with self.assertRaises(ValueError) as e:
111
            syms = elf.LookupAndWriteSymbols(elf_fname, entry, section)
112 113 114 115
        self.assertIn('has size 1: only 4 and 8 are supported',
                      str(e.exception))

    def testNoValue(self):
S
Simon Glass 已提交
116 117 118 119 120
        """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.
        """
121
        entry = FakeEntry(20)
122
        section = FakeSection(sym_value=None)
123
        elf_fname = os.path.join(binman_dir, 'test', 'u_boot_binman_syms')
124
        syms = elf.LookupAndWriteSymbols(elf_fname, entry, section)
125
        self.assertEqual(tools.GetBytes(255, 16) + 'a' * 4, entry.data)
126 127

    def testDebug(self):
S
Simon Glass 已提交
128
        """Check that enabling debug in the elf module produced debug output"""
129 130
        elf.debug = True
        entry = FakeEntry(20)
131
        section = FakeSection()
132
        elf_fname = os.path.join(binman_dir, 'test', 'u_boot_binman_syms')
133
        with test_util.capture_sys_output() as (stdout, stderr):
134
            syms = elf.LookupAndWriteSymbols(elf_fname, entry, section)
135 136 137 138
        elf.debug = False
        self.assertTrue(len(stdout.getvalue()) > 0)


139 140
if __name__ == '__main__':
    unittest.main()