builtin funcs with multiple param types

This commit is contained in:
Irmen de Jong 2018-09-30 20:19:25 +02:00
parent 02075f2cf2
commit d1589bef26
3 changed files with 132 additions and 89 deletions

View File

@ -9,12 +9,81 @@ sub start() {
str_p spvar = "spvar" str_p spvar = "spvar"
str_s ssvar = "ssvar" str_s ssvar = "ssvar"
str_ps spsvar = "spsvar" str_ps spsvar = "spsvar"
byte[2,3] matrixvar1 byte[2,3] matrixvar
byte[2,3] matrixvar2 = 2
byte[5] barrayvar byte[5] barrayvar
byte[5] barrayvar1 = 44
word[5] warrayvar word[5] warrayvar
word[5] warrayvar1 = 22
set_carry()
clear_carry()
set_irqd()
clear_irqd()
rol(bvar)
rol(wvar)
rol(fvar)
rol(svar)
rol(spvar)
rol(ssvar)
rol(spsvar)
rol(matrixvar)
rol(barrayvar)
rol(warrayvar)
rol2(bvar)
rol2(wvar)
rol2(fvar)
rol2(svar)
rol2(spvar)
rol2(ssvar)
rol2(spsvar)
rol2(matrixvar)
rol2(barrayvar)
rol2(warrayvar)
ror(bvar)
ror(wvar)
ror(fvar)
ror(svar)
ror(spvar)
ror(ssvar)
ror(spsvar)
ror(matrixvar)
ror(barrayvar)
ror(warrayvar)
ror2(bvar)
ror2(wvar)
ror2(fvar)
ror2(svar)
ror2(spvar)
ror2(ssvar)
ror2(spsvar)
ror2(matrixvar)
ror2(barrayvar)
ror2(warrayvar)
lsl(bvar)
lsl(wvar)
lsl(fvar)
lsl(svar)
lsl(spvar)
lsl(ssvar)
lsl(spsvar)
lsl(matrixvar)
lsl(barrayvar)
lsl(warrayvar)
lsr(bvar)
lsr(wvar)
lsr(fvar)
lsr(svar)
lsr(spvar)
lsr(ssvar)
lsr(spsvar)
lsr(matrixvar)
lsr(barrayvar)
lsr(warrayvar)
} }
} }

View File

