SymbolDatabase: Implement two-stage fallback mechanism for matching function overloads, increasing matching accuracy
This commit is contained in:
parent
adc659f58b
commit
4a27376694
|
@ -3751,12 +3751,36 @@ const Function* Scope::findFunction(const Token *tok, bool requireConst) const
|
||||||
// check in base classes
|
// check in base classes
|
||||||
findFunctionInBase(tok->str(), args, matches);
|
findFunctionInBase(tok->str(), args, matches);
|
||||||
|
|
||||||
|
const Function* fallback1Func = nullptr;
|
||||||
|
const Function* fallback2Func = nullptr;
|
||||||
|
|
||||||
// check each function against the arguments in the function call for a match
|
// check each function against the arguments in the function call for a match
|
||||||
for (std::size_t i = 0; i < matches.size();) {
|
for (std::size_t i = 0; i < matches.size();) {
|
||||||
bool erased = false;
|
bool erased = false;
|
||||||
const Function * func = matches[i];
|
const Function * func = matches[i];
|
||||||
size_t same = 0;
|
size_t same = 0;
|
||||||
|
|
||||||
|
if (requireConst && func->isConst())
|
||||||
|
;
|
||||||
|
else {
|
||||||
|
// get the function this call is in
|
||||||
|
const Scope * scope = tok->scope();
|
||||||
|
|
||||||
|
// check if this function is a member function
|
||||||
|
if (scope && scope->functionOf && scope->functionOf->isClassOrStruct()) {
|
||||||
|
// check if isConst mismatches
|
||||||
|
if (!(scope->function && scope->function->isConst() == func->isConst())) {
|
||||||
|
if (!erased)
|
||||||
|
++i;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t fallback1 = 0;
|
||||||
|
size_t fallback2 = 0;
|
||||||
for (std::size_t j = 0; j < args; ++j) {
|
for (std::size_t j = 0; j < args; ++j) {
|
||||||
|
|
||||||
// don't check variadic arguments
|
// don't check variadic arguments
|
||||||
if (func->isVariadic() && j > (func->argCount() - 1)) {
|
if (func->isVariadic() && j > (func->argCount() - 1)) {
|
||||||
break;
|
break;
|
||||||
|
@ -3765,11 +3789,26 @@ const Function* Scope::findFunction(const Token *tok, bool requireConst) const
|
||||||
// check for a match with a variable
|
// check for a match with a variable
|
||||||
if (Token::Match(arguments[j], "%var% ,|)")) {
|
if (Token::Match(arguments[j], "%var% ,|)")) {
|
||||||
const Variable * callarg = check->getVariableFromVarId(arguments[j]->varId());
|
const Variable * callarg = check->getVariableFromVarId(arguments[j]->varId());
|
||||||
if (callarg &&
|
|
||||||
callarg->typeStartToken()->str() == funcarg->typeStartToken()->str() &&
|
if (callarg && callarg->isArrayOrPointer() == funcarg->isArrayOrPointer()) {
|
||||||
callarg->typeStartToken()->isUnsigned() == funcarg->typeStartToken()->isUnsigned() &&
|
if (callarg->typeStartToken()->str() == funcarg->typeStartToken()->str() &&
|
||||||
callarg->typeStartToken()->isLong() == funcarg->typeStartToken()->isLong()) {
|
callarg->typeStartToken()->isUnsigned() == funcarg->typeStartToken()->isUnsigned() &&
|
||||||
same++;
|
callarg->typeStartToken()->isLong() == funcarg->typeStartToken()->isLong()) {
|
||||||
|
same++;
|
||||||
|
} else if (callarg->isArrayOrPointer()) {
|
||||||
|
if (funcarg->typeStartToken()->str() == "void")
|
||||||
|
fallback1++;
|
||||||
|
} else if ((Token::Match(funcarg->typeStartToken(), "char|short|int|long") &&
|
||||||
|
Token::Match(callarg->typeStartToken(), "char|short|int|long")) ||
|
||||||
|
(Token::Match(funcarg->typeStartToken(), "float|double") &&
|
||||||
|
Token::Match(callarg->typeStartToken(), "float|double"))) {
|
||||||
|
fallback1++;
|
||||||
|
} else if ((Token::Match(funcarg->typeStartToken(), "char|short|int|long") &&
|
||||||
|
Token::Match(callarg->typeStartToken(), "float|double")) ||
|
||||||
|
(Token::Match(funcarg->typeStartToken(), "float|double") &&
|
||||||
|
Token::Match(callarg->typeStartToken(), "char|short|int|long"))) {
|
||||||
|
fallback2++;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3777,12 +3816,14 @@ const Function* Scope::findFunction(const Token *tok, bool requireConst) const
|
||||||
else if (Token::Match(arguments[j], "& %var% ,|)")) {
|
else if (Token::Match(arguments[j], "& %var% ,|)")) {
|
||||||
const Variable * callarg = check->getVariableFromVarId(arguments[j]->next()->varId());
|
const Variable * callarg = check->getVariableFromVarId(arguments[j]->next()->varId());
|
||||||
if (callarg) {
|
if (callarg) {
|
||||||
if (funcarg->typeEndToken()->str() == "*" &&
|
bool funcargptr = (funcarg->typeEndToken()->str() == "*");
|
||||||
(funcarg->typeStartToken()->str() == "void" ||
|
if (funcargptr &&
|
||||||
(callarg->typeStartToken()->str() == funcarg->typeStartToken()->str() &&
|
(callarg->typeStartToken()->str() == funcarg->typeStartToken()->str() &&
|
||||||
callarg->typeStartToken()->isUnsigned() == funcarg->typeStartToken()->isUnsigned() &&
|
callarg->typeStartToken()->isUnsigned() == funcarg->typeStartToken()->isUnsigned() &&
|
||||||
callarg->typeStartToken()->isLong() == funcarg->typeStartToken()->isLong()))) {
|
callarg->typeStartToken()->isLong() == funcarg->typeStartToken()->isLong())) {
|
||||||
same++;
|
same++;
|
||||||
|
} else if (funcargptr && funcarg->typeStartToken()->str() == "void") {
|
||||||
|
fallback1++;
|
||||||
} else {
|
} else {
|
||||||
// can't match so remove this function from possible matches
|
// can't match so remove this function from possible matches
|
||||||
matches.erase(matches.begin() + i);
|
matches.erase(matches.begin() + i);
|
||||||
|
@ -3794,7 +3835,8 @@ const Function* Scope::findFunction(const Token *tok, bool requireConst) const
|
||||||
|
|
||||||
// check for a match with a numeric literal
|
// check for a match with a numeric literal
|
||||||
else if (Token::Match(arguments[j], "%num% ,|)")) {
|
else if (Token::Match(arguments[j], "%num% ,|)")) {
|
||||||
if (MathLib::isInt(arguments[j]->str())) {
|
if (MathLib::isInt(arguments[j]->str()) && (!funcarg->isPointer() || MathLib::isNullValue(arguments[j]->str()))) {
|
||||||
|
bool exactMatch = false;
|
||||||
if (arguments[j]->str().find("ll") != std::string::npos ||
|
if (arguments[j]->str().find("ll") != std::string::npos ||
|
||||||
arguments[j]->str().find("LL") != std::string::npos) {
|
arguments[j]->str().find("LL") != std::string::npos) {
|
||||||
if (arguments[j]->str().find('u') != std::string::npos ||
|
if (arguments[j]->str().find('u') != std::string::npos ||
|
||||||
|
@ -3802,13 +3844,13 @@ const Function* Scope::findFunction(const Token *tok, bool requireConst) const
|
||||||
if (funcarg->typeStartToken()->str() == "long" &&
|
if (funcarg->typeStartToken()->str() == "long" &&
|
||||||
funcarg->typeStartToken()->isLong() &&
|
funcarg->typeStartToken()->isLong() &&
|
||||||
funcarg->typeStartToken()->isUnsigned()) {
|
funcarg->typeStartToken()->isUnsigned()) {
|
||||||
same++;
|
exactMatch = true;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (funcarg->typeStartToken()->str() == "long" &&
|
if (funcarg->typeStartToken()->str() == "long" &&
|
||||||
funcarg->typeStartToken()->isLong() &&
|
funcarg->typeStartToken()->isLong() &&
|
||||||
!funcarg->typeStartToken()->isUnsigned()) {
|
!funcarg->typeStartToken()->isUnsigned()) {
|
||||||
same++;
|
exactMatch = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (arguments[j]->str().find('l') != std::string::npos ||
|
} else if (arguments[j]->str().find('l') != std::string::npos ||
|
||||||
|
@ -3818,46 +3860,67 @@ const Function* Scope::findFunction(const Token *tok, bool requireConst) const
|
||||||
if (funcarg->typeStartToken()->str() == "long" &&
|
if (funcarg->typeStartToken()->str() == "long" &&
|
||||||
!funcarg->typeStartToken()->isLong() &&
|
!funcarg->typeStartToken()->isLong() &&
|
||||||
funcarg->typeStartToken()->isUnsigned()) {
|
funcarg->typeStartToken()->isUnsigned()) {
|
||||||
same++;
|
exactMatch = true;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (funcarg->typeStartToken()->str() == "long" &&
|
if (funcarg->typeStartToken()->str() == "long" &&
|
||||||
!funcarg->typeStartToken()->isLong() &&
|
!funcarg->typeStartToken()->isLong() &&
|
||||||
!funcarg->typeStartToken()->isUnsigned()) {
|
!funcarg->typeStartToken()->isUnsigned()) {
|
||||||
same++;
|
exactMatch = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (arguments[j]->str().find('u') != std::string::npos ||
|
} else if (arguments[j]->str().find('u') != std::string::npos ||
|
||||||
arguments[j]->str().find('U') != std::string::npos) {
|
arguments[j]->str().find('U') != std::string::npos) {
|
||||||
if (funcarg->typeStartToken()->str() == "int" &&
|
if (funcarg->typeStartToken()->str() == "int" &&
|
||||||
funcarg->typeStartToken()->isUnsigned()) {
|
funcarg->typeStartToken()->isUnsigned()) {
|
||||||
same++;
|
exactMatch = true;
|
||||||
} else if (Token::Match(funcarg->typeStartToken(), "char|short")) {
|
} else if (Token::Match(funcarg->typeStartToken(), "char|short")) {
|
||||||
same++;
|
exactMatch = true;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (Token::Match(funcarg->typeStartToken(), "char|short|int|long")) {
|
if (Token::Match(funcarg->typeStartToken(), "char|short|int|long")) {
|
||||||
same++;
|
exactMatch = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
|
if (exactMatch)
|
||||||
|
if (funcarg->isPointer())
|
||||||
|
fallback2++;
|
||||||
|
else
|
||||||
|
same++;
|
||||||
|
else {
|
||||||
|
if (Token::Match(funcarg->typeStartToken(), "char|short|int|long"))
|
||||||
|
fallback1++;
|
||||||
|
else if (Token::Match(funcarg->typeStartToken(), "float|double"))
|
||||||
|
fallback2++;
|
||||||
|
}
|
||||||
|
} else if (!funcarg->isPointer()) {
|
||||||
|
bool exactMatch = false;
|
||||||
if (arguments[j]->str().find('f') != std::string::npos ||
|
if (arguments[j]->str().find('f') != std::string::npos ||
|
||||||
arguments[j]->str().find('F') != std::string::npos) {
|
arguments[j]->str().find('F') != std::string::npos) {
|
||||||
if (funcarg->typeStartToken()->str() == "float") {
|
if (funcarg->typeStartToken()->str() == "float") {
|
||||||
same++;
|
exactMatch = true;
|
||||||
}
|
}
|
||||||
} else if (arguments[j]->str().find('l') != std::string::npos ||
|
} else if (arguments[j]->str().find('l') != std::string::npos ||
|
||||||
arguments[j]->str().find('L') != std::string::npos) {
|
arguments[j]->str().find('L') != std::string::npos) {
|
||||||
if (funcarg->typeStartToken()->str() == "double" &&
|
if (funcarg->typeStartToken()->str() == "double" &&
|
||||||
funcarg->typeStartToken()->isLong()) {
|
funcarg->typeStartToken()->isLong()) {
|
||||||
same++;
|
exactMatch = true;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (funcarg->typeStartToken()->str() == "double" &&
|
if (funcarg->typeStartToken()->str() == "double" &&
|
||||||
!funcarg->typeStartToken()->isLong()) {
|
!funcarg->typeStartToken()->isLong()) {
|
||||||
same++;
|
exactMatch = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (exactMatch)
|
||||||
|
same++;
|
||||||
|
else {
|
||||||
|
if (Token::Match(funcarg->typeStartToken(), "float|double"))
|
||||||
|
fallback1++;
|
||||||
|
else if (Token::Match(funcarg->typeStartToken(), "char|short|int|long"))
|
||||||
|
fallback2++;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3880,27 +3943,30 @@ const Function* Scope::findFunction(const Token *tok, bool requireConst) const
|
||||||
|
|
||||||
// check if all arguments matched
|
// check if all arguments matched
|
||||||
if ((func->isVariadic() && same == (func->argCount() - 1)) ||
|
if ((func->isVariadic() && same == (func->argCount() - 1)) ||
|
||||||
(!func->isVariadic() && same == args)) {
|
(!func->isVariadic() && same == args))
|
||||||
if (requireConst && func->isConst())
|
return func;
|
||||||
return func;
|
|
||||||
|
|
||||||
// get the function this call is in
|
if (!fallback1Func) {
|
||||||
const Scope * scope = tok->scope();
|
if ((func->isVariadic() && same + fallback1 == (func->argCount() - 1)) ||
|
||||||
|
(!func->isVariadic() && same + fallback1 == args))
|
||||||
// check if this function is a member function
|
fallback1Func = func;
|
||||||
if (scope && scope->functionOf && scope->functionOf->isClassOrStruct()) {
|
else if (!fallback2Func && ((func->isVariadic() && same + fallback2 + fallback1 == (func->argCount() - 1)) ||
|
||||||
// check if isConst match
|
(!func->isVariadic() && same + fallback2 + fallback1 == args)))
|
||||||
if (scope->function && scope->function->isConst() == func->isConst())
|
fallback2Func = func;
|
||||||
return func;
|
|
||||||
} else
|
|
||||||
return func;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!erased)
|
if (!erased)
|
||||||
++i;
|
++i;
|
||||||
}
|
}
|
||||||
|
|
||||||
// no exact match, but only one candidate left
|
// Fallback cases
|
||||||
|
if (fallback1Func)
|
||||||
|
return fallback1Func;
|
||||||
|
|
||||||
|
if (fallback2Func)
|
||||||
|
return fallback2Func;
|
||||||
|
|
||||||
|
// Only one candidate left
|
||||||
if (matches.size() == 1)
|
if (matches.size() == 1)
|
||||||
return matches[0];
|
return matches[0];
|
||||||
|
|
||||||
|
|
|
@ -281,6 +281,7 @@ private:
|
||||||
TEST_CASE(findFunction10); // #7673
|
TEST_CASE(findFunction10); // #7673
|
||||||
TEST_CASE(findFunction11);
|
TEST_CASE(findFunction11);
|
||||||
TEST_CASE(findFunction12);
|
TEST_CASE(findFunction12);
|
||||||
|
TEST_CASE(findFunction13);
|
||||||
|
|
||||||
TEST_CASE(noexceptFunction1);
|
TEST_CASE(noexceptFunction1);
|
||||||
TEST_CASE(noexceptFunction2);
|
TEST_CASE(noexceptFunction2);
|
||||||
|
@ -3495,6 +3496,8 @@ private:
|
||||||
"void foo(long long a) { }\n"
|
"void foo(long long a) { }\n"
|
||||||
"void foo() {\n"
|
"void foo() {\n"
|
||||||
" foo(0);\n"
|
" foo(0);\n"
|
||||||
|
" foo(0L);\n"
|
||||||
|
" foo(0.f);\n"
|
||||||
" foo(bar());\n"
|
" foo(bar());\n"
|
||||||
"}");
|
"}");
|
||||||
|
|
||||||
|
@ -3503,8 +3506,66 @@ private:
|
||||||
const Token *f = Token::findsimplematch(tokenizer.tokens(), "foo ( 0 ) ;");
|
const Token *f = Token::findsimplematch(tokenizer.tokens(), "foo ( 0 ) ;");
|
||||||
ASSERT_EQUALS(true, db && f && f->function() && f->function()->tokenDef->linenr() == 2);
|
ASSERT_EQUALS(true, db && f && f->function() && f->function()->tokenDef->linenr() == 2);
|
||||||
|
|
||||||
|
f = Token::findsimplematch(tokenizer.tokens(), "foo ( 0L ) ;");
|
||||||
|
ASSERT_EQUALS(true, f && f->function() && f->function()->tokenDef->linenr() == 2);
|
||||||
|
|
||||||
|
f = Token::findsimplematch(tokenizer.tokens(), "foo ( 0.f ) ;");
|
||||||
|
ASSERT_EQUALS(true, f && f->function() && f->function()->tokenDef->linenr() == 2);
|
||||||
|
|
||||||
f = Token::findsimplematch(tokenizer.tokens(), "foo ( bar ( ) ) ;");
|
f = Token::findsimplematch(tokenizer.tokens(), "foo ( bar ( ) ) ;");
|
||||||
ASSERT_EQUALS(true, db && f && f->function() == nullptr);
|
ASSERT_EQUALS(true, f && f->function() == nullptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
void findFunction13() {
|
||||||
|
GET_SYMBOL_DB("void foo(std::string a) { }\n"
|
||||||
|
"void foo(double a) { }\n"
|
||||||
|
"void foo(long long a) { }\n"
|
||||||
|
"void foo(int* a) { }\n"
|
||||||
|
"void foo(void* a) { }\n"
|
||||||
|
"void func(int i, float f, int* ip, float* fp) {\n"
|
||||||
|
" foo(0);\n"
|
||||||
|
" foo(0L);\n"
|
||||||
|
" foo(0.f);\n"
|
||||||
|
" foo(bar());\n"
|
||||||
|
" foo(i);\n"
|
||||||
|
" foo(f);\n"
|
||||||
|
" foo(&i);\n"
|
||||||
|
" foo(&f);\n"
|
||||||
|
" foo(ip);\n"
|
||||||
|
" foo(fp);\n"
|
||||||
|
"}");
|
||||||
|
|
||||||
|
ASSERT_EQUALS("", errout.str());
|
||||||
|
|
||||||
|
const Token *f = Token::findsimplematch(tokenizer.tokens(), "foo ( 0 ) ;");
|
||||||
|
ASSERT_EQUALS(true, db && f && f->function() && f->function()->tokenDef->linenr() == 3);
|
||||||
|
|
||||||
|
f = Token::findsimplematch(tokenizer.tokens(), "foo ( 0L ) ;");
|
||||||
|
ASSERT_EQUALS(true, f && f->function() && f->function()->tokenDef->linenr() == 3);
|
||||||
|
|
||||||
|
f = Token::findsimplematch(tokenizer.tokens(), "foo ( 0.f ) ;");
|
||||||
|
ASSERT_EQUALS(true, f && f->function() && f->function()->tokenDef->linenr() == 2);
|
||||||
|
|
||||||
|
f = Token::findsimplematch(tokenizer.tokens(), "foo ( bar ( ) ) ;");
|
||||||
|
ASSERT_EQUALS(true, f && f->function() == nullptr);
|
||||||
|
|
||||||
|
f = Token::findsimplematch(tokenizer.tokens(), "foo ( i ) ;");
|
||||||
|
ASSERT_EQUALS(true, f && f->function() && f->function()->tokenDef->linenr() == 3);
|
||||||
|
|
||||||
|
f = Token::findsimplematch(tokenizer.tokens(), "foo ( f ) ;");
|
||||||
|
ASSERT_EQUALS(true, f && f->function() && f->function()->tokenDef->linenr() == 2);
|
||||||
|
|
||||||
|
f = Token::findsimplematch(tokenizer.tokens(), "foo ( & i ) ;");
|
||||||
|
ASSERT_EQUALS(true, f && f->function() && f->function()->tokenDef->linenr() == 4);
|
||||||
|
|
||||||
|
f = Token::findsimplematch(tokenizer.tokens(), "foo ( & f ) ;");
|
||||||
|
ASSERT_EQUALS(true, f && f->function() && f->function()->tokenDef->linenr() == 5);
|
||||||
|
|
||||||
|
f = Token::findsimplematch(tokenizer.tokens(), "foo ( ip ) ;");
|
||||||
|
ASSERT_EQUALS(true, f && f->function() && f->function()->tokenDef->linenr() == 4);
|
||||||
|
|
||||||
|
f = Token::findsimplematch(tokenizer.tokens(), "foo ( fp ) ;");
|
||||||
|
ASSERT_EQUALS(true, f && f->function() && f->function()->tokenDef->linenr() == 5);
|
||||||
}
|
}
|
||||||
|
|
||||||
#define FUNC(x) const Function *x = findFunctionByName(#x, &db->scopeList.front()); \
|
#define FUNC(x) const Function *x = findFunctionByName(#x, &db->scopeList.front()); \
|
||||||
|
|
Loading…
Reference in New Issue