blob: 4d27312c5989b7c91e0fa975d24eb7ea1cbe2691 [file] [log] [blame]
# Copyright 2022 The Chromium Authors
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
from __future__ import annotations
import argparse
import enum
import logging
from typing import TYPE_CHECKING, Dict, Optional, Sequence, Set
from crossbench import cli_helper
from crossbench import path as pth
from crossbench.config import ConfigEnum
from crossbench.helper.path_finder import TraceconvFinder
from crossbench.probes.chromium_probe import ChromiumProbe
from crossbench.probes.probe import (ProbeConfigParser, ProbeContext, ProbeKeyT,
ResultLocation)
from crossbench.probes.results import ProbeResult
if TYPE_CHECKING:
from crossbench.browsers.browser import Browser
from crossbench.runner.run import Run
# TODO: go over these again and clean the categories.
MINIMAL_CONFIG = frozenset((
"blink.user_timing",
"toplevel",
"v8",
"v8.execute",
))
DEVTOOLS_TRACE_CONFIG = frozenset((
"blink.console",
"blink.user_timing",
"devtools.timeline",
"disabled-by-default-devtools.screenshot",
"disabled-by-default-devtools.timeline",
"disabled-by-default-devtools.timeline.frame",
"disabled-by-default-devtools.timeline.layers",
"disabled-by-default-devtools.timeline.picture",
"disabled-by-default-devtools.timeline.stack",
"disabled-by-default-lighthouse",
"disabled-by-default-v8.compile",
"disabled-by-default-v8.cpu_profiler",
"disabled-by-default-v8.cpu_profiler.hires"
"latencyInfo",
"toplevel",
"v8.execute",
))
V8_TRACE_CONFIG = frozenset((
"blink",
"blink.user_timing",
"browser",
"cc",
"disabled-by-default-ipc.flow",
"disabled-by-default-power",
"disabled-by-default-v8.compile",
"disabled-by-default-v8.cpu_profiler",
"disabled-by-default-v8.cpu_profiler.hires",
"disabled-by-default-v8.gc",
"disabled-by-default-v8.inspector",
"disabled-by-default-v8.runtime",
"disabled-by-default-v8.runtime_stats",
"disabled-by-default-v8.runtime_stats_sampling",
"disabled-by-default-v8.stack_trace",
"disabled-by-default-v8.turbofan",
"disabled-by-default-v8.wasm.detailed",
"disabled-by-default-v8.wasm.turbofan",
"gpu",
"io",
"ipc",
"latency",
"latencyInfo",
"loading",
"log",
"mojom",
"navigation",
"net",
"netlog",
"toplevel",
"toplevel.flow",
"v8",
"v8.execute",
"wayland",
))
V8_GC_STATS_TRACE_CONFIG = V8_TRACE_CONFIG | frozenset(
("disabled-by-default-v8.gc_stats",))
TRACE_PRESETS: Dict[str, frozenset[str]] = {
"minimal": MINIMAL_CONFIG,
"devtools": DEVTOOLS_TRACE_CONFIG,
"v8": V8_TRACE_CONFIG,
"v8-gc-stats": V8_GC_STATS_TRACE_CONFIG,
}
@enum.unique
class RecordMode(ConfigEnum):
CONTINUOUSLY = ("record-continuously",
"Record until the trace buffer is full.")
UNTIL_FULL = ("record-until-full", "Record until the user ends the trace. "
"The trace buffer is a fixed size and we use it as "
"a ring buffer during recording.")
AS_MUCH_AS_POSSIBLE = ("record-as-much-as-possible",
"Record until the trace buffer is full, "
"but with a huge buffer size.")
TRACE_TO_CONSOLE = ("trace-to-console",
"Echo to console. Events are discarded.")
@enum.unique
class RecordFormat(ConfigEnum):
JSON = ("json", "Old about://tracing compatible file format.")
PROTO = ("proto", "New https://ui.perfetto.dev/ compatible format")
def parse_trace_config_file_path(value: str) -> pth.LocalPath:
data = cli_helper.parse_json_file(value)
if "trace_config" not in data:
raise argparse.ArgumentTypeError("Missing 'trace_config' property.")
cli_helper.parse_positive_int(
data.get("startup_duration", "0"), "for 'startup_duration'")
if "result_file" in data:
raise argparse.ArgumentTypeError(
"Explicit 'result_file' is not allowed with crossbench. "
"--probe=tracing sets a results location automatically.")
config = data["trace_config"]
if "included_categories" not in config and (
"excluded_categories" not in config) and ("memory_dump_config"
not in config):
raise argparse.ArgumentTypeError(
"Empty trace config: no trace categories or memory dumps configured.")
RecordMode.parse(config.get("record_mode", RecordMode.CONTINUOUSLY))
return pth.LocalPath(value)
ANDROID_TRACE_CONFIG_PATH = pth.RemotePath(
"/data/local/chrome-trace-config.json")
class TracingProbe(ChromiumProbe):
"""
Chromium-only Probe to collect tracing / perfetto data that can be used by
chrome://tracing or https://ui.perfetto.dev/.
Currently WIP
"""
NAME = "tracing"
RESULT_LOCATION = ResultLocation.BROWSER
CHROMIUM_FLAGS = ("--enable-perfetto",)
HELP_URL = "https://bit.ly/chrome-about-tracing"
@classmethod
def config_parser(cls) -> ProbeConfigParser:
parser = super().config_parser()
parser.add_argument(
"preset",
type=str,
default="minimal",
choices=TRACE_PRESETS.keys(),
help=("Use predefined trace categories, "
f"see source {__file__} for more details."))
parser.add_argument(
"categories",
is_list=True,
default=[],
type=str,
help=f"A list of trace categories to enable.\n{cls.HELP_URL}")
parser.add_argument(
"trace_config",
type=parse_trace_config_file_path,
help=("Sets Chromium's --trace-config-file to the given json config.\n"
"https://bit.ly/chromium-memory-startup-tracing "))
parser.add_argument(
"startup_duration",
default=0,
type=cli_helper.parse_positive_zero_int,
help=("Stop recording tracing after a given number of seconds. "
"Use 0 (default) for unlimited recording time."))
parser.add_argument(
"record_mode",
default=RecordMode.CONTINUOUSLY,
type=RecordMode,
help="")
parser.add_argument(
"record_format",
default=RecordFormat.PROTO,
type=RecordFormat,
help=("Choose between 'json' or the default 'proto' format. "
"Perfetto proto output is converted automatically to the "
"legacy json format."))
parser.add_argument(
"traceconv",
default=None,
type=cli_helper.parse_file_path,
help=(
"Path to the 'traceconv.py' helper to convert "
"'.proto' traces to legacy '.json'. "
"If not specified, tries to find it in a v8 or chromium checkout."))
return parser
def __init__(self,
preset: Optional[str] = None,
categories: Optional[Sequence[str]] = None,
trace_config: Optional[pth.LocalPath] = None,
startup_duration: int = 0,
record_mode: RecordMode = RecordMode.CONTINUOUSLY,
record_format: RecordFormat = RecordFormat.PROTO,
traceconv: Optional[pth.RemotePath] = None) -> None:
super().__init__()
self._trace_config: Optional[pth.LocalPath] = trace_config
self._categories: Set[str] = set(categories or MINIMAL_CONFIG)
self._preset: Optional[str] = preset
if preset:
self._categories.update(TRACE_PRESETS[preset])
if self._trace_config:
if self._categories != set(MINIMAL_CONFIG):
raise argparse.ArgumentTypeError(
"TracingProbe requires either a list of "
"trace categories or a trace_config file.")
self._categories = set()
self._startup_duration: int = startup_duration
self._record_mode: RecordMode = record_mode
self._record_format: RecordFormat = record_format
self._traceconv: Optional[pth.RemotePath] = traceconv
@property
def key(self) -> ProbeKeyT:
return super().key + (("preset", self._preset),
("categories", tuple(self._categories)),
("startup_duration", self._startup_duration),
("record_mode", str(self._record_mode)),
("record_format", str(self._record_format)),
("traceconv", str(self._traceconv)))
@property
def result_path_name(self) -> str:
return f"trace.{self._record_format.value}" # pylint: disable=no-member
@property
def traceconv(self) -> Optional[pth.RemotePath]:
return self._traceconv
@property
def record_format(self) -> RecordFormat:
return self._record_format
def attach(self, browser: Browser) -> None:
assert browser.attributes.is_chromium_based
flags = browser.flags
flags.update(self.CHROMIUM_FLAGS)
# Force proto file so we can convert it to legacy json as well.
flags["--trace-startup-format"] = str(self._record_format)
# pylint: disable=no-member
flags["--trace-startup-duration"] = str(self._startup_duration)
if self._trace_config:
# TODO: use ANDROID_TRACE_CONFIG_PATH
assert not browser.platform.is_android, (
"Trace config files not supported on android yet")
flags["--trace-config-file"] = str(self._trace_config.absolute())
else:
flags["--trace-startup-record-mode"] = str(self._record_mode)
assert self._categories, "No trace categories provided."
flags["--enable-tracing"] = ",".join(self._categories)
super().attach(browser)
def get_context(self, run: Run) -> TracingProbeContext:
return TracingProbeContext(self, run)
class TracingProbeContext(ProbeContext[TracingProbe]):
_traceconv: Optional[pth.RemotePath]
_record_format: RecordFormat
def setup(self) -> None:
self.session.extra_flags["--trace-startup-file"] = str(self.result_path)
self._record_format = self.probe.record_format
if self._record_format == RecordFormat.PROTO:
self._traceconv = self.probe.traceconv or TraceconvFinder(
self.browser_platform).path
else:
self._traceconv = None
def start(self) -> None:
pass
def stop(self) -> None:
pass
def teardown(self) -> ProbeResult:
if self._record_format == RecordFormat.JSON:
return self.browser_result(json=(self.result_path,))
if not self._traceconv:
logging.info(
"No traceconv binary: skipping converting proto to legacy traces")
return self.browser_result(file=(self.result_path,))
logging.info("Converting to legacy .json trace on local machine: %s",
self.result_path)
json_trace_file = self.result_path.with_suffix(".json")
self.browser_platform.sh(self._traceconv, "json", self.result_path,
json_trace_file)
return self.browser_result(
json=(json_trace_file,), trace=(self.result_path,))