]> git.ipfire.org Git - thirdparty/glibc.git/blobdiff - sysdeps/mach/hurd/mmap.c
Prefer https to http for gnu.org and fsf.org URLs
[thirdparty/glibc.git] / sysdeps / mach / hurd / mmap.c
index dc4b024bb8cdd8727935e3139426bfcfdc9edf17..1e6ab401af57e147fefa3fff6cfebb0af897cd71 100644 (file)
@@ -1,20 +1,19 @@
-/* Copyright (C) 1994, 1995, 1996, 1997 Free Software Foundation, Inc.
+/* Copyright (C) 1994-2019 Free Software Foundation, Inc.
    This file is part of the GNU C Library.
 
    The GNU C Library is free software; you can redistribute it and/or
-   modify it under the terms of the GNU Library General Public License as
-   published by the Free Software Foundation; either version 2 of the
-   License, or (at your option) any later version.
+   modify it under the terms of the GNU Lesser General Public
+   License as published by the Free Software Foundation; either
+   version 2.1 of the License, or (at your option) any later version.
 
    The GNU C Library is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
-   Library General Public License for more details.
+   Lesser General Public License for more details.
 
-   You should have received a copy of the GNU Library General Public
-   License along with the GNU C Library; see the file COPYING.LIB.  If not,
-   write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
-   Boston, MA 02111-1307, USA.  */
+   You should have received a copy of the GNU Lesser General Public
+   License along with the GNU C Library; if not, see
+   <https://www.gnu.org/licenses/>.  */
 
 #include <sys/types.h>
 #include <sys/mman.h>
    is nonzero, it is the desired mapping address.  If the MAP_FIXED bit is
    set in FLAGS, the mapping will be at ADDR exactly (which must be
    page-aligned); otherwise the system chooses a convenient nearby address.
