aboutsummaryrefslogtreecommitdiff
path: root/sysdeps/mach/hurd/mmap.c
blob: 20264a775b38757350e2b2ae8b96f046aedffb78 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
/* Copyright (C) 1994-2023 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 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
   Lesser General Public License for more details.

   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>
#include <errno.h>
#include <ldsodefs.h>
#include <hurd.h>
#include <hurd/fd.h>

/* Map addresses starting near ADDR and extending for LEN bytes.  from
   OFFSET into the file FD describes according to PROT and FLAGS.  If ADDR
   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 (void *) -1
   for errors (in which case `errno' is set).  A successful `mmap' call
   deallocates any previous mapping for the affected region.  */

void *
__mmap (void *addr, size_t len, int prot, int flags, int fd, off_t offset)
{
  error_t err;
  vm_prot_t vmprot, max_vmprot;
  memory_object_t memobj;
  vm_address_t mapaddr, mask;
  boolean_t copy, anywhere;

  mapaddr = (vm_address_t) addr;

  /* 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);

  if ((flags & MAP_EXCL) && ! (flags & MAP_FIXED))
    return (void *) (long int) __hurd_fail (EINVAL);

  vmprot = VM_PROT_NONE;
  if (prot & PROT_READ)
    vmprot |= VM_PROT_READ;
  if (prot & PROT_WRITE)
    vmprot |= VM_PROT_WRITE;
  if (prot & PROT_EXEC)
    vmprot |= VM_PROT_EXECUTE;

  copy = ! (flags & MAP_SHARED);
  anywhere = ! (flags & MAP_FIXED);

#ifdef __LP64__
  if ((addr == NULL) && (prot & PROT_EXEC)
      && HAS_ARCH_FEATURE (Prefer_MAP_32BIT_EXEC))
    flags |= MAP_32BIT;
#endif
  mask = (flags & MAP_32BIT) ? ~(vm_address_t) 0x7FFFFFFF : 0;

  switch (flags & MAP_TYPE)
    {
    default:
      return (void *) (long int) __hurd_fail (EINVAL);

    case MAP_ANON:
      memobj = MACH_PORT_NULL;
      max_vmprot = VM_PROT_ALL;
      break;

    case MAP_FILE:
    case 0:			/* Allow, e.g., just MAP_SHARED.  */
      {
	mach_port_t robj, wobj;
	if (err = HURD_DPORT_USE (fd, __io_map (port, &robj, &wobj)))
	  {
	    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:
            max_vmprot = VM_PROT_READ|VM_PROT_EXECUTE;
            if (wobj == robj)
              max_vmprot |= VM_PROT_WRITE;
	    memobj = robj;
	    if (MACH_PORT_VALID (wobj))
	      __mach_port_deallocate (__mach_task_self (), wobj);
	    break;
	  case PROT_WRITE:
            max_vmprot = VM_PROT_WRITE;
            if (robj == wobj)
              max_vmprot |= VM_PROT_READ|VM_PROT_EXECUTE;
	    memobj = wobj;
	    if (MACH_PORT_VALID (robj))
	      __mach_port_deallocate (__mach_task_self (), robj);
	    break;
	  case PROT_READ|PROT_WRITE:
            max_vmprot = VM_PROT_ALL;
	    if (robj == wobj)
	      {
		memobj = wobj;
		/* Remove extra reference.  */
		__mach_port_deallocate (__mach_task_self (), memobj);
	      }
	    else if (wobj == MACH_PORT_NULL /* Not writable by mapping.  */
		     && copy)
	      /* 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.  */
	      memobj = robj;
	    else
	      {
		__mach_port_deallocate (__mach_task_self (), wobj);
		return (void *) (long int) __hurd_fail (EACCES);
	      }
	    break;
	  default:
	    __builtin_unreachable ();
	  }
	break;
	/* XXX handle MAP_NOEXTEND */
      }
    }

  /* XXX handle MAP_INHERIT */

  if (copy)
    max_vmprot = VM_PROT_ALL;

  /* When ANYWHERE is true but the caller has provided a preferred address,
     try mapping at that address with anywhere = 0 first.  If this fails,
     we'll retry with anywhere = 1 below.  */
  err = __vm_map (__mach_task_self (),
		  &mapaddr, (vm_size_t) len, mask,
		  anywhere && (mapaddr == 0),
		  memobj, (vm_offset_t) offset,
		  copy, vmprot, max_vmprot,
		  copy ? VM_INHERIT_COPY : VM_INHERIT_SHARE);

  if (flags & MAP_FIXED)
    {
      if (err == KERN_NO_SPACE)
	{
	  if (flags & MAP_EXCL)
	    err = EEXIST;
	  else
	    {
	      /* The region is already allocated; deallocate it first.  */
	      /* XXX this is not atomic as it is in unix! */
	      err = __vm_deallocate (__mach_task_self (), mapaddr, len);
	      if (! err)
		err = __vm_map (__mach_task_self (),
				&mapaddr, (vm_size_t) len, mask,
				0, memobj, (vm_offset_t) offset,
				copy, vmprot, max_vmprot,
				copy ? VM_INHERIT_COPY : VM_INHERIT_SHARE);
	    }
	}
    }
  else
    {
      /* This mmap call is allowed to allocate anywhere,  */
      if (mapaddr != 0 && (err == KERN_NO_SPACE || err == KERN_INVALID_ADDRESS))
        /* ...but above, we tried allocating at the specific address,
           and failed to.  Now try again, with anywhere = 1 this time.  */
	err = __vm_map (__mach_task_self (),
			&mapaddr, (vm_size_t) len, mask,
			1, memobj, (vm_offset_t) offset,
			copy, vmprot, max_vmprot,
			copy ? VM_INHERIT_COPY : VM_INHERIT_SHARE);
    }

  if (MACH_PORT_VALID (memobj))
    __mach_port_deallocate (__mach_task_self (), memobj);

  if (err == KERN_PROTECTION_FAILURE)
    err = EACCES;

  if (err)
    return (void *) (long int) __hurd_fail (err);

  return (void *) mapaddr;
}

libc_hidden_def (__mmap)
weak_alias (__mmap, mmap)