blob: f00bb3ef5569542ef4703a28d90e6dffbee7a518 [file] [log] [blame]
# Copyright 2021 The Pigweed Authors
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may not
# use this file except in compliance with the License. You may obtain a copy of
# the License at
#
# https://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations under
# the License.
"""Tests for pw_symbolizer's llvm-symbolizer based symbolization."""
import os
import shutil
import subprocess
import tempfile
import unittest
import json
from pathlib import Path
import pw_symbolizer
_MODULE_PY_DIR = Path(__file__).parent.resolve()
_CPP_TEST_FILE_NAME = 'symbolizer_test.cc'
_COMPILER = 'clang++'
class TestSymbolizer(unittest.TestCase):
"""Unit tests for binary symbolization."""
def _test_symbolization_results(self, expected_symbols, symbolizer):
for expected_symbol in expected_symbols:
result = symbolizer.symbolize(expected_symbol['Address'])
self.assertEqual(result.name, expected_symbol['Expected'])
self.assertEqual(result.address, expected_symbol['Address'])
# Objects sometimes don't have a file/line number for some
# reason.
if not expected_symbol['IsObj']:
self.assertEqual(result.file, _CPP_TEST_FILE_NAME)
self.assertEqual(result.line, expected_symbol['Line'])
def _parameterized_test_symbolization(self, **llvm_symbolizer_kwargs):
"""Tests that the symbolizer can symbolize addresses properly."""
self.assertTrue('PW_PIGWEED_CIPD_INSTALL_DIR' in os.environ)
sysroot = Path(os.environ['PW_PIGWEED_CIPD_INSTALL_DIR']).joinpath(
"clang_sysroot"
)
with tempfile.TemporaryDirectory() as exe_dir:
exe_file = Path(exe_dir) / 'print_expected_symbols'
# Compiles a binary that prints symbol addresses and expected
# results as JSON.
cmd = [
_COMPILER,
_CPP_TEST_FILE_NAME,
'-gfull',
f'-ffile-prefix-map={_MODULE_PY_DIR}=',
'--sysroot=%s' % sysroot,
'-std=c++17',
'-fno-pic',
'-fno-pie',
'-no-pie',
'-o',
exe_file,
]
process = subprocess.run(
cmd,
stdout=subprocess.PIPE,
stderr=subprocess.STDOUT,
cwd=_MODULE_PY_DIR,
)
self.assertEqual(process.returncode, 0)
process = subprocess.run(
[exe_file], stdout=subprocess.PIPE, stderr=subprocess.STDOUT
)
self.assertEqual(process.returncode, 0)
expected_symbols = [
json.loads(line)
for line in process.stdout.decode().splitlines()
]
with self.subTest("non-legacy"):
symbolizer = pw_symbolizer.LlvmSymbolizer(
exe_file, **llvm_symbolizer_kwargs
)
self._test_symbolization_results(expected_symbols, symbolizer)
symbolizer.close()
with self.subTest("backwards-compability"):
# Test backwards compatibility with older versions of
# llvm-symbolizer.
symbolizer = pw_symbolizer.LlvmSymbolizer(
exe_file, force_legacy=True, **llvm_symbolizer_kwargs
)
self._test_symbolization_results(expected_symbols, symbolizer)
symbolizer.close()
def test_symbolization_default_binary(self):
self._parameterized_test_symbolization()
def test_symbolization_specified_binary(self):
location = Path(
subprocess.run(
['which', 'llvm-symbolizer'], check=True, stdout=subprocess.PIPE
)
.stdout.decode()
.strip()
)
with tempfile.TemporaryDirectory() as copy_dir:
copy_location = Path(copy_dir) / "copy-llvm-symbolizer"
shutil.copy(location, copy_location)
self._parameterized_test_symbolization(
llvm_symbolizer_binary=copy_location
)
if __name__ == '__main__':
unittest.main()