mirror of
https://github.com/mongodb/mongo.git
synced 2024-11-29 16:47:28 +01:00
128 lines
4.1 KiB
C++
128 lines
4.1 KiB
C++
// optime.h - OpTime class
|
|
|
|
/* 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/concurrency.h"
|
|
|
|
namespace mongo {
|
|
void exitCleanly( int code );
|
|
|
|
/* Operation sequence #. A combination of current second plus an ordinal value.
|
|
*/
|
|
struct ClockSkewException : public DBException {
|
|
virtual const char* what() const throw() { return "clock skew exception"; }
|
|
virtual int getCode(){ return 20001; }
|
|
};
|
|
|
|
#pragma pack(4)
|
|
class OpTime {
|
|
unsigned i;
|
|
unsigned secs;
|
|
static OpTime last;
|
|
public:
|
|
static void setLast(const Date_t &date) {
|
|
last = OpTime(date);
|
|
}
|
|
unsigned getSecs() const {
|
|
return secs;
|
|
}
|
|
OpTime(Date_t date) {
|
|
reinterpret_cast<unsigned long long&>(*this) = date.millis;
|
|
}
|
|
OpTime(unsigned long long date) {
|
|
reinterpret_cast<unsigned long long&>(*this) = date;
|
|
}
|
|
OpTime(unsigned a, unsigned b) {
|
|
secs = a;
|
|
i = b;
|
|
}
|
|
OpTime() {
|
|
secs = 0;
|
|
i = 0;
|
|
}
|
|
static OpTime now() {
|
|
unsigned t = (unsigned) time(0);
|
|
// DEV assertInWriteLock();
|
|
if ( t < last.secs ){
|
|
bool toLog = false;
|
|
ONCE toLog = true;
|
|
RARELY toLog = true;
|
|
if ( last.i & 0x80000000 )
|
|
toLog = true;
|
|
if ( toLog )
|
|
log() << "clock skew detected prev: " << last.secs << " now: " << t << " trying to handle..." << endl;
|
|
if ( last.i & 0x80000000 ) {
|
|
log() << "ERROR Large clock skew detected, shutting down" << endl;
|
|
throw ClockSkewException();
|
|
}
|
|
t = last.secs;
|
|
}
|
|
if ( last.secs == t ) {
|
|
last.i++;
|
|
return last;
|
|
}
|
|
last = OpTime(t, 1);
|
|
return last;
|
|
}
|
|
|
|
/* We store OpTime's in the database as BSON Date datatype -- we needed some sort of
|
|
64 bit "container" for these values. While these are not really "Dates", that seems a
|
|
better choice for now than say, Number, which is floating point. Note the BinData type
|
|
is perhaps the cleanest choice, lacking a true unsigned64 datatype, but BinData has 5
|
|
bytes of overhead.
|
|
*/
|
|
unsigned long long asDate() const {
|
|
return *((unsigned long long *) &i);
|
|
}
|
|
// unsigned long long& asDate() { return *((unsigned long long *) &i); }
|
|
|
|
bool isNull() {
|
|
return secs == 0;
|
|
}
|
|
|
|
string toStringLong() const {
|
|
char buf[64];
|
|
time_t_to_String(secs, buf);
|
|
stringstream ss;
|
|
ss << buf << ' ';
|
|
ss << hex << secs << ':' << i;
|
|
return ss.str();
|
|
}
|
|
|
|
string toString() const {
|
|
stringstream ss;
|
|
ss << hex << secs << ':' << i;
|
|
return ss.str();
|
|
}
|
|
operator string() const { return toString(); }
|
|
bool operator==(const OpTime& r) const {
|
|
return i == r.i && secs == r.secs;
|
|
}
|
|
bool operator!=(const OpTime& r) const {
|
|
return !(*this == r);
|
|
}
|
|
bool operator<(const OpTime& r) const {
|
|
if ( secs != r.secs )
|
|
return secs < r.secs;
|
|
return i < r.i;
|
|
}
|
|
};
|
|
#pragma pack()
|
|
|
|
} // namespace mongo
|