summaryrefslogtreecommitdiffstats
path: root/js/src/jit/PerfSpewer.h
diff options
context:
space:
mode:
Diffstat (limited to 'js/src/jit/PerfSpewer.h')
-rw-r--r--js/src/jit/PerfSpewer.h209
1 files changed, 209 insertions, 0 deletions
diff --git a/js/src/jit/PerfSpewer.h b/js/src/jit/PerfSpewer.h
new file mode 100644
index 0000000000..8b04dc4419
--- /dev/null
+++ b/js/src/jit/PerfSpewer.h
@@ -0,0 +1,209 @@
+/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*-
+ * vim: set ts=8 sts=2 et sw=2 tw=80:
+ * This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+#ifndef jit_PerfSpewer_h
+#define jit_PerfSpewer_h
+
+#ifdef JS_ION_PERF
+# include <stdio.h>
+#endif
+#include "jit/BaselineFrameInfo.h"
+#include "jit/CacheIR.h"
+#include "jit/JitCode.h"
+#include "jit/LIR.h"
+#include "js/AllocPolicy.h"
+#include "js/JitCodeAPI.h"
+#include "js/Vector.h"
+#include "vm/JSScript.h"
+
+namespace js::jit {
+
+using ProfilerJitCodeVector = Vector<JS::JitCodeRecord, 0, SystemAllocPolicy>;
+
+void ResetPerfSpewer(bool enabled);
+
+struct AutoLockPerfSpewer {
+ AutoLockPerfSpewer();
+ ~AutoLockPerfSpewer();
+};
+
+class MBasicBlock;
+class MacroAssembler;
+
+bool PerfEnabled();
+
+class PerfSpewer {
+ protected:
+ struct OpcodeEntry {
+ uint32_t offset = 0;
+ unsigned opcode = 0;
+ jsbytecode* bytecodepc = nullptr;
+
+ // This string is used to replace the opcode, to define things like
+ // Prologue/Epilogue, or to add operand info.
+ UniqueChars str;
+
+ explicit OpcodeEntry(uint32_t offset_, unsigned opcode_, UniqueChars& str_,
+ jsbytecode* pc)
+ : offset(offset_), opcode(opcode_), bytecodepc(pc) {
+ str = std::move(str_);
+ }
+
+ explicit OpcodeEntry(uint32_t offset_, unsigned opcode_, UniqueChars& str_)
+ : offset(offset_), opcode(opcode_) {
+ str = std::move(str_);
+ }
+ explicit OpcodeEntry(uint32_t offset_, UniqueChars& str_)
+ : offset(offset_) {
+ str = std::move(str_);
+ }
+ explicit OpcodeEntry(uint32_t offset_, unsigned opcode_)
+ : offset(offset_), opcode(opcode_) {}
+
+ explicit OpcodeEntry(jsbytecode* pc) : bytecodepc(pc) {}
+
+ OpcodeEntry(OpcodeEntry&& copy) {
+ offset = copy.offset;
+ opcode = copy.opcode;
+ bytecodepc = copy.bytecodepc;
+ str = std::move(copy.str);
+ }
+
+ // Do not copy the UniqueChars member.
+ OpcodeEntry(OpcodeEntry& copy) = delete;
+ };
+ Vector<OpcodeEntry, 0, SystemAllocPolicy> opcodes_;
+
+ uint32_t lir_opcode_length = 0;
+ uint32_t js_opcode_length = 0;
+
+ virtual JS::JitTier GetTier() { return JS::JitTier::Other; }
+
+ virtual const char* CodeName(unsigned op) = 0;
+
+ virtual void saveJitCodeSourceInfo(JSScript* script, JitCode* code,
+ JS::JitCodeRecord* record,
+ AutoLockPerfSpewer& lock);
+
+ void saveDebugInfo(JSScript* script, JitCode* code,
+ JS::JitCodeRecord* profilerRecord,
+ AutoLockPerfSpewer& lock);
+
+ void saveProfile(JitCode* code, UniqueChars& desc, JSScript* script);
+
+ void saveJitCodeIRInfo(JitCode* code, JS::JitCodeRecord* profilerRecord,
+ AutoLockPerfSpewer& lock);
+
+ public:
+ PerfSpewer() = default;
+
+ void recordOffset(MacroAssembler& masm, const char*);
+
+ static void Init();
+
+ static void CollectJitCodeInfo(UniqueChars& function_name, JitCode* code,
+ JS::JitCodeRecord*, AutoLockPerfSpewer& lock);
+ static void CollectJitCodeInfo(UniqueChars& function_name, void* code_addr,
+ uint64_t code_size,
+ JS::JitCodeRecord* profilerRecord,
+ AutoLockPerfSpewer& lock);
+};
+
+void CollectPerfSpewerJitCodeProfile(JitCode* code, const char* msg);
+void CollectPerfSpewerJitCodeProfile(uintptr_t base, uint64_t size,
+ const char* msg);
+
+void CollectPerfSpewerWasmMap(uintptr_t base, uintptr_t size,
+ const char* filename, const char* annotation);
+void CollectPerfSpewerWasmFunctionMap(uintptr_t base, uintptr_t size,
+ const char* filename, unsigned lineno,
+ const char* funcName);
+
+class IonPerfSpewer : public PerfSpewer {
+ JS::JitTier GetTier() override { return JS::JitTier::Ion; }
+ const char* CodeName(unsigned op) override;
+
+ public:
+ void recordInstruction(MacroAssembler& masm, LInstruction* ins);
+ void saveProfile(JSContext* cx, JSScript* script, JitCode* code);
+};
+
+class BaselineInterpreterPerfSpewer : public PerfSpewer {
+ JS::JitTier GetTier() override { return JS::JitTier::Baseline; }
+ const char* CodeName(unsigned op) override;
+
+ // Do nothing, BaselineInterpreter has no source to reference.
+ void saveJitCodeSourceInfo(JSScript* script, JitCode* code,
+ JS::JitCodeRecord* record,
+ AutoLockPerfSpewer& lock) override {}
+
+ public:
+ void recordOffset(MacroAssembler& masm, JSOp op);
+ void recordOffset(MacroAssembler& masm, const char* name);
+ void saveProfile(JitCode* code);
+};
+
+class BaselinePerfSpewer : public PerfSpewer {
+ JS::JitTier GetTier() override { return JS::JitTier::Baseline; }
+ const char* CodeName(unsigned op) override;
+
+ public:
+ void recordInstruction(JSContext* cx, MacroAssembler& masm, jsbytecode* pc,
+ CompilerFrameInfo& frame);
+ void saveProfile(JSContext* cx, JSScript* script, JitCode* code);
+};
+
+class InlineCachePerfSpewer : public PerfSpewer {
+ JS::JitTier GetTier() override { return JS::JitTier::IC; }
+ const char* CodeName(unsigned op) override;
+
+ public:
+ void recordInstruction(MacroAssembler& masm, CacheOp op);
+};
+
+class BaselineICPerfSpewer : public InlineCachePerfSpewer {
+ void saveJitCodeSourceInfo(JSScript* script, JitCode* code,
+ JS::JitCodeRecord* record,
+ AutoLockPerfSpewer& lock) override {
+ // Baseline IC stubs are shared and have no source code to reference.
+ return;
+ }
+
+ public:
+ void saveProfile(JitCode* code, const char* stubName);
+};
+
+class IonICPerfSpewer : public InlineCachePerfSpewer {
+ public:
+ explicit IonICPerfSpewer(jsbytecode* pc);
+
+ void saveJitCodeSourceInfo(JSScript* script, JitCode* code,
+ JS::JitCodeRecord* record,
+ AutoLockPerfSpewer& lock) override;
+
+ void saveProfile(JSContext* cx, JSScript* script, JitCode* code,
+ const char* stubName);
+};
+
+class PerfSpewerRangeRecorder {
+ using OffsetPair = std::tuple<uint32_t, UniqueChars>;
+ Vector<OffsetPair, 0, js::SystemAllocPolicy> ranges;
+
+ MacroAssembler& masm;
+
+ void appendEntry(UniqueChars& desc);
+
+ public:
+ explicit PerfSpewerRangeRecorder(MacroAssembler& masm_) : masm(masm_){};
+ void recordOffset(const char* name);
+ void recordOffset(const char* name, JSContext* cx, JSScript* script);
+ void recordVMWrapperOffset(const char* name);
+ void collectRangesForJitCode(JitCode* code);
+};
+
+} // namespace js::jit
+
+#endif /* jit_PerfSpewer_h */