/* * 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. */ #include "TraceBuffer.h" #include #include #include #include #include #include "android-base/utf8.h" #include "util/Files.h" namespace aapt { namespace tracebuffer { namespace { constexpr char kBegin = 'B'; constexpr char kEnd = 'E'; struct TracePoint { char type; pid_t tid; int64_t time; std::string tag; }; std::vector traces; bool enabled = true; constinit std::chrono::steady_clock::time_point startTime = {}; int64_t GetTime() noexcept { auto now = std::chrono::steady_clock::now(); if (startTime == decltype(tracebuffer::startTime){}) { startTime = now; } return std::chrono::duration_cast(now - startTime).count(); } void AddWithTime(std::string tag, char type, int64_t time) noexcept { TracePoint t = {type, getpid(), time, std::move(tag)}; traces.emplace_back(std::move(t)); } void Add(std::string tag, char type) noexcept { AddWithTime(std::move(tag), type, GetTime()); } void Flush(const std::string& basePath) { if (basePath.empty()) { return; } BeginTrace(__func__); // We can't do much here, only record that it happened. std::ostringstream s; s << basePath << aapt::file::sDirSep << "report_aapt2_" << getpid() << ".json"; FILE* f = android::base::utf8::fopen(s.str().c_str(), "a"); if (f == nullptr) { return; } // Wrap the trace in a JSON array [] to make Chrome/Perfetto UI handle it. char delimiter = '['; for (const TracePoint& trace : traces) { fprintf(f, "%c{\"ts\" : \"%" PRIu64 "\", \"ph\" : \"%c\", \"tid\" : \"%d\" , \"pid\" : \"%d\", \"name\" : \"%s\" }\n", delimiter, trace.time, trace.type, 0, trace.tid, trace.tag.c_str()); delimiter = ','; } if (!traces.empty()) { fprintf(f, "]"); } fclose(f); traces.clear(); } } // namespace } // namespace tracebuffer void BeginTrace(std::string tag) { if (!tracebuffer::enabled) return; tracebuffer::Add(std::move(tag), tracebuffer::kBegin); } void EndTrace(std::string tag) { if (!tracebuffer::enabled) return; tracebuffer::Add(std::move(tag), tracebuffer::kEnd); } bool Trace::enable(bool value) { return tracebuffer::enabled = value; } Trace::Trace(const char* tag) { if (!tracebuffer::enabled) return; tag_.assign(tag); tracebuffer::Add(tag_, tracebuffer::kBegin); } Trace::Trace(std::string tag) : tag_(std::move(tag)) { if (!tracebuffer::enabled) return; tracebuffer::Add(tag_, tracebuffer::kBegin); } template std::string makeTag(std::string_view tag, const SpanOfStrings& args) { std::ostringstream s; s << tag; if (!args.empty()) { for (const auto& arg : args) { s << ' '; s << arg; } } return std::move(s).str(); } Trace::Trace(std::string_view tag, const std::vector& args) { if (!tracebuffer::enabled) return; tag_ = makeTag(tag, args); tracebuffer::Add(tag_, tracebuffer::kBegin); } Trace::~Trace() { if (!tracebuffer::enabled) return; tracebuffer::Add(std::move(tag_), tracebuffer::kEnd); } FlushTrace::FlushTrace(std::string_view basepath, std::string_view tag) { if (!Trace::enable(!basepath.empty())) return; basepath_.assign(basepath); tag_.assign(tag); tracebuffer::Add(tag_, tracebuffer::kBegin); } FlushTrace::FlushTrace(std::string_view basepath, std::string_view tag, const std::vector& args) { if (!Trace::enable(!basepath.empty())) return; basepath_.assign(basepath); tag_ = makeTag(tag, args); tracebuffer::Add(tag_, tracebuffer::kBegin); } FlushTrace::FlushTrace(std::string_view basepath, std::string_view tag, const std::vector& args) { if (!Trace::enable(!basepath.empty())) return; basepath_.assign(basepath); tag_ = makeTag(tag, args); tracebuffer::Add(tag_, tracebuffer::kBegin); } FlushTrace::~FlushTrace() { if (!tracebuffer::enabled) return; tracebuffer::Add(std::move(tag_), tracebuffer::kEnd); tracebuffer::Flush(basepath_); } } // namespace aapt