992 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			992 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			C++
		
	
	
	
/* mmap.cc
 | 
						|
 | 
						|
   Copyright 1996, 1997, 1998, 2000, 2001, 2002 Red Hat, Inc.
 | 
						|
 | 
						|
This file is part of Cygwin.
 | 
						|
 | 
						|
This software is a copyrighted work licensed under the terms of the
 | 
						|
Cygwin license.  Please consult the file "CYGWIN_LICENSE" for
 | 
						|
details. */
 | 
						|
 | 
						|
#include "winsup.h"
 | 
						|
#include <unistd.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <stddef.h>
 | 
						|
#include <sys/mman.h>
 | 
						|
#include <errno.h>
 | 
						|
#include "security.h"
 | 
						|
#include "fhandler.h"
 | 
						|
#include "path.h"
 | 
						|
#include "dtable.h"
 | 
						|
#include "cygerrno.h"
 | 
						|
#include "cygheap.h"
 | 
						|
#include "sync.h"
 | 
						|
#include "sigproc.h"
 | 
						|
#include "pinfo.h"
 | 
						|
#include "sys/cygwin.h"
 | 
						|
 | 
						|
#define PAGE_CNT(bytes) howmany(bytes,getpagesize())
 | 
						|
 | 
						|
#define PGBITS		(sizeof(DWORD)*8)
 | 
						|
#define MAPSIZE(pages)	howmany(pages,PGBITS)
 | 
						|
 | 
						|
#define MAP_SET(n)	(map_map_[(n)/PGBITS] |= (1L << ((n) % PGBITS)))
 | 
						|
#define MAP_CLR(n)	(map_map_[(n)/PGBITS] &= ~(1L << ((n) % PGBITS)))
 | 
						|
#define MAP_ISSET(n)	(map_map_[(n)/PGBITS] & (1L << ((n) % PGBITS)))
 | 
						|
 | 
						|
/*
 | 
						|
 * Simple class used to keep a record of all current
 | 
						|
 * mmap areas in a process. Needed so that
 | 
						|
 * they can be duplicated after a fork().
 | 
						|
 */
 | 
						|
 | 
						|
class mmap_record
 | 
						|
{
 | 
						|
  private:
 | 
						|
    int fdesc_;
 | 
						|
    HANDLE mapping_handle_;
 | 
						|
    int devtype_;
 | 
						|
    DWORD access_mode_;
 | 
						|
    __off64_t offset_;
 | 
						|
    DWORD size_to_map_;
 | 
						|
    caddr_t base_address_;
 | 
						|
    DWORD *map_map_;
 | 
						|
 | 
						|
  public:
 | 
						|
    mmap_record (int fd, HANDLE h, DWORD ac, __off64_t o, DWORD s, caddr_t b) :
 | 
						|
       fdesc_ (fd),
 | 
						|
       mapping_handle_ (h),
 | 
						|
       devtype_ (0),
 | 
						|
       access_mode_ (ac),
 | 
						|
       offset_ (o),
 | 
						|
       size_to_map_ (s),
 | 
						|
       base_address_ (b),
 | 
						|
       map_map_ (NULL)
 | 
						|
      {
 | 
						|
	if (fd >= 0 && !cygheap->fdtab.not_open (fd))
 | 
						|
	  devtype_ = cygheap->fdtab[fd]->get_device ();
 | 
						|
      }
 | 
						|
 | 
						|
    /* Default Copy constructor/operator=/destructor are ok */
 | 
						|
 | 
						|
    /* Simple accessors */
 | 
						|
    int get_fd () const { return fdesc_; }
 | 
						|
    HANDLE get_handle () const { return mapping_handle_; }
 | 
						|
    DWORD get_device () const { return devtype_; }
 | 
						|
    DWORD get_access () const { return access_mode_; }
 | 
						|
    DWORD get_offset () const { return offset_; }
 | 
						|
    DWORD get_size () const { return size_to_map_; }
 | 
						|
    caddr_t get_address () const { return base_address_; }
 | 
						|
    DWORD *get_map () const { return map_map_; }
 | 
						|
 | 
						|
    void alloc_map ()
 | 
						|
      {
 | 
						|
	/* Allocate one bit per page */
 | 
						|
	map_map_ = (DWORD *) calloc (MAPSIZE(PAGE_CNT (size_to_map_)),
 | 
						|
				     sizeof (DWORD));
 | 
						|
	if (wincap.virtual_protect_works_on_shared_pages ())
 | 
						|
	  {
 | 
						|
	    DWORD old_prot;
 | 
						|
	    if (!VirtualProtect (base_address_, size_to_map_,
 | 
						|
				 PAGE_NOACCESS, &old_prot))
 | 
						|
	      syscall_printf ("-1 = alloc_map (): %E");
 | 
						|
	  }
 | 
						|
      }
 | 
						|
    void free_map () { if (map_map_) free (map_map_); }
 | 
						|
 | 
						|
    DWORD find_empty (DWORD pages);
 | 
						|
    __off64_t map_map (__off64_t off, DWORD len);
 | 
						|
    BOOL unmap_map (caddr_t addr, DWORD len);
 | 
						|
    void fixup_map (void);
 | 
						|
    int access (char *address);
 | 
						|
 | 
						|
    fhandler_base *alloc_fh ();
 | 
						|
    void free_fh (fhandler_base *fh);
 | 
						|
};
 | 
						|
 | 
						|
DWORD
 | 
						|
mmap_record::find_empty (DWORD pages)
 | 
						|
{
 | 
						|
  DWORD mapped_pages = PAGE_CNT (size_to_map_);
 | 
						|
  DWORD start;
 | 
						|
 | 
						|
  if (pages > mapped_pages)
 | 
						|
    return (DWORD)-1;
 | 
						|
  for (start = 0; start <= mapped_pages - pages; ++start)
 | 
						|
    if (!MAP_ISSET (start))
 | 
						|
      {
 | 
						|
	DWORD cnt;
 | 
						|
	for (cnt = 0; cnt < pages; ++cnt)
 | 
						|
	  if (MAP_ISSET (start + cnt))
 | 
						|
	    break;
 | 
						|
	if (cnt >= pages)
 | 
						|
	  return start;
 | 
						|
      }
 | 
						|
  return (DWORD)-1;
 | 
						|
}
 | 
						|
 | 
						|
