|
|
@ -53,48 +53,77 @@ namespace Compiler
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Extensions::registerFunction (const std::string& keyword, char returnType,
|
|
|
|
void Extensions::registerFunction (const std::string& keyword, char returnType,
|
|
|
|
const std::string& argumentType, int segment5code, int segment5codeExplicit)
|
|
|
|
const std::string& argumentType, int code, int codeExplicit)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
assert (segment5code>=33554432 && segment5code<=67108863);
|
|
|
|
Function function;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (argumentType.find ('/')==std::string::npos)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
function.mSegment = 5;
|
|
|
|
|
|
|
|
assert (code>=33554432 && code<=67108863);
|
|
|
|
|
|
|
|
assert (codeExplicit==-1 || (codeExplicit>=33554432 && codeExplicit<=67108863));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
function.mSegment = 3;
|
|
|
|
|
|
|
|
assert (code>=0x20000 && code<=0x2ffff);
|
|
|
|
|
|
|
|
assert (codeExplicit==-1 || (codeExplicit>=0x20000 && codeExplicit<=0x2ffff));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int code = mNextKeywordIndex--;
|
|
|
|
int keywordIndex = mNextKeywordIndex--;
|
|
|
|
|
|
|
|
|
|
|
|
mKeywords.insert (std::make_pair (keyword, code));
|
|
|
|
mKeywords.insert (std::make_pair (keyword, keywordIndex));
|
|
|
|
|
|
|
|
|
|
|
|
Function function;
|
|
|
|
|
|
|
|
function.mReturn = returnType;
|
|
|
|
function.mReturn = returnType;
|
|
|
|
function.mArguments = argumentType;
|
|
|
|
function.mArguments = argumentType;
|
|
|
|
function.mCode = segment5code;
|
|
|
|
function.mCode = code;
|
|
|
|
function.mCodeExplicit = segment5codeExplicit;
|
|
|
|
function.mCodeExplicit = codeExplicit;
|
|
|
|
|
|
|
|
|
|
|
|
mFunctions.insert (std::make_pair (code, function));
|
|
|
|
mFunctions.insert (std::make_pair (keywordIndex, function));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Extensions::registerInstruction (const std::string& keyword,
|
|
|
|
void Extensions::registerInstruction (const std::string& keyword,
|
|
|
|
const std::string& argumentType, int segment5code, int segment5codeExplicit)
|
|
|
|
const std::string& argumentType, int code, int codeExplicit)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
assert (segment5code>=33554432 && segment5code<=67108863);
|
|
|
|
Instruction instruction;
|
|
|
|
|
|
|
|
|
|
|
|
int code = mNextKeywordIndex--;
|
|
|
|
if (argumentType.find ('/')==std::string::npos)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
instruction.mSegment = 5;
|
|
|
|
|
|
|
|
assert (code>=33554432 && code<=67108863);
|
|
|
|
|
|
|
|
assert (codeExplicit==-1 || (codeExplicit>=33554432 && codeExplicit<=67108863));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
instruction.mSegment = 3;
|
|
|
|
|
|
|
|
assert (code>=0x20000 && code<=0x2ffff);
|
|
|
|
|
|
|
|
assert (codeExplicit==-1 || (codeExplicit>=0x20000 && codeExplicit<=0x2ffff));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
mKeywords.insert (std::make_pair (keyword, code));
|
|
|
|
int keywordIndex = mNextKeywordIndex--;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
mKeywords.insert (std::make_pair (keyword, keywordIndex));
|
|
|
|
|
|
|
|
|
|
|
|
Instruction instruction;
|
|
|
|
|
|
|
|
instruction.mArguments = argumentType;
|
|
|
|
instruction.mArguments = argumentType;
|
|
|
|
instruction.mCode = segment5code;
|
|
|
|
instruction.mCode = code;
|
|
|
|
instruction.mCodeExplicit = segment5codeExplicit;
|
|
|
|
instruction.mCodeExplicit = codeExplicit;
|
|
|
|
|
|
|
|
|
|
|
|
mInstructions.insert (std::make_pair (code, instruction));
|
|
|
|
mInstructions.insert (std::make_pair (keywordIndex, instruction));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Extensions::generateFunctionCode (int keyword, std::vector<Interpreter::Type_Code>& code,
|
|
|
|
void Extensions::generateFunctionCode (int keyword, std::vector<Interpreter::Type_Code>& code,
|
|
|
|
Literals& literals, const std::string& id) const
|
|
|
|
Literals& literals, const std::string& id, int optionalArguments) const
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
assert (optionalArguments>=0);
|
|
|
|
|
|
|
|
|
|
|
|
std::map<int, Function>::const_iterator iter = mFunctions.find (keyword);
|
|
|
|
std::map<int, Function>::const_iterator iter = mFunctions.find (keyword);
|
|
|
|
|
|
|
|
|
|
|
|
if (iter==mFunctions.end())
|
|
|
|
if (iter==mFunctions.end())
|
|
|
|
throw std::logic_error ("unknown custom function keyword");
|
|
|
|
throw std::logic_error ("unknown custom function keyword");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (optionalArguments && iter->second.mSegment!=3)
|
|
|
|
|
|
|
|
throw std::logic_error ("functions with optional arguments must be placed into segment 3");
|
|
|
|
|
|
|
|
|
|
|
|
if (!id.empty())
|
|
|
|
if (!id.empty())
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (iter->second.mCodeExplicit==-1)
|
|
|
|
if (iter->second.mCodeExplicit==-1)
|
|
|
@ -104,19 +133,46 @@ namespace Compiler
|
|
|
|
Generator::pushInt (code, literals, index);
|
|
|
|
Generator::pushInt (code, literals, index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
code.push_back (Generator::segment5 (
|
|
|
|
switch (iter->second.mSegment)
|
|
|
|
id.empty() ? iter->second.mCode : iter->second.mCodeExplicit));
|
|
|
|
{
|
|
|
|
|
|
|
|
case 3:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (optionalArguments>=256)
|
|
|
|
|
|
|
|
throw std::logic_error ("number of optional arguments is too large for segment 3");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
code.push_back (Generator::segment3 (
|
|
|
|
|
|
|
|
id.empty() ? iter->second.mCode : iter->second.mCodeExplicit,
|
|
|
|
|
|
|
|
optionalArguments));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
case 5:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
code.push_back (Generator::segment5 (
|
|
|
|
|
|
|
|
id.empty() ? iter->second.mCode : iter->second.mCodeExplicit));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
throw std::logic_error ("unsupported code segment");
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Extensions::generateInstructionCode (int keyword,
|
|
|
|
void Extensions::generateInstructionCode (int keyword,
|
|
|
|
std::vector<Interpreter::Type_Code>& code, Literals& literals, const std::string& id)
|
|
|
|
std::vector<Interpreter::Type_Code>& code, Literals& literals, const std::string& id,
|
|
|
|
const
|
|
|
|
int optionalArguments) const
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
assert (optionalArguments>=0);
|
|
|
|
|
|
|
|
|
|
|
|
std::map<int, Instruction>::const_iterator iter = mInstructions.find (keyword);
|
|
|
|
std::map<int, Instruction>::const_iterator iter = mInstructions.find (keyword);
|
|
|
|
|
|
|
|
|
|
|
|
if (iter==mInstructions.end())
|
|
|
|
if (iter==mInstructions.end())
|
|
|
|
throw std::logic_error ("unknown custom instruction keyword");
|
|
|
|
throw std::logic_error ("unknown custom instruction keyword");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (optionalArguments && iter->second.mSegment!=3)
|
|
|
|
|
|
|
|
throw std::logic_error ("instructions with optional arguments must be placed into segment 3");
|
|
|
|
|
|
|
|
|
|
|
|
if (!id.empty())
|
|
|
|
if (!id.empty())
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (iter->second.mCodeExplicit==-1)
|
|
|
|
if (iter->second.mCodeExplicit==-1)
|
|
|
@ -126,7 +182,29 @@ namespace Compiler
|
|
|
|
Generator::pushInt (code, literals, index);
|
|
|
|
Generator::pushInt (code, literals, index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
code.push_back (Generator::segment5 (
|
|
|
|
switch (iter->second.mSegment)
|
|
|
|
id.empty() ? iter->second.mCode : iter->second.mCodeExplicit));
|
|
|
|
{
|
|
|
|
|
|
|
|
case 3:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (optionalArguments>=256)
|
|
|
|
|
|
|
|
throw std::logic_error ("number of optional arguments is too large for segment 3");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
code.push_back (Generator::segment3 (
|
|
|
|
|
|
|
|
id.empty() ? iter->second.mCode : iter->second.mCodeExplicit,
|
|
|
|
|
|
|
|
optionalArguments));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
case 5:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
code.push_back (Generator::segment5 (
|
|
|
|
|
|
|
|
id.empty() ? iter->second.mCode : iter->second.mCodeExplicit));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
throw std::logic_error ("unsupported code segment");
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|