Buckets:
| # Copyright 2013 The Emscripten Authors. All rights reserved. | |
| # Emscripten is available under two separate licenses, the MIT license and the | |
| # University of Illinois/NCSA Open Source License. Both these licenses can be | |
| # found in the LICENSE file. | |
| import logging | |
| import os | |
| import shlex | |
| import tempfile | |
| from . import shared | |
| from .utils import WINDOWS | |
| DEBUG = int(os.environ.get('EMCC_DEBUG', '0')) | |
| def create_response_file_contents(args): | |
| """Create response file contents based on list of arguments.""" | |
| escape_chars = ['\\', '\"'] | |
| # When calling llvm-ar on Linux and macOS, single quote characters ' should be escaped. | |
| if not WINDOWS: | |
| escape_chars += ['\''] | |
| def escape(arg): | |
| for char in escape_chars: | |
| arg = arg.replace(char, '\\' + char) | |
| return arg | |
| args = [escape(a) for a in args] | |
| contents = '' | |
| # Arguments containing spaces need to be quoted. | |
| for arg in args: | |
| if ' ' in arg: | |
| arg = '"%s"' % arg | |
| contents += arg + '\n' | |
| return contents | |
| def create_response_file(args, directory): | |
| """Route the given cmdline into a new response file and return its name.""" | |
| # Backslashes and other special chars need to be escaped in the response file. | |
| contents = create_response_file_contents(args) | |
| response_fd, response_filename = tempfile.mkstemp(prefix='emscripten_', suffix='.rsp.utf-8', dir=directory, text=True) | |
| with os.fdopen(response_fd, 'w', encoding='utf-8') as f: | |
| f.write(contents) | |
| if DEBUG: | |
| logging.warning(f'Creating response file {response_filename} with following contents: {contents}') | |
| # Register the created .rsp file to be automatically cleaned up once this | |
| # process finishes, so that caller does not have to remember to do it. | |
| shared.get_temp_files().note(response_filename) | |
| return response_filename | |
| def expand_response_file(arg): | |
| """Read a response file, and returns the list of cmdline params found in the file. | |
| The encoding that the response filename should be read with can be specified | |
| as a suffix to the file, e.g. "foo.rsp.utf-8" or "foo.rsp.cp1252". If not | |
| specified, first UTF-8 and then Python locale.getpreferredencoding() are | |
| attempted. | |
| The parameter `arg` is the command line argument to be expanded. | |
| """ | |
| if arg.startswith('@'): | |
| response_filename = arg[1:] | |
| elif arg.startswith('-Wl,@'): | |
| response_filename = arg[5:] | |
| else: | |
| response_filename = None | |
| # Is the argument is not a response file, or if the file does not exist | |
| # just return original argument. | |
| if not response_filename or not os.path.exists(response_filename): | |
| return [arg] | |
| # Guess encoding based on the file suffix | |
| components = os.path.basename(response_filename).split('.') | |
| encoding_suffix = components[-1].lower() | |
| if len(components) > 1 and (encoding_suffix.startswith(('utf', 'cp', 'iso')) or encoding_suffix in {'ascii', 'latin-1'}): | |
| guessed_encoding = encoding_suffix | |
| else: | |
| # On windows, recent version of CMake emit rsp files containing | |
| # a BOM. Using 'utf-8-sig' works on files both with and without | |
| # a BOM. | |
| guessed_encoding = 'utf-8-sig' | |
| try: | |
| # First try with the guessed encoding | |
| with open(response_filename, encoding=guessed_encoding) as f: | |
| args = f.read() | |
| except (ValueError, LookupError): # UnicodeDecodeError is a subclass of ValueError, and Python raises either a ValueError or a UnicodeDecodeError on decode errors. LookupError is raised if guessed encoding is not an encoding. | |
| if DEBUG: | |
| logging.warning(f'failed to parse response file {response_filename} with guessed encoding "{guessed_encoding}". Trying default system encoding...') | |
| # If that fails, try with the Python default locale.getpreferredencoding() | |
| with open(response_filename) as f: # noqa: PLW1514 | |
| args = f.read() | |
| args = shlex.split(args) | |
| if DEBUG: | |
| logging.warning(f'read response file {response_filename}: {args}') | |
| # Response file can be recursive so call substitute_response_files on the arguments | |
| return substitute_response_files(args) | |
| def substitute_response_files(args): | |
| """Substitute any response files found in args with their contents.""" | |
| new_args = [] | |
| for arg in args: | |
| new_args += expand_response_file(arg) | |
| return new_args | |
Xet Storage Details
- Size:
- 4.23 kB
- Xet hash:
- 44facd720107cddd62d9945be897f798ffc59ecb5494142d1aabba456efada10
·
Xet efficiently stores files, intelligently splitting them into unique chunks and accelerating uploads and downloads. More info.