SymbolDatabase: Try to match function parameters with unknown types

This commit is contained in:
Daniel Marjamäki 2019-08-01 13:25:03 +02:00
parent f6726b76ae
commit dbddc321e8
1 changed files with 40 additions and 18 deletions

View File

@ -4186,27 +4186,49 @@ const Function* Scope::findFunction(const Token *tok, bool requireConst) const
matches.erase(matches.begin() + i); matches.erase(matches.begin() + i);
erased = true; erased = true;
break; break;
} else { } else if (!arguments[j]->valueType() && arguments[j]->astOperand1()->variable()) {
// TODO: Remove this code const Token *callArgTypeToken = arguments[j]->astOperand1()->variable()->typeStartToken();
const Variable * callarg = check->getVariableFromVarId(arguments[j]->next()->varId()); const Token *funcArgTypeToken = funcarg->typeStartToken();
if (callarg) {
const bool funcargptr = (funcarg->typeEndToken()->str() == "*"); auto parseDecl = [](const Token *typeToken) -> ValueType {
if (funcargptr && ValueType ret;
(callarg->typeStartToken()->str() == funcarg->typeStartToken()->str() && while (Token::Match(typeToken->previous(), "%name%"))
callarg->typeStartToken()->isUnsigned() == funcarg->typeStartToken()->isUnsigned() && typeToken = typeToken->previous();
callarg->typeStartToken()->isLong() == funcarg->typeStartToken()->isLong())) { while (Token::Match(typeToken, "%name%|*|&|::"))
same++; {
} else if (funcargptr && funcarg->typeStartToken()->str() == "void") { if (ret.originalTypeName.empty() && typeToken->str() == "::") {
fallback1++; while (Token::Match(typeToken, ":: %name%")) {
} else { ret.originalTypeName += "::" + typeToken->strAt(1);
// can't match so remove this function from possible matches typeToken = typeToken->tokAt(2);
matches.erase(matches.begin() + i); }
erased = true; }
break; if (typeToken->str() == "const")
ret.constness |= (1 << ret.pointer);
else if (typeToken->str() == "*")
ret.pointer++;
else if (ret.originalTypeName.empty() && Token::Match(typeToken, "%name% const| %var%|*|&"))
ret.originalTypeName = typeToken->str();
typeToken = typeToken->next();
} }
return ret;
};
ValueType callArgType = parseDecl(callArgTypeToken);
callArgType.pointer++;
ValueType funcArgType = parseDecl(funcArgTypeToken);
if (!callArgType.originalTypeName.empty() &&
callArgType.originalTypeName == funcArgType.originalTypeName) {
callArgType.sign = funcArgType.sign = ValueType::Sign::SIGNED;
callArgType.type = funcArgType.type = ValueType::Type::INT;
res = ValueType::matchParameter(&callArgType, &funcArgType);
if (res == ValueType::MatchResult::SAME)
++same;
else if (res == ValueType::MatchResult::FALLBACK1)
++fallback1;
else if (res == ValueType::MatchResult::FALLBACK2)
++fallback2;
} }
} }
} }
// check for a match with a numeric literal // check for a match with a numeric literal