0
0
mirror of https://github.com/mongodb/mongo.git synced 2024-12-01 09:32:32 +01:00
mongodb/util/mmap_posix.cpp

215 lines
6.8 KiB
C++

// mmap_posix.cpp
/* 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.
*/
#include "pch.h"
#include "mmap.h"
#include "file_allocator.h"
#include "../db/concurrency.h"
#include <errno.h>
#include <sys/mman.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include "../util/processinfo.h"
#include "mongoutils/str.h"
using namespace mongoutils;
namespace mongo {
MemoryMappedFile::MemoryMappedFile() {
fd = 0;
maphandle = 0;
len = 0;
created();
}
void MemoryMappedFile::close() {
mmmutex.assertExclusivelyLocked();
for( vector<void*>::iterator i = views.begin(); i != views.end(); i++ ) {
munmap(*i,len);
}
views.clear();
if ( fd )
::close(fd);
fd = 0;
destroyed(); // cleans up from the master list of mmaps
}
#ifndef O_NOATIME
#define O_NOATIME (0)
#endif
#ifndef MAP_NORESERVE
#define MAP_NORESERVE (0)
#endif
#if defined(__sunos__)
MAdvise::MAdvise(void *,unsigned, Advice) { }
MAdvise::~MAdvise() { }
#else
MAdvise::MAdvise(void *p, unsigned len, Advice a) : _p(p), _len(len) {
assert( a == Sequential ); // more later
madvise(_p,_len,MADV_SEQUENTIAL);
}
MAdvise::~MAdvise() {
madvise(_p,_len,MADV_NORMAL);
}
#endif
void* MemoryMappedFile::map(const char *filename, unsigned long long &length, int options) {
// length may be updated by callee.
setFilename(filename);
FileAllocator::get()->allocateAsap( filename, length );
len = length;
massert( 10446 , str::stream() << "mmap: can't map area of size 0 file: " << filename, length > 0 );
fd = open(filename, O_RDWR | O_NOATIME);
if ( fd <= 0 ) {
log() << "couldn't open " << filename << ' ' << errnoWithDescription() << endl;
fd = 0; // our sentinel for not opened
return 0;
}
unsigned long long filelen = lseek(fd, 0, SEEK_END);
uassert(10447, str::stream() << "map file alloc failed, wanted: " << length << " filelen: " << filelen << ' ' << sizeof(size_t), filelen == length );
lseek( fd, 0, SEEK_SET );
void * view = mmap(NULL, length, PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0);
if ( view == MAP_FAILED ) {
error() << " mmap() failed for " << filename << " len:" << length << " " << errnoWithDescription() << endl;
if ( errno == ENOMEM ) {
if( sizeof(void*) == 4 )
error() << "mmap failed with out of memory. You are using a 32-bit build and probably need to upgrade to 64" << endl;
else
error() << "mmap failed with out of memory. (64 bit build)" << endl;
}
return 0;
}
#if defined(__sunos__)
#warning madvise not supported on solaris yet
#else
if ( options & SEQUENTIAL ) {
if ( madvise( view , length , MADV_SEQUENTIAL ) ) {
warning() << "map: madvise failed for " << filename << ' ' << errnoWithDescription() << endl;
}
}
#endif
views.push_back( view );
DEV if (! dbMutex.info().isLocked()) {
_unlock();
}
return view;
}
void* MemoryMappedFile::createReadOnlyMap() {
void * x = mmap( /*start*/0 , len , PROT_READ , MAP_SHARED , fd , 0 );
if( x == MAP_FAILED ) {
if ( errno == ENOMEM ) {
if( sizeof(void*) == 4 )
error() << "mmap ro failed with out of memory. You are using a 32-bit build and probably need to upgrade to 64" << endl;
else
error() << "mmap ro failed with out of memory. (64 bit build)" << endl;
}
return 0;
}
return x;
}
void* MemoryMappedFile::createPrivateMap() {
void * x = mmap( /*start*/0 , len , PROT_READ|PROT_WRITE , MAP_PRIVATE|MAP_NORESERVE , fd , 0 );
if( x == MAP_FAILED ) {
if ( errno == ENOMEM ) {
if( sizeof(void*) == 4 ) {
error() << "mmap private failed with out of memory. You are using a 32-bit build and probably need to upgrade to 64" << endl;
}
else {
error() << "mmap private failed with out of memory. (64 bit build)" << endl;
}
}
else {
error() << "mmap private failed " << errnoWithDescription() << endl;
}
return 0;
}
views.push_back(x);
return x;
}
void* MemoryMappedFile::remapPrivateView(void *oldPrivateAddr) {
// don't unmap, just mmap over the old region
void * x = mmap( oldPrivateAddr, len , PROT_READ|PROT_WRITE , MAP_PRIVATE|MAP_NORESERVE|MAP_FIXED , fd , 0 );
if( x == MAP_FAILED ) {
int err = errno;
error() << "13601 Couldn't remap private view: " << errnoWithDescription(err) << endl;
log() << "aborting" << endl;
printMemInfo();
abort();
}
assert( x == oldPrivateAddr );
return x;
}
void MemoryMappedFile::flush(bool sync) {
if ( views.empty() || fd == 0 )
return;
if ( msync(viewForFlushing(), len, sync ? MS_SYNC : MS_ASYNC) )
problem() << "msync " << errnoWithDescription() << endl;
}
class PosixFlushable : public MemoryMappedFile::Flushable {
public:
PosixFlushable( void * view , HANDLE fd , long len )
: _view( view ) , _fd( fd ) , _len(len) {
}
void flush() {
if ( _view && _fd )
if ( msync(_view, _len, MS_SYNC ) )
problem() << "msync " << errnoWithDescription() << endl;
}
void * _view;
HANDLE _fd;
long _len;
};
MemoryMappedFile::Flushable * MemoryMappedFile::prepareFlush() {
return new PosixFlushable( viewForFlushing() , fd , len );
}
void MemoryMappedFile::_lock() {
if (! views.empty() && isMongoMMF() )
assert(mprotect(views[0], len, PROT_READ | PROT_WRITE) == 0);
}
void MemoryMappedFile::_unlock() {
if (! views.empty() && isMongoMMF() )
assert(mprotect(views[0], len, PROT_READ) == 0);
}
} // namespace mongo