Skip to content

Commit 27cca72

Browse files
authored
ValueFlow: extracted valueFlowSubFunction() into separate file (#6929)
1 parent 445c986 commit 27cca72

10 files changed

+332
-232
lines changed

Makefile

+5-1
Original file line numberDiff line numberDiff line change
@@ -276,6 +276,7 @@ LIBOBJ = $(libcppdir)/valueflow.o \
276276
$(libcppdir)/vf_sameexpressions.o \
277277
$(libcppdir)/vf_settokenvalue.o \
278278
$(libcppdir)/vf_string.o \
279+
$(libcppdir)/vf_subfunction.o \
279280
$(libcppdir)/vf_switchvariable.o \
280281
$(libcppdir)/vf_symbolicinfer.o \
281282
$(libcppdir)/vf_symbolicoperators.o \
@@ -480,7 +481,7 @@ validateRules:
480481

481482
###### Build
482483

483-
$(libcppdir)/valueflow.o: lib/valueflow.cpp lib/addoninfo.h lib/analyzer.h lib/astutils.h lib/check.h lib/checkuninitvar.h lib/color.h lib/config.h lib/errorlogger.h lib/errortypes.h lib/findtoken.h lib/forwardanalyzer.h lib/infer.h lib/library.h lib/mathlib.h lib/platform.h lib/programmemory.h lib/settings.h lib/smallvector.h lib/sourcelocation.h lib/standards.h lib/suppressions.h lib/symboldatabase.h lib/templatesimplifier.h lib/timer.h lib/token.h lib/tokenize.h lib/tokenlist.h lib/utils.h lib/valueflow.h lib/valueptr.h lib/vf_analyze.h lib/vf_analyzers.h lib/vf_array.h lib/vf_arraybool.h lib/vf_arrayelement.h lib/vf_bailout.h lib/vf_bitand.h lib/vf_common.h lib/vf_conditionexpressions.h lib/vf_debug.h lib/vf_enumvalue.h lib/vf_functionreturn.h lib/vf_globalconstvar.h lib/vf_globalstaticvar.h lib/vf_impossiblevalues.h lib/vf_infercondition.h lib/vf_iteratorinfer.h lib/vf_iterators.h lib/vf_number.h lib/vf_pointeralias.h lib/vf_reverse.h lib/vf_rightshift.h lib/vf_sameexpressions.h lib/vf_settokenvalue.h lib/vf_string.h lib/vf_switchvariable.h lib/vf_symbolicinfer.h lib/vf_symbolicoperators.h lib/vf_unknownfunctionreturn.h lib/vfvalue.h
484+
$(libcppdir)/valueflow.o: lib/valueflow.cpp lib/addoninfo.h lib/analyzer.h lib/astutils.h lib/color.h lib/config.h lib/errorlogger.h lib/errortypes.h lib/findtoken.h lib/forwardanalyzer.h lib/infer.h lib/library.h lib/mathlib.h lib/platform.h lib/programmemory.h lib/settings.h lib/smallvector.h lib/sourcelocation.h lib/standards.h lib/suppressions.h lib/symboldatabase.h lib/templatesimplifier.h lib/timer.h lib/token.h lib/tokenlist.h lib/utils.h lib/valueflow.h lib/valueptr.h lib/vf_analyze.h lib/vf_analyzers.h lib/vf_array.h lib/vf_arraybool.h lib/vf_arrayelement.h lib/vf_bailout.h lib/vf_bitand.h lib/vf_common.h lib/vf_conditionexpressions.h lib/vf_debug.h lib/vf_enumvalue.h lib/vf_functionreturn.h lib/vf_globalconstvar.h lib/vf_globalstaticvar.h lib/vf_impossiblevalues.h lib/vf_infercondition.h lib/vf_iteratorinfer.h lib/vf_iterators.h lib/vf_number.h lib/vf_pointeralias.h lib/vf_reverse.h lib/vf_rightshift.h lib/vf_sameexpressions.h lib/vf_settokenvalue.h lib/vf_string.h lib/vf_subfunction.h lib/vf_switchvariable.h lib/vf_symbolicinfer.h lib/vf_symbolicoperators.h lib/vf_unknownfunctionreturn.h lib/vfvalue.h
484485
$(CXX) ${INCLUDE_FOR_LIB} $(CPPFLAGS) $(CXXFLAGS) -c -o $@ $(libcppdir)/valueflow.cpp
485486

486487
$(libcppdir)/tokenize.o: lib/tokenize.cpp externals/simplecpp/simplecpp.h lib/addoninfo.h lib/astutils.h lib/color.h lib/config.h lib/errorlogger.h lib/errortypes.h lib/library.h lib/mathlib.h lib/path.h lib/platform.h lib/preprocessor.h lib/settings.h lib/smallvector.h lib/sourcelocation.h lib/standards.h lib/summaries.h lib/suppressions.h lib/symboldatabase.h lib/templatesimplifier.h lib/timer.h lib/token.h lib/tokenize.h lib/tokenlist.h lib/utils.h lib/valueflow.h lib/vfvalue.h
@@ -738,6 +739,9 @@ $(libcppdir)/vf_settokenvalue.o: lib/vf_settokenvalue.cpp lib/addoninfo.h lib/as
738739
$(libcppdir)/vf_string.o: lib/vf_string.cpp lib/config.h lib/errortypes.h lib/mathlib.h lib/sourcelocation.h lib/standards.h lib/templatesimplifier.h lib/token.h lib/tokenlist.h lib/utils.h lib/vf_settokenvalue.h lib/vf_string.h lib/vfvalue.h
739740
$(CXX) ${INCLUDE_FOR_LIB} $(CPPFLAGS) $(CXXFLAGS) -c -o $@ $(libcppdir)/vf_string.cpp
740741

742+
$(libcppdir)/vf_subfunction.o: lib/vf_subfunction.cpp lib/addoninfo.h lib/analyzer.h lib/astutils.h lib/check.h lib/checkuninitvar.h lib/config.h lib/errortypes.h lib/forwardanalyzer.h lib/library.h lib/mathlib.h lib/platform.h lib/programmemory.h lib/settings.h lib/smallvector.h lib/sourcelocation.h lib/standards.h lib/suppressions.h lib/symboldatabase.h lib/templatesimplifier.h lib/token.h lib/tokenize.h lib/tokenlist.h lib/utils.h lib/valueptr.h lib/vf_analyzers.h lib/vf_bailout.h lib/vf_common.h lib/vf_settokenvalue.h lib/vf_subfunction.h lib/vfvalue.h
743+
$(CXX) ${INCLUDE_FOR_LIB} $(CPPFLAGS) $(CXXFLAGS) -c -o $@ $(libcppdir)/vf_subfunction.cpp
744+
741745
$(libcppdir)/vf_switchvariable.o: lib/vf_switchvariable.cpp lib/addoninfo.h lib/config.h lib/errortypes.h lib/library.h lib/mathlib.h lib/platform.h lib/settings.h lib/sourcelocation.h lib/standards.h lib/suppressions.h lib/symboldatabase.h lib/templatesimplifier.h lib/token.h lib/utils.h lib/vf_bailout.h lib/vf_reverse.h lib/vf_switchvariable.h lib/vfvalue.h
742746
$(CXX) ${INCLUDE_FOR_LIB} $(CPPFLAGS) $(CXXFLAGS) -c -o $@ $(libcppdir)/vf_switchvariable.cpp
743747

lib/cppcheck.vcxproj

+2
Original file line numberDiff line numberDiff line change
@@ -111,6 +111,7 @@
111111
<ClCompile Include="vf_sameexpressions.cpp" />
112112
<ClCompile Include="vf_settokenvalue.cpp" />
113113
<ClCompile Include="vf_string.cpp" />
114+
<ClCompile Include="vf_subfunction.cpp" />
114115
<ClCompile Include="vf_switchvariable.cpp" />
115116
<ClCompile Include="vf_symbolicinfer.cpp" />
116117
<ClCompile Include="vf_symbolicoperators.cpp" />
@@ -221,6 +222,7 @@
221222
<ClInclude Include="vf_sameexpressions.h" />
222223
<ClInclude Include="vf_settokenvalue.h" />
223224
<ClInclude Include="vf_string.h" />
225+
<ClInclude Include="vf_subfunction.h" />
224226
<ClInclude Include="vf_switchvariable.h" />
225227
<ClInclude Include="vf_symbolicinfer.h" />
226228
<ClInclude Include="vf_symbolicoperators.h" />

lib/lib.pri

+2
Original file line numberDiff line numberDiff line change
@@ -104,6 +104,7 @@ HEADERS += $${PWD}/addoninfo.h \
104104
$${PWD}/vf_sameexpressions.h \
105105
$${PWD}/vf_settokenvalue.h \
106106
$${PWD}/vf_string.h \
107+
$${PWD}/vf_subfunction.h \
107108
$${PWD}/vf_switchvariable.h \
108109
$${PWD}/vf_symbolicinfer.h \
109110
$${PWD}/vf_symbolicoperators.h \
@@ -197,6 +198,7 @@ SOURCES += $${PWD}/valueflow.cpp \
197198
$${PWD}/vf_sameexpressions.cpp \
198199
$${PWD}/vf_settokenvalue.cpp \
199200
$${PWD}/vf_string.cpp \
201+
$${PWD}/vf_subfunction.cpp \
200202
$${PWD}/vf_switchvariable.cpp \
201203
$${PWD}/vf_symbolicinfer.cpp \
202204
$${PWD}/vf_symbolicoperators.cpp \

lib/valueflow.cpp

+1-230
Original file line numberDiff line numberDiff line change
@@ -79,7 +79,6 @@
7979

8080
#include "analyzer.h"
8181
#include "astutils.h"
82-
#include "checkuninitvar.h"
8382
#include "config.h"
8483
#include "errorlogger.h"
8584
#include "errortypes.h"
@@ -129,30 +128,6 @@
129128
#include <unordered_set>
130129
#include <vector>
131130

132-
static void changeKnownToPossible(std::list<ValueFlow::Value> &values, int indirect=-1)
133-
{
134-
for (ValueFlow::Value& v: values) {
135-
if (indirect >= 0 && v.indirect != indirect)
136-
continue;
137-
v.changeKnownToPossible();
138-
}
139-
}
140-
141-
static void removeImpossible(std::list<ValueFlow::Value>& values, int indirect = -1)
142-
{
143-
values.remove_if([&](const ValueFlow::Value& v) {
144-
if (indirect >= 0 && v.indirect != indirect)
145-
return false;
146-
return v.isImpossible();
147-
});
148-
}
149-
150-
static void lowerToPossible(std::list<ValueFlow::Value>& values, int indirect = -1)
151-
{
152-
changeKnownToPossible(values, indirect);
153-
removeImpossible(values, indirect);
154-
}
155-
156131
static void changePossibleToKnown(std::list<ValueFlow::Value>& values, int indirect = -1)
157132
{
158133
for (ValueFlow::Value& v : values) {
@@ -4081,85 +4056,6 @@ static void valueFlowForLoop(TokenList &tokenlist, const SymbolDatabase& symbold
40814056
}
40824057
}
40834058

4084-
template<class Key, class F>
4085-
static bool productParams(const Settings& settings, const std::unordered_map<Key, std::list<ValueFlow::Value>>& vars, F f)
4086-
{
4087-
using Args = std::vector<std::unordered_map<Key, ValueFlow::Value>>;
4088-
Args args(1);
4089-
// Compute cartesian product of all arguments
4090-
for (const auto& p:vars) {
4091-
if (p.second.empty())
4092-
continue;
4093-
args.back()[p.first] = p.second.front();
4094-
}
4095-
bool bail = false;
4096-
int max = settings.vfOptions.maxSubFunctionArgs;
4097-
for (const auto& p:vars) {
4098-
if (args.size() > max) {
4099-
bail = true;
4100-
break;
4101-
}
4102-
if (p.second.empty())
4103-
continue;
4104-
std::for_each(std::next(p.second.begin()), p.second.end(), [&](const ValueFlow::Value& value) {
4105-
Args new_args;
4106-
for (auto arg:args) {
4107-
if (value.path != 0) {
4108-
for (const auto& q:arg) {
4109-
if (q.first == p.first)
4110-
continue;
4111-
if (q.second.path == 0)
4112-
continue;
4113-
if (q.second.path != value.path)
4114-
return;
4115-
}
4116-
}
4117-
arg[p.first] = value;
4118-
new_args.push_back(std::move(arg));
4119-
}
4120-
std::copy(new_args.cbegin(), new_args.cend(), std::back_inserter(args));
4121-
});
4122-
}
4123-
4124-
if (args.size() > max) {
4125-
bail = true;
4126-
args.resize(max);
4127-
// TODO: add bailout message
4128-
}
4129-
4130-
for (const auto& arg:args) {
4131-
if (arg.empty())
4132-
continue;
4133-
// Make sure all arguments are the same path
4134-
const MathLib::bigint path = arg.cbegin()->second.path;
4135-
if (std::any_of(arg.cbegin(), arg.cend(), [&](const std::pair<Key, ValueFlow::Value>& p) {
4136-
return p.second.path != path;
4137-
}))
4138-
continue;
4139-
f(arg);
4140-
}
4141-
return !bail;
4142-
}
4143-
4144-
static void valueFlowInjectParameter(const TokenList& tokenlist,
4145-
ErrorLogger& errorLogger,
4146-
const Settings& settings,
4147-
const Scope* functionScope,
4148-
const std::unordered_map<const Variable*, std::list<ValueFlow::Value>>& vars)
4149-
{
4150-
const bool r = productParams(settings, vars, [&](const std::unordered_map<const Variable*, ValueFlow::Value>& arg) {
4151-
auto a = makeMultiValueFlowAnalyzer(arg, settings);
4152-
valueFlowGenericForward(const_cast<Token*>(functionScope->bodyStart), functionScope->bodyEnd, a, tokenlist, errorLogger, settings);
4153-
});
4154-
if (!r) {
4155-
std::string fname = "<unknown>";
4156-
if (const Function* f = functionScope->function)
4157-
fname = f->name();
4158-
if (settings.debugwarnings)
4159-
bailout(tokenlist, errorLogger, functionScope->bodyStart, "Too many argument passed to " + fname);
4160-
}
4161-
}
4162-
41634059
static void valueFlowInjectParameter(const TokenList& tokenlist,
41644060
ErrorLogger& errorLogger,
41654061
const Settings& settings,
@@ -4185,131 +4081,6 @@ static void valueFlowInjectParameter(const TokenList& tokenlist,
41854081
settings);
41864082
}
41874083

4188-
static std::list<ValueFlow::Value> getFunctionArgumentValues(const Token *argtok)
4189-
{
4190-
std::list<ValueFlow::Value> argvalues(argtok->values());
4191-
removeImpossible(argvalues);
4192-
if (argvalues.empty() && Token::Match(argtok, "%comp%|%oror%|&&|!")) {
4193-
argvalues.emplace_back(0);
4194-
argvalues.emplace_back(1);
4195-
}
4196-
return argvalues;
4197-
}
4198-
4199-
static void valueFlowLibraryFunction(Token *tok, const std::string &returnValue, const Settings &settings)
4200-
{
4201-
std::unordered_map<nonneg int, std::list<ValueFlow::Value>> argValues;
4202-
int argn = 1;
4203-
for (const Token *argtok : getArguments(tok->previous())) {
4204-
argValues[argn] = getFunctionArgumentValues(argtok);
4205-
argn++;
4206-
}
4207-
if (returnValue.find("arg") != std::string::npos && argValues.empty())
4208-
return;
4209-
productParams(settings, argValues, [&](const std::unordered_map<nonneg int, ValueFlow::Value>& arg) {
4210-
ValueFlow::Value value = evaluateLibraryFunction(arg, returnValue, settings, tok->isCpp());
4211-
if (value.isUninitValue())
4212-
return;
4213-
ValueFlow::Value::ValueKind kind = ValueFlow::Value::ValueKind::Known;
4214-
for (auto&& p : arg) {
4215-
if (p.second.isPossible())
4216-
kind = p.second.valueKind;
4217-
if (p.second.isInconclusive()) {
4218-
kind = p.second.valueKind;
4219-
break;
4220-
}
4221-
}
4222-
if (value.isImpossible() && kind != ValueFlow::Value::ValueKind::Known)
4223-
return;
4224-
if (!value.isImpossible())
4225-
value.valueKind = kind;
4226-
setTokenValue(tok, std::move(value), settings);
4227-
});
4228-
}
4229-
4230-
static void valueFlowSubFunction(const TokenList& tokenlist, SymbolDatabase& symboldatabase, ErrorLogger& errorLogger, const Settings& settings)
4231-
{
4232-
int id = 0;
4233-
for (auto it = symboldatabase.functionScopes.crbegin(); it != symboldatabase.functionScopes.crend(); ++it) {
4234-
const Scope* scope = *it;
4235-
const Function* function = scope->function;
4236-
if (!function)
4237-
continue;
4238-
for (auto* tok = const_cast<Token*>(scope->bodyStart); tok != scope->bodyEnd; tok = tok->next()) {
4239-
if (tok->isKeyword() || !Token::Match(tok, "%name% ("))
4240-
continue;
4241-
4242-
const Function * const calledFunction = tok->function();
4243-
if (!calledFunction) {
4244-
// library function?
4245-
const std::string& returnValue(settings.library.returnValue(tok));
4246-
if (!returnValue.empty())
4247-
valueFlowLibraryFunction(tok->next(), returnValue, settings);
4248-
continue;
4249-
}
4250-
4251-
const Scope * const calledFunctionScope = calledFunction->functionScope;
4252-
if (!calledFunctionScope)
4253-
continue;
4254-
4255-
id++;
4256-
std::unordered_map<const Variable*, std::list<ValueFlow::Value>> argvars;
4257-
// TODO: Rewrite this. It does not work well to inject 1 argument at a time.
4258-
const std::vector<const Token *> &callArguments = getArguments(tok);
4259-
for (int argnr = 0U; argnr < callArguments.size(); ++argnr) {
4260-
const Token *argtok = callArguments[argnr];
4261-
// Get function argument
4262-
const Variable * const argvar = calledFunction->getArgumentVar(argnr);
4263-
if (!argvar)
4264-
break;
4265-
4266-
// passing value(s) to function
4267-
std::list<ValueFlow::Value> argvalues(getFunctionArgumentValues(argtok));
4268-
4269-
// Remove non-local lifetimes
4270-
argvalues.remove_if([](const ValueFlow::Value& v) {
4271-
if (v.isLifetimeValue())
4272-
return !v.isLocalLifetimeValue() && !v.isSubFunctionLifetimeValue();
4273-
return false;
4274-
});
4275-
// Remove uninit values if argument is passed by value
4276-
if (argtok->variable() && !argtok->variable()->isPointer() && argvalues.size() == 1 && argvalues.front().isUninitValue()) {
4277-
if (CheckUninitVar::isVariableUsage(argtok, settings.library, false, CheckUninitVar::Alloc::NO_ALLOC, 0))
4278-
continue;
4279-
}
4280-
4281-
if (argvalues.empty())
4282-
continue;
4283-
4284-
// Error path..
4285-
for (ValueFlow::Value &v : argvalues) {
4286-
const std::string nr = std::to_string(argnr + 1) + getOrdinalText(argnr + 1);
4287-
4288-
v.errorPath.emplace_back(argtok,
4289-
"Calling function '" +
4290-
calledFunction->name() +
4291-
"', " +
4292-
nr +
4293-
" argument '" +
4294-
argtok->expressionString() +
4295-
"' value is " +
4296-
v.infoString());
4297-
v.path = 256 * v.path + id % 256;
4298-
// Change scope of lifetime values
4299-
if (v.isLifetimeValue())
4300-
v.lifetimeScope = ValueFlow::Value::LifetimeScope::SubFunction;
4301-
}
4302-
4303-
// passed values are not "known"..
4304-
lowerToPossible(argvalues);
4305-
4306-
argvars[argvar] = std::move(argvalues);
4307-
}
4308-
valueFlowInjectParameter(tokenlist, errorLogger, settings, calledFunctionScope, argvars);
4309-
}
4310-
}
4311-
}
4312-
43134084
static void valueFlowFunctionDefaultParameter(const TokenList& tokenlist, const SymbolDatabase& symboldatabase, ErrorLogger& errorLogger, const Settings& settings)
43144085
{
43154086
if (!tokenlist.isCPP())
@@ -5687,7 +5458,7 @@ void ValueFlow::setValues(TokenList& tokenlist,
56875458
VFA(analyzeInferCondition(tokenlist, settings)),
56885459
VFA(analyzeSwitchVariable(tokenlist, symboldatabase, errorLogger, settings)),
56895460
VFA(valueFlowForLoop(tokenlist, symboldatabase, errorLogger, settings)),
5690-
VFA(valueFlowSubFunction(tokenlist, symboldatabase, errorLogger, settings)),
5461+
VFA(analyzeSubFunction(tokenlist, symboldatabase, errorLogger, settings)),
56915462
VFA(analyzeFunctionReturn(tokenlist, errorLogger, settings)),
56925463
VFA(valueFlowLifetime(tokenlist, errorLogger, settings)),
56935464
VFA(valueFlowFunctionDefaultParameter(tokenlist, symboldatabase, errorLogger, settings)),

lib/vf_analyze.h

+1
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,7 @@
3838
#include "vf_rightshift.h" // IWYU pragma: export
3939
#include "vf_sameexpressions.h" // IWYU pragma: export
4040
#include "vf_string.h" // IWYU pragma: export
41+
#include "vf_subfunction.h" // IWYU pragma: export
4142
#include "vf_switchvariable.h" // IWYU pragma: export
4243
#include "vf_symbolicinfer.h" // IWYU pragma: export
4344
#include "vf_symbolicoperators.h" // IWYU pragma: export

lib/vf_common.cpp

+24
Original file line numberDiff line numberDiff line change
@@ -454,4 +454,28 @@ namespace ValueFlow
454454
value.intvalue = delta;
455455
return value;
456456
}
457+
458+
void removeImpossible(std::list<Value>& values, int indirect)
459+
{
460+
values.remove_if([&](const Value& v) {
461+
if (indirect >= 0 && v.indirect != indirect)
462+
return false;
463+
return v.isImpossible();
464+
});
465+
}
466+
467+
void changeKnownToPossible(std::list<Value> &values, int indirect)
468+
{
469+
for (Value& v: values) {
470+
if (indirect >= 0 && v.indirect != indirect)
471+
continue;
472+
v.changeKnownToPossible();
473+
}
474+
}
475+
476+
void lowerToPossible(std::list<Value>& values, int indirect)
477+
{
478+
changeKnownToPossible(values, indirect);
479+
removeImpossible(values, indirect);
480+
}
457481
}

lib/vf_common.h

+6
Original file line numberDiff line numberDiff line change
@@ -63,6 +63,12 @@ namespace ValueFlow
6363
void setSymbolic(Value& value, const Token* tok);
6464

6565
Value makeSymbolic(const Token* tok, MathLib::bigint delta = 0);
66+
67+
void removeImpossible(std::list<Value>& values, int indirect = -1);
68+
69+
void changeKnownToPossible(std::list<Value> &values, int indirect=-1);
70+
71+
void lowerToPossible(std::list<Value>& values, int indirect = -1);
6672
}
6773

6874
#endif // vfCommonH

0 commit comments

Comments
 (0)