/* * This file is part of the Colobot: Gold Edition source code * Copyright (C) 2001-2021, 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 */ #include "CBot/CBotInstr/CBotExprLitNum.h" #include "CBot/CBotStack.h" #include "CBot/CBotCStack.h" #include "CBot/CBotVar/CBotVar.h" #include "CBot/CBotUtils.h" #include #include namespace CBot { template <> CBotExprLitNum::CBotExprLitNum(int val) : m_numtype(CBotTypInt), m_value(val) { } template <> CBotExprLitNum::CBotExprLitNum(long val) : m_numtype(CBotTypLong), m_value(val) { } template <> CBotExprLitNum::CBotExprLitNum(float val) : m_numtype(CBotTypFloat), m_value(val) { } template <> CBotExprLitNum::CBotExprLitNum(double val) : m_numtype(CBotTypDouble), m_value(val) { } template CBotExprLitNum::~CBotExprLitNum() { } CBotInstr* CompileExprLitNum(CBotToken* &p, CBotCStack* pStack) { CBotCStack* pStk = pStack->TokenStack(); const auto& s = p->GetString(); CBotInstr* inst = nullptr; CBotType numtype = CBotTypInt; if (p->GetType() == TokenTypDef) { inst = new CBotExprLitNum(static_cast(p->GetKeywordId())); } else { if (s.find('.') != std::string::npos || ( s.find('x') == std::string::npos && ( s.find_first_of("eE") != std::string::npos ) )) { double val = GetNumFloat(s); if (val > static_cast(std::numeric_limits::max())) { numtype = CBotTypDouble; inst = new CBotExprLitNum(val); } else { numtype = CBotTypFloat; inst = new CBotExprLitNum(static_cast(val)); } } else { long val = GetNumInt(s); if (val > std::numeric_limits::max()) { numtype = CBotTypLong; inst = new CBotExprLitNum(val); } else { inst = new CBotExprLitNum(static_cast(val)); } } } inst->SetToken(p); if (pStk->NextToken(p)) { CBotVar* var = CBotVar::Create("", numtype); pStk->SetVar(var); return pStack->Return(inst, pStk); } delete inst; return pStack->Return(nullptr, pStk); } CBotInstr* CompileSizeOf(CBotToken* &p, CBotCStack* pStack) { CBotToken* pp = p; if (!IsOfType(p, TokenTypVar)) return nullptr; if (pp->GetString() == "sizeof" && IsOfType(p, ID_OPENPAR)) { CBotCStack* pStk = pStack->TokenStack(); int value; if (IsOfType(p, ID_BYTE)) value = sizeof(signed char); else if (IsOfType(p, ID_SHORT)) value = sizeof(short); else if (IsOfType(p, ID_CHAR)) value = sizeof(uint32_t); else if (IsOfType(p, ID_INT)) value = sizeof(int); else if (IsOfType(p, ID_LONG)) value = sizeof(long); else if (IsOfType(p, ID_FLOAT)) value = sizeof(float); else if (IsOfType(p, ID_DOUBLE)) value = sizeof(double); else { p = pp; return pStack->Return(nullptr, pStk); } if (IsOfType(p, ID_CLOSEPAR)) { auto inst = new CBotExprLitNum(value); inst->SetToken(pp); CBotVar* var = CBotVar::Create("", CBotTypInt); pStk->SetVar(var); return pStack->Return(inst, pStk); } pStk->SetError(CBotErrClosePar, p->GetStart()); return pStack->Return(nullptr, pStk); } p = pp; return nullptr; } template bool CBotExprLitNum::Execute(CBotStack* &pj) { CBotStack* pile = pj->AddStack(this); if (pile->IfStep()) return false; CBotVar* var = CBotVar::Create("", m_numtype); if (m_token.GetType() == TokenTypDef) { var->SetValInt(m_value, m_token.GetString()); } else { *var = m_value; } pile->SetVar(var); // place on the stack return pj->Return(pile); // it's ok } template void CBotExprLitNum::RestoreState(CBotStack* &pj, bool bMain) { if (bMain) pj->RestoreStack(this); } template std::string CBotExprLitNum::GetDebugData() { std::stringstream ss; switch (m_numtype) { case CBotTypInt : ss << "(int) "; break; case CBotTypLong : ss << "(long) "; break; case CBotTypFloat : ss << "(float) "; break; case CBotTypDouble: ss << "(double) "; break; default: assert(false); } ss << m_value; return ss.str(); } } // namespace CBot