diff --git a/src/common/scripting/vm/vm.h b/src/common/scripting/vm/vm.h index 2b7344d5dc..dba1a36de3 100644 --- a/src/common/scripting/vm/vm.h +++ b/src/common/scripting/vm/vm.h @@ -534,6 +534,236 @@ inline int VMCallAction(VMFunction *func, VMValue *params, int numparams, VMRetu return VMCall(func, params, numparams, results, numresults); } +template struct VMReturnTypeTrait { typedef T type; static const int ReturnCount = 1; }; +template<> struct VMReturnTypeTrait { typedef void type; static const int ReturnCount = 0; }; + +void VMCheckParamCount(VMFunction* func, int retcount, int argcount); + +template +void VMCheckParamCount(VMFunction* func, int argcount) { return VMCheckParamCount(func, VMReturnTypeTrait::ReturnCount, argcount); } + +// The type can't be mapped to ZScript automatically: + +template void VMCheckParam(VMFunction* func, int index) = delete; +template void VMCheckReturn(VMFunction* func) = delete; + +// Native types we support converting to/from: + +template<> void VMCheckParam(VMFunction* func, int index); +template<> void VMCheckParam(VMFunction* func, int index); +template<> void VMCheckParam(VMFunction* func, int index); +template<> void VMCheckParam(VMFunction* func, int index); + +template<> void VMCheckReturn(VMFunction* func); +template<> void VMCheckReturn(VMFunction* func); +template<> void VMCheckReturn(VMFunction* func); +template<> void VMCheckReturn(VMFunction* func); +template<> void VMCheckReturn(VMFunction* func); + +template void VMValidateSignature(VMFunction* func) +{ + VMCheckParamCount(func, 0); + VMCheckReturn(func); +} + +template void VMValidateSignature(VMFunction* func) +{ + VMCheckParamCount(func, 1); + VMCheckReturn(func); + VMCheckParam(func, 0); +} + +template void VMValidateSignature(VMFunction* func) +{ + VMCheckParamCount(func, 2); + VMCheckReturn(func); + VMCheckParam(func, 0); + VMCheckParam(func, 1); +} + +template void VMValidateSignature(VMFunction* func) +{ + VMCheckParamCount(func, 3); + VMCheckReturn(func); + VMCheckParam(func, 0); + VMCheckParam(func, 1); + VMCheckParam(func, 2); +} + +template void VMValidateSignature(VMFunction* func) +{ + VMCheckParamCount(func, 4); + VMCheckReturn(func); + VMCheckParam(func, 0); + VMCheckParam(func, 1); + VMCheckParam(func, 2); + VMCheckParam(func, 3); +} + +template void VMValidateSignature(VMFunction* func) +{ + VMCheckParamCount(func, 5); + VMCheckReturn(func); + VMCheckParam(func, 0); + VMCheckParam(func, 1); + VMCheckParam(func, 2); + VMCheckParam(func, 3); + VMCheckParam(func, 4); +} + +template void VMValidateSignature(VMFunction* func) +{ + VMCheckParamCount(func, 6); + VMCheckReturn(func); + VMCheckParam(func, 0); + VMCheckParam(func, 1); + VMCheckParam(func, 2); + VMCheckParam(func, 3); + VMCheckParam(func, 4); + VMCheckParam(func, 5); +} + +template void VMValidateSignature(VMFunction* func) +{ + VMCheckParamCount(func, 7); + VMCheckReturn(func); + VMCheckParam(func, 0); + VMCheckParam(func, 1); + VMCheckParam(func, 2); + VMCheckParam(func, 3); + VMCheckParam(func, 4); + VMCheckParam(func, 5); + VMCheckParam(func, 6); +} + +void VMCallCheckResult(VMFunction* func, VMValue* params, int numparams, VMReturn* results, int numresults); + +template +typename VMReturnTypeTrait::type VMCallScript(VMFunction* func, P1 p1) +{ + VMValidateSignature(func); + VMValue params[] = { p1 }; + RetVal resultval; VMReturn results(&resultval); + VMCallCheckResult(func, params, 1, &results, 1); + return resultval; +} + +template +typename VMReturnTypeTrait::type VMCallScript(VMFunction* func, P1 p1, P2 p2) +{ + VMValidateSignature(func); + VMValue params[] = { p1, p2 }; + RetVal resultval; VMReturn results(&resultval); + VMCallCheckResult(func, params, 2, &results, 1); + return resultval; +} + +template +typename VMReturnTypeTrait::type VMCallScript(VMFunction* func, P1 p1, P2 p2, P3 p3) +{ + VMValidateSignature(func); + VMValue params[] = { p1, p2, p3 }; + RetVal resultval; VMReturn results(&resultval); + VMCallCheckResult(func, params, 3, &results, 1); + return resultval; +} + +template +typename VMReturnTypeTrait::type VMCallScript(VMFunction* func, P1 p1, P2 p2, P3 p3, P4 p4) +{ + VMValidateSignature(func); + VMValue params[] = { p1, p2, p3, p4 }; + RetVal resultval; VMReturn results(&resultval); + VMCallCheckResult(func, params, 4, &results, 1); + return resultval; +} + +template +typename VMReturnTypeTrait::type VMCallScript(VMFunction* func, P1 p1, P2 p2, P3 p3, P4 p4, P5 p5) +{ + VMValidateSignature(func); + VMValue params[] = { p1, p2, p3, p4, p5 }; + RetVal resultval; VMReturn results(&resultval); + VMCallCheckResult(func, params, 5, &results, 1); + return resultval; +} + +template +typename VMReturnTypeTrait::type VMCallScript(VMFunction* func, P1 p1, P2 p2, P3 p3, P4 p4, P5 p5, P6 p6) +{ + VMValidateSignature(func); + VMValue params[] = { p1, p2, p3, p4, p5, p6 }; + RetVal resultval; VMReturn results(&resultval); + VMCallCheckResult(func, params, 6, &results, 1); + return resultval; +} + +template +typename VMReturnTypeTrait::type VMCallScript(VMFunction* func, P1 p1, P2 p2, P3 p3, P4 p4, P5 p5, P6 p6, P7 p7) +{ + VMValidateSignature(func); + VMValue params[] = { p1, p2, p3, p4, p5, p6, p7 }; + RetVal resultval; VMReturn results(&resultval); + VMCallCheckResult(func, params, 7, &results, 1); + return resultval; +} + +template +typename VMReturnTypeTrait::type VMCallScript(VMFunction* func, P1 p1) +{ + VMValidateSignature(func); + VMValue params[1] = { p1 }; + VMCallCheckResult(func, params, 1, nullptr, 0); +} + +template +typename VMReturnTypeTrait::type VMCallScript(VMFunction* func, P1 p1, P2 p2) +{ + VMValidateSignature(func); + VMValue params[] = { p1, p2 }; + VMCallCheckResult(func, params, 2, nullptr, 0); +} + +template +typename VMReturnTypeTrait::type VMCallScript(VMFunction* func, P1 p1, P2 p2, P3 p3) +{ + VMValidateSignature(func); + VMValue params[] = { p1, p2, p3 }; + VMCallCheckResult(func, params, 3, nullptr, 0); +} + +template +typename VMReturnTypeTrait::type VMCallScript(VMFunction* func, P1 p1, P2 p2, P3 p3, P4 p4) +{ + VMValidateSignature(func); + VMValue params[] = { p1, p2, p3, p4 }; + VMCallCheckResult(func, params, 4, nullptr, 0); +} + +template +typename VMReturnTypeTrait::type VMCallScript(VMFunction* func, P1 p1, P2 p2, P3 p3, P4 p4, P5 p5) +{ + VMValidateSignature(func); + VMValue params[] = { p1, p2, p3, p4, p5 }; + VMCallCheckResult(func, params, 5, nullptr, 0); +} + +template +typename VMReturnTypeTrait::type VMCallScript(VMFunction* func, P1 p1, P2 p2, P3 p3, P4 p4, P5 p5, P6 p6) +{ + VMValidateSignature(func); + VMValue params[] = { p1, p2, p3, p4, p5, p6 }; + VMCallCheckResult(func, params, 6, nullptr, 0); +} + +template +typename VMReturnTypeTrait::type VMCallScript(VMFunction* func, P1 p1, P2 p2, P3 p3, P4 p4, P5 p5, P6 p6, P7 p7) +{ + VMValidateSignature(func); + VMValue params[] = { p1, p2, p3, p4, p5, p6, p7 }; + VMCallCheckResult(func, params, 7, nullptr, 0); +} + // Use these to collect the parameters in a native function. // variable name at position

