oss-fuzz/infra/base-images/msan-builder/patch_build.py

139 lines
3.9 KiB
Python
Executable File

#!/usr/bin/env python
# Copyright 2017 Google Inc.
#
# 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
#
# http://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.
#
################################################################################
from __future__ import print_function
import argparse
import os
import re
import shutil
import subprocess
import sys
INSTRUMENTED_LIBRARIES_DIRNAME = 'instrumented_libraries'
MSAN_LIBS_PATH = '/msan'
def IsElf(file_path):
"""Whether if the file is an elf file."""
with open(file_path) as f:
return f.read(4) == '\x7fELF'
def Ldd(binary_path):
"""Run ldd on a file."""
try:
output = subprocess.check_output(['ldd', binary_path], stderr=subprocess.STDOUT)
except subprocess.CalledProcessError:
print('Failed to call ldd on', binary_path, file=sys.stderr)
return []
libs = []
OUTPUT_PATTERN = re.compile(r'\s*([^\s]+)\s*=>\s*([^\s]+)')
for line in output.splitlines():
match = OUTPUT_PATTERN.match(line)
if not match:
continue
libs.append((match.group(1), match.group(2)))
return libs
def FindLib(path):
"""Find instrumented version of lib."""
candidate_path = os.path.join(MSAN_LIBS_PATH, path[1:])
if os.path.exists(candidate_path):
return candidate_path
for lib_dir in os.listdir(MSAN_LIBS_PATH):
candidate_path = os.path.join(MSAN_LIBS_PATH, lib_dir, path[1:])
if os.path.exists(candidate_path):
return candidate_path
return None
def PatchBinary(binary_path, instrumented_dir):
"""Patch binary to link to instrumented libs."""
extra_rpaths = set()
for name, path in Ldd(binary_path):
if not os.path.isabs(path):
continue
instrumented_path = FindLib(path)
if not instrumented_path:
print('WARNING: Instrumented library not found for', path,
file=sys.stderr)
continue
target_path = os.path.join(instrumented_dir, path[1:])
if not os.path.exists(target_path):
print('Copying instrumented lib to', target_path)
target_dir = os.path.dirname(target_path)
if not os.path.exists(target_dir):
os.makedirs(target_dir)
shutil.copy2(instrumented_path, target_path)
extra_rpaths.add(
os.path.join('$ORIGIN', INSTRUMENTED_LIBRARIES_DIRNAME,
os.path.dirname(path[1:])))
if not extra_rpaths:
return
existing_rpaths = subprocess.check_output(
['patchelf', '--print-rpath', binary_path]).strip()
processed_rpaths = ':'.join(extra_rpaths)
if existing_rpaths:
processed_rpaths += ':' + existing_rpaths
print('Patching rpath for', binary_path, 'from', existing_rpaths, 'to',
processed_rpaths)
subprocess.check_call(
['patchelf', '--force-rpath', '--set-rpath',
processed_rpaths, binary_path])
def PatchBuild(output_directory):
"""Patch build to use msan libs."""
instrumented_dir = os.path.join(output_directory,
INSTRUMENTED_LIBRARIES_DIRNAME)
os.mkdir(instrumented_dir)
for root_dir, _, filenames in os.walk(output_directory):
for filename in filenames:
file_path = os.path.join(root_dir, filename)
if not IsElf(file_path):
continue
PatchBinary(file_path, instrumented_dir)
def main():
parser = argparse.ArgumentParser('patch_build.py', description='MSan build patcher.')
parser.add_argument('output_dir', help='Output directory.')
args = parser.parse_args()
PatchBuild(os.path.abspath(args.output_dir))
if __name__ == '__main__':
main()