1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
use super::{Context, Executable, InterpreterState};
use crate::{
syntax::ast::node::{Call, Node},
value::{Type, Value},
BoaProfiler, Result,
};
impl Executable for Call {
fn run(&self, interpreter: &mut Context) -> Result<Value> {
let _timer = BoaProfiler::global().start_event("Call", "exec");
let (this, func) = match self.expr() {
Node::GetConstField(ref get_const_field) => {
let mut obj = get_const_field.obj().run(interpreter)?;
if obj.get_type() != Type::Object {
obj = Value::Object(obj.to_object(interpreter)?);
}
(obj.clone(), obj.get_field(get_const_field.field()))
}
Node::GetField(ref get_field) => {
let obj = get_field.obj().run(interpreter)?;
let field = get_field.field().run(interpreter)?;
(
obj.clone(),
obj.get_field(field.to_property_key(interpreter)?),
)
}
_ => (
interpreter.realm().global_obj.clone(),
self.expr().run(interpreter)?,
),
};
let mut v_args = Vec::with_capacity(self.args().len());
for arg in self.args() {
if let Node::Spread(ref x) = arg {
let val = x.run(interpreter)?;
let mut vals = interpreter.extract_array_properties(&val).unwrap();
v_args.append(&mut vals);
break;
}
v_args.push(arg.run(interpreter)?);
}
let fnct_result = interpreter.call(&func, &this, &v_args);
interpreter
.executor()
.set_current_state(InterpreterState::Executing);
fnct_result
}
}