2020-02-13 16:27:06 +01:00
|
|
|
/*
|
|
|
|
* Cppcheck - A tool for static C/C++ code analysis
|
2020-05-10 11:16:32 +02:00
|
|
|
* Copyright (C) 2007-2020 Cppcheck team.
|
2020-02-13 16:27:06 +01:00
|
|
|
*
|
|
|
|
* 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://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef forwardanalyzerH
|
|
|
|
#define forwardanalyzerH
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
class Settings;
|
2020-04-13 13:44:48 +02:00
|
|
|
class Token;
|
2020-05-10 16:45:45 +02:00
|
|
|
template <class T> class ValuePtr;
|
2020-02-13 16:27:06 +01:00
|
|
|
|
|
|
|
struct ForwardAnalyzer {
|
|
|
|
struct Action {
|
|
|
|
|
|
|
|
Action() : mFlag(0) {}
|
|
|
|
|
|
|
|
// cppcheck-suppress noExplicitConstructor
|
|
|
|
Action(unsigned int f) : mFlag(f) {}
|
|
|
|
|
|
|
|
enum {
|
|
|
|
None = 0,
|
|
|
|
Read = (1 << 0),
|
|
|
|
Write = (1 << 1),
|
|
|
|
Invalid = (1 << 2),
|
|
|
|
Inconclusive = (1 << 3),
|
|
|
|
};
|
|
|
|
|
2020-02-13 17:04:05 +01:00
|
|
|
void set(unsigned int f, bool state = true) {
|
|
|
|
mFlag = state ? mFlag | f : mFlag & ~f;
|
|
|
|
}
|
2020-02-13 16:27:06 +01:00
|
|
|
|
2020-02-13 17:04:05 +01:00
|
|
|
bool get(unsigned int f) const {
|
|
|
|
return ((mFlag & f) != 0);
|
|
|
|
}
|
2020-02-13 16:27:06 +01:00
|
|
|
|
2020-02-13 17:04:05 +01:00
|
|
|
bool isRead() const {
|
|
|
|
return get(Read);
|
|
|
|
}
|
2020-02-13 16:27:06 +01:00
|
|
|
|
2020-02-13 17:04:05 +01:00
|
|
|
bool isWrite() const {
|
|
|
|
return get(Write);
|
|
|
|
}
|
2020-02-13 16:27:06 +01:00
|
|
|
|
2020-02-13 17:04:05 +01:00
|
|
|
bool isInvalid() const {
|
|
|
|
return get(Invalid);
|
|
|
|
}
|
2020-02-13 16:27:06 +01:00
|
|
|
|
2020-02-13 17:04:05 +01:00
|
|
|
bool isInconclusive() const {
|
|
|
|
return get(Inconclusive);
|
|
|
|
}
|
2020-02-13 16:27:06 +01:00
|
|
|
|
2020-02-13 17:04:05 +01:00
|
|
|
bool isNone() const {
|
|
|
|
return mFlag == None;
|
|
|
|
}
|
2020-02-13 16:27:06 +01:00
|
|
|
|
2020-02-13 17:04:05 +01:00
|
|
|
bool isModified() const {
|
|
|
|
return isWrite() || isInvalid();
|
|
|
|
}
|
2020-02-13 16:27:06 +01:00
|
|
|
|
2020-02-13 17:04:05 +01:00
|
|
|
Action& operator|=(Action a) {
|
2020-02-13 16:27:06 +01:00
|
|
|
set(a.mFlag);
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2020-02-13 17:04:05 +01:00
|
|
|
friend Action operator|(Action a, Action b) {
|
2020-02-13 16:27:06 +01:00
|
|
|
a |= b;
|
|
|
|
return a;
|
|
|
|
}
|
|
|
|
|
2020-02-13 17:04:05 +01:00
|
|
|
friend bool operator==(Action a, Action b) {
|
|
|
|
return a.mFlag == b.mFlag;
|
|
|
|
}
|
2020-02-13 16:27:06 +01:00
|
|
|
|
2020-02-13 17:04:05 +01:00
|
|
|
friend bool operator!=(Action a, Action b) {
|
|
|
|
return a.mFlag != b.mFlag;
|
|
|
|
}
|
2020-02-13 16:27:06 +01:00
|
|
|
|
2020-02-13 17:04:05 +01:00
|
|
|
private:
|
2020-02-13 16:27:06 +01:00
|
|
|
unsigned int mFlag;
|
|
|
|
};
|
|
|
|
/// Analyze a token
|
|
|
|
virtual Action analyze(const Token* tok) const = 0;
|
|
|
|
/// Update the state of the value
|
|
|
|
virtual void update(Token* tok, Action a) = 0;
|
|
|
|
/// Try to evaluate the value of a token(most likely a condition)
|
|
|
|
virtual std::vector<int> evaluate(const Token* tok) const = 0;
|
|
|
|
/// Lower any values to possible
|
|
|
|
virtual bool lowerToPossible() = 0;
|
|
|
|
/// Lower any values to inconclusive
|
|
|
|
virtual bool lowerToInconclusive() = 0;
|
|
|
|
/// If the analysis is unsure whether to update a scope, this will return true if the analysis should bifurcate the scope
|
|
|
|
virtual bool updateScope(const Token* endBlock, bool modified) const = 0;
|
|
|
|
/// If the value is conditional
|
|
|
|
virtual bool isConditional() const = 0;
|
2020-04-01 14:37:43 +02:00
|
|
|
/// The condition that will be assumed during analysis
|
2020-02-19 07:55:04 +01:00
|
|
|
virtual void assume(const Token* tok, bool state, const Token* at = nullptr) = 0;
|
2020-02-13 16:27:06 +01:00
|
|
|
virtual ~ForwardAnalyzer() {}
|
|
|
|
};
|
|
|
|
|
2020-09-10 04:32:07 +02:00
|
|
|
ForwardAnalyzer::Action valueFlowGenericForward(Token* start,
|
|
|
|
const Token* end,
|
|
|
|
const ValuePtr<ForwardAnalyzer>& fa,
|
|
|
|
const Settings* settings);
|
2020-02-13 16:27:06 +01:00
|
|
|
|
|
|
|
#endif
|