/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License, Version 1.0 only * (the "License"). You may not use this file except in compliance * with the License. * * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE * or http://www.opensolaris.org/os/licensing. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at usr/src/OPENSOLARIS.LICENSE. * If applicable, add the following below this CDDL HEADER, with the * fields enclosed by brackets "[]" replaced with your own identifying * information: Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END */ /* * Copyright 2005 Sun Microsystems, Inc. All rights reserved. * Use is subject to license terms. */ #pragma ident "%Z%%M% %I% %E% SMI" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include u_longlong_t spec_hole_start = 0x80000000000ull; u_longlong_t spec_hole_end = 0xfffff80000000000ull; /* * Count the number of available pages and the number of * chunks in the list of available memory. */ void size_physavail( u_longlong_t *physavail, size_t nelems, pgcnt_t *npages, int *memblocks) { size_t i; *npages = 0; *memblocks = 0; for (i = 0; i < nelems; i += 2) { *npages += (pgcnt_t)(physavail[i+1] >> PAGESHIFT); (*memblocks)++; } } pgcnt_t size_virtalloc(u_longlong_t *avail, size_t nelems) { u_longlong_t start, end; pgcnt_t allocpages = 0; uint_t hole_allocated = 0; uint_t i; for (i = 0; i < (nelems - 2); i += 2) { start = avail[i] + avail[i + 1]; end = avail[i + 2]; /* * Notes: * * (1) OBP on platforms with US I/II pre-allocates the hole * represented by [spec_hole_start, spec_hole_end); * pre-allocation is done to make this range unavailable * for any allocation. * * (2) OBP on starcat always pre-allocates the hole similar to * platforms with US I/II. * * (3) OBP on serengeti does _not_ pre-allocate the hole. * * (4) OBP ignores Spitfire Errata #21; i.e. it does _not_ * fill up or pre-allocate an additional 4GB on both sides * of the hole. * * (5) kernel virtual range [spec_hole_start, spec_hole_end) * is _not_ used on any platform including those with * UltraSPARC III where there is no hole. * * Algorithm: * * Check if range [spec_hole_start, spec_hole_end) is * pre-allocated by OBP; if so, subtract that range from * allocpages. */ if (end >= spec_hole_end && start <= spec_hole_start) hole_allocated = 1; allocpages += btopr(end - start); } if (hole_allocated) allocpages -= btop(spec_hole_end - spec_hole_start); return (allocpages); } /* * Returns the max contiguous physical memory present in the * memlist "physavail". */ uint64_t get_max_phys_size( struct memlist *physavail) { uint64_t max_size = 0; for (; physavail; physavail = physavail->next) { if (physavail->size > max_size) max_size = physavail->size; } return (max_size); } /* * Copy boot's physavail list deducting memory at "start" * for "size" bytes. */ int copy_physavail( u_longlong_t *src, size_t nelems, struct memlist **dstp, uint_t start, uint_t size) { struct memlist *dst, *prev; uint_t end1; int deducted = 0; size_t i; dst = *dstp; prev = dst; end1 = start + size; for (i = 0; i < nelems; i += 2) { uint64_t addr, lsize, end2; addr = src[i]; lsize = src[i+1]; end2 = addr + lsize; if ((size != 0) && start >= addr && end1 <= end2) { /* deducted range in this chunk */ deducted = 1; if (start == addr) { /* abuts start of chunk */ if (end1 == end2) /* is equal to the chunk */ continue; dst->address = end1; dst->size = lsize - size; } else if (end1 == end2) { /* abuts end of chunk */ dst->address = addr; dst->size = lsize - size; } else { /* in the middle of the chunk */ dst->address = addr; dst->size = start - addr; dst->next = 0; if (prev == dst) { dst->prev = 0; dst++; } else { dst->prev = prev; prev->next = dst; dst++; prev++; } dst->address = end1; dst->size = end2 - end1; } dst->next = 0; if (prev == dst) { dst->prev = 0; dst++; } else { dst->prev = prev; prev->next = dst; dst++; prev++; } } else { dst->address = src[i]; dst->size = src[i+1]; dst->next = 0; if (prev == dst) { dst->prev = 0; dst++; } else { dst->prev = prev; prev->next = dst; dst++; prev++; } } } *dstp = dst; return (deducted); } struct vnode prom_ppages; /* * Find the pages allocated by the prom by diffing the original * phys_avail list and the current list. In the difference, the * pages not locked belong to the PROM. (The kernel has already locked * and removed all the pages it has allocated from the freelist, this * routine removes the remaining "free" pages that really belong to the * PROM and hashs them in on the 'prom_pages' vnode.) */ void fix_prom_pages(struct memlist *orig, struct memlist *new) { struct memlist *list, *nlist; extern int kcage_on; nlist = new; for (list = orig; list; list = list->next) { uint64_t pa, end; pfn_t pfnum; page_t *pp; if (list->address == nlist->address && list->size == nlist->size) { nlist = nlist->next ? nlist->next : nlist; continue; } /* * Loop through the old list looking to * see if each page is still in the new one. * If a page is not in the new list then we * check to see if it locked permanently. * If so, the kernel allocated and owns it. * If not, then the prom must own it. We * remove any pages found to owned by the prom * from the freelist. */ end = list->address + list->size; for (pa = list->address; pa < end; pa += PAGESIZE) { if (address_in_memlist(new, pa, PAGESIZE)) continue; pfnum = (pfn_t)(pa >> PAGESHIFT); if ((pp = page_numtopp_nolock(pfnum)) == NULL) cmn_err(CE_PANIC, "missing pfnum %lx", pfnum); /* * must break up any large pages that may have * constituent pages being utilized for * BOP_ALLOC()'s. page_reclaim() can't handle * large pages. */ if (pp->p_szc != 0) page_boot_demote(pp); if (!PAGE_LOCKED(pp) && pp->p_lckcnt == 0) { /* * Ahhh yes, a prom page, * suck it off the freelist, * lock it, and hashin on prom_pages vp. */ if (page_trylock(pp, SE_EXCL) == 0) cmn_err(CE_PANIC, "prom page locked"); (void) page_reclaim(pp, NULL); /* * XXX vnode offsets on the prom_ppages vnode * are page numbers (gack) for >32 bit * physical memory machines. */ (void) page_hashin(pp, &prom_ppages, (offset_t)pfnum, NULL); if (kcage_on) { ASSERT(pp->p_szc == 0); PP_SETNORELOC(pp); } (void) page_pp_lock(pp, 0, 1); page_downgrade(pp); } } nlist = nlist->next ? nlist->next : nlist; } } /* * Find the page number of the highest installed physical * page and the number of pages installed (one cannot be * calculated from the other because memory isn't necessarily * contiguous). */ void installed_top_size_memlist_array( u_longlong_t *list, /* base of array */ size_t nelems, /* number of elements */ pfn_t *topp, /* return ptr for top value */ pgcnt_t *sumpagesp) /* return prt for sum of installed pages */ { pfn_t top = 0; pgcnt_t sumpages = 0; pfn_t highp; /* high page in a chunk */ size_t i; for (i = 0; i < nelems; i += 2) { highp = (list[i] + list[i+1] - 1) >> PAGESHIFT; if (top < highp) top = highp; sumpages += (list[i+1] >> PAGESHIFT); } *topp = top; *sumpagesp = sumpages; } /* * Copy a memory list. Used in startup() to copy boot's * memory lists to the kernel. */ void copy_memlist( u_longlong_t *src, size_t nelems, struct memlist **dstp) { struct memlist *dst, *prev; size_t i; dst = *dstp; prev = dst; for (i = 0; i < nelems; i += 2) { dst->address = src[i]; dst->size = src[i+1]; dst->next = 0; if (prev == dst) { dst->prev = 0; dst++; } else { dst->prev = prev; prev->next = dst; dst++; prev++; } } *dstp = dst; } static struct bootmem_props { char *name; u_longlong_t *ptr; size_t nelems; /* actual number of elements */ size_t bufsize; /* length of allocated buffer */ } bootmem_props[] = { { "phys-installed", NULL, 0, 0 }, { "phys-avail", NULL, 0, 0 }, { "virt-avail", NULL, 0, 0 }, { NULL, NULL, 0, 0 } }; #define PHYSINSTALLED 0 #define PHYSAVAIL 1 #define VIRTAVAIL 2 void copy_boot_memlists(u_longlong_t **physinstalled, size_t *physinstalled_len, u_longlong_t **physavail, size_t *physavail_len, u_longlong_t **virtavail, size_t *virtavail_len) { int align = BO_ALIGN_L3; size_t len; struct bootmem_props *tmp = bootmem_props; tryagain: for (tmp = bootmem_props; tmp->name != NULL; tmp++) { len = BOP_GETPROPLEN(bootops, tmp->name); if (len == 0) { panic("cannot get length of \"%s\" property", tmp->name); } tmp->nelems = len / sizeof (u_longlong_t); len = roundup(len, PAGESIZE); if (len <= tmp->bufsize) continue; /* need to allocate more */ if (tmp->ptr) { BOP_FREE(bootops, (caddr_t)tmp->ptr, tmp->bufsize); tmp->ptr = NULL; tmp->bufsize = 0; } tmp->bufsize = len; tmp->ptr = (void *)BOP_ALLOC(bootops, 0, tmp->bufsize, align); if (tmp->ptr == NULL) panic("cannot allocate %d bytes for \"%s\" property", tmp->bufsize, tmp->name); } /* * take the most current snapshot we can by calling mem-update */ if (BOP_GETPROPLEN(bootops, "memory-update") == 0) (void) BOP_GETPROP(bootops, "memory-update", NULL); /* did the sizes change? */ for (tmp = bootmem_props; tmp->name != NULL; tmp++) { len = BOP_GETPROPLEN(bootops, tmp->name); tmp->nelems = len / sizeof (u_longlong_t); len = roundup(len, PAGESIZE); if (len > tmp->bufsize) { /* ick. Free them all and try again */ for (tmp = bootmem_props; tmp->name != NULL; tmp++) { BOP_FREE(bootops, (caddr_t)tmp->ptr, tmp->bufsize); tmp->ptr = NULL; tmp->bufsize = 0; } goto tryagain; } } /* now we can retrieve the properties */ for (tmp = bootmem_props; tmp->name != NULL; tmp++) { if (BOP_GETPROP(bootops, tmp->name, tmp->ptr) == -1) { panic("cannot retrieve \"%s\" property", tmp->name); } } *physinstalled = bootmem_props[PHYSINSTALLED].ptr; *physinstalled_len = bootmem_props[PHYSINSTALLED].nelems; *physavail = bootmem_props[PHYSAVAIL].ptr; *physavail_len = bootmem_props[PHYSAVAIL].nelems; *virtavail = bootmem_props[VIRTAVAIL].ptr; *virtavail_len = bootmem_props[VIRTAVAIL].nelems; } /* * Find the page number of the highest installed physical * page and the number of pages installed (one cannot be * calculated from the other because memory isn't necessarily * contiguous). */ void installed_top_size( struct memlist *list, /* pointer to start of installed list */ pfn_t *topp, /* return ptr for top value */ pgcnt_t *sumpagesp) /* return prt for sum of installed pages */ { pfn_t top = 0; pfn_t highp; /* high page in a chunk */ pgcnt_t sumpages = 0; for (; list; list = list->next) { highp = (list->address + list->size - 1) >> PAGESHIFT; if (top < highp) top = highp; sumpages += (uint_t)(list->size >> PAGESHIFT); } *topp = top; *sumpagesp = sumpages; }