[[noreturn]] diff --git a/src/common/scripting/vm/vmframe.cpp b/src/common/scripting/vm/vmframe.cpp index f69b8ea616..e04c6f0d75 100644 --- a/src/common/scripting/vm/vmframe.cpp +++ b/src/common/scripting/vm/vmframe.cpp @@ -552,6 +552,76 @@ VMFrame *VMFrameStack::PopFrame() return parent; } +//=========================================================================== +// Templates for calling ZScript from C++ with type checks + +void VMCheckParamCount(VMFunction* func, int retcount, int argcount) +{ + if (func->Proto->ReturnTypes.Size() != retcount) + I_FatalError("Incorrect return value passed to %s", func->PrintableName); + if (func->Proto->ArgumentTypes.Size() != argcount) + I_FatalError("Incorrect parameter count passed to %s", func->PrintableName); +} + +template<> void VMCheckParam(VMFunction* func, int index) +{ + if (!func->Proto->ArgumentTypes[index]->isIntCompatible()) + I_FatalError("%s argument %d is not an integer", func->PrintableName); +} + +template<> void VMCheckParam(VMFunction* func, int index) +{ + if (func->Proto->ArgumentTypes[index] != TypeFloat64) + I_FatalError("%s argument %d is not a double", func->PrintableName); +} + +template<> void VMCheckParam(VMFunction* func, int index) +{ + if (func->Proto->ArgumentTypes[index] != TypeString) + I_FatalError("%s argument %d is not a string", func->PrintableName); +} + +template<> void VMCheckParam(VMFunction* func, int index) +{ + if (func->Proto->ArgumentTypes[index]->isObjectPointer()) + I_FatalError("%s argument %d is not an object", func->PrintableName); +} + +template<> void VMCheckReturn(VMFunction* func) +{ +} + +template<> void VMCheckReturn(VMFunction* func) +{ + if (!func->Proto->ReturnTypes[0]->isIntCompatible()) + I_FatalError("%s return value %d is not an integer", func->PrintableName); +} + +template<> void VMCheckReturn(VMFunction* func) +{ + if (func->Proto->ReturnTypes[0] != TypeFloat64) + I_FatalError("%s return value %d is not a double", func->PrintableName); +} + +template<> void VMCheckReturn(VMFunction* func) +{ + if (func->Proto->ReturnTypes[0] != TypeString) + I_FatalError("%s return value %d is not a string", func->PrintableName); +} + +template<> void VMCheckReturn(VMFunction* func) +{ + if (func->Proto->ReturnTypes[0]->isObjectPointer()) + I_FatalError("%s return value %d is not an object", func->PrintableName); +} + +void VMCallCheckResult(VMFunction* func, VMValue* params, int numparams, VMReturn* results, int numresults) +{ + int retval = VMCall(func, params, numparams, results, numresults); + if (retval != numresults) + I_FatalError("%s did not return the expected number of results", func->PrintableName); +} + //=========================================================================== // // VMFrameStack :: Call