2008-06-06 15:43:15 +02:00
|
|
|
// stdafx.h : include file for standard system include files,
|
|
|
|
// or project specific include files that are used frequently, but
|
|
|
|
// are changed infrequently
|
|
|
|
//
|
|
|
|
|
2008-07-20 23:37:33 +02:00
|
|
|
/**
|
|
|
|
* Copyright (C) 2008 10gen Inc.
|
2008-12-29 02:28:49 +01:00
|
|
|
*
|
2008-07-20 23:37:33 +02:00
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Affero General Public License, version 3,
|
|
|
|
* as published by the Free Software Foundation.
|
2008-12-29 02:28:49 +01:00
|
|
|
*
|
2008-07-20 23:37:33 +02:00
|
|
|
* 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 Affero General Public License for more details.
|
2008-12-29 02:28:49 +01:00
|
|
|
*
|
2008-07-20 23:37:33 +02:00
|
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2008-06-06 15:43:15 +02:00
|
|
|
#pragma once
|
|
|
|
|
2009-01-14 23:09:51 +01:00
|
|
|
namespace mongo {
|
|
|
|
|
2008-12-31 14:40:04 +01:00
|
|
|
#define NOMINMAX
|
|
|
|
|
2008-06-06 15:43:15 +02:00
|
|
|
#if defined(_WIN32)
|
2009-01-15 16:17:11 +01:00
|
|
|
const bool debug=true;
|
2008-06-06 15:43:15 +02:00
|
|
|
#else
|
2009-01-15 16:17:11 +01:00
|
|
|
const bool debug=false;
|
2008-06-06 15:43:15 +02:00
|
|
|
#endif
|
|
|
|
|
2009-01-14 23:09:51 +01:00
|
|
|
} // namespace mongo
|
|
|
|
|
2008-07-17 19:42:20 +02:00
|
|
|
#include <memory>
|
2008-11-17 01:11:21 +01:00
|
|
|
#include "stdlib.h"
|
|
|
|
#include "string.h"
|
|
|
|
#include "limits.h"
|
2008-07-17 19:42:20 +02:00
|
|
|
|
2009-01-14 23:09:51 +01:00
|
|
|
namespace mongo {
|
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
void sayDbContext(const char *msg = 0);
|
|
|
|
void dbexit(int returnCode, const char *whyMsg = "");
|
2009-01-26 15:18:41 +01:00
|
|
|
void exit( int status );
|
2008-07-17 19:42:20 +02:00
|
|
|
|
2009-03-17 15:22:26 +01:00
|
|
|
void printGitVersion();
|
2009-03-18 19:15:52 +01:00
|
|
|
void printSysInfo();
|
2009-03-17 15:22:26 +01:00
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
inline void * ourmalloc(size_t size) {
|
|
|
|
void *x = malloc(size);
|
|
|
|
if ( x == 0 ) dbexit(42, "malloc fails");
|
|
|
|
return x;
|
|
|
|
}
|
2008-07-17 19:42:20 +02:00
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
inline void * ourrealloc(void *ptr, size_t size) {
|
|
|
|
void *x = realloc(ptr, size);
|
|
|
|
if ( x == 0 ) dbexit(43, "realloc fails");
|
|
|
|
return x;
|
|
|
|
}
|
2008-07-17 19:42:20 +02:00
|
|
|
|
|
|
|
#define malloc ourmalloc
|
|
|
|
#define realloc ourrealloc
|
|
|
|
|
2009-01-14 23:09:51 +01:00
|
|
|
} // namespace mongo
|
|
|
|
|
2008-06-06 15:43:15 +02:00
|
|
|
#include "targetver.h"
|
|
|
|
|
2008-09-11 16:45:57 +02:00
|
|
|
#include <string>
|
2008-12-05 00:11:25 +01:00
|
|
|
#include "time.h"
|
|
|
|
|
2008-09-11 16:45:57 +02:00
|
|
|
using namespace std;
|
2008-12-05 00:11:25 +01:00
|
|
|
|
2009-01-14 23:09:51 +01:00
|
|
|
namespace mongo {
|
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
/* 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;
|
|
|
|
}
|
|
|
|
char msg[128];
|
|
|
|
char context[128];
|
|
|
|
const char *file;
|
|
|
|
unsigned line;
|
|
|
|
time_t when;
|
|
|
|
void set(const char *m, const char *ctxt, const char *f, unsigned l) {
|
|
|
|
strncpy(msg, m, 127);
|
|
|
|
strncpy(context, ctxt, 127);
|
|
|
|
file = f;
|
|
|
|
line = l;
|
|
|
|
when = time(0);
|
|
|
|
}
|
|
|
|
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];
|
2008-12-05 00:11:25 +01:00
|
|
|
|
2009-02-20 16:48:32 +01:00
|
|
|
class DBException : public exception {
|
2009-02-06 22:21:49 +01:00
|
|
|
public:
|
|
|
|
virtual const char* what() const throw() = 0;
|
2009-02-19 18:43:46 +01:00
|
|
|
virtual string toString() const {
|
|
|
|
return what();
|
|
|
|
}
|
2009-02-20 16:48:32 +01:00
|
|
|
operator string() const { return toString(); }
|
2009-02-06 22:21:49 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
class AssertionException : public DBException {
|
2009-01-15 16:17:11 +01:00
|
|
|
public:
|
|
|
|
string msg;
|
|
|
|
AssertionException() { }
|
2009-02-18 01:21:01 +01:00
|
|
|
virtual ~AssertionException() throw() { }
|
2009-01-15 16:17:11 +01:00
|
|
|
virtual bool severe() {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
virtual bool isUserAssertion() {
|
|
|
|
return false;
|
|
|
|
}
|
2009-02-06 22:21:49 +01:00
|
|
|
virtual const char* what() const throw() { return msg.c_str(); }
|
2009-01-15 16:17:11 +01:00
|
|
|
};
|
|
|
|
|
2009-02-06 22:56:14 +01:00
|
|
|
/* UserExceptions are valid errors that a user can cause, like out of disk space or duplicate key */
|
|
|
|
class UserException : public AssertionException {
|
2009-01-15 16:17:11 +01:00
|
|
|
public:
|
2009-02-06 22:56:14 +01:00
|
|
|
UserException(const char *_msg) {
|
2009-01-15 16:17:11 +01:00
|
|
|
msg = _msg;
|
|
|
|
}
|
2009-02-06 22:56:14 +01:00
|
|
|
UserException(string _msg) {
|
2009-01-15 16:17:11 +01:00
|
|
|
msg = _msg;
|
|
|
|
}
|
|
|
|
virtual bool severe() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
virtual bool isUserAssertion() {
|
|
|
|
return true;
|
|
|
|
}
|
2009-02-19 18:43:46 +01:00
|
|
|
virtual string toString() const {
|
2009-01-15 16:17:11 +01:00
|
|
|
return "userassert:" + msg;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class MsgAssertionException : public AssertionException {
|
|
|
|
public:
|
|
|
|
MsgAssertionException(const char *_msg) {
|
|
|
|
msg = _msg;
|
|
|
|
}
|
|
|
|
virtual bool severe() {
|
|
|
|
return false;
|
|
|
|
}
|
2009-02-19 18:43:46 +01:00
|
|
|
virtual string toString() const {
|
2009-01-15 16:17:11 +01:00
|
|
|
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(const char *msg);
|
2009-02-01 20:53:54 +01:00
|
|
|
inline void uasserted(string msg) { uasserted(msg.c_str()); }
|
2009-01-19 02:31:33 +01:00
|
|
|
void uassert_nothrow(const char *msg); // reported via lasterror, but don't throw exception
|
2009-01-15 16:17:11 +01:00
|
|
|
void msgasserted(const char *msg);
|
2009-02-01 20:53:54 +01:00
|
|
|
inline void msgasserted(string msg) { msgasserted(msg.c_str()); }
|
2008-06-06 15:43:15 +02:00
|
|
|
|
|
|
|
#ifdef assert
|
|
|
|
#undef assert
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define assert(_Expression) (void)( (!!(_Expression)) || (asserted(#_Expression, __FILE__, __LINE__), 0) )
|
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
/* "user assert". if asserts, user did something wrong, not our code */
|
2008-08-25 22:46:39 +02:00
|
|
|
//#define uassert(_Expression) (void)( (!!(_Expression)) || (uasserted(#_Expression, __FILE__, __LINE__), 0) )
|
|
|
|
#define uassert(msg,_Expression) (void)( (!!(_Expression)) || (uasserted(msg), 0) )
|
2008-07-28 19:51:39 +02:00
|
|
|
|
2008-06-06 15:43:15 +02:00
|
|
|
#define xassert(_Expression) (void)( (!!(_Expression)) || (asserted(#_Expression, __FILE__, __LINE__), 0) )
|
|
|
|
|
|
|
|
#define yassert 1
|
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
/* warning only - keeps going */
|
2008-06-06 15:43:15 +02:00
|
|
|
#define wassert(_Expression) (void)( (!!(_Expression)) || (wasserted(#_Expression, __FILE__, __LINE__), 0) )
|
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
/* display a message, no context, and throw assertionexception
|
2008-09-05 16:40:00 +02:00
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
easy way to throw an exception and log something without our stack trace
|
|
|
|
display happening.
|
|
|
|
*/
|
2008-07-11 18:27:23 +02:00
|
|
|
#define massert(msg,_Expression) (void)( (!!(_Expression)) || (msgasserted(msg), 0) )
|
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
/* dassert is 'debug assert' -- might want to turn off for production as these
|
|
|
|
could be slow.
|
|
|
|
*/
|
2009-02-03 00:18:22 +01:00
|
|
|
#if defined(_DEBUG)
|
2008-07-01 03:08:31 +02:00
|
|
|
#define dassert assert
|
2009-02-03 00:18:22 +01:00
|
|
|
#else
|
|
|
|
#define dassert(x)
|
|
|
|
#endif
|
2008-07-01 03:08:31 +02:00
|
|
|
|
2009-01-14 23:09:51 +01:00
|
|
|
} // namespace mongo
|
|
|
|
|
2008-06-06 15:43:15 +02:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <sstream>
|
|
|
|
#include <signal.h>
|
|
|
|
|
2009-01-14 23:09:51 +01:00
|
|
|
namespace mongo {
|
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
typedef char _TCHAR;
|
2008-06-06 15:43:15 +02:00
|
|
|
|
2009-01-14 23:09:51 +01:00
|
|
|
} // namespace mongo
|
|
|
|
|
2008-06-06 15:43:15 +02:00
|
|
|
#include <iostream>
|
|
|
|
#include <fstream>
|
|
|
|
#include <map>
|
|
|
|
#include <vector>
|
|
|
|
#include <set>
|
2009-01-14 23:09:51 +01:00
|
|
|
|
|
|
|
namespace mongo {
|
|
|
|
|
2008-09-11 16:45:57 +02:00
|
|
|
//using namespace std;
|
2008-06-06 15:43:15 +02:00
|
|
|
|
|
|
|
#if !defined(_WIN32)
|
2009-01-15 16:17:11 +01:00
|
|
|
typedef int HANDLE;
|
|
|
|
inline void strcpy_s(char *dst, unsigned len, const char *src) {
|
|
|
|
strcpy(dst, src);
|
|
|
|
}
|
2008-06-06 15:43:15 +02:00
|
|
|
#else
|
2009-01-15 16:17:11 +01:00
|
|
|
typedef void *HANDLE;
|
2008-06-06 15:43:15 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
//#if defined(CHAR)
|
|
|
|
//#error CHAR already defined?
|
|
|
|
//#endif
|
|
|
|
|
|
|
|
//#if defined(_WIN32_WINNT)
|
|
|
|
//typedef wchar_t CHAR;
|
|
|
|
//#else
|
|
|
|
// more to be done...linux unicode is 32 bit.
|
|
|
|
//typedef unsigned short CHAR; // 16 bit unicode
|
|
|
|
//#endif
|
|
|
|
|
|
|
|
#define null (0)
|
|
|
|
|
2009-02-04 19:09:00 +01:00
|
|
|
void rawOut( const string &s );
|
2009-02-02 15:52:14 +01:00
|
|
|
|
2009-01-14 23:09:51 +01:00
|
|
|
} // namespace mongo
|
|
|
|
|
2008-06-06 15:43:15 +02:00
|
|
|
#include <vector>
|
|
|
|
|
2009-01-14 23:09:51 +01:00
|
|
|
namespace mongo {
|
|
|
|
|
2008-06-06 15:43:15 +02:00
|
|
|
// for debugging
|
2009-01-15 16:17:11 +01:00
|
|
|
typedef struct _Ints {
|
|
|
|
int i[100];
|
|
|
|
} *Ints;
|
|
|
|
typedef struct _Chars {
|
|
|
|
char c[200];
|
|
|
|
} *Chars;
|
|
|
|
|
|
|
|
typedef char CHARS[400];
|
|
|
|
|
|
|
|
typedef struct _OWS {
|
|
|
|
int size;
|
|
|
|
char type;
|
|
|
|
char string[400];
|
|
|
|
} *OWS;
|
|
|
|
|
|
|
|
class Database;
|
2009-01-15 16:08:20 +01:00
|
|
|
//extern Database *database;
|
2009-01-15 16:17:11 +01:00
|
|
|
extern const char *curNs;
|
2008-06-06 15:43:15 +02:00
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
/* for now, running on win32 means development not production --
|
|
|
|
use this to log things just there.
|
|
|
|
*/
|
2008-06-10 22:57:25 +02:00
|
|
|
#if defined(_WIN32)
|
2008-12-29 02:28:49 +01:00
|
|
|
#define WIN if( 1 )
|
2008-06-08 16:58:19 +02:00
|
|
|
#else
|
2008-12-29 02:28:49 +01:00
|
|
|
#define WIN if( 0 )
|
2008-06-08 16:58:19 +02:00
|
|
|
#endif
|
|
|
|
|
2009-03-12 14:54:21 +01:00
|
|
|
#if defined(_DEBUG)
|
|
|
|
#define DEV if( 1 )
|
|
|
|
#else
|
|
|
|
#define DEV if( 0 )
|
|
|
|
#endif
|
|
|
|
|
2008-12-29 02:28:49 +01:00
|
|
|
#define DEBUGGING if( 0 )
|
2008-06-06 15:43:15 +02:00
|
|
|
|
2009-01-15 16:17:11 +01:00
|
|
|
extern unsigned occasion;
|
2009-02-11 18:25:28 +01:00
|
|
|
extern unsigned occasionR;
|
2009-01-15 16:17:11 +01:00
|
|
|
extern unsigned once;
|
2008-06-06 15:43:15 +02:00
|
|
|
|
2008-12-29 02:28:49 +01:00
|
|
|
#define OCCASIONALLY if( ++occasion % 16 == 0 )
|
2009-02-11 18:25:28 +01:00
|
|
|
#define RARELY if( ++occasionR % 128 == 0 )
|
2008-12-29 02:28:49 +01:00
|
|
|
#define ONCE if( ++once == 1 )
|
2008-06-17 22:57:12 +02:00
|
|
|
|
|
|
|
#if defined(_WIN32)
|
2009-02-05 22:17:46 +01:00
|
|
|
#define strcasecmp _stricmp
|
2009-01-15 16:17:11 +01:00
|
|
|
inline void our_debug_free(void *p) {
|
|
|
|
unsigned *u = (unsigned *) p;
|
|
|
|
u[0] = 0xEEEEEEEE;
|
|
|
|
free(p);
|
|
|
|
}
|
2008-06-17 22:57:12 +02:00
|
|
|
#define free our_debug_free
|
|
|
|
#endif
|
2008-06-23 22:28:25 +02:00
|
|
|
|
2009-01-14 23:09:51 +01:00
|
|
|
} // namespace mongo
|
|
|
|
|
2008-06-24 23:31:51 +02:00
|
|
|
#undef yassert
|
2008-12-31 17:33:41 +01:00
|
|
|
#include <boost/archive/iterators/base64_from_binary.hpp>
|
2009-01-06 19:02:09 +01:00
|
|
|
#include <boost/archive/iterators/binary_from_base64.hpp>
|
2008-12-31 17:33:41 +01:00
|
|
|
#include <boost/archive/iterators/transform_width.hpp>
|
2009-02-09 21:57:30 +01:00
|
|
|
#include <boost/date_time/posix_time/posix_time.hpp>
|
2009-01-06 19:02:09 +01:00
|
|
|
#include <boost/filesystem/convenience.hpp>
|
|
|
|
#include <boost/filesystem/operations.hpp>
|
2009-01-14 23:09:51 +01:00
|
|
|
#include <boost/program_options.hpp>
|
2009-01-06 20:27:05 +01:00
|
|
|
#include <boost/shared_ptr.hpp>
|
2009-01-06 19:02:09 +01:00
|
|
|
#define BOOST_SPIRIT_THREADSAFE
|
|
|
|
//#define BOOST_SPIRIT_DEBUG
|
|
|
|
#include <boost/spirit/core.hpp>
|
|
|
|
#include <boost/spirit/utility/loops.hpp>
|
2008-06-24 23:31:51 +02:00
|
|
|
#undef assert
|
|
|
|
#define assert xassert
|
|
|
|
#define yassert 1
|
2008-12-29 02:28:49 +01:00
|
|
|
using namespace boost::filesystem;
|
2008-06-27 20:35:05 +02:00
|
|
|
|
|
|
|
#include "util/goodies.h"
|
2008-07-09 18:32:11 +02:00
|
|
|
#include "util/log.h"
|