2013-09-07 01:18:14 +04:00
//--------------------------------------------------------------------*- C++ -*-
// CLING - the C++ LLVM-based InterpreterG :)
// author: Baozeng Ding <sploving1@gmail.com>
// author: Vassil Vassilev <vasil.georgiev.vasilev@cern.ch>
2014-01-07 14:08:37 +04:00
//
// This file is dual-licensed: you can choose to license it under the University
// of Illinois Open Source License or the GNU Lesser General Public License. See
// LICENSE.TXT for details.
2013-09-07 01:18:14 +04:00
//------------------------------------------------------------------------------
2015-11-27 15:54:20 +03:00
# include "cling/Interpreter/Exception.h"
2013-09-07 01:18:14 +04:00
2016-12-19 16:57:41 +03:00
# include "cling/Interpreter/InterpreterCallbacks.h"
# include "cling/Interpreter/Interpreter.h"
# include "cling/Utils/Validation.h"
# include "clang/Frontend/CompilerInstance.h"
2013-09-07 01:18:14 +04:00
# include "clang/Basic/SourceLocation.h"
# include "clang/Sema/Sema.h"
# include "clang/Sema/SemaDiagnostic.h"
2013-09-12 01:04:36 +04:00
2016-12-19 16:57:41 +03:00
extern " C " {
/// Throw an InvalidDerefException if the Arg pointer is invalid.
///\param Interp: The interpreter that has compiled the code.
///\param Expr: The expression corresponding determining the pointer value.
///\param Arg: The pointer to be checked.
///\returns void*, const-cast from Arg, to reduce the complexity in the
/// calling AST nodes, at the expense of possibly doing a
/// T* -> const void* -> const_cast<void*> -> T* round trip.
void * cling_runtime_internal_throwIfInvalidPointer ( void * Interp , void * Expr ,
const void * Arg ) {
const clang : : Expr * const E = ( const clang : : Expr * ) Expr ;
// The isValidAddress function return true even when the pointer is
// null thus the checks have to be done before returning successfully from the
// function in this specific order.
if ( ! Arg ) {
cling : : Interpreter * I = ( cling : : Interpreter * ) Interp ;
clang : : Sema & S = I - > getCI ( ) - > getSema ( ) ;
// Print a nice backtrace.
I - > getCallbacks ( ) - > PrintStackTrace ( ) ;
throw cling : : InvalidDerefException ( & S , E ,
cling : : InvalidDerefException : : DerefType : : NULL_DEREF ) ;
} else if ( ! cling : : utils : : isAddressValid ( Arg ) ) {
cling : : Interpreter * I = ( cling : : Interpreter * ) Interp ;
clang : : Sema & S = I - > getCI ( ) - > getSema ( ) ;
// Print a nice backtrace.
I - > getCallbacks ( ) - > PrintStackTrace ( ) ;
throw cling : : InvalidDerefException ( & S , E ,
cling : : InvalidDerefException : : DerefType : : INVALID_MEM ) ;
}
return const_cast < void * > ( Arg ) ;
}
}
2013-09-07 01:18:14 +04:00
namespace cling {
2016-12-19 16:57:41 +03:00
InterpreterException : : InterpreterException ( const std : : string & What ) :
std : : runtime_error ( What ) , m_Sema ( nullptr ) { }
InterpreterException : : InterpreterException ( const char * What , clang : : Sema * S ) :
std : : runtime_error ( What ) , m_Sema ( S ) { }
bool InterpreterException : : diagnose ( ) const { return false ; }
InterpreterException : : ~ InterpreterException ( ) LLVM_NOEXCEPT { }
2013-09-07 01:18:14 +04:00
2016-12-19 16:57:41 +03:00
InvalidDerefException : : InvalidDerefException ( clang : : Sema * S ,
const clang : : Expr * E ,
DerefType type )
: InterpreterException ( type = = INVALID_MEM ?
" Trying to access a pointer that points to an invalid memory address. " :
" Trying to dereference null pointer or trying to call routine taking "
" non-null arguments " , S ) ,
m_Arg ( E ) , m_Type ( type ) { }
InvalidDerefException : : ~ InvalidDerefException ( ) LLVM_NOEXCEPT { }
bool InvalidDerefException : : diagnose ( ) const {
2015-12-11 17:06:13 +03:00
// Construct custom diagnostic: warning for invalid memory address;
// no equivalent in clang.
2015-12-08 15:26:13 +03:00
if ( m_Type = = cling : : InvalidDerefException : : DerefType : : INVALID_MEM ) {
2016-12-19 16:57:41 +03:00
clang : : DiagnosticsEngine & Diags = m_Sema - > getDiagnostics ( ) ;
2015-12-11 17:06:13 +03:00
unsigned DiagID =
2016-12-19 16:57:41 +03:00
Diags . getCustomDiagID ( clang : : DiagnosticsEngine : : Warning ,
2015-12-11 17:06:13 +03:00
" invalid memory pointer passed to a callee: " ) ;
2016-12-19 16:57:41 +03:00
Diags . Report ( m_Arg - > getLocStart ( ) , DiagID ) < < m_Arg - > getSourceRange ( ) ;
2015-12-08 15:26:13 +03:00
}
else
m_Sema - > Diag ( m_Arg - > getLocStart ( ) , clang : : diag : : warn_null_arg )
2016-12-19 16:57:41 +03:00
< < m_Arg - > getSourceRange ( ) ;
return true ;
}
CompilationException : : CompilationException ( const std : : string & Reason ) :
InterpreterException ( Reason ) { }
CompilationException : : ~ CompilationException ( ) LLVM_NOEXCEPT { }
void CompilationException : : throwingHandler ( void * /*user_data*/ ,
const std : : string & reason ,
bool /*gen_crash_diag*/ ) {
throw cling : : CompilationException ( reason ) ;
2015-11-27 15:54:20 +03:00
}
2013-09-07 01:18:14 +04:00
} // end namespace cling