__off64_t
 | 
						|
mmap_record::map_map (__off64_t off, DWORD len)
 | 
						|
{
 | 
						|
  DWORD prot, old_prot;
 | 
						|
  switch (access_mode_)
 | 
						|
    {
 | 
						|
    case FILE_MAP_WRITE:
 | 
						|
      prot = PAGE_READWRITE;
 | 
						|
      break;
 | 
						|
    case FILE_MAP_READ:
 | 
						|
      prot = PAGE_READONLY;
 | 
						|
      break;
 | 
						|
    default:
 | 
						|
      prot = PAGE_WRITECOPY;
 | 
						|
      break;
 | 
						|
    }
 | 
						|
 | 
						|
  len = PAGE_CNT (len);
 | 
						|
  if (fdesc_ == -1 && !off)
 | 
						|
    {
 | 
						|
      off = find_empty (len);
 | 
						|
      if (off != (DWORD)-1)
 | 
						|
	{
 | 
						|
	  if (wincap.virtual_protect_works_on_shared_pages ()
 | 
						|
	      && !VirtualProtect (base_address_ + off * getpagesize (),
 | 
						|
				  len * getpagesize (), prot, &old_prot))
 | 
						|
	    {
 | 
						|
	      debug_printf ("-1 = map_map (): %E");
 | 
						|
	      return (__off64_t)-1;
 | 
						|
	    }
 | 
						|
 | 
						|
	  while (len-- > 0)
 | 
						|
	    MAP_SET (off + len);
 | 
						|
	  return off * getpagesize ();
 | 
						|
	}
 | 
						|
      return 0L;
 | 
						|
    }
 | 
						|
  off -= offset_;
 | 
						|
  DWORD start = off / getpagesize ();
 | 
						|
  if (wincap.virtual_protect_works_on_shared_pages ()
 | 
						|
      && !VirtualProtect (base_address_ + start * getpagesize (),
 | 
						|
			  len * getpagesize (), prot, &old_prot))
 | 
						|
    {
 | 
						|
      debug_printf ("-1 = map_map (): %E");
 | 
						|
      return (__off64_t)-1;
 | 
						|
    }
 | 
						|
 | 
						|
  for (; len-- > 0; ++start)
 | 
						|
    MAP_SET (start);
 | 
						|
  return off;
 | 
						|
}
 | 
						|
 | 
						|
BOOL
 | 
						|
