2024-02-24 21:00:04 +00:00
|
|
|
package otto
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"math"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/robertkrimen/otto/token"
|
|
|
|
)
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func (rt *runtime) evaluateMultiply(left float64, right float64) Value { //nolint: unused
|
2024-02-24 21:00:04 +00:00
|
|
|
// TODO 11.5.1
|
|
|
|
return Value{}
|
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func (rt *runtime) evaluateDivide(left float64, right float64) Value {
|
2024-02-24 21:00:04 +00:00
|
|
|
if math.IsNaN(left) || math.IsNaN(right) {
|
|
|
|
return NaNValue()
|
|
|
|
}
|
|
|
|
if math.IsInf(left, 0) && math.IsInf(right, 0) {
|
|
|
|
return NaNValue()
|
|
|
|
}
|
|
|
|
if left == 0 && right == 0 {
|
|
|
|
return NaNValue()
|
|
|
|
}
|
|
|
|
if math.IsInf(left, 0) {
|
|
|
|
if math.Signbit(left) == math.Signbit(right) {
|
|
|
|
return positiveInfinityValue()
|
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
return negativeInfinityValue()
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
if math.IsInf(right, 0) {
|
|
|
|
if math.Signbit(left) == math.Signbit(right) {
|
|
|
|
return positiveZeroValue()
|
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
return negativeZeroValue()
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
if right == 0 {
|
|
|
|
if math.Signbit(left) == math.Signbit(right) {
|
|
|
|
return positiveInfinityValue()
|
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
return negativeInfinityValue()
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
return float64Value(left / right)
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func (rt *runtime) evaluateModulo(left float64, right float64) Value { //nolint: unused
|
2024-02-24 21:00:04 +00:00
|
|
|
// TODO 11.5.3
|
|
|
|
return Value{}
|
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func (rt *runtime) calculateBinaryExpression(operator token.Token, left Value, right Value) Value {
|
2024-02-24 21:00:04 +00:00
|
|
|
leftValue := left.resolve()
|
|
|
|
|
|
|
|
switch operator {
|
|
|
|
// Additive
|
|
|
|
case token.PLUS:
|
2024-04-04 14:46:14 +00:00
|
|
|
leftValue = toPrimitiveValue(leftValue)
|
2024-02-24 21:00:04 +00:00
|
|
|
rightValue := right.resolve()
|
2024-04-04 14:46:14 +00:00
|
|
|
rightValue = toPrimitiveValue(rightValue)
|
2024-02-24 21:00:04 +00:00
|
|
|
|
|
|
|
if leftValue.IsString() || rightValue.IsString() {
|
2024-04-04 14:46:14 +00:00
|
|
|
return stringValue(strings.Join([]string{leftValue.string(), rightValue.string()}, ""))
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
return float64Value(leftValue.float64() + rightValue.float64())
|
2024-02-24 21:00:04 +00:00
|
|
|
case token.MINUS:
|
|
|
|
rightValue := right.resolve()
|
2024-04-04 14:46:14 +00:00
|
|
|
return float64Value(leftValue.float64() - rightValue.float64())
|
2024-02-24 21:00:04 +00:00
|
|
|
|
|
|
|
// Multiplicative
|
|
|
|
case token.MULTIPLY:
|
|
|
|
rightValue := right.resolve()
|
2024-04-04 14:46:14 +00:00
|
|
|
return float64Value(leftValue.float64() * rightValue.float64())
|
2024-02-24 21:00:04 +00:00
|
|
|
case token.SLASH:
|
|
|
|
rightValue := right.resolve()
|
2024-04-04 14:46:14 +00:00
|
|
|
return rt.evaluateDivide(leftValue.float64(), rightValue.float64())
|
2024-02-24 21:00:04 +00:00
|
|
|
case token.REMAINDER:
|
|
|
|
rightValue := right.resolve()
|
2024-04-04 14:46:14 +00:00
|
|
|
return float64Value(math.Mod(leftValue.float64(), rightValue.float64()))
|
2024-02-24 21:00:04 +00:00
|
|
|
|
|
|
|
// Logical
|
|
|
|
case token.LOGICAL_AND:
|
|
|
|
left := leftValue.bool()
|
|
|
|
if !left {
|
|
|
|
return falseValue
|
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
return boolValue(right.resolve().bool())
|
2024-02-24 21:00:04 +00:00
|
|
|
case token.LOGICAL_OR:
|
|
|
|
left := leftValue.bool()
|
|
|
|
if left {
|
|
|
|
return trueValue
|
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
return boolValue(right.resolve().bool())
|
2024-02-24 21:00:04 +00:00
|
|
|
|
|
|
|
// Bitwise
|
|
|
|
case token.AND:
|
|
|
|
rightValue := right.resolve()
|
2024-04-04 14:46:14 +00:00
|
|
|
return int32Value(toInt32(leftValue) & toInt32(rightValue))
|
2024-02-24 21:00:04 +00:00
|
|
|
case token.OR:
|
|
|
|
rightValue := right.resolve()
|
2024-04-04 14:46:14 +00:00
|
|
|
return int32Value(toInt32(leftValue) | toInt32(rightValue))
|
2024-02-24 21:00:04 +00:00
|
|
|
case token.EXCLUSIVE_OR:
|
|
|
|
rightValue := right.resolve()
|
2024-04-04 14:46:14 +00:00
|
|
|
return int32Value(toInt32(leftValue) ^ toInt32(rightValue))
|
2024-02-24 21:00:04 +00:00
|
|
|
|
|
|
|
// Shift
|
|
|
|
// (Masking of 0x1f is to restrict the shift to a maximum of 31 places)
|
|
|
|
case token.SHIFT_LEFT:
|
|
|
|
rightValue := right.resolve()
|
2024-04-04 14:46:14 +00:00
|
|
|
return int32Value(toInt32(leftValue) << (toUint32(rightValue) & 0x1f))
|
2024-02-24 21:00:04 +00:00
|
|
|
case token.SHIFT_RIGHT:
|
|
|
|
rightValue := right.resolve()
|
2024-04-04 14:46:14 +00:00
|
|
|
return int32Value(toInt32(leftValue) >> (toUint32(rightValue) & 0x1f))
|
2024-02-24 21:00:04 +00:00
|
|
|
case token.UNSIGNED_SHIFT_RIGHT:
|
|
|
|
rightValue := right.resolve()
|
|
|
|
// Shifting an unsigned integer is a logical shift
|
2024-04-04 14:46:14 +00:00
|
|
|
return uint32Value(toUint32(leftValue) >> (toUint32(rightValue) & 0x1f))
|
2024-02-24 21:00:04 +00:00
|
|
|
|
|
|
|
case token.INSTANCEOF:
|
|
|
|
rightValue := right.resolve()
|
|
|
|
if !rightValue.IsObject() {
|
2024-04-04 14:46:14 +00:00
|
|
|
panic(rt.panicTypeError("invalid kind %s for instanceof (expected object)", rightValue.kind))
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
return boolValue(rightValue.object().hasInstance(leftValue))
|
2024-02-24 21:00:04 +00:00
|
|
|
|
|
|
|
case token.IN:
|
|
|
|
rightValue := right.resolve()
|
|
|
|
if !rightValue.IsObject() {
|
2024-04-04 14:46:14 +00:00
|
|
|
panic(rt.panicTypeError("invalid kind %s for in (expected object)", rightValue.kind))
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
2024-04-04 14:46:14 +00:00
|
|
|
return boolValue(rightValue.object().hasProperty(leftValue.string()))
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
panic(hereBeDragons(operator))
|
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
type lessThanResult int
|
2024-02-24 21:00:04 +00:00
|
|
|
|
|
|
|
const (
|
2024-04-04 14:46:14 +00:00
|
|
|
lessThanFalse lessThanResult = iota
|
2024-02-24 21:00:04 +00:00
|
|
|
lessThanTrue
|
|
|
|
lessThanUndefined
|
|
|
|
)
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func calculateLessThan(left Value, right Value, leftFirst bool) lessThanResult {
|
2024-02-24 21:00:04 +00:00
|
|
|
var x, y Value
|
|
|
|
if leftFirst {
|
|
|
|
x = toNumberPrimitive(left)
|
|
|
|
y = toNumberPrimitive(right)
|
|
|
|
} else {
|
|
|
|
y = toNumberPrimitive(right)
|
|
|
|
x = toNumberPrimitive(left)
|
|
|
|
}
|
|
|
|
|
|
|
|
var result bool
|
|
|
|
if x.kind != valueString || y.kind != valueString {
|
|
|
|
x, y := x.float64(), y.float64()
|
|
|
|
if math.IsNaN(x) || math.IsNaN(y) {
|
|
|
|
return lessThanUndefined
|
|
|
|
}
|
|
|
|
result = x < y
|
|
|
|
} else {
|
|
|
|
x, y := x.string(), y.string()
|
|
|
|
result = x < y
|
|
|
|
}
|
|
|
|
|
|
|
|
if result {
|
|
|
|
return lessThanTrue
|
|
|
|
}
|
|
|
|
|
|
|
|
return lessThanFalse
|
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
// FIXME Probably a map is not the most efficient way to do this.
|
|
|
|
var lessThanTable [4](map[lessThanResult]bool) = [4](map[lessThanResult]bool){
|
2024-02-24 21:00:04 +00:00
|
|
|
// <
|
2024-04-04 14:46:14 +00:00
|
|
|
map[lessThanResult]bool{
|
2024-02-24 21:00:04 +00:00
|
|
|
lessThanFalse: false,
|
|
|
|
lessThanTrue: true,
|
|
|
|
lessThanUndefined: false,
|
|
|
|
},
|
|
|
|
|
|
|
|
// >
|
2024-04-04 14:46:14 +00:00
|
|
|
map[lessThanResult]bool{
|
2024-02-24 21:00:04 +00:00
|
|
|
lessThanFalse: false,
|
|
|
|
lessThanTrue: true,
|
|
|
|
lessThanUndefined: false,
|
|
|
|
},
|
|
|
|
|
|
|
|
// <=
|
2024-04-04 14:46:14 +00:00
|
|
|
map[lessThanResult]bool{
|
2024-02-24 21:00:04 +00:00
|
|
|
lessThanFalse: true,
|
|
|
|
lessThanTrue: false,
|
|
|
|
lessThanUndefined: false,
|
|
|
|
},
|
|
|
|
|
|
|
|
// >=
|
2024-04-04 14:46:14 +00:00
|
|
|
map[lessThanResult]bool{
|
2024-02-24 21:00:04 +00:00
|
|
|
lessThanFalse: true,
|
|
|
|
lessThanTrue: false,
|
|
|
|
lessThanUndefined: false,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
func (rt *runtime) calculateComparison(comparator token.Token, left Value, right Value) bool {
|
2024-02-24 21:00:04 +00:00
|
|
|
// FIXME Use strictEqualityComparison?
|
|
|
|
// TODO This might be redundant now (with regards to evaluateComparison)
|
|
|
|
x := left.resolve()
|
|
|
|
y := right.resolve()
|
|
|
|
|
2024-04-04 14:46:14 +00:00
|
|
|
var kindEqualKind bool
|
|
|
|
var negate bool
|
2024-02-24 21:00:04 +00:00
|
|
|
result := true
|
|
|
|
|
|
|
|
switch comparator {
|
|
|
|
case token.LESS:
|
|
|
|
result = lessThanTable[0][calculateLessThan(x, y, true)]
|
|
|
|
case token.GREATER:
|
|
|
|
result = lessThanTable[1][calculateLessThan(y, x, false)]
|
|
|
|
case token.LESS_OR_EQUAL:
|
|
|
|
result = lessThanTable[2][calculateLessThan(y, x, false)]
|
|
|
|
case token.GREATER_OR_EQUAL:
|
|
|
|
result = lessThanTable[3][calculateLessThan(x, y, true)]
|
|
|
|
case token.STRICT_NOT_EQUAL:
|
|
|
|
negate = true
|
|
|
|
fallthrough
|
|
|
|
case token.STRICT_EQUAL:
|
|
|
|
if x.kind != y.kind {
|
|
|
|
result = false
|
|
|
|
} else {
|
|
|
|
kindEqualKind = true
|
|
|
|
}
|
|
|
|
case token.NOT_EQUAL:
|
|
|
|
negate = true
|
|
|
|
fallthrough
|
|
|
|
case token.EQUAL:
|
2024-04-04 14:46:14 +00:00
|
|
|
switch {
|
|
|
|
case x.kind == y.kind:
|
2024-02-24 21:00:04 +00:00
|
|
|
kindEqualKind = true
|
2024-04-04 14:46:14 +00:00
|
|
|
case x.kind <= valueNull && y.kind <= valueNull:
|
2024-02-24 21:00:04 +00:00
|
|
|
result = true
|
2024-04-04 14:46:14 +00:00
|
|
|
case x.kind <= valueNull || y.kind <= valueNull:
|
2024-02-24 21:00:04 +00:00
|
|
|
result = false
|
2024-04-04 14:46:14 +00:00
|
|
|
case x.kind <= valueString && y.kind <= valueString:
|
2024-02-24 21:00:04 +00:00
|
|
|
result = x.float64() == y.float64()
|
2024-04-04 14:46:14 +00:00
|
|
|
case x.kind == valueBoolean:
|
|
|
|
result = rt.calculateComparison(token.EQUAL, float64Value(x.float64()), y)
|
|
|
|
case y.kind == valueBoolean:
|
|
|
|
result = rt.calculateComparison(token.EQUAL, x, float64Value(y.float64()))
|
|
|
|
case x.kind == valueObject:
|
|
|
|
result = rt.calculateComparison(token.EQUAL, toPrimitiveValue(x), y)
|
|
|
|
case y.kind == valueObject:
|
|
|
|
result = rt.calculateComparison(token.EQUAL, x, toPrimitiveValue(y))
|
|
|
|
default:
|
|
|
|
panic(fmt.Sprintf("unknown types for equal: %v ==? %v", x, y))
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
default:
|
2024-04-04 14:46:14 +00:00
|
|
|
panic(fmt.Sprintf("unknown comparator %s", comparator.String()))
|
2024-02-24 21:00:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if kindEqualKind {
|
|
|
|
switch x.kind {
|
|
|
|
case valueUndefined, valueNull:
|
|
|
|
result = true
|
|
|
|
case valueNumber:
|
|
|
|
x := x.float64()
|
|
|
|
y := y.float64()
|
|
|
|
if math.IsNaN(x) || math.IsNaN(y) {
|
|
|
|
result = false
|
|
|
|
} else {
|
|
|
|
result = x == y
|
|
|
|
}
|
|
|
|
case valueString:
|
|
|
|
result = x.string() == y.string()
|
|
|
|
case valueBoolean:
|
|
|
|
result = x.bool() == y.bool()
|
|
|
|
case valueObject:
|
2024-04-04 14:46:14 +00:00
|
|
|
result = x.object() == y.object()
|
2024-02-24 21:00:04 +00:00
|
|
|
default:
|
|
|
|
goto ERROR
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if negate {
|
|
|
|
result = !result
|
|
|
|
}
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
ERROR:
|
|
|
|
panic(hereBeDragons("%v (%v) %s %v (%v)", x, x.kind, comparator, y, y.kind))
|
|
|
|
}
|