blob: a1d7812f6fc8c220869cb5aa68202776add6d900 [file] [log] [blame]
// Copyright 2018 The CUE Authors
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package cue
import "cuelang.org/go/cue/token"
// Op indicates the operation at the top of an expression tree of the expression
// use to evaluate a value.
type Op int
func (o Op) String() string {
return opToString[o]
}
// Values of Op.
const (
NoOp Op = iota
AndOp
OrOp
SelectorOp
IndexOp
SliceOp
CallOp
BooleanAndOp
BooleanOrOp
EqualOp
NotOp
NotEqualOp
LessThanOp
LessThanEqualOp
GreaterThanOp
GreaterThanEqualOp
RegexMatchOp
NotRegexMatchOp
AddOp
SubtractOp
MultiplyOp
FloatQuotientOp
FloatRemainOp
IntQuotientOp
IntRemainderOp
IntDivideOp
IntModuloOp
InterpolationOp
)
var opToOp = map[op]Op{
opUnify: AndOp,
// TODO(eval): opUnifyUnchecked is not the same as opUnify and should have its own
// category, if needed. More likely opUnifyUnchecked, should be
// represented as a separate embedding method.
opUnifyUnchecked: AndOp,
opDisjunction: OrOp,
opLand: BooleanAndOp,
opLor: BooleanOrOp,
opEql: EqualOp,
opNot: NotOp,
opNeq: NotEqualOp,
opLss: LessThanOp,
opLeq: LessThanEqualOp,
opGtr: GreaterThanOp,
opGeq: GreaterThanEqualOp,
opMat: RegexMatchOp,
opNMat: NotRegexMatchOp,
opAdd: AddOp,
opSub: SubtractOp,
opMul: MultiplyOp,
opQuo: FloatQuotientOp,
opRem: FloatRemainOp,
opIQuo: IntQuotientOp,
opIRem: IntRemainderOp,
opIDiv: IntDivideOp,
opIMod: IntModuloOp,
}
var opToString = map[Op]string{
AndOp: "&",
OrOp: "|",
BooleanAndOp: "&&",
BooleanOrOp: "||",
EqualOp: "==",
NotOp: "!",
NotEqualOp: "!=",
LessThanOp: "<",
LessThanEqualOp: "<=",
GreaterThanOp: ">",
GreaterThanEqualOp: ">=",
RegexMatchOp: "=~",
NotRegexMatchOp: "!~",
AddOp: "+",
SubtractOp: "-",
MultiplyOp: "*",
FloatQuotientOp: "/",
FloatRemainOp: "%",
IntQuotientOp: "quo",
IntRemainderOp: "rem",
IntDivideOp: "div",
IntModuloOp: "mod",
SelectorOp: ".",
IndexOp: "[]",
SliceOp: "[:]",
CallOp: "()",
InterpolationOp: `\()`,
}
func opIn(op op, anyOf ...op) bool {
for _, o := range anyOf {
if o == op {
return true
}
}
return false
}
// isCmp reports whether an op is a comparator.
func (op op) isCmp() bool {
return opEql <= op && op <= opGeq
}
func (op op) unifyType() (unchecked, ok bool) {
if op == opUnifyUnchecked {
return true, true
}
return false, op == opUnify
}
type op uint16
const (
opUnknown op = iota
opUnify
opUnifyUnchecked
opDisjunction
opLand
opLor
opNot
opEql
opNeq
opMat
opNMat
opLss
opGtr
opLeq
opGeq
opAdd
opSub
opMul
opQuo
opRem
opIDiv
opIMod
opIQuo
opIRem
)
var opStrings = []string{
opUnknown: "??",
opUnify: "&",
// opUnifyUnchecked is internal only. Syntactically this is
// represented as embedding.
opUnifyUnchecked: "&!",
opDisjunction: "|",
opLand: "&&",
opLor: "||",
opNot: "!",
opEql: "==",
opNeq: "!=",
opMat: "=~",
opNMat: "!~",
opLss: "<",
opGtr: ">",
opLeq: "<=",
opGeq: ">=",
opAdd: "+",
opSub: "-",
opMul: "*",
opQuo: "/",
opIDiv: "div",
opIMod: "mod",
opIQuo: "quo",
opIRem: "rem",
}
func (op op) String() string { return opStrings[op] }
var tokenMap = map[token.Token]op{
token.OR: opDisjunction, // |
token.AND: opUnify, // &
token.ADD: opAdd, // +
token.SUB: opSub, // -
token.MUL: opMul, // *
token.QUO: opQuo, // /
token.IDIV: opIDiv, // div
token.IMOD: opIMod, // mod
token.IQUO: opIQuo, // quo
token.IREM: opIRem, // rem
token.LAND: opLand, // &&
token.LOR: opLor, // ||
token.EQL: opEql, // ==
token.LSS: opLss, // <
token.GTR: opGtr, // >
token.NOT: opNot, // !
token.NEQ: opNeq, // !=
token.LEQ: opLeq, // <=
token.GEQ: opGeq, // >=
token.MAT: opMat, // =~
token.NMAT: opNMat, // !~
}
var opMap = map[op]token.Token{}
func init() {
for t, o := range tokenMap {
opMap[o] = t
}
}