mmap_record::unmap_map (caddr_t addr, DWORD len)
 | 
						|
{
 | 
						|
  DWORD old_prot;
 | 
						|
  DWORD off = addr - base_address_;
 | 
						|
  off /= getpagesize ();
 | 
						|
  len = PAGE_CNT (len);
 | 
						|
  if (wincap.virtual_protect_works_on_shared_pages ()
 | 
						|
      && !VirtualProtect (base_address_ + off * getpagesize (),
 | 
						|
			  len * getpagesize (), PAGE_NOACCESS, &old_prot))
 | 
						|
    syscall_printf ("-1 = unmap_map (): %E");
 | 
						|
 | 
						|
  for (; len-- > 0; ++off)
 | 
						|
    MAP_CLR (off);
 | 
						|
  /* Return TRUE if all pages are free'd which may result in unmapping
 | 
						|
     the whole chunk. */
 | 
						|
  for (len = MAPSIZE(PAGE_CNT (size_to_map_)); len > 0; )
 | 
						|
    if (map_map_[--len])
 | 
						|
      return FALSE;
 | 
						|
  return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
mmap_record::fixup_map ()
 | 
						|
{
 | 
						|
  if (!wincap.virtual_protect_works_on_shared_pages ())
 | 
						|
    return;
 | 
						|
 | 
						|
  DWORD prot, old_prot;
 | 
						|
  switch (access_mode_)
 | 
						|
    {
 | 
						|
    case FILE_MAP_WRITE:
 | 
						|
      prot = PAGE_READWRITE;
 | 
						|
      break;
 | 
						|
    case FILE_MAP_READ:
 | 
						|
      prot = PAGE_READONLY;
 | 
						|
      break;
 | 
						|
    default:
 | 
						|
      prot = PAGE_WRITECOPY;
 | 
						|
      break;
 | 
						|
    }
 | 
						|
 | 
						|
  for (DWORD off = PAGE_CNT (size_to_map_); off > 0; --off)
 | 
						|
    VirtualProtect (base_address_ + off * getpagesize (),
 | 
						|
		    getpagesize (),
 | 
						|
		    MAP_ISSET (off - 1) ? prot : PAGE_NOACCESS,
 | 
						|
		    &old_prot);
 | 
						|
}
 | 
						|
 | 
						|
int
 | 
						|
mmap_record::access (char *address)
 | 
						|
{
 | 
						|
  if (address < base_address_ || address >= base_address_ + size_to_map_)
 | 
						|
    return 0;
 | 
						|
  DWORD off = (address - base_address_) / getpagesize ();
 | 
						|
  return MAP_ISSET (off);
 | 
						|
}
 | 
						|
 | 
						|
static fhandler_disk_file fh_paging_file;
 | 
						|
 | 
						|
fhandler_base *
 | 
						|
mmap_record::alloc_fh ()
 | 
						|
{
 | 
						|
  if (get_fd () == -1)
 | 
						|
    {
 | 
						|
      fh_paging_file.set_io_handle (INVALID_HANDLE_VALUE);
 | 
						|
      return &fh_paging_file;
 | 
						|
    }
 | 
						|
 | 
						|
  /* The file descriptor could have been closed or, even
 | 
						|
     worse, could have been reused for another file before
 | 
						|
     the call to fork(). This requires creating a fhandler
 | 
						|
     of the correct type to be sure to call the method of the
 | 
						|
     correct class. */
 | 
						|
  return cygheap->fdtab.build_fhandler (-1, get_device (), NULL);
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
mmap_record::free_fh (fhandler_base *fh)
 | 
						|
{
 | 
						|
  if (get_fd () != -1)
 | 
						|
    cfree (fh);
 | 
						|
}
 | 
						|
 | 
						|
class list {
 | 
						|
public:
 | 
						|
  mmap_record *recs;
 | 
						|
  int nrecs, maxrecs;
 | 
						|
  int fd;
 | 
						|
  DWORD hash;
 | 
						|
  list ();
 | 
						|
  ~list ();
 | 
						|
  mmap_record *add_record (mmap_record r);
 | 
						|
  void erase (int i);
 | 
						|
  void erase ();
 | 
						|
  mmap_record *match (__off64_t off, DWORD len);
 | 
						|
  long match (caddr_t addr, DWORD len, long start);
 | 
						|
};
 | 
						|
 | 
						|
list::list ()
 | 
						|
: nrecs (0), maxrecs (10), fd (0), hash (0)
 | 
						|
{
 | 
						|
  recs = (mmap_record *) malloc (10 * sizeof(mmap_record));
 | 
						|
}
 | 
						|
 | 
						|
list::~list ()
 | 
						|
{
 | 
						|
  for (mmap_record *rec = recs; nrecs-- > 0; ++rec)
 | 
						|
    rec->free_map ();
 | 
						|
  free (recs);
 | 
						|
}
 | 
						|
 | 
						|
mmap_record *
 | 
						|
list::add_record (mmap_record r)
 | 
						|
{
 | 
						|
  if (nrecs == maxrecs)
 | 
						|
    {
 | 
						|
      maxrecs += 5;
 | 
						|
      recs = (mmap_record *) realloc (recs, maxrecs * sizeof (mmap_record));
 | 
						|
    }
 | 
						|
  recs[nrecs] = r;
 | 
						|
  recs[nrecs].alloc_map ();
 | 
						|
  return recs + nrecs++;
 | 
						|
}
 | 
						|
 | 
						|
/* Used in mmap() */
 | 
						|
mmap_record *
 | 
						|
list::match (__off64_t off, DWORD len)
 | 
						|
{
 | 
						|
  if (fd == -1 && !off)
 | 
						|
    {
 | 
						|
      len = PAGE_CNT (len);
 | 
						|
      for (int i = 0; i < nrecs; ++i)
 | 
						|
	if (recs[i].find_empty (len) != (DWORD)-1)
 | 
						|
	  return recs + i;
 | 
						|
    }
 | 
						|
  else
 | 
						|
    {
 | 
						|
      for (int i = 0; i < nrecs; ++i)
 | 
						|
	if (off >= recs[i].get_offset ()
 | 
						|
	    && off + len <= recs[i].get_offset () + recs[i].get_size ())
 | 
						|
	  return recs + i;
 | 
						|
    }
 | 
						|
  return NULL;
 | 
						|
}
 | 
						|
 | 
						|
/* Used in munmap() */
 | 
						|
long
 | 
						|
list::match (caddr_t addr, DWORD len, __off32_t start)
 | 
						|
{
 | 
						|
  for (int i = start + 1; i < nrecs; ++i)
 | 
						|
    if (addr >= recs[i].get_address ()
 | 
						|
	&& addr + len <= recs[i].get_address () + recs[i].get_size ())
 | 
						|
      return i;
 | 
						|
  return -1;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
list::erase (int i)
 | 
						|
{
 | 
						|
  recs[i].free_map ();
 | 
						|
  for (; i < nrecs-1; i++)
 | 
						|
    recs[i] = recs[i+1];
 | 
						|
  nrecs--;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
list::erase ()
 | 
						|
{
 | 
						|
  erase (nrecs-1);
 | 
						|
}
 | 
						|
 | 
						|
class map {
 | 
						|
public:
 | 
						|
  list **lists;
 | 
						|
  int nlists, maxlists;
 | 
						|
  map ();
 | 
						|
  ~map ();
 | 
						|
  list *get_list_by_fd (int fd);
 | 
						|
  list *add_list (list *l, int fd);
 | 
						|
  void erase (int i);
 | 
						|
};
 | 
						|
 | 
						|
map::map ()
 | 
						|
{
 | 
						|
  lists = (list **) malloc (10 * sizeof(list *));
 | 
						|
  nlists = 0;
 | 
						|
  maxlists = 10;
 | 
						|
}
 | 
						|
 | 
						|
map::~map ()
 | 
						|
{
 | 
						|
  free (lists);
 | 
						|
}
 | 
						|
 | 
						|
list *
 | 
						|
map::get_list_by_fd (int fd)
 | 
						|
{
 | 
						|
  int i;
 | 
						|
  for (i=0; i<nlists; i++)
 | 
						|
#if 0 /* The fd isn't sufficient since it could already be another file. */
 | 
						|
    if (lists[i]->fd == fd
 | 
						|
#else /* so we use the name hash value to identify the file unless
 | 
						|
	 it's not an anonymous mapping. */
 | 
						|
    if ((fd == -1 && lists[i]->fd == -1)
 | 
						|
	|| (fd != -1 && lists[i]->hash == cygheap->fdtab[fd]->get_namehash ()))
 | 
						|
#endif
 | 
						|
      return lists[i];
 | 
						|
  return 0;
 | 
						|
}
 | 
						|
 | 
						|
list *
 | 
						|
map::add_list (list *l, int fd)
 | 
						|
{
 | 
						|
  l->fd = fd;
 | 
						|
  if (fd != -1)
 | 
						|
    l->hash = cygheap->fdtab[fd]->get_namehash ();
 | 
						|
  if (nlists == maxlists)
 | 
						|
    {
 | 
						|
      maxlists += 5;
 | 
						|
      lists = (list **) realloc (lists, maxlists * sizeof (list *));
 | 
						|
    }
 | 
						|
  lists[nlists++] = l;
 | 
						|
  return lists[nlists-1];
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
map::erase (int i)
 | 
						|
{
 | 
						|
  for (; i < nlists-1; i++)
 | 
						|
    lists[i] = lists[i+1];
 | 
						|
  nlists--;
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Code to keep a record of all mmap'ed areas in a process.
 | 
						|
 * Needed to duplicate tham in a child of fork().
 | 
						|
 * mmap_record classes are kept in an STL list in an STL map, keyed
 | 
						|
 * by file descriptor. This is *NOT* duplicated accross a fork(), it
 | 
						|
 * needs to be specially handled by the fork code.
 | 
						|
 */
 | 
						|
 | 
						|
static map *mmapped_areas;
 | 
						|
 | 
						|
extern "C"
 | 
						|
caddr_t
 | 
						|
mmap64 (caddr_t addr, size_t len, int prot, int flags, int fd, __off64_t off)
 | 
						|
{
 | 
						|
  syscall_printf ("addr %x, len %d, prot %x, flags %x, fd %d, off %D",
 | 
						|
		  addr, len, prot, flags, fd, off);
 | 
						|
 | 
						|
  static DWORD granularity;
 | 
						|
  if (!granularity)
 | 
						|
    {
 | 
						|
      SYSTEM_INFO si;
 | 
						|
      GetSystemInfo (&si);
 | 
						|
      granularity = si.dwAllocationGranularity;
 | 
						|
    }
 | 
						|
 | 
						|
  /* Error conditions according to SUSv2 */
 | 
						|
  if (off % getpagesize ()
 | 
						|
      || (!(flags & MAP_SHARED) && !(flags & MAP_PRIVATE))
 | 
						|
      || ((flags & MAP_SHARED) && (flags & MAP_PRIVATE))
 | 
						|
      || ((flags & MAP_FIXED) && ((DWORD)addr % granularity))
 | 
						|
      || !len)
 | 
						|
    {
 | 
						|
      set_errno (EINVAL);
 | 
						|
      syscall_printf ("-1 = mmap(): EINVAL");
 | 
						|
      return MAP_FAILED;
 | 
						|
    }
 | 
						|
 | 
						|
  SetResourceLock(LOCK_MMAP_LIST, READ_LOCK | WRITE_LOCK, "mmap");
 | 
						|
 | 
						|
  if (mmapped_areas == NULL)
 | 
						|
    {
 | 
						|
      /* First mmap call, create STL map */
 | 
						|
      mmapped_areas = new map;
 | 
						|
      if (mmapped_areas == NULL)
 | 
						|
	{
 | 
						|
	  set_errno (ENOMEM);
 | 
						|
	  syscall_printf ("-1 = mmap(): ENOMEM");
 | 
						|
	  ReleaseResourceLock(LOCK_MMAP_LIST, READ_LOCK | WRITE_LOCK, "mmap");
 | 
						|
	  return MAP_FAILED;
 | 
						|
	}
 | 
						|
    }
 | 
						|
 | 
						|
  if (flags & MAP_ANONYMOUS)
 | 
						|
    fd = -1;
 | 
						|
 | 
						|
  /* Map always in multipliers of `granularity'-sized chunks. */
 | 
						|
  __off64_t gran_off = off & ~(granularity - 1);
 | 
						|
  DWORD gran_len = howmany (len, granularity) * granularity;
 | 
						|
 | 
						|
  fhandler_base *fh;
 | 
						|
  caddr_t base = addr;
 | 
						|
  HANDLE h;
 | 
						|
 | 
						|
  if (fd != -1)
 | 
						|
    {
 | 
						|
      /* Ensure that fd is open */
 | 
						|
      cygheap_fdget cfd (fd);
 | 
						|
      if (cfd < 0)
 | 
						|
	{
 | 
						|
	  syscall_printf ("-1 = mmap(): EBADF");
 | 
						|
	  ReleaseResourceLock(LOCK_MMAP_LIST, READ_LOCK | WRITE_LOCK, "mmap");
 | 
						|
	  return MAP_FAILED;
 | 
						|
	}
 | 
						|
      fh = cfd;
 | 
						|
      if (fh->get_device () == FH_DISK)
 | 
						|
	{
 | 
						|
	  DWORD high;
 | 
						|
	  DWORD low = GetFileSize (fh->get_handle (), &high);
 | 
						|
	  __off64_t fsiz = ((__off64_t)high << 32) + low;
 | 
						|
	  fsiz -= gran_off;
 | 
						|
	  if (gran_len > fsiz)
 | 
						|
	    gran_len = fsiz;
 | 
						|
	}
 | 
						|
      else if (fh->get_device () == FH_ZERO)
 | 
						|
	/* mmap /dev/zero is like MAP_ANONYMOUS. */
 | 
						|
	fd = -1;
 | 
						|
    }
 | 
						|
  if (fd == -1)
 | 
						|
    {
 | 
						|
      fh_paging_file.set_io_handle (INVALID_HANDLE_VALUE);
 | 
						|
      fh = &fh_paging_file;
 | 
						|
    }
 | 
						|
 | 
						|
  list *l = mmapped_areas->get_list_by_fd (fd);
 | 
						|
 | 
						|
  /* First check if this mapping matches into the chunk of another
 | 
						|
     already performed mapping. Only valid for MAP_ANON in a special
 | 
						|
     case of MAP_PRIVATE. */
 | 
						|
  if (l && fd == -1 && off == 0 && !(flags & MAP_FIXED))
 | 
						|
    {
 | 
						|
      mmap_record *rec;
 | 
						|
      if ((rec = l->match (off, len)) != NULL)
 | 
						|
	{
 | 
						|
	  if ((off = rec->map_map (off, len)) == (__off64_t)-1)
 | 
						|
	    {
 | 
						|
	      set_errno (ENOMEM);
 | 
						|
	      syscall_printf ("-1 = mmap(): ENOMEM");
 | 
						|
	      ReleaseResourceLock(LOCK_MMAP_LIST, READ_LOCK|WRITE_LOCK, "mmap");
 | 
						|
	      return MAP_FAILED;
 | 
						|
	    }
 | 
						|
	  caddr_t ret = rec->get_address () + off;
 | 
						|
	  syscall_printf ("%x = mmap() succeeded", ret);
 | 
						|
	  ReleaseResourceLock(LOCK_MMAP_LIST, READ_LOCK | WRITE_LOCK, "mmap");
 | 
						|
	  return ret;
 | 
						|
	}
 | 
						|
    }
 | 
						|
 | 
						|
  DWORD access = (prot & PROT_WRITE) ? FILE_MAP_WRITE : FILE_MAP_READ;
 | 
						|
  /* copy-on-write doesn't work at all on 9x using anonymous maps.
 | 
						|
     Workaround: Anonymous mappings always use normal READ or WRITE
 | 
						|
		 access and don't use named file mapping.
 | 
						|
     copy-on-write doesn't also work properly on 9x with real files.
 | 
						|
     While the changes are not propagated to the file, they are
 | 
						|
     visible to other processes sharing the same file mapping object.
 | 
						|
     Workaround: Don't use named file mapping.  That should work since
 | 
						|
		 sharing file mappings only works reliable using named
 | 
						|
		 file mapping on 9x.
 | 
						|
  */
 | 
						|
  if ((flags & MAP_PRIVATE)
 | 
						|
      && (wincap.has_working_copy_on_write () || fd != -1))
 | 
						|
    access = FILE_MAP_COPY;
 | 
						|
 | 
						|
  h = fh->mmap (&base, gran_len, access, flags, gran_off);
 | 
						|
 | 
						|
  if (h == INVALID_HANDLE_VALUE)
 | 
						|
    {
 | 
						|
      ReleaseResourceLock(LOCK_MMAP_LIST, READ_LOCK | WRITE_LOCK, "mmap");
 | 
						|
      return MAP_FAILED;
 | 
						|
    }
 | 
						|
 | 
						|
  /* Now we should have a successfully mmapped area.
 | 
						|
     Need to save it so forked children can reproduce it.
 | 
						|
  */
 | 
						|
  if (fd == -1)
 | 
						|
    gran_len = PAGE_CNT (gran_len) * getpagesize ();
 | 
						|
  mmap_record mmap_rec (fd, h, access, gran_off, gran_len, base);
 | 
						|
 | 
						|
  /* Get list of mmapped areas for this fd, create a new one if
 | 
						|
     one does not exist yet.
 | 
						|
  */
 | 
						|
  if (l == 0)
 | 
						|
    {
 | 
						|
      /* Create a new one */
 | 
						|
      l = new list;
 | 
						|
      if (l == 0)
 | 
						|
	{
 | 
						|
	  fh->munmap (h, base, gran_len);
 | 
						|
	  set_errno (ENOMEM);
 | 
						|
	  syscall_printf ("-1 = mmap(): ENOMEM");
 | 
						|
	  ReleaseResourceLock(LOCK_MMAP_LIST, READ_LOCK | WRITE_LOCK, "mmap");
 | 
						|
	  return MAP_FAILED;
 | 
						|
	}
 | 
						|
      l = mmapped_areas->add_list (l, fd);
 | 
						|
  }
 | 
						|
 | 
						|
  /* Insert into the list */
 | 
						|
  mmap_record *rec = l->add_record (mmap_rec);
 | 
						|
  if ((off = rec->map_map (off, len)) == (__off64_t)-1)
 | 
						|
    {
 | 
						|
      fh->munmap (h, base, gran_len);
 | 
						|
      l->erase ();
 | 
						|
      set_errno (ENOMEM);
 | 
						|
      syscall_printf ("-1 = mmap(): ENOMEM");
 | 
						|
      ReleaseResourceLock(LOCK_MMAP_LIST, READ_LOCK | WRITE_LOCK, "mmap");
 | 
						|
      return MAP_FAILED;
 | 
						|
    }
 | 
						|
  caddr_t ret = rec->get_address () + off;
 | 
						|
  syscall_printf ("%x = mmap() succeeded", ret);
 | 
						|
  ReleaseResourceLock(LOCK_MMAP_LIST, READ_LOCK | WRITE_LOCK, "mmap");
 | 
						|
  return ret;
 | 
						|
}
 | 
						|
 | 
						|
extern "C"
 | 
						|
caddr_t
 | 
						|
mmap (caddr_t addr, size_t len, int prot, int flags, int fd, __off32_t off)
 | 
						|
{
 | 
						|
  return mmap64 (addr, len, prot, flags, fd, (__off64_t)off);
 | 
						|
}
 | 
						|
 | 
						|
/* munmap () removes an mmapped area.  It insists that base area
 | 
						|
   requested is the same as that mmapped, error if not. */
 | 
						|
 | 
						|
extern "C"
 | 
						|
int
 | 
						|
munmap (caddr_t addr, size_t len)
 | 
						|
{
 | 
						|
  syscall_printf ("munmap (addr %x, len %d)", addr, len);
 | 
						|
 | 
						|
  /* Error conditions according to SUSv2 */
 | 
						|
  if (((DWORD)addr % getpagesize ()) || !len)
 | 
						|
    {
 | 
						|
      set_errno (EINVAL);
 | 
						|
      syscall_printf ("-1 = munmap(): Invalid parameters");
 | 
						|
      return -1;
 | 
						|
    }
 | 
						|
 | 
						|
  SetResourceLock(LOCK_MMAP_LIST, WRITE_LOCK | READ_LOCK, "munmap");
 | 
						|
  /* Check if a mmap'ed area was ever created */
 | 
						|
  if (mmapped_areas == NULL)
 | 
						|
    {
 | 
						|
      syscall_printf ("-1 = munmap(): mmapped_areas == NULL");
 | 
						|
      set_errno (EINVAL);
 | 
						|
      ReleaseResourceLock(LOCK_MMAP_LIST, WRITE_LOCK | READ_LOCK, "munmap");
 | 
						|
      return -1;
 | 
						|
    }
 | 
						|
 | 
						|
  /* Iterate through the map, looking for the mmapped area.
 | 
						|
     Error if not found. */
 | 
						|
 | 
						|
  for (int it = 0; it < mmapped_areas->nlists; ++it)
 | 
						|
    {
 | 
						|
      list *l = mmapped_areas->lists[it];
 | 
						|
      if (l)
 | 
						|
	{
 | 
						|
	  long li = -1;
 | 
						|
	  if ((li = l->match(addr, len, li)) >= 0)
 | 
						|
	    {
 | 
						|
	      mmap_record *rec = l->recs + li;
 | 
						|
	      if (rec->unmap_map (addr, len))
 | 
						|
		{
 | 
						|
		  fhandler_base *fh = rec->alloc_fh ();
 | 
						|
		  fh->munmap (rec->get_handle (), addr, len);
 | 
						|
		  rec->free_fh (fh);
 | 
						|
 | 
						|
		  /* Delete the entry. */
 | 
						|
		  l->erase (li);
 | 
						|
		}
 | 
						|
	      syscall_printf ("0 = munmap(): %x", addr);
 | 
						|
	      ReleaseResourceLock(LOCK_MMAP_LIST, WRITE_LOCK | READ_LOCK, "munmap");
 | 
						|
	      return 0;
 | 
						|
	    }
 | 
						|
	}
 | 
						|
    }
 | 
						|
 | 
						|
  set_errno (EINVAL);
 | 
						|
  syscall_printf ("-1 = munmap(): EINVAL");
 | 
						|
 | 
						|
  ReleaseResourceLock(LOCK_MMAP_LIST, WRITE_LOCK | READ_LOCK, "munmap");
 | 
						|
  return -1;
 | 
						|
}
 | 
						|
 | 
						|
/* Sync file with memory. Ignore flags for now. */
 | 
						|
 | 
						|
extern "C"
 | 
						|
int
 | 
						|
msync (caddr_t addr, size_t len, int flags)
 | 
						|
{
 | 
						|
  syscall_printf ("addr = %x, len = %d, flags = %x",
 | 
						|
		  addr, len, flags);
 | 
						|
 | 
						|
  /* However, check flags for validity. */
 | 
						|
  if ((flags & ~(MS_ASYNC | MS_SYNC | MS_INVALIDATE))
 | 
						|
      || ((flags & MS_ASYNC) && (flags & MS_SYNC)))
 | 
						|
    {
 | 
						|
      syscall_printf ("-1 = msync(): Invalid flags");
 | 
						|
      set_errno (EINVAL);
 | 
						|
      return -1;
 | 
						|
    }
 | 
						|
 | 
						|
  SetResourceLock(LOCK_MMAP_LIST, WRITE_LOCK | READ_LOCK, "msync");
 | 
						|
  /* Check if a mmap'ed area was ever created */
 | 
						|
  if (mmapped_areas == NULL)
 | 
						|
    {
 | 
						|
      syscall_printf ("-1 = msync(): mmapped_areas == NULL");
 | 
						|
      set_errno (EINVAL);
 | 
						|
      ReleaseResourceLock(LOCK_MMAP_LIST, WRITE_LOCK | READ_LOCK, "msync");
 | 
						|
      return -1;
 | 
						|
    }
 | 
						|
 | 
						|
  /* Iterate through the map, looking for the mmapped area.
 | 
						|
     Error if not found. */
 | 
						|
 | 
						|
  for (int it = 0; it < mmapped_areas->nlists; ++it)
 | 
						|
    {
 | 
						|
      list *l = mmapped_areas->lists[it];
 | 
						|
      if (l != 0)
 | 
						|
	{
 | 
						|
	  for (int li = 0; li < l->nrecs; ++li)
 | 
						|
	    {
 | 
						|
	      mmap_record *rec = l->recs + li;
 | 
						|
	      if (rec->access (addr))
 | 
						|
		{
 | 
						|
		  /* Check whole area given by len. */
 | 
						|
		  for (DWORD i = getpagesize (); i < len; ++i)
 | 
						|
		    if (!rec->access (addr + i))
 | 
						|
		      goto invalid_address_range;
 | 
						|
		  fhandler_base *fh = rec->alloc_fh ();
 | 
						|
		  int ret = fh->msync (rec->get_handle (), addr, len, flags);
 | 
						|
		  rec->free_fh (fh);
 | 
						|
 | 
						|
		  if (ret)
 | 
						|
		    syscall_printf ("%d = msync(): %E", ret);
 | 
						|
		  else
 | 
						|
		    syscall_printf ("0 = msync()");
 | 
						|
 | 
						|
		  ReleaseResourceLock(LOCK_MMAP_LIST, WRITE_LOCK | READ_LOCK, "msync");
 | 
						|
		  return 0;
 | 
						|
		}
 | 
						|
	    }
 | 
						|
	}
 | 
						|
    }
 | 
						|
 | 
						|
invalid_address_range:
 | 
						|
  /* SUSv2: Return code if indicated memory was not mapped is ENOMEM. */
 | 
						|
  set_errno (ENOMEM);
 | 
						|
  syscall_printf ("-1 = msync(): ENOMEM");
 | 
						|
 | 
						|
  ReleaseResourceLock(LOCK_MMAP_LIST, WRITE_LOCK | READ_LOCK, "msync");
 | 
						|
  return -1;
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Base implementation:
 | 
						|
 *
 | 
						|
 * `mmap' returns ENODEV as documented in SUSv2.
 | 
						|
 * In contrast to the global function implementation, the member function
 | 
						|
 * `mmap' has to return the mapped base address in `addr' and the handle to
 | 
						|
 * the mapping object as return value. In case of failure, the fhandler
 | 
						|
 * mmap has to close that handle by itself and return INVALID_HANDLE_VALUE.
 | 
						|
 *
 | 
						|
 * `munmap' and `msync' get the handle to the mapping object as first parameter
 | 
						|
 * additionally.
 | 
						|
*/
 | 
						|
HANDLE
 | 
						|
fhandler_base::mmap (caddr_t *addr, size_t len, DWORD access,
 | 
						|
		     int flags, __off64_t off)
 | 
						|
{
 | 
						|
  set_errno (ENODEV);
 | 
						|
  return INVALID_HANDLE_VALUE;
 | 
						|
}
 | 
						|
 | 
						|
int
 | 
						|
fhandler_base::munmap (HANDLE h, caddr_t addr, size_t len)
 | 
						|
{
 | 
						|
  set_errno (ENODEV);
 | 
						|
  return -1;
 | 
						|
}
 | 
						|
 | 
						|
int
 | 
						|
fhandler_base::msync (HANDLE h, caddr_t addr, size_t len, int flags)
 | 
						|
{
 | 
						|
  set_errno (ENODEV);
 | 
						|
  return -1;
 | 
						|
}
 | 
						|
 | 
						|
BOOL
 | 
						|
fhandler_base::fixup_mmap_after_fork (HANDLE h, DWORD access, DWORD offset,
 | 
						|
				      DWORD size, void *address)
 | 
						|
{
 | 
						|
  set_errno (ENODEV);
 | 
						|
  return -1;
 | 
						|
}
 | 
						|
 | 
						|
/* Implementation for disk files. */
 | 
						|
HANDLE
 | 
						|
fhandler_disk_file::mmap (caddr_t *addr, size_t len, DWORD access,
 | 
						|
			  int flags, __off64_t off)
 | 
						|
{
 | 
						|
  DWORD protect;
 | 
						|
 | 
						|
  if (access & FILE_MAP_COPY)
 | 
						|
    protect = PAGE_WRITECOPY;
 | 
						|
  else if (access & FILE_MAP_WRITE)
 | 
						|
    protect = PAGE_READWRITE;
 | 
						|
  else
 | 
						|
    protect = PAGE_READONLY;
 | 
						|
 | 
						|
  HANDLE h;
 | 
						|
 | 
						|
  /* On 9x/ME try first to open the mapping by name when opening a
 | 
						|
     shared file object. This is needed since 9x/ME only shares
 | 
						|
     objects between processes by name. What a mess... */
 | 
						|
  if (wincap.share_mmaps_only_by_name ()
 | 
						|
      && get_handle () != INVALID_HANDLE_VALUE
 | 
						|
      && !(access & FILE_MAP_COPY))
 | 
						|
    {
 | 
						|
      /* Grrr, the whole stuff is just needed to try to get a reliable
 | 
						|
	 mapping of the same file. Even that uprising isn't bullet
 | 
						|
	 proof but it does it's best... */
 | 
						|
      char namebuf[MAX_PATH];
 | 
						|
      cygwin_conv_to_full_posix_path (get_name (), namebuf);
 | 
						|
      for (int i = strlen (namebuf) - 1; i >= 0; --i)
 | 
						|
	namebuf[i] = cyg_tolower (namebuf [i]);
 | 
						|
 | 
						|
      debug_printf ("named sharing");
 | 
						|
      if (!(h = OpenFileMapping (access, TRUE, namebuf)))
 | 
						|
	h = CreateFileMapping (get_handle(), &sec_none, protect, 0, 0, namebuf);
 | 
						|
    }
 | 
						|
  else
 | 
						|
    h = CreateFileMapping (get_handle (), &sec_none, protect, 0,
 | 
						|
			   get_handle () == INVALID_HANDLE_VALUE ? len : 0,
 | 
						|
			   NULL);
 | 
						|
  if (!h)
 | 
						|
    {
 | 
						|
      __seterrno ();
 | 
						|
      syscall_printf ("-1 = mmap(): CreateFileMapping failed with %E");
 | 
						|
      return INVALID_HANDLE_VALUE;
 | 
						|
    }
 | 
						|
 | 
						|
  DWORD high = off >> 32, low = off & 0xffffffff;
 | 
						|
  void *base = MapViewOfFileEx (h, access, high, low, len,
 | 
						|
			       (flags & MAP_FIXED) ? *addr : NULL);
 | 
						|
  debug_printf ("%x = MapViewOfFileEx (h:%x, access:%x, 0, off:%D, len:%d, addr:%x)", base, h, access, off, len, (flags & MAP_FIXED) ? *addr : NULL);
 | 
						|
  if (!base || ((flags & MAP_FIXED) && base != *addr))
 | 
						|
    {
 | 
						|
      if (!base)
 | 
						|
	{
 | 
						|
	  __seterrno ();
 | 
						|
	  syscall_printf ("-1 = mmap(): MapViewOfFileEx failed with %E");
 | 
						|
	}
 | 
						|
      else
 | 
						|
	{
 | 
						|
	  set_errno (EINVAL);
 | 
						|
	  syscall_printf ("-1 = mmap(): address shift with MAP_FIXED given");
 | 
						|
	}
 | 
						|
      CloseHandle (h);
 | 
						|
      return INVALID_HANDLE_VALUE;
 | 
						|
    }
 | 
						|
 | 
						|
  *addr = (caddr_t) base;
 | 
						|
  return h;
 | 
						|
}
 | 
						|
 | 
						|
int
 | 
						|
fhandler_disk_file::munmap (HANDLE h, caddr_t addr, size_t len)
 | 
						|
{
 | 
						|
  UnmapViewOfFile (addr);
 | 
						|
  CloseHandle (h);
 | 
						|
  return 0;
 | 
						|
}
 | 
						|
 | 
						|
int
 | 
						|
fhandler_disk_file::msync (HANDLE h, caddr_t addr, size_t len, int flags)
 | 
						|
{
 | 
						|
  if (FlushViewOfFile (addr, len) == 0)
 | 
						|
    {
 | 
						|
      __seterrno ();
 | 
						|
      return -1;
 | 
						|
    }
 | 
						|
  return 0;
 | 
						|
}
 | 
						|
 | 
						|
BOOL
 | 
						|
fhandler_disk_file::fixup_mmap_after_fork (HANDLE h, DWORD access, DWORD offset,
 | 
						|
					   DWORD size, void *address)
 | 
						|
{
 | 
						|
  /* Re-create the MapViewOfFileEx call */
 | 
						|
  void *base = MapViewOfFileEx (h, access, 0, offset, size, address);
 | 
						|
  return base == address;
 | 
						|
}
 | 
						|
 | 
						|
/* Set memory protection */
 | 
						|
 | 
						|
extern "C"
 | 
						|
int
 | 
						|
mprotect (caddr_t addr, size_t len, int prot)
 | 
						|
{
 | 
						|
  DWORD old_prot;
 | 
						|
  DWORD new_prot = 0;
 | 
						|
 | 
						|
  syscall_printf ("mprotect (addr %x, len %d, prot %x)", addr, len, prot);
 | 
						|
 | 
						|
  if (prot == PROT_NONE)
 | 
						|
    new_prot = PAGE_NOACCESS;
 | 
						|
  else
 | 
						|
    {
 | 
						|
      switch (prot)
 | 
						|
	{
 | 
						|
	  case PROT_READ | PROT_WRITE | PROT_EXEC:
 | 
						|
	    new_prot = PAGE_EXECUTE_READWRITE;
 | 
						|
	    break;
 | 
						|
	  case PROT_READ | PROT_WRITE:
 | 
						|
	    new_prot = PAGE_READWRITE;
 | 
						|
	    break;
 | 
						|
	  case PROT_READ | PROT_EXEC:
 | 
						|
	    new_prot = PAGE_EXECUTE_READ;
 | 
						|
	    break;
 | 
						|
	  case PROT_READ:
 | 
						|
	    new_prot = PAGE_READONLY;
 | 
						|
	    break;
 | 
						|
	  default:
 | 
						|
	    syscall_printf ("-1 = mprotect (): invalid prot value");
 | 
						|
	    set_errno (EINVAL);
 | 
						|
	    return -1;
 | 
						|
	 }
 | 
						|
     }
 | 
						|
 | 
						|
  if (VirtualProtect (addr, len, new_prot, &old_prot) == 0)
 | 
						|
    {
 | 
						|
      __seterrno ();
 | 
						|
      syscall_printf ("-1 = mprotect (): %E");
 | 
						|
      return -1;
 | 
						|
    }
 | 
						|
 | 
						|
  syscall_printf ("0 = mprotect ()");
 | 
						|
  return 0;
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Call to re-create all the file mappings in a forked
 | 
						|
 * child. Called from the child in initialization. At this
 | 
						|
 * point we are passed a valid mmapped_areas map, and all the
 | 
						|
 * HANDLE's are valid for the child, but none of the
 | 
						|
 * mapped areas are in our address space. We need to iterate
 | 
						|
 * through the map, doing the MapViewOfFile calls.
 | 
						|
 */
 | 
						|
 | 
						|
int __stdcall
 | 
						|
fixup_mmaps_after_fork (HANDLE parent)
 | 
						|
{
 | 
						|
 | 
						|
  debug_printf ("recreate_mmaps_after_fork, mmapped_areas %p", mmapped_areas);
 | 
						|
 | 
						|
  /* Check if a mmapped area was ever created */
 | 
						|
  if (mmapped_areas == NULL)
 | 
						|
    return 0;
 | 
						|
 | 
						|
  /* Iterate through the map */
 | 
						|
  for (int it = 0; it < mmapped_areas->nlists; ++it)
 | 
						|
    {
 | 
						|
      list *l = mmapped_areas->lists[it];
 | 
						|
      if (l != 0)
 | 
						|
	{
 | 
						|
	  int li;
 | 
						|
	  for (li = 0; li < l->nrecs; ++li)
 | 
						|
	    {
 | 
						|
	      mmap_record *rec = l->recs + li;
 | 
						|
 | 
						|
	      debug_printf ("fd %d, h %x, access %x, offset %d, size %d, address %p",
 | 
						|
		  rec->get_fd (), rec->get_handle (), rec->get_access (),
 | 
						|
		  rec->get_offset (), rec->get_size (), rec->get_address ());
 | 
						|
 | 
						|
	      fhandler_base *fh = rec->alloc_fh ();
 | 
						|
	      BOOL ret = fh->fixup_mmap_after_fork (rec->get_handle (),
 | 
						|
						    rec->get_access (),
 | 
						|
						    rec->get_offset (),
 | 
						|
						    rec->get_size (),
 | 
						|
						    rec->get_address ());
 | 
						|
	      rec->free_fh (fh);
 | 
						|
 | 
						|
	      if (!ret)
 | 
						|
		{
 | 
						|
		  system_printf ("base address fails to match requested address %p",
 | 
						|
				 rec->get_address ());
 | 
						|
		  return -1;
 | 
						|
		}
 | 
						|
	      if (rec->get_access () == FILE_MAP_COPY)
 | 
						|
	        {
 | 
						|
		  for (char *address = rec->get_address ();
 | 
						|
		       address < rec->get_address () + rec->get_size ();
 | 
						|
		       address += getpagesize ())
 | 
						|
		    if (rec->access (address)
 | 
						|
		        && !ReadProcessMemory (parent, address, address,
 | 
						|
					       getpagesize (), NULL))
 | 
						|
		      {
 | 
						|
			system_printf ("ReadProcessMemory failed for MAP_PRIVATE address %p, %E",
 | 
						|
				       rec->get_address ());
 | 
						|
			return -1;
 | 
						|
		      }
 | 
						|
	        }
 | 
						|
	      rec->fixup_map ();
 | 
						|
	    }
 | 
						|
	}
 | 
						|
    }
 | 
						|
 | 
						|
  debug_printf ("succeeded");
 | 
						|
  return 0;
 | 
						|
}
 |