How V8 Runs JavaScript
What's a JavaScript Engine?
APIs
document.createElement | DOM Standard |
setTimeout | HTML Standard |
console.log | Console Standard |
=> Built into browser
JavaScript Engines
Chrome | V8 |
Firefox | SpiderMonkey |
Edge | Chakra |
Safari | JavaScriptCore (Nitro) |
Node | V8 |
Opera | V8 |
Electron | V8 |
Node-ChakraCore | ChakraCore |
Execution Pipeline
Memory
var hi = "Hello"
Parsing
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!
sayHi pre-parse
sayHi full parse
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!
press Shift 6 times to see all experiments
Execution
Interpreter
Compiler
Ignition
Turbofan
$ d8 --trace_opt test.js
...
[marking 0x3830902c839 <JSFunction add (sfi = 0x3830902c489)>
for optimized recompilation, reason: small function,
ICs with typeinfo: 2/2 (100%), generic ICs: 0/2 (0%)]
[compiling method 0x3830902c839 <JSFunction add
(sfi = 0x3830902c489)> using TurboFan]
...
[optimizing 0x3830902c839 <JSFunction add (sfi = 0x3830902c489)>
[completed optimizing 0x3830902c839 <JSFunction add ...>]
function add(a, b, c){
return a + b + c
}
for (var i=0; i< 100000; i++){
add(i, 1, 2)
}
$ d8 --trace_opt --trace_deopt test.js
...
[optimizing 0x25c3063ac879 <JSFunction add (sfi =
0x25c3063ac489)> - took 1.146, 0.359, 0.014 ms]
[completed optimizing 0x25c3063ac879
<JSFunction add (sfi = 0x25c3063ac489)>]
[deoptimizing (DEOPT eager): begin 0x25c3063ac879
<JSFunction add (sfi = 0x25c3063ac489)> (opt #0) @3,
FP to SP delta: 16, caller sp: 0x7fff559a1fc8]
function add(a, b, c){
return a + b + c
}
for (var i=0; i< 100000; i++){
add(i, 1, 2)
}
add(1, 2, "a")
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 :)
Objects of the same type should have the same properties
Memory
var p1 = new Person("John", "William", "Green")
var p2 = new Person("Mary", "Lisa", "Smith")
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
Keep Types Consistent
Learn More
V8 Wiki - Design Elements
V8 Behind the Scenes and a Tale of TurboFan
JavaScript Start-up Performance
Slides
How V8 Runs JavaScript - Strikingly Tech
By Matt Zeunert
How V8 Runs JavaScript - Strikingly Tech
- 868