mirror of https://github.com/google/oss-fuzz.git
139 lines
3.9 KiB
Python
Executable File
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()
|