]> git.saurik.com Git - apple/xnu.git/blame - osfmk/vm/device_vm.c
xnu-792.22.5.tar.gz
[apple/xnu.git] / osfmk / vm / device_vm.c
CommitLineData
0b4e3aa0 1/*
91447636 2 * Copyright (c) 2000-2004 Apple Computer, Inc. All rights reserved.
0b4e3aa0 3 *
8f6c56a5 4 * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
0b4e3aa0 5 *
8f6c56a5
A
6 * This file contains Original Code and/or Modifications of Original Code
7 * as defined in and that are subject to the Apple Public Source License
8 * Version 2.0 (the 'License'). You may not use this file except in
9 * compliance with the License. The rights granted to you under the License
10 * may not be used to create, or enable the creation or redistribution of,
11 * unlawful or unlicensed copies of an Apple operating system, or to
12 * circumvent, violate, or enable the circumvention or violation of, any
13 * terms of an Apple operating system software license agreement.
14 *
15 * Please obtain a copy of the License at
16 * http://www.opensource.apple.com/apsl/ and read it before using this file.
17 *
18 * The Original Code and all software distributed under the License are
19 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23 * Please see the License for the specific language governing rights and
8ad349bb 24 * limitations under the License.
8f6c56a5
A
25 *
26 * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
0b4e3aa0
A
27 */
28
29#include <sys/errno.h>
91447636 30
0b4e3aa0 31#include <mach/mach_types.h>
0b4e3aa0 32#include <mach/kern_return.h>
91447636 33#include <mach/memory_object_control.h>
0b4e3aa0
A
34#include <mach/memory_object_types.h>
35#include <mach/port.h>
36#include <mach/policy.h>
91447636
A
37#include <mach/upl.h>
38#include <kern/kern_types.h>
39#include <kern/ipc_kobject.h>
40#include <kern/host.h>
41#include <kern/thread.h>
0b4e3aa0
A
42#include <ipc/ipc_port.h>
43#include <ipc/ipc_space.h>
0b4e3aa0 44#include <device/device_port.h>
91447636 45#include <vm/memory_object.h>
0b4e3aa0 46#include <vm/vm_pageout.h>
91447636
A
47#include <vm/vm_map.h>
48#include <vm/vm_kern.h>
49#include <vm/vm_pageout.h>
50#include <vm/vm_protos.h>
51
0b4e3aa0 52
0b4e3aa0
A
53/* Device VM COMPONENT INTERFACES */
54
55
56/*
57 * Device PAGER
58 */
59
60
61/* until component support available */
62
63
64
65/* until component support available */
4452a7af
A
66const struct memory_object_pager_ops device_pager_ops = {
67 device_pager_reference,
68 device_pager_deallocate,
69 device_pager_init,
70 device_pager_terminate,
71 device_pager_data_request,
72 device_pager_data_return,
73 device_pager_data_initialize,
74 device_pager_data_unlock,
75 device_pager_synchronize,
76 device_pager_unmap,
77 "device pager"
78};
0b4e3aa0
A
79
80typedef int device_port_t;
81
4452a7af
A
82/*
83 * The start of "struct device_pager" MUST match a "struct memory_object".
84 */
0b4e3aa0 85typedef struct device_pager {
4452a7af 86 memory_object_pager_ops_t pager_ops; /* == &device_pager_ops */
0b4e3aa0
A
87 unsigned int pager_ikot; /* fake ip_kotype() */
88 unsigned int ref_count; /* reference count */
89 memory_object_control_t control_handle; /* mem object's cntrl handle */
90 device_port_t device_handle; /* device_handle */
91 vm_size_t size;
92 int flags;
93} *device_pager_t;
94
95
96
0b4e3aa0
A
97
98device_pager_t
91447636 99device_pager_lookup( /* forward */
0b4e3aa0
A
100 memory_object_t);
101
102device_pager_t
91447636 103device_object_create(void); /* forward */
0b4e3aa0
A
104
105zone_t device_pager_zone;
106
107
108#define DEVICE_PAGER_NULL ((device_pager_t) 0)
109
110
111#define MAX_DNODE 10000
112
113
114
115
116
117/*
118 *
119 */
120void
121device_pager_bootstrap(void)
122{
123 register vm_size_t size;
124
125 size = (vm_size_t) sizeof(struct device_pager);
126 device_pager_zone = zinit(size, (vm_size_t) MAX_DNODE*size,
127 PAGE_SIZE, "device node pager structures");
128
129 return;
130}
131
132/*
133 *
134 */
135memory_object_t
136device_pager_setup(
91447636 137 __unused memory_object_t device,
0b4e3aa0
A
138 int device_handle,
139 vm_size_t size,
140 int flags)
141{
142 device_pager_t device_object;
143
144 device_object = device_object_create();
145 if (device_object == DEVICE_PAGER_NULL)
146 panic("device_pager_setup: device_object_create() failed");
147
148 device_object->device_handle = device_handle;
149 device_object->size = size;
9bccf70c 150 device_object->flags = flags;
0b4e3aa0
A
151
152 return((memory_object_t)device_object);
153}
154
155/*
156 *
157 */
158kern_return_t
159device_pager_populate_object(
160 memory_object_t device,
161 memory_object_offset_t offset,
55e303ae 162 ppnum_t page_num,
0b4e3aa0
A
163 vm_size_t size)
164{
165 device_pager_t device_object;
166 vm_object_t vm_object;
167 kern_return_t kr;
168 upl_t upl;
0b4e3aa0
A
169
170 device_object = device_pager_lookup(device);
171 if(device_object == DEVICE_PAGER_NULL)
172 return KERN_FAILURE;
173
174 vm_object = (vm_object_t)memory_object_control_to_vm_object(
175 device_object->control_handle);
176 if(vm_object == NULL)
177 return KERN_FAILURE;
178
179 kr = vm_object_populate_with_private(
55e303ae 180 vm_object, offset, page_num, size);
0b4e3aa0
A
181 if(kr != KERN_SUCCESS)
182 return kr;
183
184 if(!vm_object->phys_contiguous) {
4452a7af 185 unsigned int null_size = 0;
0b4e3aa0
A
186 kr = vm_object_upl_request(vm_object,
187 (vm_object_offset_t)offset, size, &upl, NULL,
188 &null_size, (UPL_NO_SYNC | UPL_CLEAN_IN_PLACE));
189
190 if(kr != KERN_SUCCESS)
191 panic("device_pager_populate_object: list_req failed");
192
91447636 193 upl_commit(upl, NULL, 0);
0b4e3aa0
A
194 upl_deallocate(upl);
195 }
196
197
198 return kr;
199}
200
201/*
202 *
203 */
204device_pager_t
205device_pager_lookup(
206 memory_object_t name)
207{
208 device_pager_t device_object;
209
210 device_object = (device_pager_t)name;
4452a7af 211 assert(device_object->pager_ops == &device_pager_ops);
0b4e3aa0
A
212 return (device_object);
213}
214
215/*
216 *
217 */
218kern_return_t
91447636
A
219device_pager_init(
220 memory_object_t mem_obj,
221 memory_object_control_t control,
222 __unused vm_size_t pg_size)
0b4e3aa0
A
223{
224 device_pager_t device_object;
225 kern_return_t kr;
226 memory_object_attr_info_data_t attributes;
227
228 vm_object_t vm_object;
229
230
231 if (control == MEMORY_OBJECT_CONTROL_NULL)
232 return KERN_INVALID_ARGUMENT;
233
234 device_object = device_pager_lookup(mem_obj);
235
236 memory_object_control_reference(control);
237 device_object->control_handle = control;
238
239
240/* The following settings should be done through an expanded change */
241/* attributes call */
242
243 vm_object = (vm_object_t)memory_object_control_to_vm_object(control);
244 vm_object_lock(vm_object);
245 vm_object->private = TRUE;
246 if(device_object->flags & DEVICE_PAGER_CONTIGUOUS)
247 vm_object->phys_contiguous = TRUE;
248 if(device_object->flags & DEVICE_PAGER_NOPHYSCACHE)
249 vm_object->nophyscache = TRUE;
9bccf70c
A
250
251 vm_object->wimg_bits = device_object->flags & VM_WIMG_MASK;
0b4e3aa0
A
252 vm_object_unlock(vm_object);
253
254
255 attributes.copy_strategy = MEMORY_OBJECT_COPY_DELAY;
256 /* attributes.cluster_size = (1 << (CLUSTER_SHIFT + PAGE_SHIFT));*/
257 attributes.cluster_size = (1 << (PAGE_SHIFT));
258 attributes.may_cache_object = FALSE;
259 attributes.temporary = TRUE;
260
261 kr = memory_object_change_attributes(
262 control,
263 MEMORY_OBJECT_ATTRIBUTE_INFO,
264 (memory_object_info_t) &attributes,
265 MEMORY_OBJECT_ATTR_INFO_COUNT);
266 if (kr != KERN_SUCCESS)
267 panic("device_pager_init: memory_object_change_attributes() failed");
268
269 return(KERN_SUCCESS);
270}
271
272/*
273 *
274 */
91447636 275/*ARGSUSED6*/
0b4e3aa0
A
276kern_return_t
277device_pager_data_return(
4452a7af
A
278 memory_object_t mem_obj,
279 memory_object_offset_t offset,
280 vm_size_t data_cnt,
281 __unused memory_object_offset_t *resid_offset,
282 __unused int *io_error,
91447636
A
283 __unused boolean_t dirty,
284 __unused boolean_t kernel_copy,
285 __unused int upl_flags)
0b4e3aa0
A
286{
287 device_pager_t device_object;
288
289 device_object = device_pager_lookup(mem_obj);
290 if (device_object == DEVICE_PAGER_NULL)
291 panic("device_pager_data_return: lookup failed");
292
91447636
A
293 return device_data_action(device_object->device_handle,
294 (ipc_port_t) device_object,
295 VM_PROT_READ | VM_PROT_WRITE,
296 offset, data_cnt);
0b4e3aa0
A
297}
298
299/*
300 *
301 */
302kern_return_t
303device_pager_data_request(
304 memory_object_t mem_obj,
305 memory_object_offset_t offset,
306 vm_size_t length,
91447636 307 __unused vm_prot_t protection_required)
0b4e3aa0
A
308{
309 device_pager_t device_object;
310
311 device_object = device_pager_lookup(mem_obj);
312
313 if (device_object == DEVICE_PAGER_NULL)
314 panic("device_pager_data_request: lookup failed");
315
91447636
A
316 device_data_action(device_object->device_handle,
317 (ipc_port_t) device_object,
318 VM_PROT_READ, offset, length);
0b4e3aa0
A
319 return KERN_SUCCESS;
320}
321
322/*
323 *
324 */
325void
326device_pager_reference(
327 memory_object_t mem_obj)
328{
329 device_pager_t device_object;
9bccf70c 330 unsigned int new_ref_count;
0b4e3aa0
A
331
332 device_object = device_pager_lookup(mem_obj);
9bccf70c
A
333 new_ref_count = hw_atomic_add(&device_object->ref_count, 1);
334 assert(new_ref_count > 1);
0b4e3aa0
A
335}
336
337/*
338 *
339 */
340void
341device_pager_deallocate(
342 memory_object_t mem_obj)
343{
91447636
A
344 device_pager_t device_object;
345 memory_object_control_t device_control;
0b4e3aa0
A
346
347 device_object = device_pager_lookup(mem_obj);
348
9bccf70c 349 if (hw_atomic_sub(&device_object->ref_count, 1) == 0) {
0b4e3aa0
A
350 if (device_object->device_handle != (device_port_t) NULL) {
351 device_close(device_object->device_handle);
91447636 352 device_object->device_handle = (device_port_t) NULL;
0b4e3aa0 353 }
91447636
A
354 device_control = device_object->control_handle;
355 if (device_control != MEMORY_OBJECT_CONTROL_NULL) {
356 /*
357 * The VM object should already have been disconnected
358 * from the pager at this point.
359 * We still have to release the "memory object control"
360 * handle.
361 */
4452a7af 362 assert(device_control->moc_object == VM_OBJECT_NULL);
91447636
A
363 memory_object_control_deallocate(device_control);
364 device_object->control_handle =
365 MEMORY_OBJECT_CONTROL_NULL;
366 }
367
368 zfree(device_pager_zone, device_object);
0b4e3aa0
A
369 }
370 return;
371}
372
373kern_return_t
374device_pager_data_initialize(
91447636
A
375 __unused memory_object_t mem_obj,
376 __unused memory_object_offset_t offset,
377 __unused vm_size_t data_cnt)
0b4e3aa0 378{
91447636 379 panic("device_pager_data_initialize");
0b4e3aa0
A
380 return KERN_FAILURE;
381}
382
383kern_return_t
384device_pager_data_unlock(
91447636
A
385 __unused memory_object_t mem_obj,
386 __unused memory_object_offset_t offset,
387 __unused vm_size_t size,
388 __unused vm_prot_t desired_access)
0b4e3aa0
A
389{
390 return KERN_FAILURE;
391}
392
91447636 393kern_return_t
0b4e3aa0 394device_pager_terminate(
91447636 395 __unused memory_object_t mem_obj)
0b4e3aa0
A
396{
397 return KERN_SUCCESS;
398}
399
400
401
402/*
403 *
404 */
405kern_return_t
406device_pager_synchronize(
407 memory_object_t mem_obj,
408 memory_object_offset_t offset,
409 vm_offset_t length,
91447636 410 __unused vm_sync_t sync_flags)
0b4e3aa0
A
411{
412 device_pager_t device_object;
413
414 device_object = device_pager_lookup(mem_obj);
415
416 memory_object_synchronize_completed(
417 device_object->control_handle, offset, length);
418
419 return KERN_SUCCESS;
420}
421
422/*
423 *
424 */
425kern_return_t
426device_pager_unmap(
91447636 427 __unused memory_object_t mem_obj)
0b4e3aa0
A
428{
429 return KERN_SUCCESS;
430}
431
432
433
434/*
435 *
436 */
437device_pager_t
438device_object_create()
439{
440 register device_pager_t device_object;
441
442 device_object = (struct device_pager *) zalloc(device_pager_zone);
443 if (device_object == DEVICE_PAGER_NULL)
444 return(DEVICE_PAGER_NULL);
4452a7af 445 device_object->pager_ops = &device_pager_ops;
0b4e3aa0
A
446 device_object->pager_ikot = IKOT_MEMORY_OBJECT;
447 device_object->ref_count = 1;
448 device_object->control_handle = MEMORY_OBJECT_CONTROL_NULL;
449
450
451 return(device_object);
452}
453