-   The return value is the actual mapping address chosen or (caddr_t) -1
+   The return value is the actual mapping address chosen or (void *) -1
    for errors (in which case `errno' is set).  A successful `mmap' call
    deallocates any previous mapping for the affected region.  */
 
-caddr_t
-__mmap (caddr_t addr, size_t len, int prot, int flags, int fd, off_t offset)
+void *
+__mmap (void *addr, size_t len, int prot, int flags, int fd, off_t offset)
 {
   error_t err;
   vm_prot_t vmprot;
   memory_object_t memobj;
   vm_address_t mapaddr;
-  vm_size_t pageoff;
-
-  pageoff = offset & (vm_page_size - 1);
-  offset &= ~(vm_page_size - 1);
 
   mapaddr = (vm_address_t) addr;
-  if (flags & MAP_FIXED)
-    {
-      /* A specific address is requested.  It need not be page-aligned;
-        it just needs to be congruent with the object offset.  */
-      if ((mapaddr & (vm_page_size - 1)) != pageoff)
-       return (caddr_t) (long int) __hurd_fail (EINVAL);
-      else
-       /* We will add back PAGEOFF after mapping.  */
-       mapaddr -= pageoff;
-    }
+
+  /* ADDR and OFFSET must be page-aligned.  */
+  if ((mapaddr & (__vm_page_size - 1)) || (offset & (__vm_page_size - 1)))
+    return (void *) (long int) __hurd_fail (EINVAL);
 
   vmprot = VM_PROT_NONE;
   if (prot & PROT_READ)
@@ -66,7 +55,7 @@ __mmap (caddr_t addr, size_t len, int prot, int flags, int fd, off_t offset)
   switch (flags & MAP_TYPE)
     {
     default:
-      return (caddr_t) (long int) __hurd_fail (EINVAL);
+      return (void *) (long int) __hurd_fail (EINVAL);
 
     case MAP_ANON:
       memobj = MACH_PORT_NULL;
@@ -77,9 +66,20 @@ __mmap (caddr_t addr, size_t len, int prot, int flags, int fd, off_t offset)
       {
        mach_port_t robj, wobj;
        if (err = HURD_DPORT_USE (fd, __io_map (port, &robj, &wobj)))
-         return (caddr_t) (long int) __hurd_dfail (fd, err);
+         {
+           if (err == MIG_BAD_ID || err == EOPNOTSUPP || err == ENOSYS)
+             err = ENODEV;     /* File descriptor doesn't support mmap.  */
+           return (void *) (long int) __hurd_dfail (fd, err);
+         }
        switch (prot & (PROT_READ|PROT_WRITE))
          {
+         /* Although it apparently doesn't make sense to map a file with
+            protection set to PROT_NONE, it is actually sometimes done.
+            In particular, that's how localedef reserves some space for
+            the locale archive file, the rationale being that some
+            implementations take into account whether the mapping is
+            anonymous or not when selecting addresses.  */
+         case PROT_NONE:
          case PROT_READ:
            memobj = robj;
            if (wobj != MACH_PORT_NULL)
@@ -97,8 +97,8 @@ __mmap (caddr_t addr, size_t len, int prot, int flags, int fd, off_t offset)
                /* Remove extra reference.  */
                __mach_port_deallocate (__mach_task_self (), memobj);
              }
-           else if (wobj == MACH_PORT_NULL && /* Not writable by mapping.  */
-                    !(flags & MAP_SHARED))
+           else if (wobj == MACH_PORT_NULL /* Not writable by mapping.  */
+                    && !(flags & MAP_SHARED))
              /* The file can only be mapped for reading.  Since we are
                 making a private mapping, we will never try to write the
                 object anyway, so we don't care.  */
@@ -106,9 +106,11 @@ __mmap (caddr_t addr, size_t len, int prot, int flags, int fd, off_t offset)
            else
              {
                __mach_port_deallocate (__mach_task_self (), wobj);
-               return (caddr_t) (long int) __hurd_fail (EACCES);
+               return (void *) (long int) __hurd_fail (EACCES);
              }
            break;
+         default:
+           __builtin_unreachable ();
          }
        break;
        /* XXX handle MAP_NOEXTEND */
@@ -119,21 +121,32 @@ __mmap (caddr_t addr, size_t len, int prot, int flags, int fd, off_t offset)
 
   err = __vm_map (__mach_task_self (),
                  &mapaddr, (vm_size_t) len, (vm_address_t) 0,
-                 ! (flags & MAP_FIXED),
+                 mapaddr == 0,
                  memobj, (vm_offset_t) offset,
                  ! (flags & MAP_SHARED),
                  vmprot, VM_PROT_ALL,
                  (flags & MAP_SHARED) ? VM_INHERIT_SHARE : VM_INHERIT_COPY);
 
-  if (err == KERN_NO_SPACE && (flags & MAP_FIXED))
+  if (err == KERN_NO_SPACE)
     {
-      /* XXX this is not atomic as it is in unix! */
-      /* The region is already allocated; deallocate it first.  */
-      err = __vm_deallocate (__mach_task_self (), mapaddr, len);
-      if (! err)
+      if (flags & MAP_FIXED)
+       {
+         /* XXX this is not atomic as it is in unix! */
+         /* The region is already allocated; deallocate it first.  */
+         err = __vm_deallocate (__mach_task_self (), mapaddr, len);
+         if (! err)
+           err = __vm_map (__mach_task_self (),
+                           &mapaddr, (vm_size_t) len, (vm_address_t) 0,
+                           0, memobj, (vm_offset_t) offset,
+                           ! (flags & MAP_SHARED),
+                           vmprot, VM_PROT_ALL,
+                           (flags & MAP_SHARED) ? VM_INHERIT_SHARE
+                           : VM_INHERIT_COPY);
+       }
+      else if (mapaddr != 0)
        err = __vm_map (__mach_task_self (),
                        &mapaddr, (vm_size_t) len, (vm_address_t) 0,
-                       0, memobj, (vm_offset_t) offset,
+                       1, memobj, (vm_offset_t) offset,
                        ! (flags & MAP_SHARED),
                        vmprot, VM_PROT_ALL,
                        (flags & MAP_SHARED) ? VM_INHERIT_SHARE
@@ -144,12 +157,10 @@ __mmap (caddr_t addr, size_t len, int prot, int flags, int fd, off_t offset)
     __mach_port_deallocate (__mach_task_self (), memobj);
 
   if (err)
-    return (caddr_t) (long int) __hurd_fail (err);
-
-  /* Adjust the mapping address for the offset-within-page.  */
-  mapaddr += pageoff;
+    return (void *) (long int) __hurd_fail (err);
 
-  return (caddr_t) mapaddr;
+  return (void *) mapaddr;
 }
 
+libc_hidden_def (__mmap)
 weak_alias (__mmap, mmap)