* mmap.cc: Convert usage of dynamically growing cmalloced arrays to
cmalloced linked lists throughout. (class mmap_record): Add LIST_ENTRY element. (mmap_record::match): New method, taking over match algorithm from list::search_record. (class mmap_list): Rename from class list. Add LIST_ENTRY. Convert recs to a LIST_HEAD. Drop nrecs and maxrecs members. (mmap_list::get_record): Drop entirely. (mmap_list::free_recs): Drop entirely. (mmap_list::del_record): Take mmap_record to delete as parameter. (mmap_list::search_record): Convert to mmap_record::match. (class mmap_areas): Rename from class map. Convert lists to LIST_HEAD. (mmap_areas::get_list): Drop entirely. (mmap_areas::del_list): Take mmap_list to delete as parameter. (mprotect): Fix indentation.
This commit is contained in:
parent
4694cc18c2
commit
230a3c86d1
|
@ -1,3 +1,21 @@
|
||||||
|
2007-11-27 Corinna Vinschen <corinna@vinschen.de>
|
||||||
|
|
||||||
|
* mmap.cc: Convert usage of dynamically growing cmalloced arrays to
|
||||||
|
cmalloced linked lists throughout.
|
||||||
|
(class mmap_record): Add LIST_ENTRY element.
|
||||||
|
(mmap_record::match): New method, taking over match algorithm from
|
||||||
|
list::search_record.
|
||||||
|
(class mmap_list): Rename from class list. Add LIST_ENTRY. Convert
|
||||||
|
recs to a LIST_HEAD. Drop nrecs and maxrecs members.
|
||||||
|
(mmap_list::get_record): Drop entirely.
|
||||||
|
(mmap_list::free_recs): Drop entirely.
|
||||||
|
(mmap_list::del_record): Take mmap_record to delete as parameter.
|
||||||
|
(mmap_list::search_record): Convert to mmap_record::match.
|
||||||
|
(class mmap_areas): Rename from class map. Convert lists to LIST_HEAD.
|
||||||
|
(mmap_areas::get_list): Drop entirely.
|
||||||
|
(mmap_areas::del_list): Take mmap_list to delete as parameter.
|
||||||
|
(mprotect): Fix indentation.
|
||||||
|
|
||||||
2007-11-26 Christopher Faylor <me+cygwin@cgf.cx>
|
2007-11-26 Christopher Faylor <me+cygwin@cgf.cx>
|
||||||
|
|
||||||
Change many cygheap allocation routines to their *_abort analogs.
|
Change many cygheap allocation routines to their *_abort analogs.
|
||||||
|
|
|
@ -24,6 +24,7 @@ details. */
|
||||||
#include "pinfo.h"
|
#include "pinfo.h"
|
||||||
#include "sys/cygwin.h"
|
#include "sys/cygwin.h"
|
||||||
#include "ntdll.h"
|
#include "ntdll.h"
|
||||||
|
#include <sys/queue.h>
|
||||||
|
|
||||||
/* __PROT_ATTACH indicates an anonymous mapping which is supposed to be
|
/* __PROT_ATTACH indicates an anonymous mapping which is supposed to be
|
||||||
attached to a file mapping for pages beyond the file's EOF. The idea
|
attached to a file mapping for pages beyond the file's EOF. The idea
|
||||||
|
@ -231,16 +232,20 @@ MapView (HANDLE h, void *addr, size_t len, DWORD openflags,
|
||||||
The class structure:
|
The class structure:
|
||||||
|
|
||||||
One member of class map per process, global variable mmapped_areas.
|
One member of class map per process, global variable mmapped_areas.
|
||||||
Contains a dynamic class list array. Each list entry represents all
|
Contains a singly-linked list of type class mmap_list. Each mmap_list
|
||||||
mapping to a file, keyed by file descriptor and file name hash.
|
entry represents all mapping to a file, keyed by file descriptor and
|
||||||
Each list entry contains a dynamic class mmap_record array. Each
|
file name hash.
|
||||||
mmap_record represents exactly one mapping. For each mapping, there's
|
Each list entry contains a singly-linked list of type class mmap_record.
|
||||||
|
Each mmap_record represents exactly one mapping. For each mapping, there's
|
||||||
an additional so called `page_map'. It's an array of bits, one bit
|
an additional so called `page_map'. It's an array of bits, one bit
|
||||||
per mapped memory page. The bit is set if the page is accessible,
|
per mapped memory page. The bit is set if the page is accessible,
|
||||||
unset otherwise. */
|
unset otherwise. */
|
||||||
|
|
||||||
class mmap_record
|
class mmap_record
|
||||||
{
|
{
|
||||||
|
public:
|
||||||
|
LIST_ENTRY (mmap_record) mr_next;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
int fd;
|
int fd;
|
||||||
HANDLE mapping_hdl;
|
HANDLE mapping_hdl;
|
||||||
|
@ -294,6 +299,7 @@ class mmap_record
|
||||||
void free_page_map () { if (page_map) cfree (page_map); }
|
void free_page_map () { if (page_map) cfree (page_map); }
|
||||||
|
|
||||||
DWORD find_unused_pages (DWORD pages) const;
|
DWORD find_unused_pages (DWORD pages) const;
|
||||||
|
bool match (caddr_t addr, DWORD len, caddr_t &m_addr, DWORD &m_len);
|
||||||
_off64_t map_pages (_off64_t off, DWORD len);
|
_off64_t map_pages (_off64_t off, DWORD len);
|
||||||
bool map_pages (caddr_t addr, DWORD len);
|
bool map_pages (caddr_t addr, DWORD len);
|
||||||
bool unmap_pages (caddr_t addr, DWORD len);
|
bool unmap_pages (caddr_t addr, DWORD len);
|
||||||
|
@ -309,45 +315,40 @@ class mmap_record
|
||||||
bool compatible_flags (int fl) const;
|
bool compatible_flags (int fl) const;
|
||||||
};
|
};
|
||||||
|
|
||||||
class list
|
class mmap_list
|
||||||
{
|
{
|
||||||
|
public:
|
||||||
|
LIST_ENTRY (mmap_list) ml_next;
|
||||||
|
LIST_HEAD (, mmap_record) recs;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
mmap_record *recs;
|
|
||||||
int nrecs, maxrecs;
|
|
||||||
int fd;
|
int fd;
|
||||||
__ino64_t hash;
|
__ino64_t hash;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
int get_fd () const { return fd; }
|
int get_fd () const { return fd; }
|
||||||
__ino64_t get_hash () const { return hash; }
|
__ino64_t get_hash () const { return hash; }
|
||||||
mmap_record *get_record (int i) { return i >= nrecs ? NULL : recs + i; }
|
|
||||||
|
|
||||||
bool anonymous () const { return fd == -1; }
|
bool anonymous () const { return fd == -1; }
|
||||||
void set (int nfd, struct __stat64 *st);
|
void set (int nfd, struct __stat64 *st);
|
||||||
mmap_record *add_record (mmap_record r);
|
mmap_record *add_record (mmap_record r);
|
||||||
bool del_record (int i);
|
bool del_record (mmap_record *rec);
|
||||||
void free_recs () { if (recs) cfree (recs); }
|
|
||||||
mmap_record *search_record (_off64_t off, DWORD len);
|
mmap_record *search_record (_off64_t off, DWORD len);
|
||||||
long search_record (caddr_t addr, DWORD len, caddr_t &m_addr, DWORD &m_len,
|
|
||||||
long start);
|
|
||||||
caddr_t try_map (void *addr, size_t len, int flags, _off64_t off);
|
caddr_t try_map (void *addr, size_t len, int flags, _off64_t off);
|
||||||
};
|
};
|
||||||
|
|
||||||
class map
|
class mmap_areas
|
||||||
{
|
{
|
||||||
private:
|
|
||||||
list *lists;
|
|
||||||
unsigned nlists, maxlists;
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
list *get_list (unsigned i) { return i >= nlists ? NULL : lists + i; }
|
LIST_HEAD (, mmap_list) lists;
|
||||||
list *get_list_by_fd (int fd, struct __stat64 *st);
|
|
||||||
list *add_list (int fd, struct __stat64 *st);
|
mmap_list *get_list_by_fd (int fd, struct __stat64 *st);
|
||||||
void del_list (unsigned i);
|
mmap_list *add_list (int fd, struct __stat64 *st);
|
||||||
|
void del_list (mmap_list *ml);
|
||||||
};
|
};
|
||||||
|
|
||||||
/* This is the global map structure pointer. */
|
/* This is the global map structure pointer. */
|
||||||
static map mmapped_areas;
|
static mmap_areas mmapped_areas;
|
||||||
|
|
||||||
bool
|
bool
|
||||||
mmap_record::compatible_flags (int fl) const
|
mmap_record::compatible_flags (int fl) const
|
||||||
|
@ -377,6 +378,25 @@ mmap_record::find_unused_pages (DWORD pages) const
|
||||||
return (DWORD)-1;
|
return (DWORD)-1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
mmap_record::match (caddr_t addr, DWORD len, caddr_t &m_addr, DWORD &m_len)
|
||||||
|
{
|
||||||
|
caddr_t low = (addr >= get_address ()) ? addr : get_address ();
|
||||||
|
caddr_t high = get_address ();
|
||||||
|
if (filler ())
|
||||||
|
high += get_len ();
|
||||||
|
else
|
||||||
|
high += (PAGE_CNT (get_len ()) * getsystempagesize ());
|
||||||
|
high = (addr + len < high) ? addr + len : high;
|
||||||
|
if (low < high)
|
||||||
|
{
|
||||||
|
m_addr = low;
|
||||||
|
m_len = high - low;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
bool
|
bool
|
||||||
mmap_record::alloc_page_map ()
|
mmap_record::alloc_page_map ()
|
||||||
{
|
{
|
||||||
|
@ -517,79 +537,48 @@ mmap_record::free_fh (fhandler_base *fh)
|
||||||
}
|
}
|
||||||
|
|
||||||
mmap_record *
|
mmap_record *
|
||||||
list::add_record (mmap_record r)
|
mmap_list::add_record (mmap_record r)
|
||||||
{
|
{
|
||||||
if (nrecs == maxrecs)
|
mmap_record *rec = (mmap_record *) cmalloc (HEAP_MMAP, sizeof (mmap_record));
|
||||||
{
|
if (!rec)
|
||||||
mmap_record *new_recs;
|
|
||||||
if (maxrecs == 0)
|
|
||||||
new_recs = (mmap_record *)
|
|
||||||
cmalloc (HEAP_MMAP, 5 * sizeof (mmap_record));
|
|
||||||
else
|
|
||||||
new_recs = (mmap_record *)
|
|
||||||
crealloc (recs, (maxrecs + 5) * sizeof (mmap_record));
|
|
||||||
if (!new_recs)
|
|
||||||
return NULL;
|
|
||||||
maxrecs += 5;
|
|
||||||
recs = new_recs;
|
|
||||||
}
|
|
||||||
recs[nrecs] = r;
|
|
||||||
if (!recs[nrecs].alloc_page_map ())
|
|
||||||
return NULL;
|
return NULL;
|
||||||
return recs + nrecs++;
|
*rec = r;
|
||||||
|
if (!rec->alloc_page_map ())
|
||||||
|
{
|
||||||
|
cfree (rec);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
LIST_INSERT_HEAD (&recs, rec, mr_next);
|
||||||
|
return rec;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Used in mmap() */
|
/* Used in mmap() */
|
||||||
mmap_record *
|
mmap_record *
|
||||||
list::search_record (_off64_t off, DWORD len)
|
mmap_list::search_record (_off64_t off, DWORD len)
|
||||||
{
|
{
|
||||||
|
mmap_record *rec;
|
||||||
|
|
||||||
if (anonymous () && !off)
|
if (anonymous () && !off)
|
||||||
{
|
{
|
||||||
len = PAGE_CNT (len);
|
len = PAGE_CNT (len);
|
||||||
for (int i = 0; i < nrecs; ++i)
|
LIST_FOREACH (rec, &recs, mr_next)
|
||||||
if (recs[i].find_unused_pages (len) != (DWORD)-1)
|
if (rec->find_unused_pages (len) != (DWORD)-1)
|
||||||
return recs + i;
|
return rec;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
for (int i = 0; i < nrecs; ++i)
|
LIST_FOREACH (rec, &recs, mr_next)
|
||||||
if (off >= recs[i].get_offset ()
|
if (off >= rec->get_offset ()
|
||||||
&& off + len
|
&& off + len
|
||||||
<= recs[i].get_offset ()
|
<= rec->get_offset ()
|
||||||
+ (PAGE_CNT (recs[i].get_len ()) * getsystempagesize ()))
|
+ (PAGE_CNT (rec->get_len ()) * getsystempagesize ()))
|
||||||
return recs + i;
|
return rec;
|
||||||
}
|
}
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Used in munmap() */
|
|
||||||
long
|
|
||||||
list::search_record (caddr_t addr, DWORD len, caddr_t &m_addr, DWORD &m_len,
|
|
||||||
long start)
|
|
||||||
{
|
|
||||||
caddr_t low, high;
|
|
||||||
|
|
||||||
for (long i = start + 1; i < nrecs; ++i)
|
|
||||||
{
|
|
||||||
low = (addr >= recs[i].get_address ()) ? addr : recs[i].get_address ();
|
|
||||||
high = recs[i].get_address ();
|
|
||||||
if (recs[i].filler ())
|
|
||||||
high += recs[i].get_len ();
|
|
||||||
else
|
|
||||||
high += (PAGE_CNT (recs[i].get_len ()) * getsystempagesize ());
|
|
||||||
high = (addr + len < high) ? addr + len : high;
|
|
||||||
if (low < high)
|
|
||||||
{
|
|
||||||
m_addr = low;
|
|
||||||
m_len = high - low;
|
|
||||||
return i;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
void
|
||||||
list::set (int nfd, struct __stat64 *st)
|
mmap_list::set (int nfd, struct __stat64 *st)
|
||||||
{
|
{
|
||||||
fd = nfd;
|
fd = nfd;
|
||||||
if (!anonymous ())
|
if (!anonymous ())
|
||||||
|
@ -599,27 +588,22 @@ list::set (int nfd, struct __stat64 *st)
|
||||||
the file. */
|
the file. */
|
||||||
hash = st ? st->st_ino : (__ino64_t) 0;
|
hash = st ? st->st_ino : (__ino64_t) 0;
|
||||||
}
|
}
|
||||||
nrecs = maxrecs = 0;
|
LIST_INIT (&recs);
|
||||||
recs = NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool
|
bool
|
||||||
list::del_record (int i)
|
mmap_list::del_record (mmap_record *rec)
|
||||||
{
|
{
|
||||||
if (i < nrecs)
|
rec->free_page_map ();
|
||||||
{
|
LIST_REMOVE (rec, mr_next);
|
||||||
recs[i].free_page_map ();
|
cfree (rec);
|
||||||
for (; i < nrecs - 1; i++)
|
|
||||||
recs[i] = recs[i + 1];
|
|
||||||
nrecs--;
|
|
||||||
}
|
|
||||||
/* Return true if the list is empty which allows the caller to remove
|
/* Return true if the list is empty which allows the caller to remove
|
||||||
this list from the list array. */
|
this list from the list of lists. */
|
||||||
return !nrecs;
|
return !LIST_FIRST(&recs);
|
||||||
}
|
}
|
||||||
|
|
||||||
caddr_t
|
caddr_t
|
||||||
list::try_map (void *addr, size_t len, int flags, _off64_t off)
|
mmap_list::try_map (void *addr, size_t len, int flags, _off64_t off)
|
||||||
{
|
{
|
||||||
mmap_record *rec;
|
mmap_record *rec;
|
||||||
|
|
||||||
|
@ -642,11 +626,12 @@ list::try_map (void *addr, size_t len, int flags, _off64_t off)
|
||||||
if a memory region is unmapped and remapped with MAP_FIXED. */
|
if a memory region is unmapped and remapped with MAP_FIXED. */
|
||||||
caddr_t u_addr;
|
caddr_t u_addr;
|
||||||
DWORD u_len;
|
DWORD u_len;
|
||||||
long record_idx = -1;
|
|
||||||
if ((record_idx = search_record ((caddr_t) addr, len, u_addr, u_len,
|
LIST_FOREACH (rec, &recs, mr_next)
|
||||||
record_idx)) >= 0)
|
if (rec->match ((caddr_t) addr, len, u_addr, u_len))
|
||||||
|
break;
|
||||||
|
if (rec)
|
||||||
{
|
{
|
||||||
rec = get_record (record_idx);
|
|
||||||
if (u_addr > (caddr_t) addr || u_addr + len < (caddr_t) addr + len
|
if (u_addr > (caddr_t) addr || u_addr + len < (caddr_t) addr + len
|
||||||
|| !rec->compatible_flags (flags))
|
|| !rec->compatible_flags (flags))
|
||||||
{
|
{
|
||||||
|
@ -664,52 +649,39 @@ list::try_map (void *addr, size_t len, int flags, _off64_t off)
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
list *
|
mmap_list *
|
||||||
map::get_list_by_fd (int fd, struct __stat64 *st)
|
mmap_areas::get_list_by_fd (int fd, struct __stat64 *st)
|
||||||
{
|
{
|
||||||
unsigned i;
|
mmap_list *ml;
|
||||||
for (i = 0; i < nlists; i++)
|
LIST_FOREACH (ml, &lists, ml_next)
|
||||||
{
|
{
|
||||||
if (fd == -1 && lists[i].anonymous ())
|
if (fd == -1 && ml->anonymous ())
|
||||||
return lists + i;
|
return ml;
|
||||||
/* The fd isn't sufficient since it could already be the fd of another
|
/* The fd isn't sufficient since it could already be the fd of another
|
||||||
file. So we use the inode number as evaluated by fstat to identify
|
file. So we use the inode number as evaluated by fstat to identify
|
||||||
the file. */
|
the file. */
|
||||||
if (fd != -1 && st && lists[i].get_hash () == st->st_ino)
|
if (fd != -1 && st && ml->get_hash () == st->st_ino)
|
||||||
return lists + i;
|
return ml;
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
list *
|
mmap_list *
|
||||||
map::add_list (int fd, struct __stat64 *st)
|
mmap_areas::add_list (int fd, struct __stat64 *st)
|
||||||
{
|
{
|
||||||
if (nlists == maxlists)
|
mmap_list *ml = (mmap_list *) cmalloc (HEAP_MMAP, sizeof (mmap_list));
|
||||||
{
|
if (!ml)
|
||||||
list *new_lists;
|
return NULL;
|
||||||
if (maxlists == 0)
|
ml->set (fd, st);
|
||||||
new_lists = (list *) cmalloc (HEAP_MMAP, 5 * sizeof (list));
|
LIST_INSERT_HEAD (&lists, ml, ml_next);
|
||||||
else
|
return ml;
|
||||||
new_lists = (list *) crealloc (lists, (maxlists + 5) * sizeof (list));
|
|
||||||
if (!new_lists)
|
|
||||||
return NULL;
|
|
||||||
maxlists += 5;
|
|
||||||
lists = new_lists;
|
|
||||||
}
|
|
||||||
lists[nlists].set (fd, st);
|
|
||||||
return lists + nlists++;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
map::del_list (unsigned i)
|
mmap_areas::del_list (mmap_list *ml)
|
||||||
{
|
{
|
||||||
if (i < nlists)
|
LIST_REMOVE (ml, ml_next);
|
||||||
{
|
cfree (ml);
|
||||||
lists[i].free_recs ();
|
|
||||||
for (; i < nlists - 1; i++)
|
|
||||||
lists[i] = lists[i + 1];
|
|
||||||
nlists--;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This function is called from exception_handler when a segmentation
|
/* This function is called from exception_handler when a segmentation
|
||||||
|
@ -733,7 +705,7 @@ map::del_list (unsigned i)
|
||||||
mmap_region_status
|
mmap_region_status
|
||||||
mmap_is_attached_or_noreserve (void *addr, size_t len)
|
mmap_is_attached_or_noreserve (void *addr, size_t len)
|
||||||
{
|
{
|
||||||
list *map_list = mmapped_areas.get_list_by_fd (-1, NULL);
|
mmap_list *map_list = mmapped_areas.get_list_by_fd (-1, NULL);
|
||||||
|
|
||||||
size_t pagesize = getpagesize ();
|
size_t pagesize = getpagesize ();
|
||||||
caddr_t start_addr = (caddr_t) rounddown ((uintptr_t) addr, pagesize);
|
caddr_t start_addr = (caddr_t) rounddown ((uintptr_t) addr, pagesize);
|
||||||
|
@ -743,16 +715,14 @@ mmap_is_attached_or_noreserve (void *addr, size_t len)
|
||||||
if (map_list == NULL)
|
if (map_list == NULL)
|
||||||
return MMAP_NONE;
|
return MMAP_NONE;
|
||||||
|
|
||||||
while (len > 0)
|
mmap_record *rec;
|
||||||
{
|
caddr_t u_addr;
|
||||||
caddr_t u_addr;
|
DWORD u_len;
|
||||||
DWORD u_len;
|
|
||||||
long record_idx = map_list->search_record (start_addr, len,
|
|
||||||
u_addr, u_len, -1);
|
|
||||||
if (record_idx < 0)
|
|
||||||
return MMAP_NONE;
|
|
||||||
|
|
||||||
mmap_record *rec = map_list->get_record (record_idx);
|
LIST_FOREACH (rec, &map_list->recs, mr_next)
|
||||||
|
{
|
||||||
|
if (!rec->match (start_addr, len, u_addr, u_len))
|
||||||
|
continue;
|
||||||
if (rec->attached ())
|
if (rec->attached ())
|
||||||
return MMAP_RAISE_SIGBUS;
|
return MMAP_RAISE_SIGBUS;
|
||||||
if (!rec->noreserve ())
|
if (!rec->noreserve ())
|
||||||
|
@ -768,13 +738,14 @@ mmap_is_attached_or_noreserve (void *addr, size_t len)
|
||||||
|
|
||||||
start_addr += commit_len;
|
start_addr += commit_len;
|
||||||
len -= commit_len;
|
len -= commit_len;
|
||||||
|
if (!len)
|
||||||
|
return MMAP_NORESERVE_COMMITED;
|
||||||
}
|
}
|
||||||
|
return MMAP_NONE;
|
||||||
return MMAP_NORESERVE_COMMITED;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static caddr_t
|
static caddr_t
|
||||||
mmap_worker (list *map_list, fhandler_base *fh, caddr_t base, size_t len,
|
mmap_worker (mmap_list *map_list, fhandler_base *fh, caddr_t base, size_t len,
|
||||||
int prot, int flags, int fd, _off64_t off, struct __stat64 *st)
|
int prot, int flags, int fd, _off64_t off, struct __stat64 *st)
|
||||||
{
|
{
|
||||||
HANDLE h = fh->mmap (&base, len, prot, flags, off);
|
HANDLE h = fh->mmap (&base, len, prot, flags, off);
|
||||||
|
@ -807,7 +778,7 @@ mmap64 (void *addr, size_t len, int prot, int flags, int fd, _off64_t off)
|
||||||
fhandler_base *fh = NULL;
|
fhandler_base *fh = NULL;
|
||||||
fhandler_disk_file *fh_disk_file = NULL; /* Used for reopening a disk file
|
fhandler_disk_file *fh_disk_file = NULL; /* Used for reopening a disk file
|
||||||
when necessary. */
|
when necessary. */
|
||||||
list *map_list = NULL;
|
mmap_list *map_list = NULL;
|
||||||
size_t orig_len = 0;
|
size_t orig_len = 0;
|
||||||
caddr_t base = NULL;
|
caddr_t base = NULL;
|
||||||
struct __stat64 st;
|
struct __stat64 st;
|
||||||
|
@ -1113,19 +1084,17 @@ munmap (void *addr, size_t len)
|
||||||
|
|
||||||
/* Iterate through the map, unmap pages between addr and addr+len
|
/* Iterate through the map, unmap pages between addr and addr+len
|
||||||
in all maps. */
|
in all maps. */
|
||||||
list *map_list;
|
mmap_list *map_list, *next_map_list;
|
||||||
for (unsigned list_idx = 0;
|
LIST_FOREACH_SAFE (map_list, &mmapped_areas.lists, ml_next, next_map_list)
|
||||||
(map_list = mmapped_areas.get_list (list_idx));
|
|
||||||
++list_idx)
|
|
||||||
{
|
{
|
||||||
long record_idx = -1;
|
mmap_record *rec, *next_rec;
|
||||||
caddr_t u_addr;
|
caddr_t u_addr;
|
||||||
DWORD u_len;
|
DWORD u_len;
|
||||||
|
|
||||||
while ((record_idx = map_list->search_record((caddr_t)addr, len, u_addr,
|
LIST_FOREACH_SAFE (rec, &map_list->recs, mr_next, next_rec)
|
||||||
u_len, record_idx)) >= 0)
|
{
|
||||||
{
|
if (!rec->match ((caddr_t) addr, len, u_addr, u_len))
|
||||||
mmap_record *rec = map_list->get_record (record_idx);
|
continue;
|
||||||
if (rec->unmap_pages (u_addr, u_len))
|
if (rec->unmap_pages (u_addr, u_len))
|
||||||
{
|
{
|
||||||
/* The whole record has been unmapped, so we now actually
|
/* The whole record has been unmapped, so we now actually
|
||||||
|
@ -1137,11 +1106,11 @@ munmap (void *addr, size_t len)
|
||||||
rec->free_fh (fh);
|
rec->free_fh (fh);
|
||||||
|
|
||||||
/* ...and delete the record. */
|
/* ...and delete the record. */
|
||||||
if (map_list->del_record (record_idx--))
|
if (map_list->del_record (rec))
|
||||||
{
|
{
|
||||||
/* Yay, the last record has been removed from the list,
|
/* Yay, the last record has been removed from the list,
|
||||||
we can remove the list now, too. */
|
we can remove the list now, too. */
|
||||||
mmapped_areas.del_list (list_idx--);
|
mmapped_areas.del_list (map_list);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1159,7 +1128,7 @@ extern "C" int
|
||||||
msync (void *addr, size_t len, int flags)
|
msync (void *addr, size_t len, int flags)
|
||||||
{
|
{
|
||||||
int ret = -1;
|
int ret = -1;
|
||||||
list *map_list;
|
mmap_list *map_list;
|
||||||
|
|
||||||
syscall_printf ("msync (addr: %p, len %u, flags %x)", addr, len, flags);
|
syscall_printf ("msync (addr: %p, len %u, flags %x)", addr, len, flags);
|
||||||
|
|
||||||
|
@ -1178,26 +1147,22 @@ msync (void *addr, size_t len, int flags)
|
||||||
|
|
||||||
/* Iterate through the map, looking for the mmapped area.
|
/* Iterate through the map, looking for the mmapped area.
|
||||||
Error if not found. */
|
Error if not found. */
|
||||||
for (unsigned list_idx = 0;
|
LIST_FOREACH (map_list, &mmapped_areas.lists, ml_next)
|
||||||
(map_list = mmapped_areas.get_list (list_idx));
|
|
||||||
++list_idx)
|
|
||||||
{
|
{
|
||||||
mmap_record *rec;
|
mmap_record *rec;
|
||||||
for (int record_idx = 0;
|
LIST_FOREACH (rec, &map_list->recs, mr_next)
|
||||||
(rec = map_list->get_record (record_idx));
|
|
||||||
++record_idx)
|
|
||||||
{
|
{
|
||||||
if (rec->access ((caddr_t)addr))
|
if (rec->access ((caddr_t) addr))
|
||||||
{
|
{
|
||||||
/* Check whole area given by len. */
|
/* Check whole area given by len. */
|
||||||
for (DWORD i = getpagesize (); i < len; i += getpagesize ())
|
for (DWORD i = getpagesize (); i < len; i += getpagesize ())
|
||||||
if (!rec->access ((caddr_t)addr + i))
|
if (!rec->access ((caddr_t) addr + i))
|
||||||
{
|
{
|
||||||
set_errno (ENOMEM);
|
set_errno (ENOMEM);
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
fhandler_base *fh = rec->alloc_fh ();
|
fhandler_base *fh = rec->alloc_fh ();
|
||||||
ret = fh->msync (rec->get_handle (), (caddr_t)addr, len, flags);
|
ret = fh->msync (rec->get_handle (), (caddr_t) addr, len, flags);
|
||||||
rec->free_fh (fh);
|
rec->free_fh (fh);
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
@ -1238,40 +1203,37 @@ mprotect (void *addr, size_t len, int prot)
|
||||||
|
|
||||||
/* Iterate through the map, protect pages between addr and addr+len
|
/* Iterate through the map, protect pages between addr and addr+len
|
||||||
in all maps. */
|
in all maps. */
|
||||||
list *map_list;
|
mmap_list *map_list;
|
||||||
for (unsigned list_idx = 0;
|
LIST_FOREACH (map_list, &mmapped_areas.lists, ml_next)
|
||||||
(map_list = mmapped_areas.get_list (list_idx));
|
{
|
||||||
++list_idx)
|
mmap_record *rec;
|
||||||
{
|
caddr_t u_addr;
|
||||||
long record_idx = -1;
|
DWORD u_len;
|
||||||
caddr_t u_addr;
|
|
||||||
DWORD u_len;
|
|
||||||
|
|
||||||
while ((record_idx = map_list->search_record((caddr_t)addr, len,
|
LIST_FOREACH (rec, &map_list->recs, mr_next)
|
||||||
u_addr, u_len,
|
{
|
||||||
record_idx)) >= 0)
|
if (!rec->match ((caddr_t) addr, len, u_addr, u_len))
|
||||||
{
|
continue;
|
||||||
mmap_record *rec = map_list->get_record (record_idx);
|
in_mapped = true;
|
||||||
in_mapped = true;
|
if (rec->attached ())
|
||||||
if (rec->attached ())
|
continue;
|
||||||
continue;
|
new_prot = gen_protect (prot, rec->get_flags ());
|
||||||
new_prot = gen_protect (prot, rec->get_flags ());
|
if (rec->noreserve ())
|
||||||
if (rec->noreserve ())
|
{
|
||||||
{
|
if (new_prot == PAGE_NOACCESS)
|
||||||
if (new_prot == PAGE_NOACCESS)
|
ret = VirtualFree (u_addr, u_len, MEM_DECOMMIT);
|
||||||
ret = VirtualFree (u_addr, u_len, MEM_DECOMMIT);
|
else
|
||||||
else
|
ret = !!VirtualAlloc (u_addr, u_len, MEM_COMMIT, new_prot);
|
||||||
ret = !!VirtualAlloc (u_addr, u_len, MEM_COMMIT, new_prot);
|
}
|
||||||
}
|
else
|
||||||
else
|
ret = VirtualProtect (u_addr, u_len, new_prot, &old_prot);
|
||||||
ret = VirtualProtect (u_addr, u_len, new_prot, &old_prot);
|
if (!ret)
|
||||||
if (!ret)
|
{
|
||||||
{
|
__seterrno ();
|
||||||
__seterrno ();
|
break;
|
||||||
break;
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
ReleaseResourceLock (LOCK_MMAP_LIST, WRITE_LOCK | READ_LOCK, "mprotect");
|
ReleaseResourceLock (LOCK_MMAP_LIST, WRITE_LOCK | READ_LOCK, "mprotect");
|
||||||
|
|
||||||
|
@ -1757,15 +1719,11 @@ int __stdcall
|
||||||
fixup_mmaps_after_fork (HANDLE parent)
|
fixup_mmaps_after_fork (HANDLE parent)
|
||||||
{
|
{
|
||||||
/* Iterate through the map */
|
/* Iterate through the map */
|
||||||
list *map_list;
|
mmap_list *map_list;
|
||||||
for (unsigned list_idx = 0;
|
LIST_FOREACH (map_list, &mmapped_areas.lists, ml_next)
|
||||||
(map_list = mmapped_areas.get_list (list_idx));
|
|
||||||
++list_idx)
|
|
||||||
{
|
{
|
||||||
mmap_record *rec;
|
mmap_record *rec;
|
||||||
for (int record_idx = 0;
|
LIST_FOREACH (rec, &map_list->recs, mr_next)
|
||||||
(rec = map_list->get_record (record_idx));
|
|
||||||
++record_idx)
|
|
||||||
{
|
{
|
||||||
debug_printf ("fd %d, h 0x%x, address %p, len 0x%x, prot: 0x%x, "
|
debug_printf ("fd %d, h 0x%x, address %p, len 0x%x, prot: 0x%x, "
|
||||||
"flags: 0x%x, offset %X",
|
"flags: 0x%x, offset %X",
|
||||||
|
|
Loading…
Reference in New Issue