0
0
mirror of https://github.com/mongodb/mongo.git synced 2024-11-30 17:10:48 +01:00
mongodb/util/assert_util.h
Aaron c971842a55 Revert "Revert "SERVER-695 don't destroy static global mutexes""
This reverts commit eb7cde3e75.

Conflicts:

	db/instance.h
2010-03-15 09:42:01 -07:00

220 lines
6.5 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;
}
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 msgasserted(int msgid, const char *msg);
inline void msgasserted(int msgid, string msg) { msgasserted(msgid, msg.c_str()); }
#ifdef assert
#undef assert
#endif
#define assert(_Expression) (void)( (!!(_Expression)) || (mongo::asserted(#_Expression, __FILE__, __LINE__), 0) )
/* "user assert". if asserts, user did something wrong, not our code */
//#define uassert( 10269 , _Expression) (void)( (!!(_Expression)) || (uasserted(#_Expression, __FILE__, __LINE__), 0) )
#define uassert(msgid, msg,_Expression) (void)( (!!(_Expression)) || (mongo::uasserted(msgid, msg), 0) )
#define xassert(_Expression) (void)( (!!(_Expression)) || (mongo::asserted(#_Expression, __FILE__, __LINE__), 0) )
#define yassert 1
/* warning only - keeps going */
#define wassert(_Expression) (void)( (!!(_Expression)) || (mongo::wasserted(#_Expression, __FILE__, __LINE__), 0) )
/* display a message, no context, and throw assertionexception
easy way to throw an exception and log something without our stack trace
display happening.
*/
#define massert(msgid, msg,_Expression) (void)( (!!(_Expression)) || (mongo::msgasserted(msgid, msg), 0) )
/* dassert is 'debug assert' -- might want to turn off for production as these
could be slow.
*/
#if defined(_DEBUG)
#define dassert assert
#else
#define dassert(x)
#endif
// some special ids that we want to duplicate
// > 10000 asserts
// < 10000 UserException
#define ASSERT_ID_DUPKEY 11000
void streamNotGood( int code , string msg , std::ios& myios );
#define ASSERT_STREAM_GOOD(msgid,msg,stream) (void)( (!!((stream).good())) || (mongo::streamNotGood(msgid, msg, stream), 0) )
} // namespace mongo
#define 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( 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; \
}