2014-07-30 13:54:45 +00:00
|
|
|
// Copyright 2014 the V8 project authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
|
|
#ifndef V8_COMPILER_PIPELINE_H_
|
|
|
|
#define V8_COMPILER_PIPELINE_H_
|
|
|
|
|
2019-09-13 16:11:40 +00:00
|
|
|
#include <memory>
|
|
|
|
|
2015-02-06 09:00:40 +00:00
|
|
|
// Clients of this interface shouldn't depend on lots of compiler internals.
|
|
|
|
// Do not include anything from src/compiler here!
|
2019-05-24 13:51:59 +00:00
|
|
|
#include "src/common/globals.h"
|
2017-10-13 13:24:26 +00:00
|
|
|
#include "src/objects/code.h"
|
2014-07-30 13:54:45 +00:00
|
|
|
|
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
2015-10-02 16:55:12 +00:00
|
|
|
|
2018-07-06 08:58:43 +00:00
|
|
|
struct AssemblerOptions;
|
2018-04-04 20:30:34 +00:00
|
|
|
class OptimizedCompilationInfo;
|
2022-03-17 06:30:29 +00:00
|
|
|
class TurbofanCompilationJob;
|
Profile-guided optimization of builtins
Design doc:
https://docs.google.com/document/d/1szInbXZfaErWW70d30hJsOLL0Es-l5_g8d2rXm1ZBqI/edit?usp=sharing
V8 can already collect data about how many times each basic block in the
builtins is run. This change enables using that data for profile-guided
optimization. New comments in BUILD.gn describe how to use this feature.
A few implementation details worth mentioning, which aren't covered in
the design doc:
- BasicBlockProfilerData currently contains an array of RPO numbers.
However, this array is always just [0, 1, 2, 3, ...], so this change
removes that array. A new DCHECK in BasicBlockInstrumentor::Instrument
ensures that the removal is valid.
- RPO numbers, while useful for printing data that matches with the
stringified schedule, are not useful for matching profiling data with
blocks that haven't been scheduled yet. This change adds a new array
of block IDs in BasicBlockProfilerData, so that block counters can be
used for PGO.
- Basic block counters need to be written to a file so that they can be
provided to a subsequent run of mksnapshot, but the design doc doesn't
specify the transfer format or what file is used. In this change, I
propose using the existing v8.log file for that purpose. Block count
records look like this:
block,TestLessThanHandler,37,29405
This line indicates that block ID 37 in TestLessThanHandler was run
29405 times. If multiple lines refer to the same block, the reader
adds them all together. I like this format because it's easy to use:
- V8 already has robust logic for creating the log file, naming it to
avoid conflicts in multi-process situations, etc.
- Line order doesn't matter, and interleaved writes from various
logging sources are fine, given that V8 writes each line atomically.
- Combining multiple sources of profiling data is as simple as
concatenating their v8.log files together.
- It is a good idea to avoid making any changes based on profiling data
if the function being compiled doesn't match the one that was
profiled, since it is common to use profiling data downloaded from a
central lab which is updated only periodically. To check whether a
function matches, I propose using a hash of the Graph state right
before scheduling. This might be stricter than necessary, as some
changes to the function might be small enough that the profile data is
still relevant, but I'd rather err on the side of not making incorrect
changes. This hash is also written to the v8.log file, in a line that
looks like this:
builtin_hash,LdaZeroHandler,3387822046
Bug: v8:10470
Change-Id: I429e5ce5efa94e01e7489deb3996012cf860cf13
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2220765
Commit-Queue: Seth Brenith <seth.brenith@microsoft.com>
Reviewed-by: Ross McIlroy <rmcilroy@chromium.org>
Reviewed-by: Tobias Tebbi <tebbi@chromium.org>
Cr-Commit-Position: refs/heads/master@{#69008}
2020-07-16 16:37:08 +00:00
|
|
|
class ProfileDataFromFile;
|
2015-10-02 16:55:12 +00:00
|
|
|
class RegisterConfiguration;
|
|
|
|
|
2017-06-29 01:14:44 +00:00
|
|
|
namespace wasm {
|
2022-04-20 17:00:04 +00:00
|
|
|
class AssemblerBufferCache;
|
2021-09-07 10:14:39 +00:00
|
|
|
struct CompilationEnv;
|
2018-05-24 09:52:43 +00:00
|
|
|
struct FunctionBody;
|
2019-03-12 11:00:28 +00:00
|
|
|
struct WasmCompilationResult;
|
2019-01-21 11:57:22 +00:00
|
|
|
struct WasmModule;
|
2021-09-07 10:14:39 +00:00
|
|
|
class WireBytesStorage;
|
2017-06-29 01:14:44 +00:00
|
|
|
} // namespace wasm
|
|
|
|
|
2014-07-30 13:54:45 +00:00
|
|
|
namespace compiler {
|
|
|
|
|
2014-11-17 14:46:41 +00:00
|
|
|
class CallDescriptor;
|
2014-07-30 13:54:45 +00:00
|
|
|
class Graph;
|
2014-11-17 12:36:58 +00:00
|
|
|
class InstructionSequence;
|
2019-12-14 19:04:20 +00:00
|
|
|
class JSGraph;
|
2019-01-29 12:50:53 +00:00
|
|
|
class JSHeapBroker;
|
2018-05-08 16:09:47 +00:00
|
|
|
class MachineGraph;
|
2018-05-18 14:04:36 +00:00
|
|
|
class NodeOriginTable;
|
2014-07-30 13:54:45 +00:00
|
|
|
class Schedule;
|
2016-04-26 12:46:03 +00:00
|
|
|
class SourcePositionTable;
|
2021-02-24 13:49:09 +00:00
|
|
|
struct WasmLoopInfo;
|
2014-07-30 13:54:45 +00:00
|
|
|
|
2016-05-03 14:15:03 +00:00
|
|
|
class Pipeline : public AllStatic {
|
2014-07-30 13:54:45 +00:00
|
|
|
public:
|
2018-07-03 09:13:55 +00:00
|
|
|
// Returns a new compilation job for the given JavaScript function.
|
2022-03-17 06:30:29 +00:00
|
|
|
static V8_EXPORT_PRIVATE std::unique_ptr<TurbofanCompilationJob>
|
2020-11-30 10:19:47 +00:00
|
|
|
NewCompilationJob(Isolate* isolate, Handle<JSFunction> function,
|
|
|
|
CodeKind code_kind, bool has_script,
|
2021-01-20 12:07:43 +00:00
|
|
|
BytecodeOffset osr_offset = BytecodeOffset::None(),
|
2020-11-30 10:19:47 +00:00
|
|
|
JavaScriptFrame* osr_frame = nullptr);
|
2014-07-30 13:54:45 +00:00
|
|
|
|
2018-11-14 16:38:59 +00:00
|
|
|
// Run the pipeline for the WebAssembly compilation info.
|
2019-01-21 11:57:22 +00:00
|
|
|
static void GenerateCodeForWasmFunction(
|
2021-09-07 10:14:39 +00:00
|
|
|
OptimizedCompilationInfo* info, wasm::CompilationEnv* env,
|
|
|
|
const wasm::WireBytesStorage* wire_bytes_storage, MachineGraph* mcgraph,
|
2021-06-21 11:20:13 +00:00
|
|
|
CallDescriptor* call_descriptor, SourcePositionTable* source_positions,
|
|
|
|
NodeOriginTable* node_origins, wasm::FunctionBody function_body,
|
|
|
|
const wasm::WasmModule* module, int function_index,
|
2022-04-20 17:00:04 +00:00
|
|
|
std::vector<compiler::WasmLoopInfo>* loop_infos,
|
|
|
|
wasm::AssemblerBufferCache* buffer_cache);
|
2014-07-30 13:54:45 +00:00
|
|
|
|
2018-09-04 09:41:36 +00:00
|
|
|
// Run the pipeline on a machine graph and generate code.
|
2019-03-12 11:00:28 +00:00
|
|
|
static wasm::WasmCompilationResult GenerateCodeForWasmNativeStub(
|
2021-06-21 11:20:13 +00:00
|
|
|
CallDescriptor* call_descriptor, MachineGraph* mcgraph, CodeKind kind,
|
2021-10-04 15:49:12 +00:00
|
|
|
const char* debug_name, const AssemblerOptions& assembler_options,
|
2018-11-14 15:23:40 +00:00
|
|
|
SourcePositionTable* source_positions = nullptr);
|
|
|
|
|
2019-06-18 14:24:59 +00:00
|
|
|
// Returns a new compilation job for a wasm heap stub.
|
2022-03-17 06:30:29 +00:00
|
|
|
static std::unique_ptr<TurbofanCompilationJob> NewWasmHeapStubCompilationJob(
|
2021-06-21 11:20:13 +00:00
|
|
|
Isolate* isolate, CallDescriptor* call_descriptor,
|
|
|
|
std::unique_ptr<Zone> zone, Graph* graph, CodeKind kind,
|
|
|
|
std::unique_ptr<char[]> debug_name, const AssemblerOptions& options,
|
2018-09-04 09:41:36 +00:00
|
|
|
SourcePositionTable* source_positions = nullptr);
|
|
|
|
|
2018-12-12 14:25:17 +00:00
|
|
|
// Run the pipeline on a machine graph and generate code.
|
2018-06-19 08:09:09 +00:00
|
|
|
static MaybeHandle<Code> GenerateCodeForCodeStub(
|
2017-10-26 13:00:43 +00:00
|
|
|
Isolate* isolate, CallDescriptor* call_descriptor, Graph* graph,
|
2020-08-05 11:48:03 +00:00
|
|
|
JSGraph* jsgraph, SourcePositionTable* source_positions, CodeKind kind,
|
2021-08-12 11:17:00 +00:00
|
|
|
const char* debug_name, Builtin builtin, const AssemblerOptions& options,
|
Profile-guided optimization of builtins
Design doc:
https://docs.google.com/document/d/1szInbXZfaErWW70d30hJsOLL0Es-l5_g8d2rXm1ZBqI/edit?usp=sharing
V8 can already collect data about how many times each basic block in the
builtins is run. This change enables using that data for profile-guided
optimization. New comments in BUILD.gn describe how to use this feature.
A few implementation details worth mentioning, which aren't covered in
the design doc:
- BasicBlockProfilerData currently contains an array of RPO numbers.
However, this array is always just [0, 1, 2, 3, ...], so this change
removes that array. A new DCHECK in BasicBlockInstrumentor::Instrument
ensures that the removal is valid.
- RPO numbers, while useful for printing data that matches with the
stringified schedule, are not useful for matching profiling data with
blocks that haven't been scheduled yet. This change adds a new array
of block IDs in BasicBlockProfilerData, so that block counters can be
used for PGO.
- Basic block counters need to be written to a file so that they can be
provided to a subsequent run of mksnapshot, but the design doc doesn't
specify the transfer format or what file is used. In this change, I
propose using the existing v8.log file for that purpose. Block count
records look like this:
block,TestLessThanHandler,37,29405
This line indicates that block ID 37 in TestLessThanHandler was run
29405 times. If multiple lines refer to the same block, the reader
adds them all together. I like this format because it's easy to use:
- V8 already has robust logic for creating the log file, naming it to
avoid conflicts in multi-process situations, etc.
- Line order doesn't matter, and interleaved writes from various
logging sources are fine, given that V8 writes each line atomically.
- Combining multiple sources of profiling data is as simple as
concatenating their v8.log files together.
- It is a good idea to avoid making any changes based on profiling data
if the function being compiled doesn't match the one that was
profiled, since it is common to use profiling data downloaded from a
central lab which is updated only periodically. To check whether a
function matches, I propose using a hash of the Graph state right
before scheduling. This might be stricter than necessary, as some
changes to the function might be small enough that the profile data is
still relevant, but I'd rather err on the side of not making incorrect
changes. This hash is also written to the v8.log file, in a line that
looks like this:
builtin_hash,LdaZeroHandler,3387822046
Bug: v8:10470
Change-Id: I429e5ce5efa94e01e7489deb3996012cf860cf13
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2220765
Commit-Queue: Seth Brenith <seth.brenith@microsoft.com>
Reviewed-by: Ross McIlroy <rmcilroy@chromium.org>
Reviewed-by: Tobias Tebbi <tebbi@chromium.org>
Cr-Commit-Position: refs/heads/master@{#69008}
2020-07-16 16:37:08 +00:00
|
|
|
const ProfileDataFromFile* profile_data);
|
2015-08-25 12:56:50 +00:00
|
|
|
|
2018-07-03 09:13:55 +00:00
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
// The following methods are for testing purposes only. Avoid production use.
|
|
|
|
// ---------------------------------------------------------------------------
|
2016-04-27 12:38:19 +00:00
|
|
|
|
2019-05-02 12:17:06 +00:00
|
|
|
// Run the pipeline on JavaScript bytecode and generate code. If requested,
|
|
|
|
// hands out the heap broker on success, transferring its ownership to the
|
|
|
|
// caller.
|
2019-03-27 11:56:38 +00:00
|
|
|
V8_EXPORT_PRIVATE static MaybeHandle<Code> GenerateCodeForTesting(
|
2019-01-29 12:50:53 +00:00
|
|
|
OptimizedCompilationInfo* info, Isolate* isolate,
|
2019-05-02 12:17:06 +00:00
|
|
|
std::unique_ptr<JSHeapBroker>* out_broker = nullptr);
|
2014-11-17 12:36:58 +00:00
|
|
|
|
2015-08-28 09:02:09 +00:00
|
|
|
// Run the pipeline on a machine graph and generate code. If {schedule} is
|
|
|
|
// {nullptr}, then compute a new schedule for code generation.
|
2018-06-19 08:09:09 +00:00
|
|
|
V8_EXPORT_PRIVATE static MaybeHandle<Code> GenerateCodeForTesting(
|
2018-04-04 20:30:34 +00:00
|
|
|
OptimizedCompilationInfo* info, Isolate* isolate,
|
|
|
|
CallDescriptor* call_descriptor, Graph* graph,
|
2018-09-04 11:55:58 +00:00
|
|
|
const AssemblerOptions& options, Schedule* schedule = nullptr);
|
2014-11-17 14:46:41 +00:00
|
|
|
|
2018-07-03 09:13:55 +00:00
|
|
|
// Run just the register allocator phases.
|
2022-06-10 14:25:33 +00:00
|
|
|
V8_EXPORT_PRIVATE static void AllocateRegistersForTesting(
|
2018-07-03 09:13:55 +00:00
|
|
|
const RegisterConfiguration* config, InstructionSequence* sequence,
|
2020-07-14 19:57:45 +00:00
|
|
|
bool use_fast_register_allocator, bool run_verifier);
|
2018-07-03 09:13:55 +00:00
|
|
|
|
2015-08-28 09:02:09 +00:00
|
|
|
private:
|
2016-05-03 14:15:03 +00:00
|
|
|
DISALLOW_IMPLICIT_CONSTRUCTORS(Pipeline);
|
2014-07-30 13:54:45 +00:00
|
|
|
};
|
2014-11-14 16:44:38 +00:00
|
|
|
|
|
|
|
} // namespace compiler
|
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|
2014-07-30 13:54:45 +00:00
|
|
|
|
|
|
|
#endif // V8_COMPILER_PIPELINE_H_
|