gn: fix incremental build of descriptor when changing transitive deps
This CL makes it so that GN is aware of the transitive deps of a proto
descriptor rule so that it rebuilds the descriptor when the non-root
files are touched.
We need to wrap protoc because of a bug in protoc where the depfile is
not in the right output format when generating descriptors.
Change-Id: Ib0fb95351849f0ca802e87ea54329430ec6b1ffd
diff --git a/gn/standalone/proto_library.gni b/gn/standalone/proto_library.gni
index 3c73926..f79ae07 100644
--- a/gn/standalone/proto_library.gni
+++ b/gn/standalone/proto_library.gni
@@ -125,9 +125,12 @@
sources = proto_sources
outputs = get_path_info(protogens, "abspath")
+ protoc_script = "//gn/standalone/protoc.py"
protoc_label = "//gn:protoc($host_toolchain)"
protoc_path = get_label_info(protoc_label, "root_out_dir") + "/protoc"
args = [
+ "./" + rebase_path(protoc_script, root_build_dir),
+
# Path should be rebased because |root_build_dir| for current toolchain
# may be different from |root_out_dir| of protoc built on host toolchain.
"./" + rebase_path(protoc_path, root_build_dir),
@@ -141,10 +144,14 @@
]
}
if (generate_descriptor != "") {
+ depfile = "$target_gen_dir/" +
+ rebase_path("$generate_descriptor.d", root_gen_dir)
args += [
"--include_imports",
"--descriptor_set_out",
rebase_path("$root_gen_dir/" + generate_descriptor, root_build_dir),
+ "--dependency_out",
+ rebase_path("$root_gen_dir/$generate_descriptor.d", root_build_dir),
]
}
diff --git a/gn/standalone/protoc.py b/gn/standalone/protoc.py
new file mode 100755
index 0000000..7f83bef
--- /dev/null
+++ b/gn/standalone/protoc.py
@@ -0,0 +1,54 @@
+#!/usr/bin/env python
+# Copyright (C) 2019 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.
+
+"""Script to wrap protoc execution.
+
+This script exists to work-around the bad depfile generation by protoc when
+generating descriptors."""
+
+from __future__ import print_function
+import argparse
+import os
+import sys
+import subprocess
+import tempfile
+
+def main():
+ parser = argparse.ArgumentParser()
+ parser.add_argument('--descriptor_set_out', default=None)
+ parser.add_argument('--dependency_out', default=None)
+ parser.add_argument('protoc')
+ args, remaining = parser.parse_known_args()
+
+ if args.dependency_out and args.descriptor_set_out:
+ with tempfile.NamedTemporaryFile() as t:
+ custom = [
+ '--descriptor_set_out',
+ args.descriptor_set_out,
+ '--dependency_out',
+ t.name
+ ]
+ subprocess.call([args.protoc] + custom + remaining)
+
+ dependency_data = t.read()
+
+ with open(args.dependency_out, 'w') as f:
+ f.write(args.descriptor_set_out + ":")
+ f.write(dependency_data)
+ else:
+ subprocess.call(sys.argv[1:])
+
+if __name__ == '__main__':
+ sys.exit(main())