@ -1152,11 +1152,11 @@ class FunctionCall(override var target: IdentifierReference,
var resultValue: LiteralValue? = null var resultValue: LiteralValue? = null
val func = BuiltinFunctions[target.nameInSource[0]] val func = BuiltinFunctions[target.nameInSource[0]]
if(func!=null) { if(func!=null) {
val exprfunc = func.expressionFunc val exprfunc = func.constExpressionFunc
if(exprfunc!=null) if(exprfunc!=null)
resultValue = exprfunc(arglist, position, namespace, heap) resultValue = exprfunc(arglist, position, namespace, heap)
else if(func.returnvalues.isEmpty()) else if(func.returntype==null)
throw ExpressionError("builtin function ${target.nameInSource[0]} can't be used in expressions because it doesn't return a value", position) throw ExpressionError("builtin function ${target.nameInSource[0]} can't be used here because it doesn't return a value", position)
} }
if(withDatatypeCheck) { if(withDatatypeCheck) {

View File

@ -5,87 +5,64 @@ import prog8.compiler.HeapValues
import kotlin.math.log2 import kotlin.math.log2
class FunctionSignature(val pure: Boolean, // does it have side effects?
val parameters: List<SubroutineParameter>,
val paramTypesVariable: Boolean,
val returnvalues: List<DataType>,
val type: DataType?,
val expressionFunc: ((args: List<IExpression>, position: Position, namespace: INameScope, heap: HeapValues) -> LiteralValue)?) {
companion object {
private val dummyPos = Position("dummy", 0, 0, 0)
fun sig(pure: Boolean, class BuiltinFunctionParam(val name: String, possibleDatatypes: List<DataType>)
args: List<Pair<String, DataType?>>,
hasReturnValue: Boolean, class FunctionSignature(val pure: Boolean, // does it have side effects?
type: DataType?, val parameters: List<BuiltinFunctionParam>,
expressionFunc: ((args: List<IExpression>, position: Position, namespace: INameScope, heap: HeapValues) -> LiteralValue)? = null val returntype: DataType?,
) : FunctionSignature { val constExpressionFunc: ((args: List<IExpression>, position: Position, namespace: INameScope, heap: HeapValues) -> LiteralValue)? = null)
if(!hasReturnValue && expressionFunc!=null)
throw IllegalArgumentException("can't have expression func when hasReturnValue is false")
return FunctionSignature(pure,
args.map { SubroutineParameter(it.first, it.second ?: DataType.BYTE, dummyPos) },
args.any { it.second==null },
if(hasReturnValue && type!=null)
listOf(type)
else
emptyList(),
type,
expressionFunc
)
}
}
}
val BuiltinFunctions = mapOf( val BuiltinFunctions = mapOf(
"set_carry" to FunctionSignature.sig(false, emptyList(), false, null), "rol" to FunctionSignature(false, listOf(BuiltinFunctionParam("item", listOf(DataType.BYTE, DataType.WORD))), null),
"clear_carry" to FunctionSignature.sig(false, emptyList(), false, null), "ror" to FunctionSignature(false, listOf(BuiltinFunctionParam("item", listOf(DataType.BYTE, DataType.WORD))), null),
"set_irqd" to FunctionSignature.sig(false, emptyList(), false, null), "rol2" to FunctionSignature(false, listOf(BuiltinFunctionParam("item", listOf(DataType.BYTE, DataType.WORD))), null),
"clear_irqd" to FunctionSignature.sig(false, emptyList(), false, null), "ror2" to FunctionSignature(false, listOf(BuiltinFunctionParam("item", listOf(DataType.BYTE, DataType.WORD))), null),
"rol" to FunctionSignature.sig(false, listOf(Pair("item", null)), false, null), "lsl" to FunctionSignature(false, listOf(BuiltinFunctionParam("item", listOf(DataType.BYTE, DataType.WORD))), null),
"ror" to FunctionSignature.sig(false, listOf(Pair("item", null)), false, null), "lsr" to FunctionSignature(false, listOf(BuiltinFunctionParam("item", listOf(DataType.BYTE, DataType.WORD))), null),
"rol2" to FunctionSignature.sig(false, listOf(Pair("item", null)), false, null), "sin" to FunctionSignature(true, listOf(BuiltinFunctionParam("rads", listOf(DataType.FLOAT))), DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::sin) },
"ror2" to FunctionSignature.sig(false, listOf(Pair("item", null)), false, null), "cos" to FunctionSignature(true, listOf(BuiltinFunctionParam("rads", listOf(DataType.FLOAT))), DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::cos) },
"lsl" to FunctionSignature.sig(false, listOf(Pair("item", null)), false, null), "acos" to FunctionSignature(true, listOf(BuiltinFunctionParam("rads", listOf(DataType.FLOAT))), DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::acos) },
"lsr" to FunctionSignature.sig(false, listOf(Pair("item", null)), false, null), "asin" to FunctionSignature(true, listOf(BuiltinFunctionParam("rads", listOf(DataType.FLOAT))), DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::asin) },
"sin" to FunctionSignature.sig(true, listOf(Pair("rads", DataType.FLOAT)), true, DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::sin) }, "tan" to FunctionSignature(true, listOf(BuiltinFunctionParam("rads", listOf(DataType.FLOAT))), DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::tan) },
"cos" to FunctionSignature.sig(true, listOf(Pair("rads", DataType.FLOAT)), true, DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::cos) }, "atan" to FunctionSignature(true, listOf(BuiltinFunctionParam("rads", listOf(DataType.FLOAT))), DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::atan) },
"acos" to FunctionSignature.sig(true, listOf(Pair("rads", DataType.FLOAT)), true, DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::acos) }, "ln" to FunctionSignature(true, listOf(BuiltinFunctionParam("value", listOf(DataType.FLOAT))), DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::log) },
"asin" to FunctionSignature.sig(true, listOf(Pair("rads", DataType.FLOAT)), true, DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::asin) }, "log2" to FunctionSignature(true, listOf(BuiltinFunctionParam("value", listOf(DataType.FLOAT))), DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, ::log2) },
"tan" to FunctionSignature.sig(true, listOf(Pair("rads", DataType.FLOAT)), true, DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::tan) }, "log10" to FunctionSignature(true, listOf(BuiltinFunctionParam("value", listOf(DataType.FLOAT))), DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::log10) },
"atan" to FunctionSignature.sig(true, listOf(Pair("rads", DataType.FLOAT)), true, DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::atan) }, "sqrt" to FunctionSignature(true, listOf(BuiltinFunctionParam("value", listOf(DataType.FLOAT))), DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::sqrt) },
"rnd" to FunctionSignature.sig(true, emptyList(), true, DataType.BYTE), "rad" to FunctionSignature(true, listOf(BuiltinFunctionParam("value", listOf(DataType.FLOAT))), DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::toRadians) },
"rndw" to FunctionSignature.sig(true, emptyList(), true, DataType.WORD), "deg" to FunctionSignature(true, listOf(BuiltinFunctionParam("value", listOf(DataType.FLOAT))), DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::toDegrees) },
"rndf" to FunctionSignature.sig(true, emptyList(), true, DataType.FLOAT), "avg" to FunctionSignature(true, listOf(BuiltinFunctionParam("values", listOf(DataType.ARRAY, DataType.ARRAY_W, DataType.MATRIX))), DataType.FLOAT, ::builtinAvg),
"ln" to FunctionSignature.sig(true, listOf(Pair("value", DataType.FLOAT)), true, DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::log) }, "abs" to FunctionSignature(true, listOf(BuiltinFunctionParam("value", listOf(DataType.FLOAT))), DataType.FLOAT, ::builtinAbs),
"log2" to FunctionSignature.sig(true, listOf(Pair("value", DataType.FLOAT)), true, DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, ::log2) }, "round" to FunctionSignature(true, listOf(BuiltinFunctionParam("value", listOf(DataType.FLOAT))), null) { a, p, n, h -> oneDoubleArgOutputInt(a, p, n, h, Math::round) }, // type depends on arg
"log10" to FunctionSignature.sig(true, listOf(Pair("value", DataType.FLOAT)), true, DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::log10) }, "floor" to FunctionSignature(true, listOf(BuiltinFunctionParam("value", listOf(DataType.FLOAT))), null) { a, p, n, h -> oneDoubleArgOutputInt(a, p, n, h, Math::floor) }, // type depends on arg
"sqrt" to FunctionSignature.sig(true, listOf(Pair("value", DataType.FLOAT)), true, DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::sqrt) }, "ceil" to FunctionSignature(true, listOf(BuiltinFunctionParam("value", listOf(DataType.FLOAT))), null) { a, p, n, h -> oneDoubleArgOutputInt(a, p, n, h, Math::ceil) }, // type depends on arg
"rad" to FunctionSignature.sig(true, listOf(Pair("value", DataType.FLOAT)), true, DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::toRadians) }, "max" to FunctionSignature(true, listOf(BuiltinFunctionParam("values", listOf(DataType.ARRAY, DataType.ARRAY_W, DataType.MATRIX))), null) { a, p, n, h -> collectionArgOutputNumber(a, p, n, h) { it.max()!! }}, // type depends on args
"deg" to FunctionSignature.sig(true, listOf(Pair("value", DataType.FLOAT)), true, DataType.FLOAT) { a, p, n, h -> oneDoubleArg(a, p, n, h, Math::toDegrees) }, "min" to FunctionSignature(true, listOf(BuiltinFunctionParam("values", listOf(DataType.ARRAY, DataType.ARRAY_W, DataType.MATRIX))), null) { a, p, n, h -> collectionArgOutputNumber(a, p, n, h) { it.min()!! }}, // type depends on args
"avg" to FunctionSignature.sig(true, listOf(Pair("values", null)), true, DataType.FLOAT, ::builtinAvg), "sum" to FunctionSignature(true, listOf(BuiltinFunctionParam("values", listOf(DataType.ARRAY, DataType.ARRAY_W, DataType.MATRIX))), null) { a, p, n, h -> collectionArgOutputNumber(a, p, n, h) { it.sum() }}, // type depends on args
"abs" to FunctionSignature.sig(true, listOf(Pair("value", null)), true, null, ::builtinAbs), // type depends on arg "len" to FunctionSignature(true, listOf(BuiltinFunctionParam("values", listOf(DataType.STR, DataType.STR_P, DataType.STR_S, DataType.STR_PS, DataType.ARRAY, DataType.ARRAY_W, DataType.MATRIX))), null, ::builtinLen), // type depends on args
"round" to FunctionSignature.sig(true, listOf(Pair("value", DataType.FLOAT)), true, null) { a, p, n, h -> oneDoubleArgOutputInt(a, p, n, h, Math::round) }, // type depends on arg "any" to FunctionSignature(true, listOf(BuiltinFunctionParam("values", listOf(DataType.ARRAY, DataType.ARRAY_W, DataType.MATRIX))), DataType.BYTE) { a, p, n, h -> collectionArgOutputBoolean(a, p, n, h) { it.any { v -> v != 0.0} }},
"floor" to FunctionSignature.sig(true, listOf(Pair("value", DataType.FLOAT)), true, null) { a, p, n, h -> oneDoubleArgOutputInt(a, p, n, h, Math::floor) }, // type depends on arg "all" to FunctionSignature(true, listOf(BuiltinFunctionParam("values", listOf(DataType.ARRAY, DataType.ARRAY_W, DataType.MATRIX))), DataType.BYTE) { a, p, n, h -> collectionArgOutputBoolean(a, p, n, h) { it.all { v -> v != 0.0} }},
"ceil" to FunctionSignature.sig(true, listOf(Pair("value", DataType.FLOAT)), true, null) { a, p, n, h -> oneDoubleArgOutputInt(a, p, n, h, Math::ceil) }, // type depends on arg "lsb" to FunctionSignature(true, listOf(BuiltinFunctionParam("value", listOf(DataType.WORD))), DataType.BYTE) { a, p, n, h -> oneIntArgOutputInt(a, p, n, h) { x: Int -> x and 255 }},
"max" to FunctionSignature.sig(true, listOf(Pair("values", null)), true, null) { a, p, n, h -> collectionArgOutputNumber(a, p, n, h) { it.max()!! }}, // type depends on args "msb" to FunctionSignature(true, listOf(BuiltinFunctionParam("value", listOf(DataType.WORD))), DataType.BYTE) { a, p, n, h -> oneIntArgOutputInt(a, p, n, h) { x: Int -> x ushr 8 and 255}},
"min" to FunctionSignature.sig(true, listOf(Pair("values", null)), true, null) { a, p, n, h -> collectionArgOutputNumber(a, p, n, h) { it.min()!! }}, // type depends on args "flt" to FunctionSignature(true, listOf(BuiltinFunctionParam("value", listOf(DataType.BYTE, DataType.WORD))), DataType.FLOAT, ::builtinFlt),
"sum" to FunctionSignature.sig(true, listOf(Pair("values", null)), true, null) { a, p, n, h -> collectionArgOutputNumber(a, p, n, h) { it.sum() }}, // type depends on args "rnd" to FunctionSignature(true, emptyList(), DataType.BYTE),
"len" to FunctionSignature.sig(true, listOf(Pair("values", null)), true, null, ::builtinLen), // type depends on args "rndw" to FunctionSignature(true, emptyList(), DataType.WORD),
"any" to FunctionSignature.sig(true, listOf(Pair("values", null)), true, DataType.BYTE) { a, p, n, h -> collectionArgOutputBoolean(a, p, n, h) { it.any { v -> v != 0.0} }}, "rndf" to FunctionSignature(true, emptyList(), DataType.FLOAT),
"all" to FunctionSignature.sig(true, listOf(Pair("values", null)), true, DataType.BYTE) { a, p, n, h -> collectionArgOutputBoolean(a, p, n, h) { it.all { v -> v != 0.0} }}, "set_carry" to FunctionSignature(false, emptyList(), null),
"lsb" to FunctionSignature.sig(true, listOf(Pair("value", DataType.WORD)), true, DataType.BYTE) { a, p, n, h -> oneIntArgOutputInt(a, p, n, h) { x: Int -> x and 255 }}, "clear_carry" to FunctionSignature(false, emptyList(), null),
"msb" to FunctionSignature.sig(true, listOf(Pair("value", DataType.WORD)), true, DataType.BYTE) { a, p, n, h -> oneIntArgOutputInt(a, p, n, h) { x: Int -> x ushr 8 and 255}}, "set_irqd" to FunctionSignature(false, emptyList(), null),
"flt" to FunctionSignature.sig(true, listOf(Pair("value", null)), true, DataType.FLOAT, ::builtinFlt), "clear_irqd" to FunctionSignature(false, emptyList(), null),
"_vm_write_memchr" to FunctionSignature.sig(false, emptyList(), false, null), "_vm_write_memchr" to FunctionSignature(false, emptyList(), null),
"_vm_write_memstr" to FunctionSignature.sig(false, emptyList(), false, null), "_vm_write_memstr" to FunctionSignature(false, emptyList(), null),
"_vm_write_num" to FunctionSignature.sig(false, emptyList(), false, null), "_vm_write_num" to FunctionSignature(false, emptyList(), null),
"_vm_write_char" to FunctionSignature.sig(false, emptyList(), false, null), "_vm_write_char" to FunctionSignature(false, emptyList(), null),
"_vm_write_str" to FunctionSignature.sig(false, emptyList(), false, null), "_vm_write_str" to FunctionSignature(false, emptyList(), null),
"_vm_input_str" to FunctionSignature.sig(false, emptyList(), false, null), "_vm_input_str" to FunctionSignature(false, emptyList(), null),
"_vm_gfx_clearscr" to FunctionSignature.sig(false, emptyList(), false, null), "_vm_gfx_clearscr" to FunctionSignature(false, emptyList(), null),
"_vm_gfx_pixel" to FunctionSignature.sig(false, emptyList(), false, null), "_vm_gfx_pixel" to FunctionSignature(false, emptyList(), null),
"_vm_gfx_text" to FunctionSignature.sig(false, emptyList(), false, null) "_vm_gfx_text" to FunctionSignature(false, emptyList(), null)
) )
@ -127,14 +104,11 @@ fun builtinFunctionReturnType(function: String, args: List<IExpression>, namespa
} }
val func = BuiltinFunctions[function]!! val func = BuiltinFunctions[function]!!
if(func.returnvalues.isEmpty()) if(func.returntype!=null)
return null return func.returntype
if(func.type!=null)
return func.type
// function has return values, but the return type depends on the arguments // function has return values, but the return type depends on the arguments
return when (function) { return when (function) {
"abs" -> args.single().resultingDatatype(namespace, heap)
"max", "min" -> { "max", "min" -> {
val dt = datatypeFromListArg(args.single()) val dt = datatypeFromListArg(args.single())
when(dt) { when(dt) {