blob: c2c639f5b0aa243697a559c2a534a5fb0cfeec43 [file] [log] [blame] [edit]
#!/usr/bin/env python
# Copyright 2019 Google LLC
#
# This source code is licensed under the BSD-style license found in the
# LICENSE file in the root directory of this source tree.
import argparse
import codecs
import io
import os
import re
import sys
from itertools import chain
def key_value_pair(line):
key, value = line.split("=", 1)
# represent value as integer, if possible, otherwise as str
try:
value = int(value)
except ValueError:
pass
return key, value
parser = argparse.ArgumentParser(description='XNNPACK generator')
parser.add_argument("input", metavar="FILE", nargs=1,
help="Input file")
parser.add_argument("-D", dest="defines", metavar="KEY=VALUE", nargs="*",
type=key_value_pair, action="append",
help="Predefined variables")
parser.add_argument("-o", "--output",
help='Output file')
parser.set_defaults(defines=list())
LEADING_WHITESPACE_REGEX = re.compile(r"^\s*", flags=0)
def extract_leading_whitespace(line):
match = re.match(r"\s*", line)
return match.group(0) if match else ""
def escape(line):
output_parts = []
while "${" in line:
start_pos = line.index("${")
end_pos = line.index("}", start_pos + 2)
if start_pos != 0:
output_parts.append("\"" + line[:start_pos].replace("\"", "\\\"") + "\"")
output_parts.append("str(" + line[start_pos+2:end_pos] + ")")
line = line[end_pos+1:]
if line:
output_parts.append("\"" + line.replace("\"", "\\\"") + "\"")
return " + ".join(output_parts)
def preprocess(input_text, input_globals, input_path="codegen"):
input_lines = input_text.splitlines()
python_lines = ["from __future__ import print_function"]
blank_lines = 0
last_line = ""
last_indent = ""
# List of tuples (total_index, python_indent)
indent_stack = [("", "")]
# Indicates whether this is the first line inside Python
# code block (i.e. for, while, if, elif, else)
python_block_start = True
for i, input_line in enumerate(input_lines):
if input_line == "":
blank_lines += 1
continue
# Skip lint markers.
if 'LINT' in input_line:
continue
input_indent = extract_leading_whitespace(input_line)
if python_block_start:
assert input_indent.startswith(last_indent)
extra_python_indent = input_indent[len(last_indent):]
python_indent = indent_stack[-1][1] + extra_python_indent
indent_stack.append((input_indent, python_indent))
assert input_indent.startswith(indent_stack[-1][0])
else:
while not input_indent.startswith(indent_stack[-1][0]):
del indent_stack[-1]
python_block_start = False
python_indent = indent_stack[-1][1]
stripped_input_line = input_line.strip()
if stripped_input_line.startswith("$") and not stripped_input_line.startswith("${"):
if stripped_input_line.endswith(":"):
python_block_start = True
while blank_lines != 0:
python_lines.append(python_indent + "print(file=OUT_STREAM)")
blank_lines -= 1
python_lines.append(python_indent + stripped_input_line.replace("$", ""))
else:
assert input_line.startswith(python_indent)
while blank_lines != 0:
python_lines.append(python_indent + "print(file=OUT_STREAM)")
blank_lines -= 1
python_lines.append(python_indent + "print(%s, file=OUT_STREAM)" % escape(input_line[len(python_indent):]))
last_line = input_line
last_indent = input_indent
while blank_lines != 0:
python_lines.append(python_indent + "print(file=OUT_STREAM)")
blank_lines -= 1
exec_globals = dict(input_globals)
if sys.version_info > (3, 0):
output_stream = io.StringIO()
else:
output_stream = io.BytesIO()
exec_globals["OUT_STREAM"] = output_stream
python_bytecode = compile("\n".join(python_lines), input_path, 'exec')
exec(python_bytecode, exec_globals)
return output_stream.getvalue()
PREAMBLE = """\
// Auto-generated file. Do not edit!
// Template: {template}
// Generator: {generator}
//
"""
def main(args):
options = parser.parse_args(args)
input_text = codecs.open(options.input[0], "r", encoding="utf-8").read()
python_globals = dict(chain(*options.defines))
output_text = PREAMBLE.format(template=options.input[0], generator=sys.argv[0]) + preprocess(input_text, python_globals, options.input[0])
txt_changed = True
if os.path.exists(options.output):
with codecs.open(options.output, "r", encoding="utf-8") as output_file:
txt_changed = output_file.read() != output_text
if txt_changed:
with codecs.open(options.output, "w", encoding="utf-8") as output_file:
output_file.write(output_text)
if __name__ == "__main__":
main(sys.argv[1:])