blob: a110618df2b30f50525ad33db6119b7ca86eec3d [file] [log] [blame]
#!/usr/bin/env python3
#
# Copyright (C) 2020 The Android Open Source Project
#
# 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.
#
import argparse
import collections
import os
import pathlib
import sys
class RustModule:
def __init__(self):
self.files = []
self.nested = collections.defaultdict(RustModule)
def emit(self, output_file, indent=""):
for (input_name, input_path) in self.files:
output_file.write(indent)
output_file.write("pub mod %s {\n" % input_name)
# Copy the contents of the input file into the output
with open(input_path, "r") as input_file:
for l in input_file:
output_file.write(indent)
output_file.write(" ")
output_file.write(l)
output_file.write(indent)
output_file.write("}\n")
for name, mod in self.nested.items():
output_file.write(indent)
output_file.write("pub mod %s {\n" % name)
mod.emit(output_file, indent + " ")
output_file.write(indent)
output_file.write("}\n")
def emit_mangled(self, output_file, indent="", prefix=""):
for (input_name, _) in self.files:
output_file.write(indent)
output_file.write("pub use %s::%s::mangled::*;\n" % (prefix, input_name))
for name, mod in self.nested.items():
new_prefix = prefix + "::" + name
mod.emit_mangled(output_file, indent, prefix=new_prefix)
def main(output, root, inputs, imports):
root_module = RustModule()
for inp in inputs:
in_rel = os.path.relpath(inp, root)
in_path = pathlib.PurePath(in_rel)
node = root_module
for part in in_path.parts[:-1]:
node = node.nested[part]
if os.path.isfile(inp):
in_name, in_ext = os.path.splitext(in_path.parts[-1])
node.files.append((in_name, inp))
with open(output, "w") as lib_rs_file:
# Enable custom attributes for #![rustfmt::skip]
lib_rs_file.write("#![feature(custom_inner_attributes)]\n")
lib_rs_file.write("#![allow(non_snake_case)]\n")
lib_rs_file.write("#![allow(missing_docs)]\n")
lib_rs_file.write("pub use binder;\n")
lib_rs_file.write("pub mod aidl {\n")
root_module.emit(lib_rs_file, indent=" ")
lib_rs_file.write("}\n")
lib_rs_file.write("pub mod mangled {\n")
root_module.emit_mangled(lib_rs_file, indent=" ", prefix="super::aidl")
for imp in imports:
lib_rs_file.write(" pub(crate) use %s::mangled::*;\n" % imp)
lib_rs_file.write("}\n")
def execute():
parser = argparse.ArgumentParser(description='Generate the top-level lib.rs.')
parser.add_argument('output', help='Path to output .rs file')
parser.add_argument('root', help='Common ancestor of all input files')
parser.add_argument('inputs', nargs='+', help='Input .rs files')
parser.add_argument('-I', '--import', action='append', dest='imports',
default=[], help='Crates to import')
args = parser.parse_args()
if args is None:
sys.exit(1)
sys.exit(main(args.output, args.root, args.inputs, args.imports))
if __name__ == "__main__":
execute()