How V8 Runs JavaScript
What is JavaScript?
Execution
Memory
What is JavaScript?
APIs
document.createElement | DOM Standard |
setTimeout | HTML Standard |
console.log | Console Standard |
Environments
Web Pages
Node
Web Worker
Service Worker
Chrome Extension
D8
JavaScript Engines
Chrome | V8 |
Firefox | SpiderMonkey |
Edge | Chakra |
Safari | JavaScriptCore (Nitro) |
Node | V8 |
Opera | V8 |
Electron | V8 |
Node-ChakraCore | ChakraCore |
Execution
Parsing
Compilation
var hi = "Hello"
Parsing
function sayHi(name){
var message = "Hi " + name + "!"
console.log(message)
}
sayHi("Sparkle")
$ node --trace_parse test.js
$ node --trace_parse test.js
[parsing script: native harmony-regexp-exec.js - took 0.165 ms]
[parsing function: ImportNow - took 0.020 ms]
[parsing function: OverrideFunction - took 0.028 ms]
[parsing function: SetFunctionName - took 0.022 ms]
[parsing script: native harmony-species.js - took 0.052 ms]
[parsing function: get __proto__ - took 0.012 ms]
[parsing function: InstallGetter - took 0.020 ms]
[parsing script: native harmony-unicode-regexps.js - took 0.043 ms]
[parsing function: Import - took 0.008 ms]
[parsing script: native promise-extra.js - took 0.023 ms]
[parsing function: InstallFunctions - took 0.032 ms]
[parsing function: PostExperimentals - took 0.027 ms]
[parsing function: - took 0.007 ms]
[parsing function: - took 0.008 ms]
[parsing function: - took 0.005 ms]
[parsing function: b.CreateDoubleResultArray - took 0.008 ms]
[parsing function: Float64Array - took 0.040 ms]
[parsing function: Float64ArrayConstructByLength - took 0.024 ms]
[parsing function: ToPositiveInteger - took 0.015 ms]
[parsing script: bootstrap_node.js - took 0.611 ms]
[parsing function: some - took 0.036 ms]
[parsing function: InnerArraySome - took 0.034 ms]
[parsing function: - took 0.010 ms]
[parsing function: match - took 0.050 ms]
[parsing function: [Symbol.match] - took 0.050 ms]
[parsing function: get global - took 0.040 ms]
[parsing function: RegExpSubclassExec - took 0.034 ms]
[parsing function: exec - took 0.059 ms]
[parsing function: startup - took 0.160 ms]
[parsing function: NativeModule.require - took 0.028 ms]
[parsing function: NativeModule.getCached - took 0.008 ms]
[parsing function: NativeModule.exists - took 0.006 ms]
[parsing function: NativeModule - took 0.013 ms]
[parsing function: NativeModule.cache - took 0.008 ms]
[parsing function: NativeModule.compile - took 0.022 ms]
[parsing function: NativeModule.getSource - took 0.007 ms]
[parsing function: NativeModule.wrap - took 0.008 ms]
[parsing function: runInThisContext - took 0.011 ms]
[parsing script: events.js - took 0.428 ms]
[parsing function: defineProperty - took 0.036 ms]
[parsing function: setPrototypeOf - took 0.025 ms]
[parsing function: EventEmitter - took 0.009 ms]
[parsing function: EventEmitter.init - took 0.025 ms]
[parsing function: getPrototypeOf - took 0.041 ms]
[parsing function: setupProcessObject - took 0.023 ms]
[parsing function: setupProcessFatal - took 0.033 ms]
[parsing function: setupGlobalVariables - took 0.038 ms]
[parsing script: util.js - took 1.090 ms]
[parsing script: buffer.js - took 1.063 ms]
[parsing script: internal/util.js - took 0.172 ms]
[parsing function: createPool - took 0.013 ms]
[parsing function: createUnsafeBuffer - took 0.013 ms]
[parsing function: FastBuffer - took 0.009 ms]
[parsing function: Uint8Array - took 0.048 ms]
[parsing function: Uint8ArrayConstructByLength - took 0.035 ms]
[parsing function: exports.deprecate - took 0.011 ms]
[parsing function: exports._deprecate - took 0.031 ms]
[parsing function: forEach - took 0.032 ms]
[parsing function: InnerArrayForEach - took 0.044 ms]
[parsing function: - took 0.038 ms]
[parsing function: setupGlobalTimeouts - took 0.019 ms]
[parsing script: timers.js - took 0.463 ms]
[parsing script: internal/linkedlist.js - took 0.056 ms]
[parsing script: assert.js - took 0.286 ms]
[parsing function: exports.inherits - took 0.022 ms]
[parsing function: exports.debuglog - took 0.053 ms]
[parsing function: toUpperCase - took 0.019 ms]
[parsing function: RegExp - took 0.032 ms]
[parsing function: IsRegExp - took 0.014 ms]
[parsing function: RegExpInitialize - took 0.014 ms]
[parsing function: test - took 0.015 ms]
[parsing function: ImmediateList - took 0.019 ms]
[parsing function: setupGlobalConsole - took 0.038 ms]
[parsing script: internal/process.js - took 0.217 ms]
[parsing function: setup_hrtime - took 0.042 ms]
[parsing function: Uint32Array - took 0.056 ms]
[parsing function: Uint32ArrayConstructByLength - took 0.042 ms]
[parsing function: setup_cpuUsage - took 0.056 ms]
[parsing function: setupConfig - took 0.052 ms]
[parsing function: split - took 0.048 ms]
[parsing function: join - took 0.021 ms]
[parsing function: InnerArrayJoin - took 0.020 ms]
[parsing function: Join - took 0.018 ms]
[parsing function: StackHas - took 0.015 ms]
[parsing function: StackPush - took 0.008 ms]
[parsing function: DoJoin - took 0.057 ms]
[parsing function: UseSparseVariant - took 0.018 ms]
[parsing function: StackPop - took 0.009 ms]
[parsing function: replace - took 0.047 ms]
[parsing function: [Symbol.replace] - took 0.096 ms]
[parsing function: get unicode - took 0.022 ms]
[parsing function: [Symbol.replace] - took 0.045 ms]
[parsing function: parse - took 0.021 ms]
[parsing function: InternalizeJSONProperty - took 0.042 ms]
[parsing function: - took 0.012 ms]
[parsing function: CreateDataProperty - took 0.015 ms]
[parsing script: internal/process/warning.js - took 0.084 ms]
[parsing function: setupProcessWarnings - took 0.057 ms]
[parsing function: addListener - took 0.010 ms]
[parsing function: _addListener - took 0.058 ms]
[parsing script: internal/process/next_tick.js - took 0.133 ms]
[parsing function: setupNextTick - took 0.115 ms]
[parsing script: internal/process/promises.js - took 0.096 ms]
[parsing function: WeakMap - took 0.038 ms]
[parsing function: setupPromises - took 0.078 ms]
[parsing script: internal/process/stdio.js - took 0.141 ms]
[parsing function: setupStdio - took 0.096 ms]
[parsing function: setupKillAndExit - took 0.044 ms]
[parsing function: setupSignalHandlers - took 0.046 ms]
[parsing function: emit - took 0.072 ms]
[parsing function: emitTwo - took 0.023 ms]
[parsing function: - took 0.031 ms]
[parsing function: isSignal - took 0.010 ms]
[parsing function: lazyConstants - took 0.010 ms]
[parsing function: setupChannel - took 0.023 ms]
[parsing function: setupRawDebug - took 0.017 ms]
[parsing script: path.js - took 2.163 ms]
[parsing function: resolve - took 0.045 ms]
[parsing function: assertPath - took 0.012 ms]
[parsing function: charCodeAt - took 0.023 ms]
[parsing function: normalizeStringPosix - took 0.066 ms]
[parsing function: slice - took 0.037 ms]
[parsing script: module.js - took 0.585 ms]
[parsing script: internal/module.js - took 0.088 ms]
[parsing script: vm.js - took 0.134 ms]
[parsing script: fs.js - took 2.119 ms]
[parsing function: Script.runInThisContext - took 0.018 ms]
[parsing script: stream.js - took 0.105 ms]
[parsing script: _stream_readable.js - took 0.747 ms]
[parsing script: internal/streams/BufferList.js - took 0.082 ms]
[parsing script: _stream_writable.js - took 0.416 ms]
[parsing script: _stream_duplex.js - took 0.061 ms]
[parsing script: _stream_transform.js - took 0.136 ms]
[parsing script: _stream_passthrough.js - took 0.042 ms]
[parsing function: - took 0.015 ms]
[parsing function: Map - took 0.039 ms]
[parsing function: Module._initPaths - took 0.039 ms]
[parsing function: preloadModules - took 0.012 ms]
[parsing function: run - took 0.026 ms]
[parsing function: Module.runMain - took 0.011 ms]
[parsing function: Module._load - took 0.040 ms]
[parsing function: Module._resolveFilename - took 0.035 ms]
[parsing function: NativeModule.nonInternalExists - took 0.025 ms]
[parsing function: Module._resolveLookupPaths - took 0.099 ms]
[parsing function: debugs.(anonymous function) - took 0.004 ms]
[parsing function: Module._findPath - took 0.102 ms]
[parsing function: isAbsolute - took 0.022 ms]
[parsing function: stringify - took 0.100 ms]
[parsing function: stat - took 0.030 ms]
[parsing function: _makeLong - took 0.006 ms]
[parsing function: toRealPath - took 0.021 ms]
[parsing function: realpathSync - took 0.131 ms]
[parsing function: nullCheck - took 0.030 ms]
[parsing function: indexOf - took 0.033 ms]
[parsing function: toString - took 0.013 ms]
[parsing function: get - took 0.037 ms]
[parsing function: GetExistingHash - took 0.023 ms]
[parsing function: MapFindEntry - took 0.038 ms]
[parsing function: HashToEntry - took 0.010 ms]
[parsing function: start - took 0.019 ms]
[parsing function: fs.lstatSync - took 0.011 ms]
[parsing function: fs.Stats - took 0.039 ms]
[parsing function: fs.Stats.isSymbolicLink - took 0.018 ms]
[parsing function: fs.Stats._checkModeProperty - took 0.008 ms]
[parsing function: set - took 0.065 ms]
[parsing function: GetHash - took 0.017 ms]
[parsing function: encodeRealpathResult - took 0.030 ms]
[parsing function: Module - took 0.026 ms]
[parsing function: tryModuleLoad - took 0.026 ms]
[parsing function: Module.load - took 0.031 ms]
[parsing function: ok - took 0.019 ms]
[parsing function: dirname - took 0.030 ms]
[parsing function: Module._nodeModulePaths - took 0.055 ms]
[parsing function: extname - took 0.044 ms]
[parsing function: Module._extensions..js - took 0.013 ms]
[parsing function: fs.readFileSync - took 0.062 ms]
[parsing function: assertEncoding - took 0.012 ms]
[parsing function: Buffer.isEncoding - took 0.009 ms]
[parsing function: normalizeEncoding - took 0.025 ms]
[parsing function: isFd - took 0.007 ms]
[parsing function: fs.openSync - took 0.015 ms]
[parsing function: modeNum - took 0.070 ms]
[parsing function: stringToFlags - took 0.054 ms]
[parsing function: tryStatSync - took 0.022 ms]
[parsing function: fs.fstatSync - took 0.007 ms]
[parsing function: fs.Stats.isFile - took 0.007 ms]
[parsing function: tryCreateBuffer - took 0.027 ms]
[parsing function: Buffer.allocUnsafe - took 0.007 ms]
[parsing function: assertSize - took 0.023 ms]
[parsing function: isNaN - took 0.024 ms]
[parsing function: get buffer - took 0.013 ms]
[parsing function: get byteOffset - took 0.009 ms]
[parsing function: Uint8ArrayConstructByArrayBuffer - took 0.069 ms]
[parsing function: alignPool - took 0.011 ms]
[parsing function: tryReadSync - took 0.019 ms]
[parsing function: fs.readSync - took 0.051 ms]
[parsing function: fs.closeSync - took 0.008 ms]
[parsing function: Buffer.toString - took 0.017 ms]
[parsing function: slowToString - took 0.054 ms]
[parsing function: stripBOM - took 0.012 ms]
[parsing function: Module._compile - took 0.084 ms]
[parsing function: exports.runInThisContext - took 0.031 ms]
[parsing script: /Users/mattzeunert/test.js - took 0.019 ms]
[parsing function: makeRequireFunction - took 0.037 ms]
[parsing function: sayHi - took 0.022 ms]
[parsing function: get - took 0.013 ms]
[parsing script: console.js - took 0.333 ms]
[parsing function: getStdout - took 0.108 ms]
[parsing function: createWritableStdioStream - took 0.098 ms]
[parsing script: tty.js - took 0.131 ms]
[parsing script: net.js - took 1.197 ms]
[parsing script: internal/net.js - took 0.032 ms]
[parsing function: protoGetter - took 0.021 ms]
[parsing function: WriteStream - took 0.040 ms]
[parsing function: Socket - took 0.083 ms]
[parsing function: Duplex - took 0.035 ms]
[parsing function: Readable - took 0.018 ms]
[parsing function: ReadableState - took 0.077 ms]
[parsing function: BufferList - took 0.010 ms]
[parsing function: Stream - took 0.013 ms]
[parsing function: EventHandlers - took 0.005 ms]
[parsing function: Writable - took 0.032 ms]
[parsing function: WritableState - took 0.066 ms]
[parsing function: CorkedRequest - took 0.029 ms]
[parsing function: once - took 0.018 ms]
[parsing function: _onceWrap - took 0.021 ms]
[parsing function: Readable.on - took 0.032 ms]
[parsing function: initSocketHandle - took 0.020 ms]
[parsing function: getStderr - took 0.031 ms]
[parsing function: $getMaxListeners - took 0.014 ms]
[parsing function: get - took 0.006 ms]
[parsing function: Console - took 0.052 ms]
[parsing function: Console.log - took 0.016 ms]
[parsing function: Console.warn - took 0.012 ms]
[parsing function: Console.dir - took 0.016 ms]
[parsing function: Console.time - took 0.009 ms]
[parsing function: Console.timeEnd - took 0.025 ms]
[parsing function: trace - took 0.018 ms]
[parsing function: Console.assert - took 0.014 ms]
[parsing function: installInspectorConsoleIfNeeded - took 0.036 ms]
[parsing function: exports.format - took 0.073 ms]
[parsing function: Socket.write - took 0.020 ms]
[parsing function: Writable.write - took 0.031 ms]
[parsing function: validChunk - took 0.029 ms]
[parsing function: writeOrBuffer - took 0.038 ms]
[parsing function: decodeChunk - took 0.015 ms]
[parsing function: doWrite - took 0.020 ms]
[parsing function: Socket._write - took 0.008 ms]
[parsing function: Socket._writeGeneric - took 0.063 ms]
[parsing function: _unrefTimer - took 0.012 ms]
[parsing function: exports._unrefActive - took 0.007 ms]
[parsing function: insert - took 0.028 ms]
[parsing function: createWriteReq - took 0.027 ms]
Hi Sparkle!
[parsing function: WritableState.onwrite - took 0.008 ms]
[parsing function: onwrite - took 0.026 ms]
[parsing function: onwriteStateUpdate - took 0.011 ms]
[parsing function: needFinish - took 0.010 ms]
[parsing function: nextTick - took 0.026 ms]
[parsing function: TickObject - took 0.011 ms]
[parsing function: _tickCallback - took 0.027 ms]
[parsing function: _combinedTickCallback - took 0.020 ms]
[parsing function: afterWrite - took 0.013 ms]
[parsing function: onwriteDrain - took 0.011 ms]
[parsing function: nop - took 0.003 ms]
[parsing function: finishMaybe - took 0.016 ms]
[parsing function: tickDone - took 0.017 ms]
[parsing function: emitPendingUnhandledRejections - took 0.034 ms]
$ d8 --trace-parse test.js
test.js:3: ReferenceError: console is not defined
console.log(message)
^
ReferenceError: console is not defined
at sayHi (test.js:3:5)
at test.js:6:1
function sayHi(name){
var message = "Hi " + name + "!"
console.log(message)
}
sayHi("Sparkle")
$ d8 --trace-parse test.js
[parsing script: native datetime-format-...js - took 3.453 ms]
[parsing function: ImportNow - took 0.025 ms]
[parsing function: InstallFunctions - took 0.035 ms]
[parsing function: SetFunctionName - took 0.020 ms]
[parsing script: native icu-case-mapping.js - took 0.029 ms]
[parsing function: OverrideFunction - took 0.311 ms]
[parsing function: PostExperimentals - took 0.024 ms]
[parsing script: test.js - took 1.248 ms]
[parsing function: sayHi - took 0.012 ms]
Hi Sparkle!
function sayHi(name){
var message = "Hi " + name + "!"
print(message)
}
sayHi("Sparkle")
function sayHi(name){
var message = "Hi " + name + "!"
print(message)
}
sayHi("Sparkle")
$ d8 --trace-parse test.js
[parsing script: test.js - took 1.248 ms]
[parsing function: sayHi - took 0.012 ms]
Hi Sparkle!
sayHi pre-parse
sayHi full parse
function sayHi(name){
var message = "Hi " + name + "!"
print(message)
}
function add(a, b){
return a + b
}
sayHi("Sparkle")
$ d8 --trace-parse test.js
[parsing script: test.js - took 0.651 ms]
[parsing function: sayHi - took 0.009 ms]
Hi Sparkle!
function sayHi(name){
var message = "Hi " + name + "!"
print(message)
}
sayHi("Sparkle")
$ d8 --trace-parse test.js
[parsing script: test.js - took 1.248 ms]
[parsing function: sayHi - took 0.012 ms]
Hi Sparkle!
var constants = (function(){
return {pi: 3.14}
})()
$ d8 --trace-parse test.js
[parsing script: test.js - took 0.024 ms]
var sayHi = (function sayHi(name){
var message = "Hi " + name + "!"
print(message)
})
sayHi("Sparkle")
$ d8 --trace-parse test.js
[parsing script: test.js - took 0.019 ms]
Hi Sparkle!
Create React App Bundle:
24ms => 18ms
press Shift 6 times to see all experiments
Compilation
Baseline Compiler
Optimizing Compiler
Full-codegen
Crankshaft / Turbofan
$ d8 --trace_opt test.js
[marking 0x2b71a8bd <JS Function add (SharedFunctionInfo 0x2b71a6d9)> for optimized recompilation, reason: small function, ICs with typeinfo: 1/1 (100%), generic ICs: 0/1 (0%)]
[compiling method 0x2b71a8bd <JS Function add
(SharedFunctionInfo 0x2b71a6d9)> using Crankshaft]
[optimizing 0x2b71a8bd <JS Function add (SharedFunctionInfo 0x2b71a6d9)> - took 0.085, 0.112, 0.044 ms]
[completed optimizing 0x2b71a8bd <JS Function add (SharedFunctionInfo 0x2b71a6d9)>]
function add(a,b){
return a + b;
}
for (var i=0; i< 10000; i++){
add(i, 1);
}
$ d8 --trace_opt --trace_deopt test.js
...
[optimizing 0x4761a8d5 <JS Function add (SharedFunctionInfo 0x4761a6d9)> - took 0.087, 0.105, 0.054 ms]
...
[deoptimizing (DEOPT eager): begin 0x4761a8d5 <JS Function add (SharedFunctionInfo 0x4761a6d9)> (opt #0) @2, FP to SP delta: 12, caller sp: 0xbffa1044]
...
[removing optimized code for: add]
function add(a,b){
return a + b
}
for (var i=0; i< 10000; i++){
add(i, 1)
}
add(i, "a")
function addInteger(a,b){
return a + b;
}
function addString(a,b){
return a + b;
}
console.time("Integer")
for (var i=0; i< 10000000; i++){
addInteger(i, 1);
}
console.timeEnd("Integer")
console.time("String")
for (var i=0; i< 10000000; i++){
addString(i, "a");
}
console.timeEnd("String")
$ node test.js
Integer: 11.545ms
String: 792.009ms
$ node test.js
Integer: 10.061ms
String: 1031.975ms
add
addInteger/addString
Objects and Properties
person.age
age = person.age
age = [objAddress + offset]
age = [0x674836362 + 2]
Person |
---|
[0] firstName |
[1] lastName |
[2] age |
[3] location |
Hidden Classes
var p1 = new Person("John", "William", "Green")
var p2 = new Person("Mary", "Lisa", "Smith")
Hidden Classes
var p1 = new Person("John", "William", "Green")
var p2 = new Person("Mary", "Lisa", "Smith")
p2.location = "London"
function getName(person){
return person.firstName + person.lastName
}
var p1 = new Person("John", "William", "Green")
var p2 = new Person("Mary", "Lisa", "Smith")
for (var i=0;i<10000;i++){ getName(p1) }
for (var i=0;i<10000;i++){ getName(p2) }
var p1 = new Person("John", "William", "Green")
var p2 = new Person("Mary", "Lisa", "Smith")
p2.location = "London"
for (var i=0;i<10000;i++){ getName(p1) }
for (var i=0;i<10000;i++){ getName(p2) }
$ d8 --trace_deopt test.js
[deoptimizing (DEOPT eager): begin 0x52f9ab45 <JS Function getName (SharedFunctionInfo 0x52f9a829)> (opt #0) @2, FP to SP delta: 12, caller sp: 0xbff65004]
...
[deoptimizing (eager): end 0x52f9ab45 <JS Function getName (SharedFunctionInfo 0x52f9a829)> @2 => node=4, pc=0x3ad0756f, caller sp=0xbff65004, state=NO_REGISTERS, took 0.104 ms]
[removing optimized code for: getName]
var p1 = new Person("John", "William", "Green")
p1.location = null
var p2 = new Person("Mary", "Lisa", "Smith")
p2.location = "London"
for (var i=0;i<10000;i++){ getName(p1) }
for (var i=0;i<10000;i++){ getName(p2) }
$ d8 --trace_deopt test.js
No output :)
Assign All Properties In The Constructor
Bailouts
function doSth(){
debugger;
// code here isn't optimized
}
function debuggerFunction(){
debugger
}
function doSth(){
debuggerFunction()
// code here is optimized
}
function sum(){
var args = toArray(arguments)
return args.reduce((sum, val) => sum + val, 0)
}
sum(1,2,3) // 6
"Bad value context for arguments value"
function sum(){
var args = toArray(arguments)
return args.reduce((sum, val) => sum + val, 0)
}
sum(1,2,3) // 6
function sum(){
var sum = 0
for (var i=0; i<arguments.length; i++){
sum += arguments[i]
}
return sum
}
Baseline Compiler
Optimizing Compiler
Full-codegen
Crankshaft / Turbofan
Interpreter
Ignition
</Execution>
Memory
var p1 = new Person("John", "William", "Green")
var p2 = new Person("Mary", "Lisa", "Smith")
Objects
1
2
3
4
5
5 * (64 / 8) = 40
var p = [];
for (var i=0; i<100000;i++){
p.push(new Person("a", "b", "c"))
}
p1.location = "London"
p1.otherProp = "sth"
Extra Properties
p1[0] = "zero"
p1[1] = "one"
Numbered Properties
Map |
extraProperties |
elements |
firstName |
middleName |
lastName |
location |
otherProp |
Person Object
0 |
1 |
1
2
3
4
5
6
6 * (64/8) = 48
delete p2.middleName
Arrays
var arr = []
for (var i=0; i< 1000000;i++){
arr.push(Math.random())
}
9.7MB ≈ 8MB
0.2 | 1.8 | 1.4 | |
0x8789 | 0x37b2 | 0xae72 |
"abc"
(reference)
0xc79d
(reference) | (reference) | (reference) | |
0x127b | 0x7a55 | 0x98ee |
"abc"
(reference)
0xc79d
{ num: 0.2 }
{ num: 1.8 }
{ num: 1.4 }
arr.push("abc")
arr.push("abc")
25.7MB ≈ 24MB
Strings
Keep things uniform and predictable
Learn More
V8 Wiki - Design Elements
Breaking the JavaScript Speed Limit with V8
JavaScript Start-up Performance
Performance Profiling for V8
Garbage Collection?
Getting Garbage Collection For Free
Twitter: @mattzeunert
Thank You
Slides
How V8 runs JavaScript
By Matt Zeunert
How V8 runs JavaScript
- 1,673