| // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file |
| // for details. All rights reserved. Use of this source code is governed by a |
| // BSD-style license that can be found in the LICENSE file. |
| |
| #ifndef RUNTIME_VM_COMPILER_H_ |
| #define RUNTIME_VM_COMPILER_H_ |
| |
| #include "vm/allocation.h" |
| #include "vm/growable_array.h" |
| #include "vm/runtime_entry.h" |
| #include "vm/thread_pool.h" |
| |
| namespace dart { |
| |
| // Forward declarations. |
| class BackgroundCompilationQueue; |
| class Class; |
| class Code; |
| class CompilationWorkQueue; |
| class FlowGraph; |
| class Function; |
| class IndirectGotoInstr; |
| class Library; |
| class ParsedFunction; |
| class QueueElement; |
| class RawInstance; |
| class Script; |
| class SequenceNode; |
| |
| bool UseKernelFrontEndFor(ParsedFunction* parsed_function); |
| |
| class CompilationPipeline : public ZoneAllocated { |
| public: |
| static CompilationPipeline* New(Zone* zone, const Function& function); |
| |
| virtual void ParseFunction(ParsedFunction* parsed_function) = 0; |
| virtual FlowGraph* BuildFlowGraph( |
| Zone* zone, |
| ParsedFunction* parsed_function, |
| const ZoneGrowableArray<const ICData*>& ic_data_array, |
| intptr_t osr_id) = 0; |
| virtual void FinalizeCompilation(FlowGraph* flow_graph) = 0; |
| virtual ~CompilationPipeline() {} |
| }; |
| |
| |
| class DartCompilationPipeline : public CompilationPipeline { |
| public: |
| virtual void ParseFunction(ParsedFunction* parsed_function); |
| |
| virtual FlowGraph* BuildFlowGraph( |
| Zone* zone, |
| ParsedFunction* parsed_function, |
| const ZoneGrowableArray<const ICData*>& ic_data_array, |
| intptr_t osr_id); |
| |
| virtual void FinalizeCompilation(FlowGraph* flow_graph); |
| }; |
| |
| |
| class IrregexpCompilationPipeline : public CompilationPipeline { |
| public: |
| IrregexpCompilationPipeline() : backtrack_goto_(NULL) {} |
| |
| virtual void ParseFunction(ParsedFunction* parsed_function); |
| |
| virtual FlowGraph* BuildFlowGraph( |
| Zone* zone, |
| ParsedFunction* parsed_function, |
| const ZoneGrowableArray<const ICData*>& ic_data_array, |
| intptr_t osr_id); |
| |
| virtual void FinalizeCompilation(FlowGraph* flow_graph); |
| |
| private: |
| IndirectGotoInstr* backtrack_goto_; |
| }; |
| |
| |
| class Compiler : public AllStatic { |
| public: |
| static const intptr_t kNoOSRDeoptId = Thread::kNoDeoptId; |
| |
| static bool IsBackgroundCompilation(); |
| // The result for a function may change if debugging gets turned on/off. |
| static bool CanOptimizeFunction(Thread* thread, const Function& function); |
| |
| // Extracts top level entities from the script and populates |
| // the class dictionary of the library. |
| // |
| // Returns Error::null() if there is no compilation error. |
| static RawError* Compile(const Library& library, const Script& script); |
| |
| // Extracts function and field symbols from the class and populates |
| // the class. |
| // |
| // Returns Error::null() if there is no compilation error. |
| static RawError* CompileClass(const Class& cls); |
| |
| // Generates code for given function and sets its code field. |
| // |
| // Returns Error::null() if there is no compilation error. |
| static RawError* CompileFunction(Thread* thread, const Function& function); |
| static RawError* ParseFunction(Thread* thread, const Function& function); |
| |
| // Generates unoptimized code if not present, current code is unchanged. |
| static RawError* EnsureUnoptimizedCode(Thread* thread, |
| const Function& function); |
| |
| // Generates optimized code for function. |
| // |
| // Returns Error::null() if there is no compilation error. |
| // If 'result_code' is not NULL, then the generated code is returned but |
| // not installed. |
| static RawError* CompileOptimizedFunction(Thread* thread, |
| const Function& function, |
| intptr_t osr_id = kNoOSRDeoptId); |
| |
| // Generates code for given parsed function (without parsing it again) and |
| // sets its code field. |
| // |
| // Returns Error::null() if there is no compilation error. |
| static RawError* CompileParsedFunction(ParsedFunction* parsed_function); |
| |
| // Generates and executes code for a given code fragment, e.g. a |
| // compile time constant expression. Returns the result returned |
| // by the fragment. |
| // |
| // The return value is either a RawInstance on success or a RawError |
| // on compilation failure. |
| static RawObject* ExecuteOnce(SequenceNode* fragment); |
| |
| // Evaluates the initializer expression of the given static field. |
| // |
| // The return value is either a RawInstance on success or a RawError |
| // on compilation failure. |
| static RawObject* EvaluateStaticInitializer(const Field& field); |
| |
| // Generates local var descriptors and sets it in 'code'. Do not call if the |
| // local var descriptor already exists. |
| static void ComputeLocalVarDescriptors(const Code& code); |
| |
| // Eagerly compiles all functions in a class. |
| // |
| // Returns Error::null() if there is no compilation error. |
| static RawError* CompileAllFunctions(const Class& cls); |
| static RawError* ParseAllFunctions(const Class& cls); |
| |
| // Notify the compiler that background (optimized) compilation has failed |
| // because the mutator thread changed the state (e.g., deoptimization, |
| // deferred loading). The background compilation may retry to compile |
| // the same function later. |
| static void AbortBackgroundCompilation(intptr_t deopt_id, const char* msg); |
| }; |
| |
| |
| // Class to run optimizing compilation in a background thread. |
| // Current implementation: one task per isolate, it dies with the owning |
| // isolate. |
| // No OSR compilation in the background compiler. |
| class BackgroundCompiler : public ThreadPool::Task { |
| public: |
| virtual ~BackgroundCompiler(); |
| |
| static void EnsureInit(Thread* thread); |
| |
| // Stops background compiler of the given isolate. |
| // TODO(turnidge): Give Stop and Disable more distinct names. |
| static void Stop(Isolate* isolate); |
| |
| static void Disable(); |
| |
| static void Enable(); |
| |
| static bool IsDisabled(); |
| |
| // Call to optimize a function in the background, enters the function in the |
| // compilation queue. |
| void CompileOptimized(const Function& function); |
| |
| void VisitPointers(ObjectPointerVisitor* visitor); |
| |
| BackgroundCompilationQueue* function_queue() const { return function_queue_; } |
| bool is_running() const { return running_; } |
| |
| private: |
| explicit BackgroundCompiler(Isolate* isolate); |
| |
| virtual void Run(); |
| |
| Isolate* isolate_; |
| bool running_; // While true, will try to read queue and compile. |
| bool* done_; // True if the thread is done. |
| Monitor* queue_monitor_; // Controls access to the queue. |
| Monitor* done_monitor_; // Notify/wait that the thread is done. |
| |
| BackgroundCompilationQueue* function_queue_; |
| |
| DISALLOW_IMPLICIT_CONSTRUCTORS(BackgroundCompiler); |
| }; |
| |
| } // namespace dart |
| |
| #endif // RUNTIME_VM_COMPILER_H_ |