2019-09-17 21:00:59 +02:00
/*
* Cppcheck - A tool for static C / C + + code analysis
* Copyright ( C ) 2007 - 2019 Cppcheck team .
*
* 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 exprengineH
# define exprengineH
//---------------------------------------------------------------------------
2019-09-19 19:40:00 +02:00
# include "config.h"
2019-09-17 21:00:59 +02:00
# include <functional>
# include <map>
# include <memory>
# include <set>
# include <string>
# include <vector>
# include <stdint.h>
class ErrorLogger ;
class Tokenizer ;
class Scope ;
class Settings ;
class Token ;
# ifdef __GNUC__
typedef __int128_t int128_t ;
# else
typedef long long int128_t ;
2019-09-18 21:50:23 +02:00
# ifdef _MSC_VER
# pragma message(__FILE__ "(" _CRT_STRINGIZE(__LINE__) ")" ": warning: TODO No 128-bit integer type is available => Limited analysis of large integers...")
# else
2019-09-17 21:00:59 +02:00
# warning TODO No 128-bit integer type is available => Limited analysis of large integers
# endif
2019-09-18 21:50:23 +02:00
# endif
2019-09-17 21:00:59 +02:00
namespace ExprEngine {
std : : string str ( int128_t ) ;
// TODO we need to handle floats, containers, pointers, aliases and structs and stuff
2019-09-22 21:14:20 +02:00
enum class ValueType { UninitValue , IntRange , FloatRange , PointerValue , ArrayValue , StringLiteralValue , StructValue , AddressOfValue , BinOpResult } ;
2019-09-17 21:00:59 +02:00
class Value ;
typedef std : : shared_ptr < Value > ValuePtr ;
class Value {
public :
Value ( const std : : string & name ) : name ( name ) { }
virtual ~ Value ( ) { }
virtual ValueType type ( ) const = 0 ;
virtual std : : string getRange ( ) const = 0 ;
virtual bool isIntValueInRange ( int value ) const {
( void ) value ;
return false ;
}
const std : : string name ;
} ;
class UninitValue : public Value {
public :
UninitValue ( ) : Value ( " ? " ) { }
ValueType type ( ) const override {
return ValueType : : UninitValue ;
}
std : : string getRange ( ) const override {
return " ? " ;
}
} ;
class IntRange : public Value {
public :
IntRange ( const std : : string & name , int128_t minValue , int128_t maxValue )
: Value ( name )
, minValue ( minValue )
, maxValue ( maxValue ) {
}
2019-09-20 21:57:16 +02:00
~ IntRange ( ) OVERRIDE { }
2019-09-17 21:00:59 +02:00
ValueType type ( ) const override {
return ValueType : : IntRange ;
}
std : : string getRange ( ) const override {
2019-09-21 14:17:16 +02:00
if ( minValue = = maxValue )
return str ( minValue ) ;
return str ( minValue ) + " : " + str ( maxValue ) ;
2019-09-17 21:00:59 +02:00
}
bool isIntValueInRange ( int value ) const override {
return value > = minValue & & value < = maxValue ;
}
int128_t minValue ;
int128_t maxValue ;
} ;
2019-09-22 21:14:20 +02:00
class FloatRange : public Value {
public :
FloatRange ( const std : : string & name , long double minValue , long double maxValue )
: Value ( name )
, minValue ( minValue )
, maxValue ( maxValue ) {
}
~ FloatRange ( ) OVERRIDE { }
ValueType type ( ) const override {
return ValueType : : FloatRange ;
}
std : : string getRange ( ) const override {
return std : : to_string ( minValue ) + " : " + std : : to_string ( maxValue ) ;
}
long double minValue ;
long double maxValue ;
} ;
2019-09-17 21:00:59 +02:00
class PointerValue : public Value {
public :
2019-09-22 10:56:49 +02:00
PointerValue ( const std : : string & name , ValuePtr data , bool null , bool uninitData )
: Value ( name )
, data ( data )
, null ( null )
, uninitData ( uninitData ) {
}
2019-09-17 21:00:59 +02:00
ValueType type ( ) const override {
return ValueType : : PointerValue ;
}
2019-09-22 10:56:49 +02:00
std : : string getRange ( ) const override ;
2019-09-17 21:00:59 +02:00
ValuePtr data ;
2019-09-22 10:56:49 +02:00
bool null ;
bool uninitData ;
2019-09-17 21:00:59 +02:00
} ;
class ArrayValue : public Value {
public :
const int MAXSIZE = 0x100000 ;
ArrayValue ( const std : : string & name , size_t size )
: Value ( name ) {
data . resize ( ( size < MAXSIZE ) ? size : MAXSIZE ,
std : : make_shared < UninitValue > ( ) ) ;
}
ValueType type ( ) const override {
return ValueType : : ArrayValue ;
}
2019-09-21 14:17:16 +02:00
std : : string getRange ( ) const override ;
2019-09-17 21:00:59 +02:00
void assign ( ValuePtr index , ValuePtr value ) ;
ValuePtr read ( ValuePtr index ) ;
std : : vector < ValuePtr > data ;
} ;
2019-09-21 11:36:34 +02:00
class StringLiteralValue : public Value {
public :
StringLiteralValue ( const std : : string & name , const std : : string & s )
: Value ( name )
, string ( s ) {
}
ValueType type ( ) const override {
return ValueType : : StringLiteralValue ;
}
std : : string getRange ( ) const override {
return " \" " + string + " \" " ;
}
int size ( ) const {
return string . size ( ) ;
}
const std : : string string ;
} ;
2019-09-17 21:00:59 +02:00
class StructValue : public Value {
public :
2019-09-17 22:28:36 +02:00
explicit StructValue ( const std : : string & name ) : Value ( name ) { }
2019-09-17 21:00:59 +02:00
ValueType type ( ) const override {
return ValueType : : StructValue ;
}
std : : string getRange ( ) const override {
return name ;
}
ValuePtr getValueOfMember ( const std : : string & name ) const {
auto it = member . find ( name ) ;
return ( it = = member . end ( ) ) ? ValuePtr ( ) : it - > second ;
}
std : : map < std : : string , ValuePtr > member ;
} ;
class AddressOfValue : public Value {
public :
AddressOfValue ( const std : : string & name , int varId )
: Value ( name )
, varId ( varId )
{ }
ValueType type ( ) const override {
return ValueType : : AddressOfValue ;
}
std : : string getRange ( ) const override {
return " (&@ " + std : : to_string ( varId ) ;
}
int varId ;
} ;
class BinOpResult : public Value {
public :
BinOpResult ( const std : : string & binop , ValuePtr op1 , ValuePtr op2 )
: Value ( " ( " + op1 - > name + " ) " + binop + " ( " + op2 - > name + " ) " )
, binop ( binop )
, op1 ( op1 )
, op2 ( op2 ) {
auto b1 = std : : dynamic_pointer_cast < BinOpResult > ( op1 ) ;
if ( b1 )
mLeafs = b1 - > mLeafs ;
else
mLeafs . insert ( op1 ) ;
auto b2 = std : : dynamic_pointer_cast < BinOpResult > ( op2 ) ;
if ( b2 )
mLeafs . insert ( b2 - > mLeafs . begin ( ) , b2 - > mLeafs . end ( ) ) ;
else
mLeafs . insert ( op2 ) ;
}
ValueType type ( ) const override {
return ValueType : : BinOpResult ;
}
std : : string getRange ( ) const override ;
2019-09-22 21:14:20 +02:00
struct IntOrFloatValue {
void setIntValue ( int128_t v ) {
type = INT ;
intValue = v ;
floatValue = 0 ;
}
void setFloatValue ( long double v ) {
type = FLOAT ;
intValue = 0 ;
floatValue = v ;
}
enum { INT , FLOAT } type ;
bool isFloat ( ) const {
return type = = FLOAT ;
}
int128_t intValue ;
long double floatValue ;
} ;
void getRange ( IntOrFloatValue * minValue , IntOrFloatValue * maxValue ) const ;
2019-09-17 21:00:59 +02:00
bool isIntValueInRange ( int value ) const override ;
std : : string binop ;
ValuePtr op1 ;
ValuePtr op2 ;
private :
2019-09-22 21:14:20 +02:00
IntOrFloatValue evaluate ( int test , const std : : map < ValuePtr , int > & valueBit ) const ;
IntOrFloatValue evaluateOperand ( int test , const std : : map < ValuePtr , int > & valueBit , ValuePtr value ) const ;
2019-09-17 21:00:59 +02:00
std : : set < ValuePtr > mLeafs ;
} ;
typedef std : : function < void ( const Token * , const ExprEngine : : Value & ) > Callback ;
/** Execute all functions */
2019-09-19 19:40:00 +02:00
void CPPCHECKLIB executeAllFunctions ( const Tokenizer * tokenizer , const Settings * settings , const std : : vector < Callback > & callbacks ) ;
2019-09-17 21:00:59 +02:00
void executeFunction ( const Scope * functionScope , const Tokenizer * tokenizer , const Settings * settings , const std : : vector < Callback > & callbacks ) ;
void runChecks ( ErrorLogger * errorLogger , const Tokenizer * tokenizer , const Settings * settings ) ;
}
# endif // exprengineH