You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			150 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			150 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| //===--- Trace.cpp - Performance tracing facilities -----------------------===//
 | |
| //
 | |
| //                     The LLVM Compiler Infrastructure
 | |
| //
 | |
| // This file is distributed under the University of Illinois Open Source
 | |
| // License. See LICENSE.TXT for details.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #include "Trace.h"
 | |
| #include "llvm/ADT/DenseSet.h"
 | |
| #include "llvm/Support/Chrono.h"
 | |
| #include "llvm/Support/FormatProviders.h"
 | |
| #include "llvm/Support/FormatVariadic.h"
 | |
| #include "llvm/Support/Threading.h"
 | |
| #include <mutex>
 | |
| 
 | |
| namespace clang {
 | |
| namespace clangd {
 | |
| namespace trace {
 | |
| using namespace llvm;
 | |
| 
 | |
| namespace {
 | |
| // The current implementation is naive: each thread writes to Out guarded by Mu.
 | |
| // Perhaps we should replace this by something that disturbs performance less.
 | |
| class JSONTracer : public EventTracer {
 | |
| public:
 | |
|   JSONTracer(raw_ostream &Out, bool Pretty)
 | |
|       : Out(Out), Sep(""), Start(std::chrono::system_clock::now()),
 | |
|         JSONFormat(Pretty ? "{0:2}" : "{0}") {
 | |
|     // The displayTimeUnit must be ns to avoid low-precision overlap
 | |
|     // calculations!
 | |
|     Out << R"({"displayTimeUnit":"ns","traceEvents":[)"
 | |
|         << "\n";
 | |
|     rawEvent("M", json::obj{
 | |
|                       {"name", "process_name"},
 | |
|                       {"args", json::obj{{"name", "clangd"}}},
 | |
|                   });
 | |
|   }
 | |
| 
 | |
|   ~JSONTracer() {
 | |
|     Out << "\n]}";
 | |
|     Out.flush();
 | |
|   }
 | |
| 
 | |
|   EndEventCallback beginSpan(const Context &Ctx,
 | |
|                              llvm::StringRef Name) override {
 | |
|     jsonEvent("B", json::obj{{"name", Name}});
 | |
| 
 | |
|     // The callback that will run when event ends.
 | |
|     return [this](json::Expr &&Args) {
 | |
|       jsonEvent("E", json::obj{{"args", std::move(Args)}});
 | |
|     };
 | |
|   }
 | |
| 
 | |
|   void instant(const Context &Ctx, llvm::StringRef Name,
 | |
|                json::obj &&Args) override {
 | |
|     jsonEvent("i", json::obj{{"name", Name}, {"args", std::move(Args)}});
 | |
|   }
 | |
| 
 | |
|   // Record an event on the current thread. ph, pid, tid, ts are set.
 | |
|   // Contents must be a list of the other JSON key/values.
 | |
|   void jsonEvent(StringRef Phase, json::obj &&Contents) {
 | |
|     uint64_t TID = get_threadid();
 | |
|     std::lock_guard<std::mutex> Lock(Mu);
 | |
|     // If we haven't already, emit metadata describing this thread.
 | |
|     if (ThreadsWithMD.insert(TID).second) {
 | |
|       SmallString<32> Name;
 | |
|       get_thread_name(Name);
 | |
|       if (!Name.empty()) {
 | |
|         rawEvent("M", json::obj{
 | |
|                           {"tid", TID},
 | |
|                           {"name", "thread_name"},
 | |
|                           {"args", json::obj{{"name", Name}}},
 | |
|                       });
 | |
|       }
 | |
|     }
 | |
|     Contents["ts"] = timestamp();
 | |
|     Contents["tid"] = TID;
 | |
|     rawEvent(Phase, std::move(Contents));
 | |
|   }
 | |
| 
 | |
| private:
 | |
|   // Record an event. ph and pid are set.
 | |
|   // Contents must be a list of the other JSON key/values.
 | |
|   void rawEvent(StringRef Phase, json::obj &&Event) /*REQUIRES(Mu)*/ {
 | |
|     // PID 0 represents the clangd process.
 | |
|     Event["pid"] = 0;
 | |
|     Event["ph"] = Phase;
 | |
|     Out << Sep << formatv(JSONFormat, json::Expr(std::move(Event)));
 | |
|     Sep = ",\n";
 | |
|   }
 | |
| 
 | |
|   double timestamp() {
 | |
|     using namespace std::chrono;
 | |
|     return duration<double, std::micro>(system_clock::now() - Start).count();
 | |
|   }
 | |
| 
 | |
|   std::mutex Mu;
 | |
|   raw_ostream &Out /*GUARDED_BY(Mu)*/;
 | |
|   const char *Sep /*GUARDED_BY(Mu)*/;
 | |
|   DenseSet<uint64_t> ThreadsWithMD /*GUARDED_BY(Mu)*/;
 | |
|   const sys::TimePoint<> Start;
 | |
|   const char *JSONFormat;
 | |
| };
 | |
| 
 | |
| EventTracer *T = nullptr;
 | |
| } // namespace
 | |
| 
 | |
| Session::Session(EventTracer &Tracer) {
 | |
|   assert(!T && "Resetting global tracer is not allowed.");
 | |
|   T = &Tracer;
 | |
| }
 | |
| 
 | |
| Session::~Session() { T = nullptr; }
 | |
| 
 | |
| std::unique_ptr<EventTracer> createJSONTracer(llvm::raw_ostream &OS,
 | |
|                                               bool Pretty) {
 | |
|   return llvm::make_unique<JSONTracer>(OS, Pretty);
 | |
| }
 | |
| 
 | |
| void log(const Context &Ctx, const Twine &Message) {
 | |
|   if (!T)
 | |
|     return;
 | |
|   T->instant(Ctx, "Log", json::obj{{"Message", Message.str()}});
 | |
| }
 | |
| 
 | |
| Span::Span(const Context &Ctx, llvm::StringRef Name) {
 | |
|   if (!T)
 | |
|     return;
 | |
| 
 | |
|   Callback = T->beginSpan(Ctx, Name);
 | |
|   if (!Callback)
 | |
|     return;
 | |
| 
 | |
|   Args = llvm::make_unique<json::obj>();
 | |
| }
 | |
| 
 | |
| Span::~Span() {
 | |
|   if (!Callback)
 | |
|     return;
 | |
| 
 | |
|   assert(Args && "Args must be non-null if Callback is defined");
 | |
|   Callback(std::move(*Args));
 | |
| }
 | |
| 
 | |
| } // namespace trace
 | |
| } // namespace clangd
 | |
| } // namespace clang
 |