]> git.saurik.com Git - apt.git/blame - apt-pkg/contrib/mmap.cc
Fixed size mismatch
[apt.git] / apt-pkg / contrib / mmap.cc
CommitLineData
578bfd0a
AL
1// -*- mode: cpp; mode: fold -*-
2// Description /*{{{*/
35c22def 3// $Id: mmap.cc,v 1.14 1999/03/18 04:32:46 jgg Exp $
578bfd0a
AL
4/* ######################################################################
5
6 MMap Class - Provides 'real' mmap or a faked mmap using read().
7
8 MMap cover class.
9
10 Some broken versions of glibc2 (libc6) have a broken definition
11 of mmap that accepts a char * -- all other systems (and libc5) use
12 void *. We can't safely do anything here that would be portable, so
13 libc6 generates warnings -- which should be errors, g++ isn't properly
14 strict.
15
16 The configure test notes that some OS's have broken private mmap's
17 so on those OS's we can't use mmap. This means we have to use
18 configure to test mmap and can't rely on the POSIX
19 _POSIX_MAPPED_FILES test.
20
21 ##################################################################### */
22 /*}}}*/
23// Include Files /*{{{*/
6c139d6e 24#ifdef __GNUG__
094a497d 25#pragma implementation "apt-pkg/mmap.h"
6c139d6e
AL
26#endif
27
578bfd0a 28#define _BSD_SOURCE
094a497d
AL
29#include <apt-pkg/mmap.h>
30#include <apt-pkg/error.h>
578bfd0a
AL
31
32#include <sys/mman.h>
33#include <sys/stat.h>
578bfd0a
AL
34#include <unistd.h>
35#include <fcntl.h>
36 /*}}}*/
37
38// MMap::MMap - Constructor /*{{{*/
39// ---------------------------------------------------------------------
40/* */
8e06abb2 41MMap::MMap(FileFd &F,unsigned long Flags) : Fd(F), Flags(Flags), iSize(0),
578bfd0a
AL
42 Base(0)
43{
44 if ((Flags & NoImmMap) != NoImmMap)
45 Map();
46}
47 /*}}}*/
48// MMap::~MMap - Destructor /*{{{*/
49// ---------------------------------------------------------------------
50/* */
51MMap::~MMap()
52{
53 Close(true);
54}
55 /*}}}*/
56// MMap::Map - Perform the mapping /*{{{*/
57// ---------------------------------------------------------------------
58/* */
59bool MMap::Map()
60{
61 iSize = Fd.Size();
62
63 // Set the permissions.
64 int Prot = PROT_READ;
65 int Map = MAP_SHARED;
66 if ((Flags & ReadOnly) != ReadOnly)
67 Prot |= PROT_WRITE;
68 if ((Flags & Public) != Public)
69 Map = MAP_PRIVATE;
70
b35d2f5f
AL
71 if (iSize == 0)
72 return _error->Error("Can't mmap an empty file");
73
578bfd0a
AL
74 // Map it.
75 Base = mmap(0,iSize,Prot,Map,Fd.Fd(),0);
76 if (Base == (void *)-1)
77 return _error->Errno("mmap","Couldn't make mmap of %u bytes",iSize);
78
79 return true;
80}
81 /*}}}*/
82// MMap::Close - Close the map /*{{{*/
83// ---------------------------------------------------------------------
84/* */
1164783d 85bool MMap::Close(bool DoClose, bool DoSync)
578bfd0a
AL
86{
87 if (Fd.IsOpen() == false)
88 return true;
89
1164783d
AL
90 if (DoSync == true)
91 Sync();
578bfd0a
AL
92
93 if (munmap((char *)Base,iSize) != 0)
94 _error->Warning("Unable to munmap");
95
96 iSize = 0;
97 if (DoClose == true)
98 Fd.Close();
99 return true;
100}
101 /*}}}*/
102// MMap::Sync - Syncronize the map with the disk /*{{{*/
103// ---------------------------------------------------------------------
0149949b
AL
104/* This is done in syncronous mode - the docs indicate that this will
105 not return till all IO is complete */
578bfd0a
AL
106bool MMap::Sync()
107{
de3c15ea 108#ifdef _POSIX_SYNCHRONIZED_IO
1164783d 109 if ((Flags & ReadOnly) != ReadOnly)
578bfd0a
AL
110 if (msync((char *)Base,iSize,MS_SYNC) != 0)
111 return _error->Error("msync","Unable to write mmap");
de3c15ea 112#endif
578bfd0a
AL
113 return true;
114}
115 /*}}}*/
116// MMap::Sync - Syncronize a section of the file to disk /*{{{*/
117// ---------------------------------------------------------------------
118/* */
119bool MMap::Sync(unsigned long Start,unsigned long Stop)
120{
35c22def
AL
121#ifdef _POSIX_SYNCHRONIZED_IO
122 unsigned long PSize = sysconf(_SC_PAGESIZE);
1164783d 123 if ((Flags & ReadOnly) != ReadOnly)
35c22def 124 if (msync((char *)Base+(int)(Start/PSize)*PSize,Stop - Start,MS_SYNC) != 0)
578bfd0a 125 return _error->Error("msync","Unable to write mmap");
de3c15ea 126#endif
578bfd0a
AL
127 return true;
128}
129 /*}}}*/
130
131// DynamicMMap::DynamicMMap - Constructor /*{{{*/
132// ---------------------------------------------------------------------
133/* */
8e06abb2 134DynamicMMap::DynamicMMap(FileFd &F,unsigned long Flags,unsigned long WorkSpace) :
578bfd0a
AL
135 MMap(F,Flags | NoImmMap), WorkSpace(WorkSpace)
136{
d38b7b3d
AL
137 if (_error->PendingError() == true)
138 return;
139
578bfd0a
AL
140 unsigned long EndOfFile = Fd.Size();
141 Fd.Seek(WorkSpace);
142 char C = 0;
143 Fd.Write(&C,sizeof(C));
144 Map();
145 iSize = EndOfFile;
146}
147 /*}}}*/
148// DynamicMMap::~DynamicMMap - Destructor /*{{{*/
149// ---------------------------------------------------------------------
150/* We truncate the file to the size of the memory data set */
151DynamicMMap::~DynamicMMap()
152{
153 unsigned long EndOfFile = iSize;
1164783d
AL
154 Sync();
155 iSize = WorkSpace;
156 Close(false,false);
578bfd0a
AL
157 ftruncate(Fd.Fd(),EndOfFile);
158 Fd.Close();
159}
160 /*}}}*/
161// DynamicMMap::RawAllocate - Allocate a raw chunk of unaligned space /*{{{*/
162// ---------------------------------------------------------------------
f55a958f
AL
163/* This allocates a block of memory aligned to the given size */
164unsigned long DynamicMMap::RawAllocate(unsigned long Size,unsigned long Aln)
578bfd0a
AL
165{
166 unsigned long Result = iSize;
f55a958f
AL
167 if (Aln != 0)
168 Result += Aln - (iSize%Aln);
169
170 iSize = Result + Size;
578bfd0a
AL
171
172 // Just in case error check
e5eebd12 173 if (Result + Size > WorkSpace)
578bfd0a
AL
174 {
175 _error->Error("Dynamic MMap ran out of room");
176 return 0;
177 }
578bfd0a
AL
178 return Result;
179}
180 /*}}}*/
181// DynamicMMap::Allocate - Pooled aligned allocation /*{{{*/
182// ---------------------------------------------------------------------
183/* This allocates an Item of size ItemSize so that it is aligned to its
184 size in the file. */
185unsigned long DynamicMMap::Allocate(unsigned long ItemSize)
186{
187 // Look for a matching pool entry
188 Pool *I;
189 Pool *Empty = 0;
190 for (I = Pools; I != Pools + PoolCount; I++)
191 {
192 if (I->ItemSize == 0)
193 Empty = I;
194 if (I->ItemSize == ItemSize)
195 break;
196 }
197
198 // No pool is allocated, use an unallocated one
199 if (I == Pools + PoolCount)
200 {
201 // Woops, we ran out, the calling code should allocate more.
202 if (Empty == 0)
203 {
204 _error->Error("Ran out of allocation pools");
205 return 0;
206 }
207
208 I = Empty;
209 I->ItemSize = ItemSize;
210 I->Count = 0;
211 }
212
213 // Out of space, allocate some more
214 if (I->Count == 0)
215 {
216 I->Count = 20*1024/ItemSize;
f55a958f 217 I->Start = RawAllocate(I->Count*ItemSize,ItemSize);
578bfd0a 218 }
f55a958f 219
578bfd0a
AL
220 I->Count--;
221 unsigned long Result = I->Start;
222 I->Start += ItemSize;
223 return Result/ItemSize;
224}
225 /*}}}*/
226// DynamicMMap::WriteString - Write a string to the file /*{{{*/
227// ---------------------------------------------------------------------
228/* Strings are not aligned to anything */
229unsigned long DynamicMMap::WriteString(const char *String,
230 unsigned long Len)
231{
232 unsigned long Result = iSize;
233 // Just in case error check
234 if (Result > WorkSpace)
235 {
236 _error->Error("Dynamic MMap ran out of room");
237 return 0;
238 }
239
240 if (Len == 0)
241 Len = strlen(String);
242 iSize += Len + 1;
243 memcpy((char *)Base + Result,String,Len);
244 ((char *)Base)[Result + Len] = 0;
245 return Result;
246}
247 /*}}}*/