2019-10-30 17:57:46 +01:00
|
|
|
|
|
|
|
#include "programmemory.h"
|
|
|
|
#include "token.h"
|
|
|
|
#include "astutils.h"
|
|
|
|
#include "symboldatabase.h"
|
|
|
|
#include <cassert>
|
|
|
|
|
2019-10-30 17:58:19 +01:00
|
|
|
void ProgramMemory::setValue(nonneg int varid, const ValueFlow::Value &value)
|
|
|
|
{
|
2019-10-30 17:57:46 +01:00
|
|
|
values[varid] = value;
|
|
|
|
}
|
|
|
|
|
2019-10-30 17:58:19 +01:00
|
|
|
bool ProgramMemory::getIntValue(nonneg int varid, MathLib::bigint* result) const
|
|
|
|
{
|
2019-10-30 17:57:46 +01:00
|
|
|
const std::map<int, ValueFlow::Value>::const_iterator it = values.find(varid);
|
|
|
|
const bool found = it != values.end() && it->second.isIntValue();
|
|
|
|
if (found)
|
|
|
|
*result = it->second.intvalue;
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
2019-10-30 17:58:19 +01:00
|
|
|
void ProgramMemory::setIntValue(nonneg int varid, MathLib::bigint value)
|
|
|
|
{
|
2019-10-30 17:57:46 +01:00
|
|
|
values[varid] = ValueFlow::Value(value);
|
|
|
|
}
|
|
|
|
|
2019-10-30 17:58:19 +01:00
|
|
|
bool ProgramMemory::getTokValue(nonneg int varid, const Token** result) const
|
|
|
|
{
|
2019-10-30 17:57:46 +01:00
|
|
|
const std::map<int, ValueFlow::Value>::const_iterator it = values.find(varid);
|
|
|
|
const bool found = it != values.end() && it->second.isTokValue();
|
|
|
|
if (found)
|
|
|
|
*result = it->second.tokvalue;
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
2020-02-13 17:04:05 +01:00
|
|
|
void ProgramMemory::setUnknown(nonneg int varid)
|
|
|
|
{
|
|
|
|
values[varid].valueType = ValueFlow::Value::ValueType::UNINIT;
|
|
|
|
}
|
2020-02-13 16:27:06 +01:00
|
|
|
|
2019-10-30 17:58:19 +01:00
|
|
|
bool ProgramMemory::hasValue(nonneg int varid)
|
|
|
|
{
|
2019-10-30 17:57:46 +01:00
|
|
|
return values.find(varid) != values.end();
|
|
|
|
}
|
|
|
|
|
2019-10-30 17:58:19 +01:00
|
|
|
void ProgramMemory::swap(ProgramMemory &pm)
|
|
|
|
{
|
2019-10-30 17:57:46 +01:00
|
|
|
values.swap(pm.values);
|
|
|
|
}
|
|
|
|
|
2019-10-30 17:58:19 +01:00
|
|
|
void ProgramMemory::clear()
|
|
|
|
{
|
2019-10-30 17:57:46 +01:00
|
|
|
values.clear();
|
|
|
|
}
|
|
|
|
|
2019-10-30 17:58:19 +01:00
|
|
|
bool ProgramMemory::empty() const
|
|
|
|
{
|
2019-10-30 17:57:46 +01:00
|
|
|
return values.empty();
|
|
|
|
}
|
|
|
|
|
2019-10-30 17:58:19 +01:00
|
|
|
void ProgramMemory::replace(const ProgramMemory &pm)
|
|
|
|
{
|
2019-10-30 17:57:46 +01:00
|
|
|
for (auto&& p:pm.values)
|
|
|
|
values[p.first] = p.second;
|
|
|
|
}
|
|
|
|
|
2019-10-30 17:58:19 +01:00
|
|
|
void ProgramMemory::insert(const ProgramMemory &pm)
|
|
|
|
{
|
2019-10-30 17:57:46 +01:00
|
|
|
for (auto&& p:pm.values)
|
|
|
|
values.insert(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool conditionIsFalse(const Token *condition, const ProgramMemory &programMemory)
|
|
|
|
{
|
|
|
|
if (!condition)
|
|
|
|
return false;
|
|
|
|
if (condition->str() == "&&") {
|
|
|
|
return conditionIsFalse(condition->astOperand1(), programMemory) ||
|
|
|
|
conditionIsFalse(condition->astOperand2(), programMemory);
|
|
|
|
}
|
|
|
|
ProgramMemory progmem(programMemory);
|
|
|
|
MathLib::bigint result = 0;
|
|
|
|
bool error = false;
|
|
|
|
execute(condition, &progmem, &result, &error);
|
|
|
|
return !error && result == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool conditionIsTrue(const Token *condition, const ProgramMemory &programMemory)
|
|
|
|
{
|
|
|
|
if (!condition)
|
|
|
|
return false;
|
|
|
|
if (condition->str() == "||") {
|
|
|
|
return conditionIsTrue(condition->astOperand1(), programMemory) ||
|
|
|
|
conditionIsTrue(condition->astOperand2(), programMemory);
|
|
|
|
}
|
|
|
|
ProgramMemory progmem(programMemory);
|
|
|
|
bool error = false;
|
|
|
|
MathLib::bigint result = 0;
|
|
|
|
execute(condition, &progmem, &result, &error);
|
|
|
|
return !error && result == 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void programMemoryParseCondition(ProgramMemory& pm, const Token* tok, const Token* endTok, const Settings* settings, bool then)
|
|
|
|
{
|
|
|
|
if (Token::Match(tok, "==|>=|<=|<|>|!=")) {
|
|
|
|
if (then && !Token::Match(tok, "==|>=|<="))
|
|
|
|
return;
|
|
|
|
if (!then && !Token::Match(tok, "<|>|!="))
|
|
|
|
return;
|
|
|
|
ValueFlow::Value truevalue;
|
|
|
|
ValueFlow::Value falsevalue;
|
|
|
|
const Token* vartok = parseCompareInt(tok, truevalue, falsevalue);
|
|
|
|
if (!vartok)
|
|
|
|
return;
|
|
|
|
if (vartok->varId() == 0)
|
|
|
|
return;
|
|
|
|
if (!truevalue.isIntValue())
|
|
|
|
return;
|
|
|
|
if (isVariableChanged(tok->next(), endTok, vartok->varId(), false, settings, true))
|
|
|
|
return;
|
|
|
|
pm.setIntValue(vartok->varId(), then ? truevalue.intvalue : falsevalue.intvalue);
|
|
|
|
} else if (Token::Match(tok, "%var%")) {
|
|
|
|
if (tok->varId() == 0)
|
|
|
|
return;
|
|
|
|
if (then && !astIsPointer(tok) && !astIsBool(tok))
|
|
|
|
return;
|
|
|
|
if (isVariableChanged(tok->next(), endTok, tok->varId(), false, settings, true))
|
|
|
|
return;
|
|
|
|
pm.setIntValue(tok->varId(), then);
|
|
|
|
} else if (Token::simpleMatch(tok, "!")) {
|
|
|
|
programMemoryParseCondition(pm, tok->astOperand1(), endTok, settings, !then);
|
|
|
|
} else if (then && Token::simpleMatch(tok, "&&")) {
|
|
|
|
programMemoryParseCondition(pm, tok->astOperand1(), endTok, settings, then);
|
|
|
|
programMemoryParseCondition(pm, tok->astOperand2(), endTok, settings, then);
|
|
|
|
} else if (!then && Token::simpleMatch(tok, "||")) {
|
|
|
|
programMemoryParseCondition(pm, tok->astOperand1(), endTok, settings, then);
|
|
|
|
programMemoryParseCondition(pm, tok->astOperand2(), endTok, settings, then);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void fillProgramMemoryFromConditions(ProgramMemory& pm, const Scope* scope, const Token* endTok, const Settings* settings)
|
|
|
|
{
|
|
|
|
if (!scope)
|
|
|
|
return;
|
|
|
|
if (!scope->isLocal())
|
|
|
|
return;
|
|
|
|
assert(scope != scope->nestedIn);
|
|
|
|
fillProgramMemoryFromConditions(pm, scope->nestedIn, endTok, settings);
|
|
|
|
if (scope->type == Scope::eIf || scope->type == Scope::eWhile || scope->type == Scope::eElse) {
|
|
|
|
const Token * bodyStart = scope->bodyStart;
|
|
|
|
if (scope->type == Scope::eElse) {
|
|
|
|
if (!Token::simpleMatch(bodyStart->tokAt(-2), "} else {"))
|
|
|
|
return;
|
|
|
|
bodyStart = bodyStart->linkAt(-2);
|
|
|
|
}
|
|
|
|
const Token * condEndTok = bodyStart->previous();
|
|
|
|
if (!Token::simpleMatch(condEndTok, ") {"))
|
|
|
|
return;
|
|
|
|
const Token * condStartTok = condEndTok->link();
|
|
|
|
if (!condStartTok)
|
|
|
|
return;
|
|
|
|
if (!Token::Match(condStartTok->previous(), "if|while ("))
|
|
|
|
return;
|
|
|
|
const Token * condTok = condStartTok->astOperand2();
|
|
|
|
programMemoryParseCondition(pm, condTok, endTok, settings, scope->type != Scope::eElse);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void fillProgramMemoryFromConditions(ProgramMemory& pm, const Token* tok, const Settings* settings)
|
|
|
|
{
|
|
|
|
fillProgramMemoryFromConditions(pm, tok->scope(), tok, settings);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void fillProgramMemoryFromAssignments(ProgramMemory& pm, const Token* tok, const ProgramMemory& state, std::unordered_map<nonneg int, ValueFlow::Value> vars)
|
|
|
|
{
|
|
|
|
int indentlevel = 0;
|
|
|
|
for (const Token *tok2 = tok; tok2; tok2 = tok2->previous()) {
|
|
|
|
bool setvar = false;
|
|
|
|
if (Token::Match(tok2, "[;{}] %var% = %var% ;")) {
|
|
|
|
for (auto&& p:vars) {
|
|
|
|
if (p.first != tok2->next()->varId())
|
|
|
|
continue;
|
|
|
|
const Token *vartok = tok2->tokAt(3);
|
|
|
|
pm.setValue(vartok->varId(), p.second);
|
|
|
|
setvar = true;
|
|
|
|
}
|
|
|
|
}
|
2019-11-04 07:02:25 +01:00
|
|
|
if (!setvar && (Token::Match(tok2, ";|{|}|%type% %var% =") ||
|
2019-10-30 17:57:46 +01:00
|
|
|
Token::Match(tok2, "[;{}] const| %type% %var% ("))) {
|
|
|
|
const Token *vartok = tok2->next();
|
|
|
|
while (vartok->next()->isName())
|
|
|
|
vartok = vartok->next();
|
|
|
|
if (!pm.hasValue(vartok->varId())) {
|
|
|
|
MathLib::bigint result = 0;
|
|
|
|
bool error = false;
|
|
|
|
execute(vartok->next()->astOperand2(), &pm, &result, &error);
|
|
|
|
if (!error)
|
|
|
|
pm.setIntValue(vartok->varId(), result);
|
2020-02-13 16:27:06 +01:00
|
|
|
else
|
|
|
|
pm.setUnknown(vartok->varId());
|
2019-10-30 17:57:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tok2->str() == "{") {
|
|
|
|
if (indentlevel <= 0)
|
|
|
|
break;
|
|
|
|
--indentlevel;
|
|
|
|
}
|
|
|
|
if (tok2->str() == "}") {
|
|
|
|
const Token *cond = tok2->link();
|
|
|
|
cond = Token::simpleMatch(cond->previous(), ") {") ? cond->linkAt(-1) : nullptr;
|
|
|
|
if (cond && conditionIsFalse(cond->astOperand2(), state))
|
|
|
|
tok2 = cond->previous();
|
|
|
|
else if (cond && conditionIsTrue(cond->astOperand2(), state)) {
|
|
|
|
++indentlevel;
|
|
|
|
continue;
|
|
|
|
} else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-05 16:25:33 +01:00
|
|
|
static void removeModifiedVars(ProgramMemory& pm, const Token* tok, const Token* origin)
|
|
|
|
{
|
|
|
|
for (auto i = pm.values.begin(), last = pm.values.end(); i != last;) {
|
|
|
|
if (isVariableChanged(origin, tok, i->first, false, nullptr, true)) {
|
|
|
|
i = pm.values.erase(i);
|
|
|
|
} else {
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static ProgramMemory getInitialProgramState(const Token* tok,
|
2020-01-05 17:00:34 +01:00
|
|
|
const Token* origin,
|
|
|
|
const std::unordered_map<nonneg int, ValueFlow::Value>& vars = std::unordered_map<nonneg int, ValueFlow::Value> {})
|
2020-01-05 16:25:33 +01:00
|
|
|
{
|
|
|
|
ProgramMemory pm;
|
|
|
|
if (origin) {
|
|
|
|
fillProgramMemoryFromConditions(pm, origin, nullptr);
|
|
|
|
const ProgramMemory state = pm;
|
|
|
|
fillProgramMemoryFromAssignments(pm, tok, state, vars);
|
|
|
|
removeModifiedVars(pm, tok, origin);
|
|
|
|
}
|
|
|
|
return pm;
|
|
|
|
}
|
|
|
|
|
2019-10-30 17:57:46 +01:00
|
|
|
ProgramMemory getProgramMemory(const Token *tok, nonneg int varid, const ValueFlow::Value &value)
|
|
|
|
{
|
|
|
|
ProgramMemory programMemory;
|
2020-01-05 16:25:33 +01:00
|
|
|
programMemory.replace(getInitialProgramState(tok, value.tokvalue));
|
|
|
|
programMemory.replace(getInitialProgramState(tok, value.condition));
|
2020-02-13 16:27:06 +01:00
|
|
|
fillProgramMemoryFromConditions(programMemory, tok, nullptr);
|
2019-10-30 17:57:46 +01:00
|
|
|
programMemory.setValue(varid, value);
|
|
|
|
if (value.varId)
|
|
|
|
programMemory.setIntValue(value.varId, value.varvalue);
|
|
|
|
const ProgramMemory state = programMemory;
|
|
|
|
fillProgramMemoryFromAssignments(programMemory, tok, state, {{varid, value}});
|
|
|
|
return programMemory;
|
|
|
|
}
|
|
|
|
|
|
|
|
void execute(const Token *expr,
|
2019-10-30 17:58:19 +01:00
|
|
|
ProgramMemory * const programMemory,
|
|
|
|
MathLib::bigint *result,
|
|
|
|
bool *error)
|
2019-10-30 17:57:46 +01:00
|
|
|
{
|
|
|
|
if (!expr)
|
|
|
|
*error = true;
|
|
|
|
|
|
|
|
else if (expr->hasKnownIntValue()) {
|
|
|
|
*result = expr->values().front().intvalue;
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (expr->isNumber()) {
|
|
|
|
*result = MathLib::toLongNumber(expr->str());
|
|
|
|
if (MathLib::isFloat(expr->str()))
|
|
|
|
*error = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (expr->varId() > 0) {
|
|
|
|
if (!programMemory->getIntValue(expr->varId(), result))
|
|
|
|
*error = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (expr->isComparisonOp()) {
|
|
|
|
MathLib::bigint result1(0), result2(0);
|
|
|
|
execute(expr->astOperand1(), programMemory, &result1, error);
|
|
|
|
execute(expr->astOperand2(), programMemory, &result2, error);
|
|
|
|
if (expr->str() == "<")
|
|
|
|
*result = result1 < result2;
|
|
|
|
else if (expr->str() == "<=")
|
|
|
|
*result = result1 <= result2;
|
|
|
|
else if (expr->str() == ">")
|
|
|
|
*result = result1 > result2;
|
|
|
|
else if (expr->str() == ">=")
|
|
|
|
*result = result1 >= result2;
|
|
|
|
else if (expr->str() == "==")
|
|
|
|
*result = result1 == result2;
|
|
|
|
else if (expr->str() == "!=")
|
|
|
|
*result = result1 != result2;
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (expr->isAssignmentOp()) {
|
|
|
|
execute(expr->astOperand2(), programMemory, result, error);
|
|
|
|
if (!expr->astOperand1() || !expr->astOperand1()->varId())
|
|
|
|
*error = true;
|
|
|
|
if (*error)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (expr->str() == "=") {
|
|
|
|
programMemory->setIntValue(expr->astOperand1()->varId(), *result);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
long long intValue;
|
|
|
|
if (!programMemory->getIntValue(expr->astOperand1()->varId(), &intValue)) {
|
|
|
|
*error = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (expr->str() == "+=")
|
|
|
|
programMemory->setIntValue(expr->astOperand1()->varId(), intValue + *result);
|
|
|
|
else if (expr->str() == "-=")
|
|
|
|
programMemory->setIntValue(expr->astOperand1()->varId(), intValue - *result);
|
|
|
|
else if (expr->str() == "*=")
|
|
|
|
programMemory->setIntValue(expr->astOperand1()->varId(), intValue * *result);
|
|
|
|
else if (expr->str() == "/=" && *result != 0)
|
|
|
|
programMemory->setIntValue(expr->astOperand1()->varId(), intValue / *result);
|
|
|
|
else if (expr->str() == "%=" && *result != 0)
|
|
|
|
programMemory->setIntValue(expr->astOperand1()->varId(), intValue % *result);
|
|
|
|
else if (expr->str() == "&=")
|
|
|
|
programMemory->setIntValue(expr->astOperand1()->varId(), intValue & *result);
|
|
|
|
else if (expr->str() == "|=")
|
|
|
|
programMemory->setIntValue(expr->astOperand1()->varId(), intValue | *result);
|
|
|
|
else if (expr->str() == "^=")
|
|
|
|
programMemory->setIntValue(expr->astOperand1()->varId(), intValue ^ *result);
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (Token::Match(expr, "++|--")) {
|
|
|
|
if (!expr->astOperand1() || expr->astOperand1()->varId() == 0U)
|
|
|
|
*error = true;
|
|
|
|
else {
|
|
|
|
long long intValue;
|
|
|
|
if (!programMemory->getIntValue(expr->astOperand1()->varId(), &intValue))
|
|
|
|
*error = true;
|
|
|
|
else {
|
|
|
|
if (intValue == 0 &&
|
|
|
|
expr->str() == "--" &&
|
|
|
|
expr->astOperand1()->variable() &&
|
2020-01-11 14:00:41 +01:00
|
|
|
expr->astOperand1()->variable()->isUnsigned())
|
2019-10-30 17:57:46 +01:00
|
|
|
*error = true; // overflow
|
|
|
|
*result = intValue + (expr->str() == "++" ? 1 : -1);
|
|
|
|
programMemory->setIntValue(expr->astOperand1()->varId(), *result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (expr->isArithmeticalOp() && expr->astOperand1() && expr->astOperand2()) {
|
|
|
|
MathLib::bigint result1(0), result2(0);
|
|
|
|
execute(expr->astOperand1(), programMemory, &result1, error);
|
|
|
|
execute(expr->astOperand2(), programMemory, &result2, error);
|
|
|
|
if (expr->str() == "+")
|
|
|
|
*result = result1 + result2;
|
|
|
|
else if (expr->str() == "-")
|
|
|
|
*result = result1 - result2;
|
|
|
|
else if (expr->str() == "*") {
|
|
|
|
if (result2 && (result1 > std::numeric_limits<MathLib::bigint>::max()/result2))
|
|
|
|
*error = true;
|
|
|
|
else
|
|
|
|
*result = result1 * result2;
|
|
|
|
} else if (result2 == 0)
|
|
|
|
*error = true;
|
|
|
|
else if (expr->str() == "/")
|
|
|
|
*result = result1 / result2;
|
|
|
|
else if (expr->str() == "%")
|
|
|
|
*result = result1 % result2;
|
|
|
|
else if (expr->str() == "<<") {
|
|
|
|
if (result2 < 0 || result1 < 0 || result2 >= MathLib::bigint_bits) { // don't perform UB
|
|
|
|
*error= true;
|
|
|
|
} else {
|
|
|
|
*result = result1 << result2;
|
|
|
|
}
|
|
|
|
} else if (expr->str() == ">>") {
|
|
|
|
if (result2 < 0) { // don't perform UB
|
|
|
|
*error=true;
|
|
|
|
} else {
|
|
|
|
*result = result1 >> result2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (expr->str() == "&&") {
|
|
|
|
bool error1 = false;
|
|
|
|
execute(expr->astOperand1(), programMemory, result, &error1);
|
|
|
|
if (!error1 && *result == 0)
|
|
|
|
*result = 0;
|
|
|
|
else {
|
|
|
|
bool error2 = false;
|
|
|
|
execute(expr->astOperand2(), programMemory, result, &error2);
|
|
|
|
if (error1 && error2)
|
|
|
|
*error = true;
|
|
|
|
if (error2)
|
|
|
|
*result = 1;
|
|
|
|
else
|
|
|
|
*result = !!*result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (expr->str() == "||") {
|
|
|
|
execute(expr->astOperand1(), programMemory, result, error);
|
|
|
|
if (*result == 0 && *error == false)
|
|
|
|
execute(expr->astOperand2(), programMemory, result, error);
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (expr->str() == "!") {
|
|
|
|
execute(expr->astOperand1(), programMemory, result, error);
|
|
|
|
*result = !(*result);
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (expr->str() == "," && expr->astOperand1() && expr->astOperand2()) {
|
|
|
|
execute(expr->astOperand1(), programMemory, result, error);
|
|
|
|
execute(expr->astOperand2(), programMemory, result, error);
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (expr->str() == "[" && expr->astOperand1() && expr->astOperand2()) {
|
|
|
|
const Token *tokvalue = nullptr;
|
|
|
|
if (!programMemory->getTokValue(expr->astOperand1()->varId(), &tokvalue)) {
|
|
|
|
auto tokvalue_it = std::find_if(expr->astOperand1()->values().begin(),
|
|
|
|
expr->astOperand1()->values().end(),
|
|
|
|
std::mem_fn(&ValueFlow::Value::isTokValue));
|
|
|
|
if (tokvalue_it == expr->astOperand1()->values().end()) {
|
|
|
|
*error = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
tokvalue = tokvalue_it->tokvalue;
|
|
|
|
}
|
|
|
|
if (!tokvalue || !tokvalue->isLiteral()) {
|
|
|
|
*error = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const std::string strValue = tokvalue->strValue();
|
|
|
|
MathLib::bigint index = 0;
|
|
|
|
execute(expr->astOperand2(), programMemory, &index, error);
|
|
|
|
if (index >= 0 && index < strValue.size())
|
|
|
|
*result = strValue[index];
|
|
|
|
else if (index == strValue.size())
|
|
|
|
*result = 0;
|
|
|
|
else
|
|
|
|
*error = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
*error = true;
|
|
|
|
}
|