]> git.saurik.com Git - apple/xnu.git/blob - bsd/kern/kern_symfile.c
xnu-4570.41.2.tar.gz
[apple/xnu.git] / bsd / kern / kern_symfile.c
1 /*
2 * Copyright (c) 2000-2006 Apple Computer, Inc. All rights reserved.
3 *
4 * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
5 *
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
24 * limitations under the License.
25 *
26 * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
27 */
28 /* Copyright (c) 1998 Apple Computer, Inc. All rights reserved.
29 *
30 * File: bsd/kern/kern_symfile.c
31 *
32 * HISTORY
33 */
34
35 #include <mach/vm_param.h>
36
37 #include <sys/param.h>
38 #include <sys/systm.h>
39 #include <sys/signalvar.h>
40 #include <sys/resourcevar.h>
41 #include <sys/namei.h>
42 #include <sys/vnode_internal.h>
43 #include <sys/proc_internal.h>
44 #include <sys/kauth.h>
45 #include <sys/timeb.h>
46 #include <sys/times.h>
47 #include <sys/acct.h>
48 #include <sys/file_internal.h>
49 #include <sys/uio.h>
50 #include <sys/kernel.h>
51 #include <sys/stat.h>
52 #include <sys/disk.h>
53 #include <sys/conf.h>
54 #include <sys/content_protection.h>
55
56 #include <mach-o/loader.h>
57 #include <mach-o/nlist.h>
58
59 #include <kern/kalloc.h>
60 #include <vm/vm_kern.h>
61 #include <pexpert/pexpert.h>
62 #include <IOKit/IOPolledInterface.h>
63
64 /* This function is called from kern_sysctl in the current process context;
65 * it is exported with the System6.0.exports, but this appears to be a legacy
66 * export, as there are no internal consumers.
67 */
68 int
69 get_kernel_symfile(__unused proc_t p, __unused char const **symfile);
70 int
71 get_kernel_symfile(__unused proc_t p, __unused char const **symfile)
72 {
73 return KERN_FAILURE;
74 }
75
76 struct kern_direct_file_io_ref_t
77 {
78 vfs_context_t ctx;
79 struct vnode * vp;
80 dev_t device;
81 uint32_t blksize;
82 off_t filelength;
83 char cf;
84 char pinned;
85 };
86
87
88 static int file_ioctl(void * p1, void * p2, u_long theIoctl, caddr_t result)
89 {
90 dev_t device = *(dev_t*) p1;
91
92 return ((*bdevsw[major(device)].d_ioctl)
93 (device, theIoctl, result, S_IFBLK, p2));
94 }
95
96 static int device_ioctl(void * p1, __unused void * p2, u_long theIoctl, caddr_t result)
97 {
98 return (VNOP_IOCTL(p1, theIoctl, result, 0, p2));
99 }
100
101 static int
102 kern_ioctl_file_extents(struct kern_direct_file_io_ref_t * ref, u_long theIoctl, off_t offset, off_t end)
103 {
104 int error = 0;
105 int (*do_ioctl)(void * p1, void * p2, u_long theIoctl, caddr_t result);
106 void * p1;
107 void * p2;
108 uint64_t fileblk;
109 size_t filechunk;
110 dk_extent_t extent;
111 dk_unmap_t unmap;
112 _dk_cs_pin_t pin;
113
114 bzero(&extent, sizeof(dk_extent_t));
115 bzero(&unmap, sizeof(dk_unmap_t));
116 bzero(&pin, sizeof(pin));
117 if (ref->vp->v_type == VREG)
118 {
119 p1 = &ref->device;
120 p2 = kernproc;
121 do_ioctl = &file_ioctl;
122 }
123 else
124 {
125 /* Partition. */
126 p1 = ref->vp;
127 p2 = ref->ctx;
128 do_ioctl = &device_ioctl;
129 }
130
131 if (_DKIOCCSPINEXTENT == theIoctl) {
132 /* Tell CS the image size, so it knows whether to place the subsequent pins SSD/HDD */
133 pin.cp_extent.length = end;
134 pin.cp_flags = _DKIOCCSHIBERNATEIMGSIZE;
135 (void) do_ioctl(p1, p2, _DKIOCCSPINEXTENT, (caddr_t)&pin);
136 } else if (_DKIOCCSUNPINEXTENT == theIoctl) {
137 /* Tell CS hibernation is done, so it can stop blocking overlapping writes */
138 pin.cp_flags = _DKIOCCSPINDISCARDBLACKLIST;
139 (void) do_ioctl(p1, p2, _DKIOCCSUNPINEXTENT, (caddr_t)&pin);
140 }
141
142 for (; offset < end; offset += filechunk)
143 {
144 if (ref->vp->v_type == VREG)
145 {
146 daddr64_t blkno;
147 filechunk = 1*1024*1024*1024;
148 if (filechunk > (size_t)(end - offset))
149 filechunk = (size_t)(end - offset);
150 error = VNOP_BLOCKMAP(ref->vp, offset, filechunk, &blkno,
151 &filechunk, NULL, VNODE_WRITE | VNODE_BLOCKMAP_NO_TRACK, NULL);
152 if (error) break;
153 if (-1LL == blkno) continue;
154 fileblk = blkno * ref->blksize;
155 }
156 else if ((ref->vp->v_type == VBLK) || (ref->vp->v_type == VCHR))
157 {
158 fileblk = offset;
159 filechunk = ref->filelength;
160 }
161
162 if (DKIOCUNMAP == theIoctl)
163 {
164 extent.offset = fileblk;
165 extent.length = filechunk;
166 unmap.extents = &extent;
167 unmap.extentsCount = 1;
168 error = do_ioctl(p1, p2, theIoctl, (caddr_t)&unmap);
169 // printf("DKIOCUNMAP(%d) 0x%qx, 0x%qx\n", error, extent.offset, extent.length);
170 }
171 else if (_DKIOCCSPINEXTENT == theIoctl)
172 {
173 pin.cp_extent.offset = fileblk;
174 pin.cp_extent.length = filechunk;
175 pin.cp_flags = _DKIOCCSPINFORHIBERNATION;
176 error = do_ioctl(p1, p2, theIoctl, (caddr_t)&pin);
177 if (error && (ENOTTY != error))
178 {
179 printf("_DKIOCCSPINEXTENT(%d) 0x%qx, 0x%qx\n", error, pin.cp_extent.offset, pin.cp_extent.length);
180 }
181 }
182 else if (_DKIOCCSUNPINEXTENT == theIoctl)
183 {
184 pin.cp_extent.offset = fileblk;
185 pin.cp_extent.length = filechunk;
186 pin.cp_flags = _DKIOCCSPINFORHIBERNATION;
187 error = do_ioctl(p1, p2, theIoctl, (caddr_t)&pin);
188 if (error && (ENOTTY != error))
189 {
190 printf("_DKIOCCSUNPINEXTENT(%d) 0x%qx, 0x%qx\n", error, pin.cp_extent.offset, pin.cp_extent.length);
191 }
192 }
193 else error = EINVAL;
194
195 if (error) break;
196 }
197 return (error);
198 }
199
200 extern uint32_t freespace_mb(vnode_t vp);
201
202 struct kern_direct_file_io_ref_t *
203 kern_open_file_for_direct_io(const char * name,
204 boolean_t create_file,
205 kern_get_file_extents_callback_t callback,
206 void * callback_ref,
207 off_t set_file_size,
208 off_t fs_free_size,
209 off_t write_file_offset,
210 void * write_file_addr,
211 size_t write_file_len,
212 dev_t * partition_device_result,
213 dev_t * image_device_result,
214 uint64_t * partitionbase_result,
215 uint64_t * maxiocount_result,
216 uint32_t * oflags)
217 {
218 struct kern_direct_file_io_ref_t * ref;
219
220 proc_t p;
221 struct vnode_attr va;
222 int error;
223 off_t f_offset;
224 uint64_t fileblk;
225 size_t filechunk;
226 uint64_t physoffset;
227 dev_t device;
228 dev_t target = 0;
229 int isssd = 0;
230 uint32_t flags = 0;
231 uint32_t blksize;
232 off_t maxiocount, count, segcount;
233 boolean_t locked = FALSE;
234 int fmode, cmode;
235 struct nameidata nd;
236 u_int32_t ndflags;
237 off_t mpFree;
238
239 int (*do_ioctl)(void * p1, void * p2, u_long theIoctl, caddr_t result);
240 void * p1 = NULL;
241 void * p2 = NULL;
242
243 error = EFAULT;
244
245 ref = (struct kern_direct_file_io_ref_t *) kalloc(sizeof(struct kern_direct_file_io_ref_t));
246 if (!ref)
247 {
248 error = EFAULT;
249 goto out;
250 }
251
252 bzero(ref, sizeof(*ref));
253 p = kernproc;
254 ref->ctx = vfs_context_kernel();
255
256 fmode = (create_file) ? (O_CREAT | FWRITE) : FWRITE;
257 cmode = S_IRUSR | S_IWUSR;
258 ndflags = NOFOLLOW;
259 NDINIT(&nd, LOOKUP, OP_OPEN, ndflags, UIO_SYSSPACE, CAST_USER_ADDR_T(name), ref->ctx);
260 VATTR_INIT(&va);
261 VATTR_SET(&va, va_mode, cmode);
262 VATTR_SET(&va, va_dataprotect_flags, VA_DP_RAWENCRYPTED);
263 VATTR_SET(&va, va_dataprotect_class, PROTECTION_CLASS_D);
264 if ((error = vn_open_auth(&nd, &fmode, &va))) {
265 kprintf("vn_open_auth(fmode: %d, cmode: %d) failed with error: %d\n", fmode, cmode, error);
266 goto out;
267 }
268
269 ref->vp = nd.ni_vp;
270 if (ref->vp->v_type == VREG)
271 {
272 vnode_lock_spin(ref->vp);
273 SET(ref->vp->v_flag, VSWAP);
274 vnode_unlock(ref->vp);
275 }
276
277 if (write_file_addr && write_file_len)
278 {
279 if ((error = kern_write_file(ref, write_file_offset, write_file_addr, write_file_len, IO_SKIP_ENCRYPTION))) {
280 kprintf("kern_write_file() failed with error: %d\n", error);
281 goto out;
282 }
283 }
284
285 VATTR_INIT(&va);
286 VATTR_WANTED(&va, va_rdev);
287 VATTR_WANTED(&va, va_fsid);
288 VATTR_WANTED(&va, va_devid);
289 VATTR_WANTED(&va, va_data_size);
290 VATTR_WANTED(&va, va_data_alloc);
291 VATTR_WANTED(&va, va_nlink);
292 error = EFAULT;
293 if (vnode_getattr(ref->vp, &va, ref->ctx)) goto out;
294
295 mpFree = freespace_mb(ref->vp);
296 mpFree <<= 20;
297 kprintf("kern_direct_file(%s): vp size %qd, alloc %qd, mp free %qd, keep free %qd\n",
298 name, va.va_data_size, va.va_data_alloc, mpFree, fs_free_size);
299
300 if (ref->vp->v_type == VREG)
301 {
302 /* Don't dump files with links. */
303 if (va.va_nlink != 1) goto out;
304
305 device = (VATTR_IS_SUPPORTED(&va, va_devid)) ? va.va_devid : va.va_fsid;
306 ref->filelength = va.va_data_size;
307
308 p1 = &device;
309 p2 = p;
310 do_ioctl = &file_ioctl;
311
312 if (set_file_size)
313 {
314 if (fs_free_size)
315 {
316 mpFree += va.va_data_alloc;
317 if ((mpFree < set_file_size) || ((mpFree - set_file_size) < fs_free_size))
318 {
319 error = ENOSPC;
320 goto out;
321 }
322 }
323 error = vnode_setsize(ref->vp, set_file_size, IO_NOZEROFILL | IO_NOAUTH, ref->ctx);
324 if (error) goto out;
325 ref->filelength = set_file_size;
326 }
327 }
328 else if ((ref->vp->v_type == VBLK) || (ref->vp->v_type == VCHR))
329 {
330 /* Partition. */
331 device = va.va_rdev;
332
333 p1 = ref->vp;
334 p2 = ref->ctx;
335 do_ioctl = &device_ioctl;
336 }
337 else
338 {
339 /* Don't dump to non-regular files. */
340 error = EFAULT;
341 goto out;
342 }
343 ref->device = device;
344
345 // probe for CF
346 dk_corestorage_info_t cs_info;
347 memset(&cs_info, 0, sizeof(dk_corestorage_info_t));
348 error = do_ioctl(p1, p2, DKIOCCORESTORAGE, (caddr_t)&cs_info);
349 ref->cf = (error == 0) && (cs_info.flags & DK_CORESTORAGE_ENABLE_HOTFILES);
350
351 // get block size
352
353 error = do_ioctl(p1, p2, DKIOCGETBLOCKSIZE, (caddr_t) &ref->blksize);
354 if (error)
355 goto out;
356
357 if (ref->vp->v_type != VREG)
358 {
359 error = do_ioctl(p1, p2, DKIOCGETBLOCKCOUNT, (caddr_t) &fileblk);
360 if (error) goto out;
361 ref->filelength = fileblk * ref->blksize;
362 }
363
364 // pin logical extents
365
366 error = kern_ioctl_file_extents(ref, _DKIOCCSPINEXTENT, 0, ref->filelength);
367 if (error && (ENOTTY != error)) goto out;
368 ref->pinned = (error == 0);
369
370 // generate the block list
371
372 error = do_ioctl(p1, p2, DKIOCLOCKPHYSICALEXTENTS, NULL);
373 if (error) goto out;
374 locked = TRUE;
375
376 f_offset = 0;
377 for (; f_offset < ref->filelength; f_offset += filechunk)
378 {
379 if (ref->vp->v_type == VREG)
380 {
381 filechunk = 1*1024*1024*1024;
382 daddr64_t blkno;
383
384 error = VNOP_BLOCKMAP(ref->vp, f_offset, filechunk, &blkno,
385 &filechunk, NULL, VNODE_WRITE | VNODE_BLOCKMAP_NO_TRACK, NULL);
386 if (error) goto out;
387 if (-1LL == blkno) continue;
388 fileblk = blkno * ref->blksize;
389 }
390 else if ((ref->vp->v_type == VBLK) || (ref->vp->v_type == VCHR))
391 {
392 fileblk = f_offset;
393 filechunk = f_offset ? 0 : ref->filelength;
394 }
395
396 physoffset = 0;
397 while (physoffset < filechunk)
398 {
399 dk_physical_extent_t getphysreq;
400 bzero(&getphysreq, sizeof(getphysreq));
401
402 getphysreq.offset = fileblk + physoffset;
403 getphysreq.length = (filechunk - physoffset);
404 error = do_ioctl(p1, p2, DKIOCGETPHYSICALEXTENT, (caddr_t) &getphysreq);
405 if (error) goto out;
406 if (!target)
407 {
408 target = getphysreq.dev;
409 }
410 else if (target != getphysreq.dev)
411 {
412 error = ENOTSUP;
413 goto out;
414 }
415 #if HIBFRAGMENT
416 uint64_t rev;
417 for (rev = 4096; rev <= getphysreq.length; rev += 4096)
418 {
419 callback(callback_ref, getphysreq.offset + getphysreq.length - rev, 4096);
420 }
421 #else
422 callback(callback_ref, getphysreq.offset, getphysreq.length);
423 #endif
424 physoffset += getphysreq.length;
425 }
426 }
427 callback(callback_ref, 0ULL, 0ULL);
428
429 if (ref->vp->v_type == VREG) p1 = &target;
430 else
431 {
432 p1 = &target;
433 p2 = p;
434 do_ioctl = &file_ioctl;
435 }
436
437 // get partition base
438
439 if (partitionbase_result)
440 {
441 error = do_ioctl(p1, p2, DKIOCGETBASE, (caddr_t) partitionbase_result);
442 if (error)
443 goto out;
444 }
445
446 // get block size & constraints
447
448 error = do_ioctl(p1, p2, DKIOCGETBLOCKSIZE, (caddr_t) &blksize);
449 if (error)
450 goto out;
451
452 maxiocount = 1*1024*1024*1024;
453
454 error = do_ioctl(p1, p2, DKIOCGETMAXBLOCKCOUNTREAD, (caddr_t) &count);
455 if (error)
456 count = 0;
457 count *= blksize;
458 if (count && (count < maxiocount))
459 maxiocount = count;
460
461 error = do_ioctl(p1, p2, DKIOCGETMAXBLOCKCOUNTWRITE, (caddr_t) &count);
462 if (error)
463 count = 0;
464 count *= blksize;
465 if (count && (count < maxiocount))
466 maxiocount = count;
467
468 error = do_ioctl(p1, p2, DKIOCGETMAXBYTECOUNTREAD, (caddr_t) &count);
469 if (error)
470 count = 0;
471 if (count && (count < maxiocount))
472 maxiocount = count;
473
474 error = do_ioctl(p1, p2, DKIOCGETMAXBYTECOUNTWRITE, (caddr_t) &count);
475 if (error)
476 count = 0;
477 if (count && (count < maxiocount))
478 maxiocount = count;
479
480 error = do_ioctl(p1, p2, DKIOCGETMAXSEGMENTBYTECOUNTREAD, (caddr_t) &count);
481 if (!error)
482 error = do_ioctl(p1, p2, DKIOCGETMAXSEGMENTCOUNTREAD, (caddr_t) &segcount);
483 if (error)
484 count = segcount = 0;
485 count *= segcount;
486 if (count && (count < maxiocount))
487 maxiocount = count;
488
489 error = do_ioctl(p1, p2, DKIOCGETMAXSEGMENTBYTECOUNTWRITE, (caddr_t) &count);
490 if (!error)
491 error = do_ioctl(p1, p2, DKIOCGETMAXSEGMENTCOUNTWRITE, (caddr_t) &segcount);
492 if (error)
493 count = segcount = 0;
494 count *= segcount;
495 if (count && (count < maxiocount))
496 maxiocount = count;
497
498 kprintf("max io 0x%qx bytes\n", maxiocount);
499 if (maxiocount_result)
500 *maxiocount_result = maxiocount;
501
502 error = do_ioctl(p1, p2, DKIOCISSOLIDSTATE, (caddr_t)&isssd);
503 if (!error && isssd)
504 flags |= kIOPolledFileSSD;
505
506 if (partition_device_result)
507 *partition_device_result = device;
508 if (image_device_result)
509 *image_device_result = target;
510 if (oflags)
511 *oflags = flags;
512
513 if ((ref->vp->v_type == VBLK) || (ref->vp->v_type == VCHR))
514 {
515 vnode_close(ref->vp, FWRITE, ref->ctx);
516 ref->vp = NULLVP;
517 ref->ctx = NULL;
518 }
519
520 out:
521 printf("kern_open_file_for_direct_io(%p, %d)\n", ref, error);
522
523
524 if (error && locked)
525 {
526 p1 = &device;
527 (void) do_ioctl(p1, p2, DKIOCUNLOCKPHYSICALEXTENTS, NULL);
528 }
529
530 if (error && ref)
531 {
532 if (ref->vp)
533 {
534 (void) kern_ioctl_file_extents(ref, _DKIOCCSUNPINEXTENT, 0, (ref->pinned && ref->cf) ? ref->filelength : 0);
535 vnode_close(ref->vp, FWRITE, ref->ctx);
536 ref->vp = NULLVP;
537 }
538 ref->ctx = NULL;
539 kfree(ref, sizeof(struct kern_direct_file_io_ref_t));
540 ref = NULL;
541 }
542
543 return(ref);
544 }
545
546 int
547 kern_write_file(struct kern_direct_file_io_ref_t * ref, off_t offset, void * addr, size_t len, int ioflag)
548 {
549 return (vn_rdwr(UIO_WRITE, ref->vp,
550 addr, len, offset,
551 UIO_SYSSPACE, ioflag|IO_SYNC|IO_NODELOCKED|IO_UNIT,
552 vfs_context_ucred(ref->ctx), (int *) 0,
553 vfs_context_proc(ref->ctx)));
554 }
555
556 int
557 kern_read_file(struct kern_direct_file_io_ref_t * ref, off_t offset, void * addr, size_t len, int ioflag)
558 {
559 return (vn_rdwr(UIO_READ, ref->vp,
560 addr, len, offset,
561 UIO_SYSSPACE, ioflag|IO_SYNC|IO_NODELOCKED|IO_UNIT,
562 vfs_context_ucred(ref->ctx), (int *) 0,
563 vfs_context_proc(ref->ctx)));
564 }
565
566
567 struct mount *
568 kern_file_mount(struct kern_direct_file_io_ref_t * ref)
569 {
570 return (ref->vp->v_mount);
571 }
572
573 void
574 kern_close_file_for_direct_io(struct kern_direct_file_io_ref_t * ref,
575 off_t write_offset, void * addr, size_t write_length,
576 off_t discard_offset, off_t discard_end)
577 {
578 int error;
579 printf("kern_close_file_for_direct_io(%p)\n", ref);
580
581 if (!ref) return;
582
583 if (ref->vp)
584 {
585 int (*do_ioctl)(void * p1, void * p2, u_long theIoctl, caddr_t result);
586 void * p1;
587 void * p2;
588
589 if (ref->vp->v_type == VREG)
590 {
591 p1 = &ref->device;
592 p2 = kernproc;
593 do_ioctl = &file_ioctl;
594 }
595 else
596 {
597 /* Partition. */
598 p1 = ref->vp;
599 p2 = ref->ctx;
600 do_ioctl = &device_ioctl;
601 }
602 (void) do_ioctl(p1, p2, DKIOCUNLOCKPHYSICALEXTENTS, NULL);
603
604 //XXX If unmapping extents then don't also need to unpin; except ...
605 //XXX if file unaligned (HFS 4k / Fusion 128k) then pin is superset and
606 //XXX unmap is subset, so save extra walk over file extents (and the risk
607 //XXX that CF drain starts) vs leaving partial units pinned to SSD
608 //XXX (until whatever was sharing also unmaps). Err on cleaning up fully.
609 boolean_t will_unmap = (!ref->pinned || ref->cf) && (discard_end > discard_offset);
610 boolean_t will_unpin = (ref->pinned && ref->cf /* && !will_unmap */);
611
612 (void) kern_ioctl_file_extents(ref, _DKIOCCSUNPINEXTENT, 0, (will_unpin) ? ref->filelength : 0);
613
614 if (will_unmap)
615 {
616 (void) kern_ioctl_file_extents(ref, DKIOCUNMAP, discard_offset, (ref->cf) ? ref->filelength : discard_end);
617 }
618
619 if (addr && write_length)
620 {
621 (void) kern_write_file(ref, write_offset, addr, write_length, IO_SKIP_ENCRYPTION);
622 }
623
624 error = vnode_close(ref->vp, FWRITE, ref->ctx);
625
626 ref->vp = NULLVP;
627 kprintf("vnode_close(%d)\n", error);
628
629 }
630
631 ref->ctx = NULL;
632
633 kfree(ref, sizeof(struct kern_direct_file_io_ref_t));
634 }