]> git.saurik.com Git - apple/xnu.git/blame - osfmk/kern/kalloc.c
xnu-2422.1.72.tar.gz
[apple/xnu.git] / osfmk / kern / kalloc.c
CommitLineData
1c79356b 1/*
316670eb 2 * Copyright (c) 2000-2011 Apple Computer, Inc. All rights reserved.
1c79356b 3 *
2d21ac55 4 * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
1c79356b 5 *
2d21ac55
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.
8f6c56a5 14 *
2d21ac55
A
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
8f6c56a5
A
20 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
2d21ac55
A
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.
8f6c56a5 25 *
2d21ac55 26 * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
1c79356b
A
27 */
28/*
29 * @OSF_COPYRIGHT@
30 */
1c79356b
A
31/*
32 * Mach Operating System
33 * Copyright (c) 1991,1990,1989,1988,1987 Carnegie Mellon University
34 * All Rights Reserved.
35 *
36 * Permission to use, copy, modify and distribute this software and its
37 * documentation is hereby granted, provided that both the copyright
38 * notice and this permission notice appear in all copies of the
39 * software, derivative works or modified versions, and any portions
40 * thereof, and that both notices appear in supporting documentation.
41 *
42 * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
43 * CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
44 * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
45 *
46 * Carnegie Mellon requests users of this software to return to
47 *
48 * Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
49 * School of Computer Science
50 * Carnegie Mellon University
51 * Pittsburgh PA 15213-3890
52 *
53 * any improvements or extensions that they make and grant Carnegie Mellon
54 * the rights to redistribute these changes.
55 */
56/*
57 */
58/*
59 * File: kern/kalloc.c
60 * Author: Avadis Tevanian, Jr.
61 * Date: 1985
62 *
63 * General kernel memory allocator. This allocator is designed
64 * to be used by the kernel to manage dynamic memory fast.
65 */
66
67#include <zone_debug.h>
68
69#include <mach/boolean.h>
70#include <mach/machine/vm_types.h>
71#include <mach/vm_param.h>
72#include <kern/misc_protos.h>
73#include <kern/zalloc.h>
74#include <kern/kalloc.h>
75#include <kern/lock.h>
316670eb 76#include <kern/ledger.h>
1c79356b
A
77#include <vm/vm_kern.h>
78#include <vm/vm_object.h>
79#include <vm/vm_map.h>
91447636 80#include <libkern/OSMalloc.h>
1c79356b
A
81
82#ifdef MACH_BSD
83zone_t kalloc_zone(vm_size_t);
84#endif
85
2d21ac55
A
86#define KALLOC_MAP_SIZE_MIN (16 * 1024 * 1024)
87#define KALLOC_MAP_SIZE_MAX (128 * 1024 * 1024)
1c79356b 88vm_map_t kalloc_map;
1c79356b
A
89vm_size_t kalloc_max;
90vm_size_t kalloc_max_prerounded;
0c530ab8 91vm_size_t kalloc_kernmap_size; /* size of kallocs that can come from kernel map */
1c79356b
A
92
93unsigned int kalloc_large_inuse;
94vm_size_t kalloc_large_total;
95vm_size_t kalloc_large_max;
6d2010ae
A
96vm_size_t kalloc_largest_allocated = 0;
97uint64_t kalloc_large_sum;
98
99int kalloc_fake_zone_index = -1; /* index of our fake zone in statistics arrays */
b0d623f7
A
100
101vm_offset_t kalloc_map_min;
102vm_offset_t kalloc_map_max;
1c79356b 103
6d2010ae
A
104#ifdef MUTEX_ZONE
105/*
106 * Diagnostic code to track mutexes separately rather than via the 2^ zones
107 */
108 zone_t lck_mtx_zone;
109#endif
110
111static void
112KALLOC_ZINFO_SALLOC(vm_size_t bytes)
113{
114 thread_t thr = current_thread();
115 task_t task;
116 zinfo_usage_t zinfo;
117
316670eb
A
118 ledger_debit(thr->t_ledger, task_ledgers.tkm_shared, bytes);
119
6d2010ae
A
120 if (kalloc_fake_zone_index != -1 &&
121 (task = thr->task) != NULL && (zinfo = task->tkm_zinfo) != NULL)
122 zinfo[kalloc_fake_zone_index].alloc += bytes;
123}
124
125static void
126KALLOC_ZINFO_SFREE(vm_size_t bytes)
127{
128 thread_t thr = current_thread();
129 task_t task;
130 zinfo_usage_t zinfo;
131
316670eb
A
132 ledger_credit(thr->t_ledger, task_ledgers.tkm_shared, bytes);
133
6d2010ae
A
134 if (kalloc_fake_zone_index != -1 &&
135 (task = thr->task) != NULL && (zinfo = task->tkm_zinfo) != NULL)
136 zinfo[kalloc_fake_zone_index].free += bytes;
137}
138
1c79356b
A
139/*
140 * All allocations of size less than kalloc_max are rounded to the
316670eb 141 * next nearest sized zone. This allocator is built on top of
1c79356b
A
142 * the zone allocator. A zone is created for each potential size
143 * that we are willing to get in small blocks.
144 *
145 * We assume that kalloc_max is not greater than 64K;
1c79356b
A
146 *
147 * Note that kalloc_max is somewhat confusingly named.
148 * It represents the first power of two for which no zone exists.
149 * kalloc_max_prerounded is the smallest allocation size, before
150 * rounding, for which no zone exists.
316670eb
A
151 *
152 * Also if the allocation size is more than kalloc_kernmap_size
153 * then allocate from kernel map rather than kalloc_map.
154 */
155
156#if KALLOC_MINSIZE == 16 && KALLOC_LOG2_MINALIGN == 4
157
158/*
159 * "Legacy" aka "power-of-2" backing zones with 16-byte minimum
160 * size and alignment. Users of this profile would probably
161 * benefit from some tuning.
162 */
163
164#define K_ZONE_SIZES \
165 16, \
166 32, \
167/* 6 */ 64, \
168 128, \
169 256, \
170/* 9 */ 512, \
171 1024, \
172 2048, \
173/* C */ 4096
174
175
176#define K_ZONE_NAMES \
177 "kalloc.16", \
178 "kalloc.32", \
179/* 6 */ "kalloc.64", \
180 "kalloc.128", \
181 "kalloc.256", \
182/* 9 */ "kalloc.512", \
183 "kalloc.1024", \
184 "kalloc.2048", \
185/* C */ "kalloc.4096"
186
187#define K_ZONE_MAXIMA \
188 1024, \
189 4096, \
190/* 6 */ 4096, \
191 4096, \
192 4096, \
193/* 9 */ 1024, \
194 1024, \
195 1024, \
196/* C */ 1024
197
198#elif KALLOC_MINSIZE == 8 && KALLOC_LOG2_MINALIGN == 3
199
200/*
201 * Tweaked for ARM (and x64) in 04/2011
1c79356b
A
202 */
203
316670eb
A
204#define K_ZONE_SIZES \
205/* 3 */ 8, \
206 16, 24, \
207 32, 40, 48, \
208/* 6 */ 64, 88, 112, \
209 128, 192, \
210 256, 384, \
211/* 9 */ 512, 768, \
212 1024, 1536, \
213 2048, 3072, \
214 4096, 6144
215
216#define K_ZONE_NAMES \
217/* 3 */ "kalloc.8", \
218 "kalloc.16", "kalloc.24", \
219 "kalloc.32", "kalloc.40", "kalloc.48", \
220/* 6 */ "kalloc.64", "kalloc.88", "kalloc.112", \
221 "kalloc.128", "kalloc.192", \
222 "kalloc.256", "kalloc.384", \
223/* 9 */ "kalloc.512", "kalloc.768", \
224 "kalloc.1024", "kalloc.1536", \
225 "kalloc.2048", "kalloc.3072", \
226 "kalloc.4096", "kalloc.6144"
227
228#define K_ZONE_MAXIMA \
229/* 3 */ 1024, \
230 1024, 1024, \
231 4096, 4096, 4096, \
232/* 6 */ 4096, 4096, 4096, \
233 4096, 4096, \
234 4096, 4096, \
235/* 9 */ 1024, 1024, \
236 1024, 1024, \
237 1024, 1024, \
238/* C */ 1024, 64
239
240#else
241#error missing zone size parameters for kalloc
242#endif
243
244#define KALLOC_MINALIGN (1 << KALLOC_LOG2_MINALIGN)
245
246static const int k_zone_size[] = {
247 K_ZONE_SIZES,
248 8192,
249 16384,
250/* F */ 32768
251};
252
253#define N_K_ZONE (sizeof (k_zone_size) / sizeof (k_zone_size[0]))
254
255/*
256 * Many kalloc() allocations are for small structures containing a few
257 * pointers and longs - the k_zone_dlut[] direct lookup table, indexed by
258 * size normalized to the minimum alignment, finds the right zone index
259 * for them in one dereference.
260 */
261
262#define INDEX_ZDLUT(size) \
263 (((size) + KALLOC_MINALIGN - 1) / KALLOC_MINALIGN)
264#define N_K_ZDLUT (2048 / KALLOC_MINALIGN)
265 /* covers sizes [0 .. 2048 - KALLOC_MINALIGN] */
266#define MAX_SIZE_ZDLUT ((N_K_ZDLUT - 1) * KALLOC_MINALIGN)
267
268static int8_t k_zone_dlut[N_K_ZDLUT]; /* table of indices into k_zone[] */
269
270/*
271 * If there's no hit in the DLUT, then start searching from k_zindex_start.
272 */
273static int k_zindex_start;
274
275static zone_t k_zone[N_K_ZONE];
276
277static const char *k_zone_name[N_K_ZONE] = {
278 K_ZONE_NAMES,
279 "kalloc.8192",
280 "kalloc.16384",
281/* F */ "kalloc.32768"
1c79356b
A
282};
283
284/*
285 * Max number of elements per zone. zinit rounds things up correctly
286 * Doing things this way permits each zone to have a different maximum size
287 * based on need, rather than just guessing; it also
288 * means its patchable in case you're wrong!
289 */
316670eb
A
290unsigned int k_zone_max[N_K_ZONE] = {
291 K_ZONE_MAXIMA,
292 4096,
293 64,
294/* F */ 64
1c79356b
A
295};
296
316670eb
A
297/* #define KALLOC_DEBUG 1 */
298
91447636
A
299/* forward declarations */
300void * kalloc_canblock(
301 vm_size_t size,
302 boolean_t canblock);
303
304
6d2010ae
A
305lck_grp_t *kalloc_lck_grp;
306lck_mtx_t kalloc_lock;
307
308#define kalloc_spin_lock() lck_mtx_lock_spin(&kalloc_lock)
309#define kalloc_unlock() lck_mtx_unlock(&kalloc_lock)
310
311
91447636
A
312/* OSMalloc local data declarations */
313static
314queue_head_t OSMalloc_tag_list;
315
6d2010ae
A
316lck_grp_t *OSMalloc_tag_lck_grp;
317lck_mtx_t OSMalloc_tag_lock;
318
319#define OSMalloc_tag_spin_lock() lck_mtx_lock_spin(&OSMalloc_tag_lock)
320#define OSMalloc_tag_unlock() lck_mtx_unlock(&OSMalloc_tag_lock)
321
91447636
A
322
323/* OSMalloc forward declarations */
324void OSMalloc_init(void);
325void OSMalloc_Tagref(OSMallocTag tag);
326void OSMalloc_Tagrele(OSMallocTag tag);
327
1c79356b
A
328/*
329 * Initialize the memory allocator. This should be called only
330 * once on a system wide basis (i.e. first processor to get here
331 * does the initialization).
332 *
333 * This initializes all of the zones.
334 */
335
336void
337kalloc_init(
338 void)
339{
340 kern_return_t retval;
341 vm_offset_t min;
2d21ac55 342 vm_size_t size, kalloc_map_size;
1c79356b
A
343 register int i;
344
2d21ac55
A
345 /*
346 * Scale the kalloc_map_size to physical memory size: stay below
b0d623f7 347 * 1/8th the total zone map size, or 128 MB (for a 32-bit kernel).
2d21ac55 348 */
b0d623f7
A
349 kalloc_map_size = (vm_size_t)(sane_size >> 5);
350#if !__LP64__
2d21ac55
A
351 if (kalloc_map_size > KALLOC_MAP_SIZE_MAX)
352 kalloc_map_size = KALLOC_MAP_SIZE_MAX;
b0d623f7 353#endif /* !__LP64__ */
2d21ac55
A
354 if (kalloc_map_size < KALLOC_MAP_SIZE_MIN)
355 kalloc_map_size = KALLOC_MAP_SIZE_MIN;
356
1c79356b 357 retval = kmem_suballoc(kernel_map, &min, kalloc_map_size,
b0d623f7
A
358 FALSE, VM_FLAGS_ANYWHERE | VM_FLAGS_PERMANENT,
359 &kalloc_map);
91447636 360
1c79356b
A
361 if (retval != KERN_SUCCESS)
362 panic("kalloc_init: kmem_suballoc failed");
363
b0d623f7
A
364 kalloc_map_min = min;
365 kalloc_map_max = min + kalloc_map_size - 1;
366
1c79356b
A
367 /*
368 * Ensure that zones up to size 8192 bytes exist.
369 * This is desirable because messages are allocated
370 * with kalloc, and messages up through size 8192 are common.
371 */
372
373 if (PAGE_SIZE < 16*1024)
374 kalloc_max = 16*1024;
375 else
376 kalloc_max = PAGE_SIZE;
377 kalloc_max_prerounded = kalloc_max / 2 + 1;
0c530ab8
A
378 /* size it to be more than 16 times kalloc_max (256k) for allocations from kernel map */
379 kalloc_kernmap_size = (kalloc_max * 16) + 1;
b0d623f7 380 kalloc_largest_allocated = kalloc_kernmap_size;
1c79356b
A
381
382 /*
383 * Allocate a zone for each size we are going to handle.
6d2010ae
A
384 * We specify non-paged memory. Don't charge the caller
385 * for the allocation, as we aren't sure how the memory
386 * will be handled.
1c79356b 387 */
316670eb 388 for (i = 0; (size = k_zone_size[i]) < kalloc_max; i++) {
1c79356b
A
389 k_zone[i] = zinit(size, k_zone_max[i] * size, size,
390 k_zone_name[i]);
6d2010ae 391 zone_change(k_zone[i], Z_CALLERACCT, FALSE);
1c79356b 392 }
316670eb
A
393
394 /*
395 * Build the Direct LookUp Table for small allocations
396 */
397 for (i = 0, size = 0; i <= N_K_ZDLUT; i++, size += KALLOC_MINALIGN) {
398 int zindex = 0;
399
400 while ((vm_size_t)k_zone_size[zindex] < size)
401 zindex++;
402
403 if (i == N_K_ZDLUT) {
404 k_zindex_start = zindex;
405 break;
406 }
407 k_zone_dlut[i] = (int8_t)zindex;
408 }
409
410#ifdef KALLOC_DEBUG
411 printf("kalloc_init: k_zindex_start %d\n", k_zindex_start);
412
413 /*
414 * Do a quick synthesis to see how well/badly we can
415 * find-a-zone for a given size.
416 * Useful when debugging/tweaking the array of zone sizes.
417 * Cache misses probably more critical than compare-branches!
418 */
419 for (i = 0; i < (int)N_K_ZONE; i++) {
420 vm_size_t testsize = (vm_size_t)k_zone_size[i] - 1;
421 int compare = 0;
422 int zindex;
423
424 if (testsize < MAX_SIZE_ZDLUT) {
425 compare += 1; /* 'if' (T) */
426
427 long dindex = INDEX_ZDLUT(testsize);
428 zindex = (int)k_zone_dlut[dindex];
429
430 } else if (testsize < kalloc_max_prerounded) {
431
432 compare += 2; /* 'if' (F), 'if' (T) */
433
434 zindex = k_zindex_start;
435 while ((vm_size_t)k_zone_size[zindex] < testsize) {
436 zindex++;
437 compare++; /* 'while' (T) */
438 }
439 compare++; /* 'while' (F) */
440 } else
441 break; /* not zone-backed */
442
443 zone_t z = k_zone[zindex];
444 printf("kalloc_init: req size %4lu: %11s took %d compare%s\n",
445 (unsigned long)testsize, z->zone_name, compare,
446 compare == 1 ? "" : "s");
447 }
448#endif
6d2010ae
A
449 kalloc_lck_grp = lck_grp_alloc_init("kalloc.large", LCK_GRP_ATTR_NULL);
450 lck_mtx_init(&kalloc_lock, kalloc_lck_grp, LCK_ATTR_NULL);
91447636 451 OSMalloc_init();
6d2010ae
A
452#ifdef MUTEX_ZONE
453 lck_mtx_zone = zinit(sizeof(struct _lck_mtx_), 1024*256, 4096, "lck_mtx");
454#endif
316670eb 455}
6d2010ae 456
316670eb
A
457/*
458 * Given an allocation size, return the kalloc zone it belongs to.
459 * Direct LookUp Table variant.
460 */
461static __inline zone_t
462get_zone_dlut(vm_size_t size)
463{
464 long dindex = INDEX_ZDLUT(size);
465 int zindex = (int)k_zone_dlut[dindex];
466 return (k_zone[zindex]);
467}
468
469/* As above, but linear search k_zone_size[] for the next zone that fits. */
470
471static __inline zone_t
472get_zone_search(vm_size_t size, int zindex)
473{
474 assert(size < kalloc_max_prerounded);
475
476 while ((vm_size_t)k_zone_size[zindex] < size)
477 zindex++;
478
479 assert((unsigned)zindex < N_K_ZONE &&
480 (vm_size_t)k_zone_size[zindex] < kalloc_max);
481
482 return (k_zone[zindex]);
1c79356b
A
483}
484
91447636 485void *
1c79356b
A
486kalloc_canblock(
487 vm_size_t size,
488 boolean_t canblock)
489{
316670eb
A
490 zone_t z;
491
492 if (size < MAX_SIZE_ZDLUT)
493 z = get_zone_dlut(size);
494 else if (size < kalloc_max_prerounded)
495 z = get_zone_search(size, k_zindex_start);
496 else {
497 /*
498 * If size is too large for a zone, then use kmem_alloc.
499 * (We use kmem_alloc instead of kmem_alloc_kobject so that
500 * krealloc can use kmem_realloc.)
501 */
502 vm_map_t alloc_map;
91447636 503 void *addr;
1c79356b
A
504
505 /* kmem_alloc could block so we return if noblock */
506 if (!canblock) {
6d2010ae 507 return(NULL);
1c79356b 508 }
0c530ab8 509
6d2010ae 510 if (size >= kalloc_kernmap_size)
2d21ac55 511 alloc_map = kernel_map;
6d2010ae 512 else
0c530ab8
A
513 alloc_map = kalloc_map;
514
b0d623f7
A
515 if (kmem_alloc(alloc_map, (vm_offset_t *)&addr, size) != KERN_SUCCESS) {
516 if (alloc_map != kernel_map) {
517 if (kmem_alloc(kernel_map, (vm_offset_t *)&addr, size) != KERN_SUCCESS)
518 addr = NULL;
6d2010ae 519 }
b0d623f7
A
520 else
521 addr = NULL;
522 }
1c79356b 523
b0d623f7 524 if (addr != NULL) {
6d2010ae
A
525 kalloc_spin_lock();
526 /*
527 * Thread-safe version of the workaround for 4740071
528 * (a double FREE())
529 */
530 if (size > kalloc_largest_allocated)
531 kalloc_largest_allocated = size;
532
1c79356b
A
533 kalloc_large_inuse++;
534 kalloc_large_total += size;
6d2010ae 535 kalloc_large_sum += size;
1c79356b
A
536
537 if (kalloc_large_total > kalloc_large_max)
538 kalloc_large_max = kalloc_large_total;
6d2010ae
A
539
540 kalloc_unlock();
541
542 KALLOC_ZINFO_SALLOC(size);
1c79356b
A
543 }
544 return(addr);
545 }
316670eb
A
546#ifdef KALLOC_DEBUG
547 if (size > z->elem_size)
548 panic("%s: z %p (%s) but requested size %lu", __func__,
549 z, z->zone_name, (unsigned long)size);
550#endif
551 assert(size <= z->elem_size);
552 return (zalloc_canblock(z, canblock));
1c79356b
A
553}
554
91447636 555void *
1c79356b
A
556kalloc(
557 vm_size_t size)
558{
91447636 559 return( kalloc_canblock(size, TRUE) );
1c79356b
A
560}
561
91447636 562void *
1c79356b
A
563kalloc_noblock(
564 vm_size_t size)
565{
91447636 566 return( kalloc_canblock(size, FALSE) );
1c79356b
A
567}
568
b0d623f7
A
569volatile SInt32 kfree_nop_count = 0;
570
1c79356b
A
571void
572kfree(
91447636 573 void *data,
1c79356b
A
574 vm_size_t size)
575{
316670eb
A
576 zone_t z;
577
578 if (size < MAX_SIZE_ZDLUT)
579 z = get_zone_dlut(size);
580 else if (size < kalloc_max_prerounded)
581 z = get_zone_search(size, k_zindex_start);
582 else {
583 /* if size was too large for a zone, then use kmem_free */
1c79356b 584
316670eb 585 vm_map_t alloc_map = kernel_map;
1c79356b 586
b0d623f7
A
587 if ((((vm_offset_t) data) >= kalloc_map_min) && (((vm_offset_t) data) <= kalloc_map_max))
588 alloc_map = kalloc_map;
589 if (size > kalloc_largest_allocated) {
0c530ab8
A
590 /*
591 * work around double FREEs of small MALLOCs
316670eb 592 * this used to end up being a nop
0c530ab8
A
593 * since the pointer being freed from an
594 * alloc backed by the zalloc world could
595 * never show up in the kalloc_map... however,
596 * the kernel_map is a different issue... since it
597 * was released back into the zalloc pool, a pointer
598 * would have gotten written over the 'size' that
599 * the MALLOC was retaining in the first 4 bytes of
600 * the underlying allocation... that pointer ends up
601 * looking like a really big size on the 2nd FREE and
602 * pushes the kfree into the kernel_map... we
316670eb 603 * end up removing a ton of virtual space before we panic
0c530ab8
A
604 * this check causes us to ignore the kfree for a size
605 * that must be 'bogus'... note that it might not be due
606 * to the above scenario, but it would still be wrong and
607 * cause serious damage.
608 */
b0d623f7
A
609
610 OSAddAtomic(1, &kfree_nop_count);
0c530ab8 611 return;
b0d623f7 612 }
0c530ab8 613 kmem_free(alloc_map, (vm_offset_t)data, size);
1c79356b 614
6d2010ae
A
615 kalloc_spin_lock();
616
1c79356b
A
617 kalloc_large_total -= size;
618 kalloc_large_inuse--;
619
6d2010ae
A
620 kalloc_unlock();
621
622 KALLOC_ZINFO_SFREE(size);
1c79356b
A
623 return;
624 }
625
1c79356b 626 /* free to the appropriate zone */
316670eb
A
627#ifdef KALLOC_DEBUG
628 if (size > z->elem_size)
629 panic("%s: z %p (%s) but requested size %lu", __func__,
630 z, z->zone_name, (unsigned long)size);
631#endif
632 assert(size <= z->elem_size);
633 zfree(z, data);
1c79356b
A
634}
635
636#ifdef MACH_BSD
637zone_t
638kalloc_zone(
639 vm_size_t size)
640{
316670eb
A
641 if (size < MAX_SIZE_ZDLUT)
642 return (get_zone_dlut(size));
643 if (size <= kalloc_max)
644 return (get_zone_search(size, k_zindex_start));
1c79356b
A
645 return (ZONE_NULL);
646}
647#endif
648
6d2010ae
A
649void
650kalloc_fake_zone_init(int zone_index)
651{
652 kalloc_fake_zone_index = zone_index;
653}
1c79356b 654
91447636 655void
6d2010ae
A
656kalloc_fake_zone_info(int *count,
657 vm_size_t *cur_size, vm_size_t *max_size, vm_size_t *elem_size, vm_size_t *alloc_size,
658 uint64_t *sum_size, int *collectable, int *exhaustable, int *caller_acct)
1c79356b 659{
91447636 660 *count = kalloc_large_inuse;
1c79356b
A
661 *cur_size = kalloc_large_total;
662 *max_size = kalloc_large_max;
6d2010ae
A
663
664 if (kalloc_large_inuse) {
665 *elem_size = kalloc_large_total / kalloc_large_inuse;
666 *alloc_size = kalloc_large_total / kalloc_large_inuse;
667 } else {
668 *elem_size = 0;
669 *alloc_size = 0;
670 }
671 *sum_size = kalloc_large_sum;
1c79356b
A
672 *collectable = 0;
673 *exhaustable = 0;
6d2010ae 674 *caller_acct = 0;
1c79356b
A
675}
676
91447636
A
677
678void
679OSMalloc_init(
680 void)
681{
682 queue_init(&OSMalloc_tag_list);
6d2010ae
A
683
684 OSMalloc_tag_lck_grp = lck_grp_alloc_init("OSMalloc_tag", LCK_GRP_ATTR_NULL);
685 lck_mtx_init(&OSMalloc_tag_lock, OSMalloc_tag_lck_grp, LCK_ATTR_NULL);
91447636
A
686}
687
688OSMallocTag
689OSMalloc_Tagalloc(
690 const char *str,
691 uint32_t flags)
692{
693 OSMallocTag OSMTag;
694
695 OSMTag = (OSMallocTag)kalloc(sizeof(*OSMTag));
696
697 bzero((void *)OSMTag, sizeof(*OSMTag));
698
699 if (flags & OSMT_PAGEABLE)
700 OSMTag->OSMT_attr = OSMT_ATTR_PAGEABLE;
701
702 OSMTag->OSMT_refcnt = 1;
703
704 strncpy(OSMTag->OSMT_name, str, OSMT_MAX_NAME);
705
6d2010ae 706 OSMalloc_tag_spin_lock();
91447636 707 enqueue_tail(&OSMalloc_tag_list, (queue_entry_t)OSMTag);
6d2010ae 708 OSMalloc_tag_unlock();
91447636
A
709 OSMTag->OSMT_state = OSMT_VALID;
710 return(OSMTag);
711}
712
713void
714OSMalloc_Tagref(
715 OSMallocTag tag)
716{
717 if (!((tag->OSMT_state & OSMT_VALID_MASK) == OSMT_VALID))
316670eb 718 panic("OSMalloc_Tagref():'%s' has bad state 0x%08X\n", tag->OSMT_name, tag->OSMT_state);
91447636 719
2d21ac55 720 (void)hw_atomic_add(&tag->OSMT_refcnt, 1);
91447636
A
721}
722
723void
724OSMalloc_Tagrele(
725 OSMallocTag tag)
726{
727 if (!((tag->OSMT_state & OSMT_VALID_MASK) == OSMT_VALID))
316670eb 728 panic("OSMalloc_Tagref():'%s' has bad state 0x%08X\n", tag->OSMT_name, tag->OSMT_state);
91447636 729
2d21ac55 730 if (hw_atomic_sub(&tag->OSMT_refcnt, 1) == 0) {
91447636 731 if (hw_compare_and_store(OSMT_VALID|OSMT_RELEASED, OSMT_VALID|OSMT_RELEASED, &tag->OSMT_state)) {
6d2010ae 732 OSMalloc_tag_spin_lock();
91447636 733 (void)remque((queue_entry_t)tag);
6d2010ae 734 OSMalloc_tag_unlock();
91447636
A
735 kfree((void*)tag, sizeof(*tag));
736 } else
316670eb 737 panic("OSMalloc_Tagrele():'%s' has refcnt 0\n", tag->OSMT_name);
91447636
A
738 }
739}
740
741void
742OSMalloc_Tagfree(
743 OSMallocTag tag)
744{
745 if (!hw_compare_and_store(OSMT_VALID, OSMT_VALID|OSMT_RELEASED, &tag->OSMT_state))
316670eb 746 panic("OSMalloc_Tagfree():'%s' has bad state 0x%08X \n", tag->OSMT_name, tag->OSMT_state);
91447636 747
2d21ac55 748 if (hw_atomic_sub(&tag->OSMT_refcnt, 1) == 0) {
6d2010ae 749 OSMalloc_tag_spin_lock();
91447636 750 (void)remque((queue_entry_t)tag);
6d2010ae 751 OSMalloc_tag_unlock();
91447636
A
752 kfree((void*)tag, sizeof(*tag));
753 }
754}
755
756void *
757OSMalloc(
758 uint32_t size,
759 OSMallocTag tag)
760{
761 void *addr=NULL;
762 kern_return_t kr;
763
764 OSMalloc_Tagref(tag);
765 if ((tag->OSMT_attr & OSMT_PAGEABLE)
766 && (size & ~PAGE_MASK)) {
767
768 if ((kr = kmem_alloc_pageable(kernel_map, (vm_offset_t *)&addr, size)) != KERN_SUCCESS)
2d21ac55 769 addr = NULL;
91447636
A
770 } else
771 addr = kalloc((vm_size_t)size);
772
2d21ac55
A
773 if (!addr)
774 OSMalloc_Tagrele(tag);
775
91447636
A
776 return(addr);
777}
778
779void *
780OSMalloc_nowait(
781 uint32_t size,
782 OSMallocTag tag)
783{
784 void *addr=NULL;
785
786 if (tag->OSMT_attr & OSMT_PAGEABLE)
787 return(NULL);
788
789 OSMalloc_Tagref(tag);
790 /* XXX: use non-blocking kalloc for now */
791 addr = kalloc_noblock((vm_size_t)size);
792 if (addr == NULL)
793 OSMalloc_Tagrele(tag);
794
795 return(addr);
796}
797
798void *
799OSMalloc_noblock(
800 uint32_t size,
801 OSMallocTag tag)
802{
803 void *addr=NULL;
804
805 if (tag->OSMT_attr & OSMT_PAGEABLE)
806 return(NULL);
807
808 OSMalloc_Tagref(tag);
809 addr = kalloc_noblock((vm_size_t)size);
810 if (addr == NULL)
811 OSMalloc_Tagrele(tag);
812
813 return(addr);
814}
815
816void
817OSFree(
818 void *addr,
819 uint32_t size,
820 OSMallocTag tag)
821{
822 if ((tag->OSMT_attr & OSMT_PAGEABLE)
823 && (size & ~PAGE_MASK)) {
824 kmem_free(kernel_map, (vm_offset_t)addr, size);
825 } else
316670eb 826 kfree((void *)addr, size);
91447636
A
827
828 OSMalloc_Tagrele(tag);
829}