Changes between Version 6 and Version 7 of JavaScriptCore


Ignore:
Timestamp:
Feb 15, 2013, 11:35:05 PM (12 years ago)
Author:
fpizlo@apple.com
Comment:

--

Legend:

Unmodified
Added
Removed
Modified
  • JavaScriptCore

    v6 v7  
    2222'''Baseline JIT''' kicks in for functions that are invoked at least 6 times, or take a loop at least 100 times (or some combination - like 3 invocations with 50 loop iterations total).  Note, these numbers are approximate; the actual heuristics depend on function size and current memory pressure.  The LLInt will on-stack-replace (OSR) to the JIT if it is stuck in a loop; as well all callers of the function are relinked to point to the compiled code as opposed to the LLInt prologue.  The Baseline JIT also acts as a fall-back for functions that are compiled by the optimizing JIT: if the optimized code encounters a case it cannot handle, it bails (via an OSR exit) to the Baseline JIT.  The Baseline JIT is in [http://trac.webkit.org/browser/trunk/Source/JavaScriptCore/jit jit/].  The Baseline JIT also performs sophisticated polymorphic inline caching for almost all heap accesses.
    2323
    24 Both the LLInt and Baseline JIT collect light-weight profiling information to enable speculative execution by the next tier of execution (the DFG).  Information collected includes recent values loaded into arguments, loaded from the heap, or loaded from a call return.  Additionally, all inline caching in the LLInt and Baseline JIT is engineered to enable the DFG to scrape type information easily: for example the DFG can detect that a heap access sometimes, often, or always sees a particular type just by looking at the current state of an inline cache; this can be used to determine the most profitable level of speculation.
     24Both the LLInt and Baseline JIT collect light-weight profiling information to enable speculative execution by the next tier of execution (the DFG).  Information collected includes recent values loaded into arguments, loaded from the heap, or loaded from a call return.  Additionally, all inline caching in the LLInt and Baseline JIT is engineered to enable the DFG to scrape type information easily: for example the DFG can detect that a heap access sometimes, often, or always sees a particular type just by looking at the current state of an inline cache; this can be used to determine the most profitable level of speculation.  A more thorough overview of type inference in JavaScriptCore is provided in the next section.
    2525
    2626'''DFG JIT''' kicks in for functions that are invoked at least 60 times, or that took a loop at least 1000 times.  Again, these numbers are approximate and are subject to additional heuristics.  The DFG performs aggressive type speculation based on profiling information collected by the lower tiers.  This allows it to forward-propagate type information, eliding many type checks.  Sometimes the DFG goes further and speculates on values themselves - for example it may speculate that a value loaded from the heap is always some known function in order to enable inlining.  The DFG uses deoptimization (we call it "OSR exit") to handle cases where speculation fails.  Deoptimization may be synchronous (for example, a branch that checks that the type of a value is that which was expected) or asynchronous (for example, the runtime may observe that the shape or value of some object or variable has changed in a way that contravenes assumptions made by the DFG).  The latter is referred to as "watchpointing" in the DFG codebase.  Altogether, the Baseline JIT and the DFG JIT share a two-way OSR relationship: the Baseline JIT may OSR into the DFG when a function gets hot, and the DFG may OSR to the Baseline JIT in the case of deoptimization.  Repeated OSR exits from the DFG serve as an additional profiling hint: the DFG OSR exit machinery records the reason of the exit (including potentially the values that failed speculation) as well as the frequency with which it occurred; if an exit is taken often enough then reoptimization kicks in: callers are relinked to the Baseline JIT for the affected function, more profiling is gathered, and then the DFG may be later reinvoked.  Reoptimization uses exponential back-off to defend against pathological code.  The DFG is in [http://trac.webkit.org/browser/trunk/Source/JavaScriptCore/dfg dfg/].
     
    2828At any time, functions, eval blocks, and global code in JavaScriptCore may be executing in a mix of the LLInt, Baseline JIT, and DFG.  In the extreme case of a recursive function, there may even be multiple stack frames where one frame is in the LLInt, another is in the Baseline JIT, while another still is in the DFG; even more extreme are cases where one stack frame is executing an old DFG compilation and another is executing a new DFG compilation because recompilation kicked in but execution did not yet return to the old DFG code.  These three engines are designed to maintain identical execution semantics, and so even if multiple functions in a JavaScript program are executing in a mix of these engines the only perceptible effect ought to be execution performance.
    2929
     30== Type Inference ==
     31
     32Type inference is achieved by profiling values, predicting what types operations will produce based on those profiles, inserting type checks based on the type predictions, and then attempting to construct type proofs about the types of values based on the type checks.
     33
     34Consider this example to motivate type inference, and optimization, of JavaScript:
     35
     36{{{
     37o.x * o.x + o.y * o.y
     38}}}
     39
     40Say that in the context where this code is used, 'o' is an object, and it indeed has properties 'x' and 'y', and those properties are nothing special - in particular, they are not accessors.  Let's also say that 'o.x' and 'o.y' usually return doubles, but may sometimes return integers - JavaScript does not have a built-in notion of an integer value, but for efficiency JavaScriptCore will represent most integers as int32 rather than as double.  To understand both the problem of type inference, and its solution in JavaScriptCore, it's useful to first consider what a JavaScript engine would have to do if it had no information about 'o', 'o.x', or 'o.y':
     41
     42* The expression 'o.x' has to first check if 'o' has any special handling of property access.  It may be a DOM object, and DOM objects may intercept accesses to their properties in non-obvious ways.  If it doesn't have special handling, the engine must look up the property named "x" (where "x" is literally a string) in the object.  Objects are just tables mapping strings to either values or accessors.  If it maps to an accessor, the accessor must be called.  If it isn't an accessor, then the value is returned.  If "x" is not found in 'o', then the process repeats for o's prototype.  The inference required for optimizing the object access is not covered in this section.
     43* The binary multiply operation in the expression 'o.x * o.x' has to first check the types of its operands.  Either operand may be an object, in which case its 'valueOf' method must be called.  Either operand may be a string, in which case it must be converted to a number.  Once both operands are appropriately converted to numbers (or if they were numbers already), the engine must check if they are both integers; if so, then an integer multiply is performed.  This may overflow, in which case a double multiply is performed instead.  If either operand is a double, then both are converted to doubles, and a double multiply is performed.  Thus 'o.x * o.x' may return either an integer or a double.  There is no way of proving, for the generic JavaScript multiply, what kind of number it will return and how that number will be represented.
     44* The binary addition operation in the expression 'o.x * o.x + o.y * o.y' has to proceed roughly as the multiply did, except it has to consider the possibility that its operands are strings, in which case a string concatenation is performed.  In this case, we could statically prove that this isn't the case - multiply must have returned a number.  But still, addition must perform checks for integers versus doubles on both operands, since we do not know which of those types was returned by the multiplication expressions.  As a result, the addition may also return either an integer, or a double.
     45
     46The intuition behind JavaScriptCore's type inference is that we can say with good probability what type a numerical operation (such as addition or multiplication) will return, and which path it will take, if we could guess what types flowed into that operation.  This forms a kind of induction step that applies to operations that don't generally operate over the heap: if we can predict their inputs, then we can predict their outputs.  But induction requires a base case.  In the case of JavaScript operations, the base cases are operations that get non-local values: for example, loading a value from the heap (as in 'o.x'), accessing an argument to a function, or using a value returned from a function call.  For simplicity, we refer to all such non-local values as heap values, and all operations that place heap values into local variables as heap operations.  For arguments, we treat the function prologue as a "heap operation" of sorts, which "loads" the arguments into the argument variables.  We bootstrap our inductive reasoning about type predictions by using ''value profiling'': both the LLInt and Baseline JIT will record the most recent value seen at any heap operation.  Each heap operation has exactly one value profile bucket associated with it, and each value profile bucket will store exactly one recent value.
     47
     48A straw-man view of JavaScriptCore's type inference is that we convert each value profile's most recent value into a type, and then apply the induction step to propagate this type through all operations in the function.  This gives us ''type predictions'' at each value-producing operation in the function.  All variables become predictively typed as well.
     49
     50In reality, JavaScriptCore includes in each value profile a second field, which is a type that bounds a random subset of the values seen.  This type uses the SpeculatedType (or SpecType for short) type system, which is implemented in [http://trac.webkit.org/browser/trunk/Source/JavaScriptCore/bytecode/SpeculatedType.h SpeculatedType.h].  The type of each value profile starts out as SpecNone - i.e. the type corresponding to no values.  You can also think of this as ''bottom'' (see [http://en.wikipedia.org/wiki/Lattice_(order) Lattice]), or the "contradiction".  When the Baseline JIT's execution counter exceeds a threshold (see JIT::emitOptimizationCheck in [http://trac.webkit.org/browser/trunk/Source/JavaScriptCore/jit/JIT.cpp JIT.cpp]), it will produce a new type to bound both the last type, and the most recent value.  It may also then choose to invoke the DFG, or it may choose to let the baseline code run more.  Our heuristics favor the latter, meaning that when DFG compilation kicks in, each value profile will typically have a type that bounds multiple different values.
     51
     52The propagation of SpecTypes derived at value profiles to all operations and variables in a function is performed using a standard [http://en.wikipedia.org/wiki/Data-flow_analysis forward data flow] formulation, implemented as a flow-insensitive fixpoint.  This is one of the first phases of DFG compilation, and is only activated once the Baseline JIT decides, based on its execution count, that a function would be better off executing optimized code.  See [http://trac.webkit.org/browser/trunk/Source/JavaScriptCore/dfg/DFGPredictionPropagationPhase.cpp DFGPredictionPropagationPhase.cpp].
     53
     54After each value in a function is labeled with a predicted type, we insert speculative type checks based on those predictions.  For example, in a numeric operation (like 'o.x * o.y'), we insert speculate-double checks on the operands of the multiplication.  If a speculation check fails, execution is diverted from optimized DFG code back to the Baseline JIT.  This has the effect of ''proving'' the type for subsequent code within the DFG.  Consider how the simple addition operation 'a + b' will be decomposed, if 'a' and 'b' both had SpecInt32 as their predicted type:
     55
     56{{{
     57check if a is Int32 -> else OSR exit to Baseline JIT
     58check if b is Int32 -> else OSR exit to Baseline JIT
     59result = a + b // integer addition
     60check if overflow -> else OSR exit to Baseline JIT
     61}}}
     62
     63After this operation completes, we know that:
     64
     65* 'a' is an integer.
     66* 'b' is an integer.
     67* the result is an integer.
     68
     69Any subsequent operations on either 'a' or 'b' do not need to check their types.  Likewise for operations on the result.  The elimination of subsequent checks is achieved by a second data flow analysis, called simply the DFG CFA.  Unlike the prediction propagation phase, which is concerned with constructing ''type predictions'', the CFA is concerned with constructing ''type proofs''.  The CFA, found in [http://trac.webkit.org/browser/trunk/Source/JavaScriptCore/dfg/DFGCFAPhase.cpp DFGCFAPhase.cpp] and [http://trac.webkit.org/browser/trunk/Source/JavaScriptCore/dfg/DFGAbstractState.cpp DFGAbstractState.cpp], follows a ''flow-sensitive'' forward data flow formulation.  It also implements sparse conditional constant propagation, which gives it the ability to sometimes prove that values are constants, as well as proving their types.
     70
     71Putting this together, the expression 'o.x * o.x + o.y * o.y' will only require type checks on the value loaded from 'o.x' and the value loaded from 'o.y'.  After that, we know that the values are doubles, and we know that we only have to emit a double multiply path followed by a double addition.  When combined with type check hoisting, DFG code will usually execute a type check at most once per heap load.
     72
     73== Object Access Optimization ==
     74
     75(to be written)
     76
    3077Attached are some slides from a recent talk about JavaScriptCore.