blob: 2aec99a9ac6d6b1df5d64c9a30cc9a29d35748d4 [file] [log] [blame]
# Copyright (C) 2018 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("../gn/gen_perfetto_version_header.gni")
import("../gn/perfetto.gni")
import("../gn/perfetto_check_build_deps.gni")
import("../gn/wasm.gni")
import("../protos/perfetto/trace_processor/proto_files.gni")
# Prevent that this file is accidentally included in embedder builds.
assert(enable_perfetto_ui)
ui_dir = "$root_build_dir/ui"
chrome_extension_dir = "$root_build_dir/chrome_extension"
ui_gen_dir = "$target_out_dir/gen"
nodejs_root = "../buildtools/nodejs"
nodejs_bin = rebase_path("$nodejs_root/bin", root_build_dir)
# +----------------------------------------------------------------------------+
# | The outer "ui" target to just ninja -C out/xxx ui |
# +----------------------------------------------------------------------------+
group("ui") {
deps = [
":chrome_extension_assets_dist",
":chrome_extension_bundle_dist",
":dist",
":gen_dist_file_map",
":service_worker_bundle_dist",
":test_scripts",
# IMPORTANT: Only add deps here if they are NOT part of the production UI
# (e.g., tests, extensions, ...). Any UI dep should go in the
# |ui_dist_targets| list below. The only exception is the service worker
# target, that depends on that list.
]
}
# The list of targets that produces dist/ files for the UI. This list is used
# also by the gen_dist_file_map to generate the map of hashes of all UI files,
# which is turn used by the service worker code for the offline caching.
ui_dist_targets = [
":assets_dist",
":catapult_dist",
":controller_bundle_dist",
":engine_bundle_dist",
":frontend_bundle_dist",
":index_dist",
":scss",
":typefaces_dist",
":wasm_dist",
]
# Builds the ui, but not service worker, tests and extensions.
group("dist") {
deps = ui_dist_targets
}
# A minimal page to profile the WASM engine without the all UI.
group("query") {
deps = [
":query_bundle_dist",
":query_dist",
":ui",
]
}
# +----------------------------------------------------------------------------+
# | Template used to run node binaries using the hermetic node toolchain. |
# +----------------------------------------------------------------------------+
template("node_bin") {
action(target_name) {
forward_variables_from(invoker,
[
"inputs",
"outputs",
"depfile",
])
deps = [ ":node_modules" ]
if (defined(invoker.deps)) {
deps += invoker.deps
}
script = "../gn/standalone/build_tool_wrapper.py"
_node_cmd = invoker.node_cmd
args = []
if (defined(invoker.suppress_stdout) && invoker.suppress_stdout) {
args += [ "--suppress_stdout" ]
}
if (defined(invoker.suppress_stderr) && invoker.suppress_stderr) {
args += [ "--suppress_stderr" ]
}
# Some of the node_bin rules *cough*transpile_all_ts*cough* don't
# accuratly report the output files. This means if they can run,
# change something, and not cause their dependees to rerun causing
# bugs. Adding a stamp file which is always rewritten avoids this.
# See also b/120010518
stamp_path = "$ui_dir/$target_name.node_bin.stamp"
outputs += [ stamp_path ]
args += [
"--stamp=" + rebase_path(stamp_path, root_build_dir),
"--path=$nodejs_bin",
"node",
rebase_path("node_modules/.bin/$_node_cmd", root_build_dir),
] + invoker.args
}
}
# +----------------------------------------------------------------------------+
# | Template for "sorcery" the source map resolver. |
# +----------------------------------------------------------------------------+
template("sorcery") {
node_bin(target_name) {
assert(defined(invoker.input))
assert(defined(invoker.output))
forward_variables_from(invoker, [ "deps" ])
inputs = [ invoker.input ]
outputs = [
invoker.output,
invoker.output + ".map",
]
node_cmd = "sorcery"
args = [
"-i",
rebase_path(invoker.input, root_build_dir),
"-o",
rebase_path(invoker.output, root_build_dir),
]
}
}
# +----------------------------------------------------------------------------+
# | Template for bundling js |
# +----------------------------------------------------------------------------+
template("bundle") {
node_bin(target_name) {
assert(defined(invoker.input))
assert(defined(invoker.output))
forward_variables_from(invoker, [ "deps" ])
inputs = [
invoker.input,
"rollup.config.js",
]
outputs = [
invoker.output,
invoker.output + ".map",
]
node_cmd = "rollup"
args = [
"-c",
rebase_path("rollup.config.js", root_build_dir),
rebase_path(invoker.input, root_build_dir),
"-o",
rebase_path(invoker.output, root_build_dir),
"-f",
"iife",
"-m",
"--silent",
]
}
}
# +----------------------------------------------------------------------------+
# | Bundles all *.js files together resolving CommonJS require() deps. |
# +----------------------------------------------------------------------------+
# Bundle together all js sources into a bundle.js file, that will ultimately be
# included by the .html files.
bundle("frontend_bundle") {
deps = [ ":transpile_all_ts" ]
input = "$target_out_dir/frontend/index.js"
output = "$target_out_dir/frontend_bundle.js"
}
bundle("chrome_extension_bundle") {
deps = [ ":transpile_all_ts" ]
input = "$target_out_dir/chrome_extension/index.js"
output = "$target_out_dir/chrome_extension_bundle.js"
}
bundle("controller_bundle") {
deps = [ ":transpile_all_ts" ]
input = "$target_out_dir/controller/index.js"
output = "$target_out_dir/controller_bundle.js"
}
bundle("engine_bundle") {
deps = [ ":transpile_all_ts" ]
input = "$target_out_dir/engine/index.js"
output = "$target_out_dir/engine_bundle.js"
}
bundle("service_worker_bundle") {
deps = [ ":transpile_service_worker_ts" ]
input = "$target_out_dir/service_worker/service_worker.js"
output = "$target_out_dir/service_worker.js"
}
bundle("query_bundle") {
deps = [ ":transpile_all_ts" ]
input = "$target_out_dir/query/index.js"
output = "$target_out_dir/query_bundle.js"
}
# +----------------------------------------------------------------------------+
# | Protobuf: gen rules to create .js and .d.ts files from protos. |
# +----------------------------------------------------------------------------+
node_bin("protos_to_js") {
inputs = []
foreach(proto, trace_processor_protos) {
inputs += [ "../protos/perfetto/trace_processor/$proto.proto" ]
}
inputs += [
"../protos/perfetto/common/trace_stats.proto",
"../protos/perfetto/common/tracing_service_capabilities.proto",
"../protos/perfetto/config/perfetto_config.proto",
"../protos/perfetto/ipc/consumer_port.proto",
"../protos/perfetto/ipc/wire_protocol.proto",
"../protos/perfetto/metrics/metrics.proto",
]
outputs = [ "$ui_gen_dir/protos.js" ]
node_cmd = "pbjs"
args = [
"--force-number",
"-t",
"static-module",
"-w",
"commonjs",
"-p",
rebase_path("..", root_build_dir),
"-o",
rebase_path(outputs[0], root_build_dir),
] + rebase_path(inputs, root_build_dir)
}
# Protobuf.js requires to first generate .js files from the .proto and then
# create .ts definitions for them.
node_bin("protos_to_ts") {
deps = [ ":protos_to_js" ]
inputs = [ "$ui_gen_dir/protos.js" ]
outputs = [ "$ui_gen_dir/protos.d.ts" ]
node_cmd = "pbts"
args = [
"-p",
rebase_path("..", root_build_dir),
"-o",
rebase_path(outputs[0], root_build_dir),
rebase_path(inputs[0], root_build_dir),
]
}
# +----------------------------------------------------------------------------+
# | TypeScript: transpiles all *.ts into .js |
# +----------------------------------------------------------------------------+
# Builds all .ts sources in the repo under |src|.
node_bin("transpile_all_ts") {
deps = [
":dist_symlink",
":protos_to_ts",
":version_ts_gen",
":wasm_gen",
]
inputs = [ "tsconfig.json" ]
outputs = [
"$target_out_dir/frontend/index.js",
"$target_out_dir/engine/index.js",
"$target_out_dir/controller/index.js",
"$target_out_dir/query/index.js",
"$target_out_dir/chrome_extension/index.js",
]
depfile = root_out_dir + "/tsc.d"
exec_script("../gn/standalone/glob.py",
[
"--root=" + rebase_path(".", root_build_dir),
"--filter=*.ts",
"--exclude=node_modules",
"--exclude=dist",
"--exclude=service_worker",
"--deps=obj/ui/frontend/index.js",
"--output=" + rebase_path(depfile),
],
"")
node_cmd = "tsc"
args = [
"--incremental",
"--project",
rebase_path(".", root_build_dir),
"--outDir",
rebase_path(target_out_dir, root_build_dir),
]
}
node_bin("transpile_service_worker_ts") {
deps = [
":dist_symlink",
":gen_dist_file_map",
]
inputs = [
"tsconfig.json",
"src/service_worker/service_worker.ts",
]
outputs = [ "$target_out_dir/service_worker/service_worker.js" ]
node_cmd = "tsc"
args = [
"--project",
rebase_path("src/service_worker", root_build_dir),
"--outDir",
rebase_path(target_out_dir, root_build_dir),
]
}
# +----------------------------------------------------------------------------+
# | Build css. |
# +----------------------------------------------------------------------------+
scss_root = "src/assets/perfetto.scss"
scss_srcs = [
"src/assets/analyze_page.scss",
"src/assets/common.scss",
"src/assets/details.scss",
"src/assets/metrics_page.scss",
"src/assets/modal.scss",
"src/assets/record.scss",
"src/assets/sidebar.scss",
"src/assets/topbar.scss",
"src/assets/trace_info_page.scss",
"src/assets/typefaces.scss",
]
# Build css.
node_bin("scss") {
deps = [ ":dist_symlink" ]
inputs = [ scss_root ] + scss_srcs
outputs = [ "$ui_dir/perfetto.css" ]
node_cmd = "node-sass"
args = [
"--quiet",
rebase_path(scss_root, root_build_dir),
rebase_path(outputs[0], root_build_dir),
]
}
# +----------------------------------------------------------------------------+
# | Copy rules: create the final output directory. |
# +----------------------------------------------------------------------------+
copy("index_dist") {
sources = [ "index.html" ]
outputs = [ "$ui_dir/index.html" ]
}
copy("typefaces_dist") {
sources = [
"../buildtools/typefaces/MaterialIcons.woff2",
"../buildtools/typefaces/Raleway-Regular.woff2",
"../buildtools/typefaces/Raleway-Thin.woff2",
"../buildtools/typefaces/RobotoCondensed-Light.woff2",
"../buildtools/typefaces/RobotoCondensed-Regular.woff2",
"../buildtools/typefaces/RobotoMono-Regular.woff2",
]
outputs = [ "$ui_dir/assets/{{source_file_part}}" ]
}
copy("query_dist") {
sources = [ "query.html" ]
outputs = [ "$ui_dir/query.html" ]
}
copy("assets_dist") {
sources = [
"src/assets/brand.png",
"src/assets/favicon.png",
"src/assets/logo-3d.png",
"src/assets/rec_atrace.png",
"src/assets/rec_battery_counters.png",
"src/assets/rec_board_voltage.png",
"src/assets/rec_cpu_coarse.png",
"src/assets/rec_cpu_fine.png",
"src/assets/rec_cpu_freq.png",
"src/assets/rec_cpu_voltage.png",
"src/assets/rec_ftrace.png",
"src/assets/rec_gpu_mem_total.png",
"src/assets/rec_java_heap_dump.png",
"src/assets/rec_lmk.png",
"src/assets/rec_logcat.png",
"src/assets/rec_long_trace.png",
"src/assets/rec_mem_hifreq.png",
"src/assets/rec_meminfo.png",
"src/assets/rec_native_heap_profiler.png",
"src/assets/rec_one_shot.png",
"src/assets/rec_ps_stats.png",
"src/assets/rec_ring_buf.png",
"src/assets/rec_vmstat.png",
] + [ scss_root ] + scss_srcs
outputs = [ "$ui_dir/assets/{{source_file_part}}" ]
}
copy("chrome_extension_assets_dist") {
sources = [
"src/assets/logo-128.png",
"src/chrome_extension/manifest.json",
]
outputs = [ "$chrome_extension_dir/{{source_file_part}}" ]
}
sorcery("frontend_bundle_dist") {
deps = [ ":frontend_bundle" ]
input = "$target_out_dir/frontend_bundle.js"
output = "$ui_dir/frontend_bundle.js"
}
sorcery("chrome_extension_bundle_dist") {
deps = [ ":chrome_extension_bundle" ]
input = "$target_out_dir/chrome_extension_bundle.js"
output = "$chrome_extension_dir/chrome_extension_bundle.js"
}
sorcery("controller_bundle_dist") {
deps = [ ":controller_bundle" ]
input = "$target_out_dir/controller_bundle.js"
output = "$ui_dir/controller_bundle.js"
}
sorcery("engine_bundle_dist") {
deps = [ ":engine_bundle" ]
input = "$target_out_dir/engine_bundle.js"
output = "$ui_dir/engine_bundle.js"
}
sorcery("service_worker_bundle_dist") {
deps = [ ":service_worker_bundle" ]
input = "$target_out_dir/service_worker.js"
output = "$ui_dir/service_worker.js"
}
sorcery("query_bundle_dist") {
deps = [ ":query_bundle" ]
input = "$target_out_dir/query_bundle.js"
output = "$ui_dir/query_bundle.js"
}
copy("wasm_dist") {
deps = [
"//src/trace_processor:trace_processor.wasm($wasm_toolchain)",
"//tools/trace_to_text:trace_to_text.wasm($wasm_toolchain)",
]
sources = [
"$root_build_dir/wasm/trace_processor.wasm",
"$root_build_dir/wasm/trace_to_text.wasm",
]
outputs = [ "$ui_dir/{{source_file_part}}" ]
}
copy("wasm_gen") {
deps = [
":dist_symlink",
# trace_processor
"//src/trace_processor:trace_processor.d.ts($wasm_toolchain)",
"//src/trace_processor:trace_processor.js($wasm_toolchain)",
"//src/trace_processor:trace_processor.wasm($wasm_toolchain)",
# trace_to_text
"//tools/trace_to_text:trace_to_text.d.ts($wasm_toolchain)",
"//tools/trace_to_text:trace_to_text.js($wasm_toolchain)",
"//tools/trace_to_text:trace_to_text.wasm($wasm_toolchain)",
]
sources = [
# trace_processor
"$root_build_dir/wasm/trace_processor.d.ts",
"$root_build_dir/wasm/trace_processor.js",
"$root_build_dir/wasm/trace_processor.wasm",
# trace_to_text
"$root_build_dir/wasm/trace_to_text.d.ts",
"$root_build_dir/wasm/trace_to_text.js",
"$root_build_dir/wasm/trace_to_text.wasm",
]
if (is_debug) {
sources += [
"$root_build_dir/wasm/trace_processor.wasm.map",
"$root_build_dir/wasm/trace_to_text.wasm.map",
]
}
outputs = [ "$ui_gen_dir/{{source_file_part}}" ]
}
# Copy over the vulcanized legacy trace viewer.
copy("catapult_dist") {
sources = [
"../buildtools/catapult_trace_viewer/catapult_trace_viewer.html",
"../buildtools/catapult_trace_viewer/catapult_trace_viewer.js",
]
outputs = [ "$ui_dir/assets/{{source_file_part}}" ]
}
# +----------------------------------------------------------------------------+
# | Node JS: Creates a symlink in the out directory to node_modules. |
# +----------------------------------------------------------------------------+
perfetto_check_build_deps("check_ui_deps") {
args = [ "--ui" ]
inputs = [ "package-lock.json" ]
}
# Creates a symlink from out/xxx/ui/node_modules -> ../../../ui/node_modules.
# This allows to run rollup and other node tools from the out/xxx directory.
action("node_modules_symlink") {
deps = [ ":check_ui_deps" ]
script = "../gn/standalone/build_tool_wrapper.py"
stamp_file = "$target_out_dir/.$target_name.stamp"
args = [
"--stamp",
rebase_path(stamp_file, root_build_dir),
"/bin/ln",
"-fns",
rebase_path("node_modules", target_out_dir),
rebase_path("$target_out_dir/node_modules", root_build_dir),
]
outputs = [ stamp_file ]
}
group("node_modules") {
deps = [ ":node_modules_symlink" ]
}
# Creates a symlink from //ui/dist -> ../../out/xxx/ui. Used only for
# autocompletion in IDEs. The problem this is solving is that in tsconfig.json
# we can't possibly know the path to ../../out/xxx for outDir. Instead, we set
# outDir to "./dist" and create a symlink on the first build.
action("dist_symlink") {
script = "../gn/standalone/build_tool_wrapper.py"
stamp_file = "$target_out_dir/.$target_name.stamp"
args = [
"--stamp",
rebase_path(stamp_file, root_build_dir),
"/bin/ln",
"-fns",
rebase_path(target_out_dir, "."),
rebase_path("dist", root_build_dir),
]
inputs = [ "$root_build_dir" ]
outputs = [ stamp_file ]
}
group("test_scripts") {
deps = [
":copy_tests_script",
":copy_unittests_script",
]
}
copy("copy_unittests_script") {
sources = [ "config/ui_unittests_template" ]
outputs = [ "$root_build_dir/ui_unittests" ]
}
copy("copy_tests_script") {
sources = [ "config/ui_tests_template" ]
outputs = [ "$root_build_dir/ui_tests" ]
}
# This target generates an map containing all the UI subresources and their
# hashes. This is used by the service worker code for offline caching.
# This target needs to be kept at the end of the BUILD.gn file, because of the
# get_target_outputs() call (fails otherwise due to GN's evaluation order).
action("gen_dist_file_map") {
out_file_path = "$ui_gen_dir/dist_file_map.ts"
dist_files = []
foreach(target, ui_dist_targets) {
foreach(dist_file, get_target_outputs(target)) {
dist_files += [ rebase_path(dist_file, root_build_dir) ]
}
}
deps = [ ":dist" ]
script = "../gn/standalone/write_ui_dist_file_map.py"
inputs = []
outputs = [ out_file_path ]
args = [
"--out",
rebase_path(out_file_path, root_build_dir),
"--strip",
rebase_path(ui_dir, root_build_dir),
] + dist_files
}
gen_perfetto_version_header("version_ts_gen") {
ts_out = "$ui_gen_dir/perfetto_version.ts"
}