167 строки
5.8 KiB
Python
167 строки
5.8 KiB
Python
#!/usr/bin/env python
|
|
"""wrapper around emcc link step.
|
|
|
|
This wrapper currently serves the following purposes.
|
|
|
|
1. When building with --config=wasm the final output is multiple files, usually
|
|
at least one .js and one .wasm file. Since the cc_binary link step only
|
|
allows a single output, we must tar up the outputs into a single file.
|
|
|
|
2. Add quotes around arguments that need them in the response file to work
|
|
around a bazel quirk.
|
|
|
|
3. Ensure the external_debug_info section of the wasm points at the correct
|
|
bazel path.
|
|
"""
|
|
|
|
from __future__ import print_function
|
|
|
|
import argparse
|
|
import os
|
|
import subprocess
|
|
import sys
|
|
|
|
# Only argument should be @path/to/parameter/file
|
|
assert sys.argv[1][0] == '@', sys.argv
|
|
param_filename = sys.argv[1][1:]
|
|
param_file_args = [l.strip() for l in open(param_filename, 'r').readlines()]
|
|
|
|
# Re-write response file if needed.
|
|
if any(' ' in a for a in param_file_args):
|
|
new_param_filename = param_filename + '.modified'
|
|
with open(new_param_filename, 'w') as f:
|
|
for param in param_file_args:
|
|
if ' ' in param:
|
|
f.write('"%s"' % param)
|
|
else:
|
|
f.write(param)
|
|
f.write('\n')
|
|
sys.argv[1] = '@' + new_param_filename
|
|
|
|
emcc_py = os.path.join(os.environ['EMSCRIPTEN'], 'emcc.py')
|
|
rtn = subprocess.call([sys.executable, emcc_py] + sys.argv[1:])
|
|
if rtn != 0:
|
|
sys.exit(1)
|
|
|
|
# Parse the arguments that we gave to the linker to determine what the output
|
|
# file is named and what the output format is.
|
|
parser = argparse.ArgumentParser(add_help=False)
|
|
parser.add_argument('-o')
|
|
parser.add_argument('--oformat')
|
|
options = parser.parse_known_args(param_file_args)[0]
|
|
output_file = options.o
|
|
oformat = options.oformat
|
|
outdir = os.path.normpath(os.path.dirname(output_file))
|
|
base_name = os.path.basename(output_file)
|
|
|
|
# The output file name is the name of the build rule that was built.
|
|
# Add an appropriate file extension based on --oformat.
|
|
if oformat is not None:
|
|
base_name_split = os.path.splitext(base_name)
|
|
|
|
# If the output name has no extension, give it the appropriate extension.
|
|
if not base_name_split[1]:
|
|
os.replace(output_file, output_file + '.' + oformat)
|
|
|
|
# If the output name does have an extension and it matches the output format,
|
|
# change the base_name so it doesn't have an extension.
|
|
elif base_name_split[1] == '.' + oformat:
|
|
base_name = base_name_split[0]
|
|
|
|
# If the output name does have an extension and it does not match the output
|
|
# format, change the base_name so it doesn't have an extension and rename
|
|
# the output_file so it has the proper extension.
|
|
# Note that if you do something like name your build rule "foo.js" and pass
|
|
# "--oformat=html", emscripten will write to the same file for both the js and
|
|
# html output, overwriting the js output entirely with the html.
|
|
# Please don't do that.
|
|
else:
|
|
base_name = base_name_split[0]
|
|
os.replace(output_file, os.path.join(outdir, base_name + '.' + oformat))
|
|
|
|
files = []
|
|
extensions = [
|
|
'.js',
|
|
'.wasm',
|
|
'.wasm.map',
|
|
'.js.mem',
|
|
'.fetch.js',
|
|
'.worker.js',
|
|
'.data',
|
|
'.js.symbols',
|
|
'.wasm.debug.wasm',
|
|
'.html'
|
|
]
|
|
|
|
for ext in extensions:
|
|
filename = base_name + ext
|
|
if os.path.exists(os.path.join(outdir, filename)):
|
|
files.append(filename)
|
|
|
|
wasm_base = os.path.join(outdir, base_name + '.wasm')
|
|
if os.path.exists(wasm_base + '.debug.wasm') and os.path.exists(wasm_base):
|
|
# If we have a .wasm.debug.wasm file and a .wasm file, we need to rewrite the
|
|
# section in the .wasm file that refers to it. The path that's in there
|
|
# is the blaze output path; we want it to be just the filename.
|
|
|
|
llvm_objcopy = os.path.join(
|
|
os.environ['EM_BIN_PATH'], 'bin/llvm-objcopy')
|
|
# First, check to make sure the .wasm file has the header that needs to be
|
|
# rewritten.
|
|
rtn = subprocess.call([
|
|
llvm_objcopy,
|
|
'--dump-section=external_debug_info=/dev/null',
|
|
wasm_base], stdout=subprocess.PIPE, stderr=subprocess.PIPE)
|
|
if rtn == 0:
|
|
# If llvm-objcopy did not return an error, the external_debug_info section
|
|
# must exist, so we're good to continue.
|
|
|
|
# Next we need to convert length of the filename to LEB128.
|
|
# Start by converting the length of the filename to a bit string.
|
|
bit_string = '{0:b}'.format(len(base_name + '.wasm.debug.wasm'))
|
|
|
|
# Pad the bit string with 0s so that its length is a multiple of 7.
|
|
while len(bit_string) % 7 != 0:
|
|
bit_string = '0' + bit_string
|
|
|
|
# Break up our bit string into chunks of 7.
|
|
# We do this backwards because the final format is little-endian.
|
|
final_bytes = bytearray()
|
|
for i in reversed(range(0, len(bit_string), 7)):
|
|
binary_part = bit_string[i:i + 7]
|
|
if i != 0:
|
|
# Every chunk except the last one needs to be prepended with '1'.
|
|
# The length of each chunk is 7, so that one has an implicit '0'.
|
|
binary_part = '1' + binary_part
|
|
final_bytes.append(int(binary_part, 2))
|
|
# Finally, add the actual filename.
|
|
final_bytes.extend((base_name + '.wasm.debug.wasm').encode())
|
|
|
|
# Write our length + filename bytes to a temp file.
|
|
with open('debugsection.tmp', 'wb+') as f:
|
|
f.write(final_bytes)
|
|
f.close()
|
|
|
|
# First delete the old section.
|
|
subprocess.check_call([
|
|
llvm_objcopy,
|
|
wasm_base,
|
|
'--remove-section=external_debug_info'])
|
|
# Rewrite section with the new size and filename from the temp file.
|
|
subprocess.check_call([
|
|
llvm_objcopy,
|
|
wasm_base,
|
|
'--add-section=external_debug_info=debugsection.tmp'])
|
|
|
|
# Make sure we have at least one output file.
|
|
if not len(files):
|
|
print('emcc.py did not appear to output any known files!')
|
|
sys.exit(1)
|
|
|
|
# cc_binary must output exactly one file; put all the output files in a tarball.
|
|
cmd = ['tar', 'cf', 'tmp.tar'] + files
|
|
subprocess.check_call(cmd, cwd=outdir)
|
|
os.replace(os.path.join(outdir, 'tmp.tar'), output_file)
|
|
|
|
sys.exit(0)
|