2015-11-15 15:49:06 +00:00
|
|
|
/*
|
|
|
|
* This file is part of the Colobot: Gold Edition source code
|
|
|
|
* Copyright (C) 2001-2015, Daniel Roux, EPSITEC SA & TerranovaTeam
|
|
|
|
* http://epsitec.ch; http://colobot.info; http://github.com/colobot
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
* See the GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see http://gnu.org/licenses
|
|
|
|
*/
|
|
|
|
|
|
|
|
// Modules inlcude
|
2015-11-23 20:59:56 +00:00
|
|
|
#include "CBot/CBotUtils.h"
|
2015-11-15 15:49:06 +00:00
|
|
|
|
2015-11-23 20:59:56 +00:00
|
|
|
#include "CBot/CBotToken.h"
|
|
|
|
#include "CBot/CBotClass.h"
|
|
|
|
#include "CBot/CBotStack.h"
|
|
|
|
#include "CBot/CBotCStack.h"
|
2015-11-15 17:40:16 +00:00
|
|
|
|
2015-11-23 20:59:56 +00:00
|
|
|
#include "CBot/CBotVar/CBotVar.h"
|
2015-11-15 22:18:47 +00:00
|
|
|
|
2015-11-15 15:49:06 +00:00
|
|
|
// Local include
|
|
|
|
|
|
|
|
// Global include
|
2015-11-22 15:42:51 +00:00
|
|
|
#include <cstring>
|
2015-11-15 15:49:06 +00:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
CBotVar* MakeListVars(CBotVar** ppVars, bool bSetVal)
|
|
|
|
{
|
|
|
|
int i = 0;
|
|
|
|
CBotVar* pVar = nullptr;
|
|
|
|
|
|
|
|
while( true )
|
|
|
|
{
|
|
|
|
// ppVars[i];
|
|
|
|
if ( ppVars[i] == nullptr ) break;
|
|
|
|
|
|
|
|
CBotVar* pp = CBotVar::Create(ppVars[i]);
|
|
|
|
if (bSetVal) pp->Copy(ppVars[i]);
|
|
|
|
else
|
|
|
|
if ( ppVars[i]->GetType() == CBotTypPointer )
|
|
|
|
pp->SetClass( ppVars[i]->GetClass());
|
|
|
|
// copy the pointer according to indirections
|
|
|
|
if (pVar == nullptr) pVar = pp;
|
|
|
|
else pVar->AddNext(pp);
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
return pVar;
|
|
|
|
}
|
2015-11-15 17:40:16 +00:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
CBotTypResult TypeParam(CBotToken* &p, CBotCStack* pile)
|
|
|
|
{
|
|
|
|
CBotClass* pClass = nullptr;
|
|
|
|
|
|
|
|
switch (p->GetType())
|
|
|
|
{
|
|
|
|
case ID_INT:
|
|
|
|
p = p->GetNext();
|
|
|
|
return ArrayType(p, pile, CBotTypResult( CBotTypInt ));
|
|
|
|
case ID_FLOAT:
|
|
|
|
p = p->GetNext();
|
|
|
|
return ArrayType(p, pile, CBotTypResult( CBotTypFloat ));
|
|
|
|
case ID_BOOLEAN:
|
|
|
|
case ID_BOOL:
|
|
|
|
p = p->GetNext();
|
|
|
|
return ArrayType(p, pile, CBotTypResult( CBotTypBoolean ));
|
|
|
|
case ID_STRING:
|
|
|
|
p = p->GetNext();
|
|
|
|
return ArrayType(p, pile, CBotTypResult( CBotTypString ));
|
|
|
|
case ID_VOID:
|
|
|
|
p = p->GetNext();
|
|
|
|
return CBotTypResult( 0 );
|
|
|
|
|
|
|
|
case TokenTypVar:
|
|
|
|
pClass = CBotClass::Find(p);
|
|
|
|
if ( pClass != nullptr)
|
|
|
|
{
|
|
|
|
p = p->GetNext();
|
|
|
|
return ArrayType(p, pile,
|
|
|
|
pClass->IsIntrinsic() ?
|
|
|
|
CBotTypResult( CBotTypIntrinsic, pClass ) :
|
|
|
|
CBotTypResult( CBotTypPointer, pClass ) );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return CBotTypResult( -1 );
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
CBotTypResult ArrayType(CBotToken* &p, CBotCStack* pile, CBotTypResult type)
|
|
|
|
{
|
|
|
|
while ( IsOfType( p, ID_OPBRK ) )
|
|
|
|
{
|
|
|
|
if ( !IsOfType( p, ID_CLBRK ) )
|
|
|
|
{
|
2015-12-20 18:01:03 +00:00
|
|
|
pile->SetError(CBotErrCloseIndex, p->GetStart());
|
2015-11-15 17:40:16 +00:00
|
|
|
return CBotTypResult( -1 );
|
|
|
|
}
|
|
|
|
type = CBotTypResult( CBotTypArrayPointer, type );
|
|
|
|
}
|
|
|
|
return type;
|
|
|
|
}
|
2015-11-15 20:13:50 +00:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
bool WriteWord(FILE* pf, unsigned short w)
|
|
|
|
{
|
|
|
|
size_t lg;
|
|
|
|
|
|
|
|
lg = fwrite(&w, sizeof( unsigned short ), 1, pf );
|
|
|
|
|
|
|
|
return (lg == 1);
|
|
|
|
}
|
2015-11-15 20:24:54 +00:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
2015-12-20 15:19:10 +00:00
|
|
|
bool WriteString(FILE* pf, std::string s)
|
2015-11-15 20:24:54 +00:00
|
|
|
{
|
|
|
|
size_t lg1, lg2;
|
|
|
|
|
2015-12-20 15:19:10 +00:00
|
|
|
lg1 = s.size();
|
2015-11-15 20:24:54 +00:00
|
|
|
if (!WriteWord(pf, lg1)) return false;
|
|
|
|
|
2015-12-20 15:19:10 +00:00
|
|
|
lg2 = fwrite(s.c_str(), 1, lg1, pf );
|
2015-11-15 20:24:54 +00:00
|
|
|
return (lg1 == lg2);
|
|
|
|
}
|
2015-11-15 20:37:56 +00:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
bool WriteFloat(FILE* pf, float w)
|
|
|
|
{
|
|
|
|
size_t lg;
|
|
|
|
|
|
|
|
lg = fwrite(&w, sizeof( float ), 1, pf );
|
|
|
|
|
|
|
|
return (lg == 1);
|
|
|
|
}
|
2015-11-16 21:17:33 +00:00
|
|
|
|
2015-11-16 21:34:30 +00:00
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
2015-12-20 15:19:10 +00:00
|
|
|
long GetNumInt(const std::string& str)
|
2015-11-22 16:25:46 +00:00
|
|
|
{
|
2015-12-20 15:19:10 +00:00
|
|
|
const char* p = str.c_str();
|
2015-11-22 16:25:46 +00:00
|
|
|
long num = 0;
|
|
|
|
while (*p >= '0' && *p <= '9')
|
|
|
|
{
|
|
|
|
num = num * 10 + *p - '0';
|
|
|
|
p++;
|
|
|
|
}
|
|
|
|
if (*p == 'x' || *p == 'X')
|
|
|
|
{
|
|
|
|
while (*++p != 0)
|
|
|
|
{
|
|
|
|
if (*p >= '0' && *p <= '9')
|
|
|
|
{
|
|
|
|
num = num * 16 + *p - '0';
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (*p >= 'A' && *p <= 'F')
|
|
|
|
{
|
|
|
|
num = num * 16 + *p - 'A' + 10;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (*p >= 'a' && *p <= 'f')
|
|
|
|
{
|
|
|
|
num = num * 16 + *p - 'a' + 10;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return num;
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
2015-12-20 15:19:10 +00:00
|
|
|
float GetNumFloat(const std::string& str)
|
2015-11-22 16:25:46 +00:00
|
|
|
{
|
2015-12-20 15:19:10 +00:00
|
|
|
const char* p = str.c_str();
|
2015-11-22 16:25:46 +00:00
|
|
|
double num = 0;
|
|
|
|
double div = 10;
|
|
|
|
bool bNeg = false;
|
|
|
|
|
|
|
|
if (*p == '-')
|
|
|
|
{
|
|
|
|
bNeg = true;
|
|
|
|
p++;
|
|
|
|
}
|
|
|
|
while (*p >= '0' && *p <= '9')
|
|
|
|
{
|
|
|
|
num = num * 10. + (*p - '0');
|
|
|
|
p++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*p == '.')
|
|
|
|
{
|
|
|
|
p++;
|
|
|
|
while (*p >= '0' && *p <= '9')
|
|
|
|
{
|
|
|
|
num = num + (*p - '0') / div;
|
|
|
|
div = div * 10;
|
|
|
|
p++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int exp = 0;
|
|
|
|
if (*p == 'e' || *p == 'E')
|
|
|
|
{
|
|
|
|
char neg = 0;
|
|
|
|
p++;
|
|
|
|
if (*p == '-' || *p == '+') neg = *p++;
|
|
|
|
|
|
|
|
while (*p >= '0' && *p <= '9')
|
|
|
|
{
|
|
|
|
exp = exp * 10 + (*p - '0');
|
|
|
|
p++;
|
|
|
|
}
|
|
|
|
if (neg == '-') exp = -exp;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (exp > 0)
|
|
|
|
{
|
|
|
|
num *= 10.0;
|
|
|
|
exp--;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (exp < 0)
|
|
|
|
{
|
|
|
|
num /= 10.0;
|
|
|
|
exp++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bNeg) num = -num;
|
|
|
|
return static_cast<float>(num);
|
|
|
|
}
|