diff --git a/apps/openmw/mwscript/compilercontext.cpp b/apps/openmw/mwscript/compilercontext.cpp index bcd88b620..1e5cdaf4a 100644 --- a/apps/openmw/mwscript/compilercontext.cpp +++ b/apps/openmw/mwscript/compilercontext.cpp @@ -18,7 +18,16 @@ namespace MWScript char CompilerContext::getGlobalType (const std::string& name) const { if (const ESM::Global *global = mEnvironment.mWorld->getStore().globals.find (name)) - return global->type; + { + switch (global->type) + { + case ESM::VT_Short: return 's'; + case ESM::VT_Int: return 'l'; + case ESM::VT_Float: return 'f'; + + default: return ' '; + } + } return ' '; } diff --git a/components/compiler/exprparser.cpp b/components/compiler/exprparser.cpp index 47e8eadaa..6eee56e8d 100644 --- a/components/compiler/exprparser.cpp +++ b/components/compiler/exprparser.cpp @@ -304,6 +304,17 @@ namespace Compiler mNextOperand = false; return true; } + else if (keyword==Scanner::K_random) + { + mTokenLoc = loc; + parseArguments ("l", scanner); + + Generator::random (mCode); + mOperands.push_back ('l'); + + mNextOperand = false; + return true; + } else { // check for custom extensions diff --git a/components/compiler/generator.cpp b/components/compiler/generator.cpp index a11e0969f..692a94348 100644 --- a/components/compiler/generator.cpp +++ b/components/compiler/generator.cpp @@ -254,6 +254,11 @@ namespace { code.push_back (Compiler::Generator::segment5 (44)); } + + void opRandom (Compiler::Generator::CodeContainer& code) + { + code.push_back (Compiler::Generator::segment5 (45)); + } } namespace Compiler @@ -625,6 +630,11 @@ namespace Compiler assert (0); } } + + void random (CodeContainer& code) + { + opRandom (code); + } } } diff --git a/components/compiler/generator.hpp b/components/compiler/generator.hpp index 4d5f46cf4..5115d7134 100644 --- a/components/compiler/generator.hpp +++ b/components/compiler/generator.hpp @@ -98,6 +98,8 @@ namespace Compiler void fetchGlobal (CodeContainer& code, Literals& literals, char localType, const std::string& name); + + void random (CodeContainer& code); } } diff --git a/components/compiler/scanner.cpp b/components/compiler/scanner.cpp index 72fe0610e..7370e5c29 100644 --- a/components/compiler/scanner.cpp +++ b/components/compiler/scanner.cpp @@ -243,6 +243,7 @@ namespace Compiler "set", "to", "getsquareroot", "menumode", + "random", 0 }; diff --git a/components/compiler/scanner.hpp b/components/compiler/scanner.hpp index d00f2b065..1af454740 100644 --- a/components/compiler/scanner.hpp +++ b/components/compiler/scanner.hpp @@ -49,7 +49,8 @@ namespace Compiler K_messagebox, K_set, K_to, K_getsquareroot, - K_menumode + K_menumode, + K_random }; enum special diff --git a/components/interpreter/docs/vmformat.txt b/components/interpreter/docs/vmformat.txt index b984c5abe..458268e10 100644 --- a/components/interpreter/docs/vmformat.txt +++ b/components/interpreter/docs/vmformat.txt @@ -103,6 +103,7 @@ op 41: store stack[0] in global float stack[1] and pop twice op 42: replace stack[0] with global short stack[0] op 43: replace stack[0] with global long stack[0] op 44: replace stack[0] with global float stack[0] -opcodes 45-33554431 unused +op 45: replace stack[0] with a random integer value in the range [0, stack[0]-1] +opcodes 46-33554431 unused opcodes 33554432-67108863 reserved for extensions diff --git a/components/interpreter/installopcodes.cpp b/components/interpreter/installopcodes.cpp index 8b96e4bc5..46740a997 100644 --- a/components/interpreter/installopcodes.cpp +++ b/components/interpreter/installopcodes.cpp @@ -85,6 +85,7 @@ namespace Interpreter // misc interpreter.installSegment3 (0, new OpMessageBox); interpreter.installSegment5 (38, new OpMenuMode); + interpreter.installSegment5 (45, new OpRandom); } } diff --git a/components/interpreter/miscopcodes.hpp b/components/interpreter/miscopcodes.hpp index 2ab8f890b..36061247d 100644 --- a/components/interpreter/miscopcodes.hpp +++ b/components/interpreter/miscopcodes.hpp @@ -1,6 +1,7 @@ #ifndef INTERPRETER_MISCOPCODES_H_INCLUDED #define INTERPRETER_MISCOPCODES_H_INCLUDED +#include #include #include #include @@ -97,6 +98,26 @@ namespace Interpreter runtime.push (runtime.getContext().menuMode()); } }; + + class OpRandom : public Opcode0 + { + public: + + virtual void execute (Runtime& runtime) + { + double r = static_cast (std::rand()) / RAND_MAX; // [0, 1) + + Type_Integer limit = *reinterpret_cast (&runtime[0]); + + if (limit<0) + throw std::runtime_error ( + "random: argument out of range (Don't be so negative!)"); + + Type_Integer value = static_cast (r*limit); // [o, limit) + + runtime[0] = *reinterpret_cast (&value); + } + }; } #endif