mirror of
https://github.com/mongodb/mongo.git
synced 2024-11-29 00:32:18 +01:00
251 lines
7.3 KiB
C++
251 lines
7.3 KiB
C++
// assert_util.h
|
|
|
|
/* Copyright 2009 10gen Inc.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
|
|
#pragma once
|
|
|
|
#include "../db/lasterror.h"
|
|
|
|
namespace mongo {
|
|
|
|
/* these are manipulated outside of mutexes, so be careful */
|
|
struct Assertion {
|
|
Assertion() {
|
|
msg[0] = msg[127] = 0;
|
|
context[0] = context[127] = 0;
|
|
file = "";
|
|
line = 0;
|
|
when = 0;
|
|
}
|
|
private:
|
|
static mongo::mutex *_mutex;
|
|
char msg[128];
|
|
char context[128];
|
|
const char *file;
|
|
unsigned line;
|
|
time_t when;
|
|
public:
|
|
void set(const char *m, const char *ctxt, const char *f, unsigned l) {
|
|
if( _mutex == 0 ) {
|
|
/* asserted during global variable initialization */
|
|
return;
|
|
}
|
|
scoped_lock lk(*_mutex);
|
|
strncpy(msg, m, 127);
|
|
strncpy(context, ctxt, 127);
|
|
file = f;
|
|
line = l;
|
|
when = time(0);
|
|
}
|
|
std::string toString();
|
|
bool isSet() {
|
|
return when != 0;
|
|
}
|
|
};
|
|
|
|
enum {
|
|
AssertRegular = 0,
|
|
AssertW = 1,
|
|
AssertMsg = 2,
|
|
AssertUser = 3
|
|
};
|
|
|
|
/* last assert of diff types: regular, wassert, msgassert, uassert: */
|
|
extern Assertion lastAssert[4];
|
|
|
|
class AssertionCount {
|
|
public:
|
|
AssertionCount();
|
|
void rollover();
|
|
void condrollover( int newValue );
|
|
|
|
int regular;
|
|
int warning;
|
|
int msg;
|
|
int user;
|
|
int rollovers;
|
|
};
|
|
|
|
extern AssertionCount assertionCount;
|
|
|
|
class DBException : public std::exception {
|
|
public:
|
|
virtual const char* what() const throw() = 0;
|
|
virtual string toString() const {
|
|
return what();
|
|
}
|
|
virtual int getCode() = 0;
|
|
operator string() const { return toString(); }
|
|
};
|
|
|
|
class AssertionException : public DBException {
|
|
public:
|
|
int code;
|
|
string msg;
|
|
AssertionException() { code = 0; }
|
|
virtual ~AssertionException() throw() { }
|
|
virtual bool severe() {
|
|
return true;
|
|
}
|
|
virtual bool isUserAssertion() {
|
|
return false;
|
|
}
|
|
virtual int getCode(){ return code; }
|
|
virtual const char* what() const throw() { return msg.c_str(); }
|
|
|
|
/* true if an interrupted exception - see KillCurrentOp */
|
|
bool interrupted() {
|
|
return code == 11600 || code == 11601;
|
|
}
|
|
};
|
|
|
|
/* UserExceptions are valid errors that a user can cause, like out of disk space or duplicate key */
|
|
class UserException : public AssertionException {
|
|
public:
|
|
UserException(int c , const string& m) {
|
|
code = c;
|
|
msg = m;
|
|
}
|
|
virtual bool severe() {
|
|
return false;
|
|
}
|
|
virtual bool isUserAssertion() {
|
|
return true;
|
|
}
|
|
virtual string toString() const {
|
|
return "userassert:" + msg;
|
|
}
|
|
};
|
|
|
|
class MsgAssertionException : public AssertionException {
|
|
public:
|
|
MsgAssertionException(int c, const char *m) {
|
|
code = c;
|
|
msg = m;
|
|
}
|
|
MsgAssertionException(int c, const string& m) {
|
|
code = c;
|
|
msg = m;
|
|
}
|
|
virtual bool severe() {
|
|
return false;
|
|
}
|
|
virtual string toString() const {
|
|
return "massert:" + msg;
|
|
}
|
|
};
|
|
|
|
void asserted(const char *msg, const char *file, unsigned line);
|
|
void wasserted(const char *msg, const char *file, unsigned line);
|
|
void uasserted(int msgid, const char *msg);
|
|
inline void uasserted(int msgid , string msg) { uasserted(msgid, msg.c_str()); }
|
|
void uassert_nothrow(const char *msg); // reported via lasterror, but don't throw exception
|
|
void msgassertedNoTrace(int msgid, const char *msg);
|
|
void msgasserted(int msgid, const char *msg);
|
|
inline void msgasserted(int msgid, string msg) { msgasserted(msgid, msg.c_str()); }
|
|
|
|
#ifdef assert
|
|
#undef assert
|
|
#endif
|
|
|
|
#define MONGO_assert(_Expression) (void)( (!!(_Expression)) || (mongo::asserted(#_Expression, __FILE__, __LINE__), 0) )
|
|
#define assert MONGO_assert
|
|
|
|
/* "user assert". if asserts, user did something wrong, not our code */
|
|
inline void uassert(unsigned msgid, string msg, bool expr) {
|
|
if( !expr ) uasserted(msgid, msg.c_str());
|
|
}
|
|
|
|
template<class T>
|
|
inline void uassert(unsigned msgid, string msg, const T&t , bool expr) {
|
|
if( !expr ){
|
|
stringstream ss;
|
|
ss << msg << " " << t.toString();
|
|
uasserted(msgid, ss.str());
|
|
}
|
|
}
|
|
|
|
|
|
inline void uassert(unsigned msgid, const char * msg, bool expr) {
|
|
if( !expr ) uasserted(msgid, msg);
|
|
}
|
|
|
|
/* warning only - keeps going */
|
|
#define MONGO_wassert(_Expression) (void)( (!!(_Expression)) || (mongo::wasserted(#_Expression, __FILE__, __LINE__), 0) )
|
|
#define wassert MONGO_wassert
|
|
|
|
/* display a message, no context, and throw assertionexception
|
|
|
|
easy way to throw an exception and log something without our stack trace
|
|
display happening.
|
|
*/
|
|
inline void massert(unsigned msgid, string msg, bool expr) {
|
|
if( !expr) msgasserted(msgid, msg.c_str());
|
|
}
|
|
inline void massert(unsigned msgid, const char * msg, bool expr) {
|
|
if( !expr) msgasserted(msgid, msg);
|
|
}
|
|
|
|
/* dassert is 'debug assert' -- might want to turn off for production as these
|
|
could be slow.
|
|
*/
|
|
#if defined(_DEBUG)
|
|
# define MONGO_dassert assert
|
|
#else
|
|
# define MONGO_dassert(x)
|
|
#endif
|
|
#define dassert MONGO_dassert
|
|
|
|
// some special ids that we want to duplicate
|
|
|
|
// > 10000 asserts
|
|
// < 10000 UserException
|
|
|
|
enum { ASSERT_ID_DUPKEY = 11000 };
|
|
|
|
/* throws a uassertion with an appropriate msg */
|
|
void streamNotGood( int code , string msg , std::ios& myios );
|
|
|
|
inline void assertStreamGood(unsigned msgid, string msg, std::ios& myios) {
|
|
if( !myios.good() ) streamNotGood(msgid, msg, myios);
|
|
}
|
|
|
|
string demangleName( const type_info& typeinfo );
|
|
|
|
} // namespace mongo
|
|
|
|
#define BOOST_CHECK_EXCEPTION MONGO_BOOST_CHECK_EXCEPTION
|
|
#define MONGO_BOOST_CHECK_EXCEPTION( expression ) \
|
|
try { \
|
|
expression; \
|
|
} catch ( const std::exception &e ) { \
|
|
problem() << "caught boost exception: " << e.what() << endl; \
|
|
assert( false ); \
|
|
} catch ( ... ) { \
|
|
massert( 10437 , "unknown boost failed" , false ); \
|
|
}
|
|
|
|
#define DESTRUCTOR_GUARD MONGO_DESTRUCTOR_GUARD
|
|
#define MONGO_DESTRUCTOR_GUARD( expression ) \
|
|
try { \
|
|
expression; \
|
|
} catch ( const std::exception &e ) { \
|
|
problem() << "caught exception (" << e.what() << ") in destructor (" << __FUNCTION__ << ")" << endl; \
|
|
} catch ( ... ) { \
|
|
problem() << "caught unknown exception in destructor (" << __FUNCTION__ << ")" << endl; \
|
|
}
|