123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281 |
- // -*- mode: cpp; mode: fold -*-
- // Description /*{{{*/
- // $Id: mmap.cc,v 1.3 1999/10/24 06:53:12 jgg Exp $
- /* ######################################################################
-
- MMap Class - Provides 'real' mmap or a faked mmap using read().
- MMap cover class.
- Some broken versions of glibc2 (libc6) have a broken definition
- of mmap that accepts a char * -- all other systems (and libc5) use
- void *. We can't safely do anything here that would be portable, so
- libc6 generates warnings -- which should be errors, g++ isn't properly
- strict.
-
- The configure test notes that some OS's have broken private mmap's
- so on those OS's we can't use mmap. This means we have to use
- configure to test mmap and can't rely on the POSIX
- _POSIX_MAPPED_FILES test.
-
- ##################################################################### */
- /*}}}*/
- // Include Files /*{{{*/
- #ifdef __GNUG__
- #pragma implementation "dsync/mmap.h"
- #endif
- #define _BSD_SOURCE
- #include <dsync/mmap.h>
- #include <dsync/error.h>
- #include <sys/mman.h>
- #include <sys/stat.h>
- #include <unistd.h>
- #include <fcntl.h>
- #include <string.h>
- /*}}}*/
- // MMap::MMap - Constructor /*{{{*/
- // ---------------------------------------------------------------------
- /* */
- MMap::MMap(FileFd &F,unsigned long Flags) : Flags(Flags), iSize(0),
- Base(0)
- {
- if ((Flags & NoImmMap) != NoImmMap)
- Map(F);
- }
- /*}}}*/
- // MMap::MMap - Constructor /*{{{*/
- // ---------------------------------------------------------------------
- /* */
- MMap::MMap(unsigned long Flags) : Flags(Flags), iSize(0),
- Base(0)
- {
- }
- /*}}}*/
- // MMap::~MMap - Destructor /*{{{*/
- // ---------------------------------------------------------------------
- /* */
- MMap::~MMap()
- {
- Close();
- }
- /*}}}*/
- // MMap::Map - Perform the mapping /*{{{*/
- // ---------------------------------------------------------------------
- /* */
- bool MMap::Map(FileFd &Fd)
- {
- iSize = Fd.Size();
-
- // Set the permissions.
- int Prot = PROT_READ;
- int Map = MAP_SHARED;
- if ((Flags & ReadOnly) != ReadOnly)
- Prot |= PROT_WRITE;
- if ((Flags & Public) != Public)
- Map = MAP_PRIVATE;
-
- if (iSize == 0)
- return _error->Error("Can't mmap an empty file");
-
- // Map it.
- Base = mmap(0,iSize,Prot,Map,Fd.Fd(),0);
- if (Base == (void *)-1)
- return _error->Errno("mmap","Couldn't make mmap of %u bytes",iSize);
- return true;
- }
- /*}}}*/
- // MMap::Close - Close the map /*{{{*/
- // ---------------------------------------------------------------------
- /* */
- bool MMap::Close(bool DoSync)
- {
- if ((Flags & UnMapped) == UnMapped || Base == 0 || iSize == 0)
- return true;
-
- if (DoSync == true)
- Sync();
-
- if (munmap((char *)Base,iSize) != 0)
- _error->Warning("Unable to munmap");
-
- iSize = 0;
- return true;
- }
- /*}}}*/
- // MMap::Sync - Syncronize the map with the disk /*{{{*/
- // ---------------------------------------------------------------------
- /* This is done in syncronous mode - the docs indicate that this will
- not return till all IO is complete */
- bool MMap::Sync()
- {
- if ((Flags & UnMapped) == UnMapped)
- return true;
-
- #ifdef _POSIX_SYNCHRONIZED_IO
- if ((Flags & ReadOnly) != ReadOnly)
- if (msync((char *)Base,iSize,MS_SYNC) != 0)
- return _error->Errno("msync","Unable to write mmap");
- #endif
- return true;
- }
- /*}}}*/
- // MMap::Sync - Syncronize a section of the file to disk /*{{{*/
- // ---------------------------------------------------------------------
- /* */
- bool MMap::Sync(unsigned long Start,unsigned long Stop)
- {
- if ((Flags & UnMapped) == UnMapped)
- return true;
-
- #ifdef _POSIX_SYNCHRONIZED_IO
- unsigned long PSize = sysconf(_SC_PAGESIZE);
- if ((Flags & ReadOnly) != ReadOnly)
- if (msync((char *)Base+(int)(Start/PSize)*PSize,Stop - Start,MS_SYNC) != 0)
- return _error->Errno("msync","Unable to write mmap");
- #endif
- return true;
- }
- /*}}}*/
- // DynamicMMap::DynamicMMap - Constructor /*{{{*/
- // ---------------------------------------------------------------------
- /* */
- DynamicMMap::DynamicMMap(FileFd &F,unsigned long Flags,unsigned long WorkSpace) :
- MMap(F,Flags | NoImmMap), Fd(&F), WorkSpace(WorkSpace)
- {
- if (_error->PendingError() == true)
- return;
-
- unsigned long EndOfFile = Fd->Size();
- Fd->Seek(WorkSpace);
- char C = 0;
- Fd->Write(&C,sizeof(C));
- Map(F);
- iSize = EndOfFile;
- }
- /*}}}*/
- // DynamicMMap::DynamicMMap - Constructor for a non-file backed map /*{{{*/
- // ---------------------------------------------------------------------
- /* This is just a fancy malloc really.. */
- DynamicMMap::DynamicMMap(unsigned long Flags,unsigned long WorkSpace) :
- MMap(Flags | NoImmMap | UnMapped), Fd(0), WorkSpace(WorkSpace)
- {
- if (_error->PendingError() == true)
- return;
-
- Base = new unsigned char[WorkSpace];
- iSize = 0;
- }
- /*}}}*/
- // DynamicMMap::~DynamicMMap - Destructor /*{{{*/
- // ---------------------------------------------------------------------
- /* We truncate the file to the size of the memory data set */
- DynamicMMap::~DynamicMMap()
- {
- if (Fd == 0)
- {
- delete [] (unsigned char *)Base;
- return;
- }
-
- unsigned long EndOfFile = iSize;
- Sync();
- iSize = WorkSpace;
- Close(false);
- ftruncate(Fd->Fd(),EndOfFile);
- Fd->Close();
- }
- /*}}}*/
- // DynamicMMap::RawAllocate - Allocate a raw chunk of unaligned space /*{{{*/
- // ---------------------------------------------------------------------
- /* This allocates a block of memory aligned to the given size */
- unsigned long DynamicMMap::RawAllocate(unsigned long Size,unsigned long Aln)
- {
- unsigned long Result = iSize;
- if (Aln != 0)
- Result += Aln - (iSize%Aln);
-
- iSize = Result + Size;
-
- // Just in case error check
- if (Result + Size > WorkSpace)
- {
- _error->Error("Dynamic MMap ran out of room");
- return 0;
- }
- return Result;
- }
- /*}}}*/
- // DynamicMMap::Allocate - Pooled aligned allocation /*{{{*/
- // ---------------------------------------------------------------------
- /* This allocates an Item of size ItemSize so that it is aligned to its
- size in the file. */
- unsigned long DynamicMMap::Allocate(unsigned long ItemSize)
- {
- // Look for a matching pool entry
- Pool *I;
- Pool *Empty = 0;
- for (I = Pools; I != Pools + PoolCount; I++)
- {
- if (I->ItemSize == 0)
- Empty = I;
- if (I->ItemSize == ItemSize)
- break;
- }
- // No pool is allocated, use an unallocated one
- if (I == Pools + PoolCount)
- {
- // Woops, we ran out, the calling code should allocate more.
- if (Empty == 0)
- {
- _error->Error("Ran out of allocation pools");
- return 0;
- }
-
- I = Empty;
- I->ItemSize = ItemSize;
- I->Count = 0;
- }
-
- // Out of space, allocate some more
- if (I->Count == 0)
- {
- I->Count = 20*1024/ItemSize;
- I->Start = RawAllocate(I->Count*ItemSize,ItemSize);
- }
- I->Count--;
- unsigned long Result = I->Start;
- I->Start += ItemSize;
- return Result/ItemSize;
- }
- /*}}}*/
- // DynamicMMap::WriteString - Write a string to the file /*{{{*/
- // ---------------------------------------------------------------------
- /* Strings are not aligned to anything */
- unsigned long DynamicMMap::WriteString(const char *String,
- unsigned long Len)
- {
- unsigned long Result = iSize;
- // Just in case error check
- if (Result + Len > WorkSpace)
- {
- _error->Error("Dynamic MMap ran out of room");
- return 0;
- }
-
- if (Len == (unsigned long)-1)
- Len = strlen(String);
- iSize += Len + 1;
- memcpy((char *)Base + Result,String,Len);
- ((char *)Base)[Result + Len] = 0;
- return Result;
- }
- /*}}}*/
|