summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/arch/i386/i386/machdep.c136
1 files changed, 95 insertions, 41 deletions
diff --git a/sys/arch/i386/i386/machdep.c b/sys/arch/i386/i386/machdep.c
index c4b01b7534f..90c45de024f 100644
--- a/sys/arch/i386/i386/machdep.c
+++ b/sys/arch/i386/i386/machdep.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: machdep.c,v 1.76 1998/01/20 18:40:14 niklas Exp $ */
+/* $OpenBSD: machdep.c,v 1.77 1998/01/22 02:30:46 niklas Exp $ */
/* $NetBSD: machdep.c,v 1.214 1996/11/10 03:16:17 thorpej Exp $ */
/*-
@@ -218,6 +218,7 @@ struct extent *iomem_ex;
static ioport_malloc_safe;
caddr_t allocsys __P((caddr_t));
+void setup_buffers __P((vm_offset_t *));
void dumpsys __P((void));
void identifycpu __P((void));
void init386 __P((vm_offset_t));
@@ -257,9 +258,7 @@ cpu_startup()
unsigned i;
caddr_t v;
int sz;
- int base, residual;
vm_offset_t minaddr, maxaddr, pa;
- vm_size_t size;
struct pcb *pcb;
int x;
@@ -306,58 +305,27 @@ cpu_startup()
* Now allocate buffers proper. They are different than the above
* in that they usually occupy more virtual memory than physical.
*/
- size = MAXBSIZE * nbuf;
- buffer_map = kmem_suballoc(kernel_map, (vm_offset_t *)&buffers,
- &maxaddr, size, TRUE);
- minaddr = (vm_offset_t)buffers;
- if (vm_map_find(buffer_map, vm_object_allocate(size), (vm_offset_t)0,
- &minaddr, size, FALSE) != KERN_SUCCESS)
- panic("startup: cannot allocate buffers");
-
- base = bufpages / nbuf;
- residual = bufpages % nbuf;
- if (base >= MAXBSIZE / CLBYTES) {
- /* don't want to alloc more physical mem than needed */
- base = MAXBSIZE / CLBYTES;
- residual = 0;
- }
-
- for (i = 0; i < nbuf; i++) {
- vm_size_t curbufsize;
- vm_offset_t curbuf;
-
- /*
- * First <residual> buffers get (base+1) physical pages
- * allocated for them. The rest get (base) physical pages.
- *
- * The rest of each buffer occupies virtual space,
- * but has no physical memory allocated for it.
- */
- curbuf = (vm_offset_t)buffers + i * MAXBSIZE;
- curbufsize = CLBYTES * (i < residual ? base+1 : base);
- vm_map_pageable(buffer_map, curbuf, curbuf+curbufsize, FALSE);
- vm_map_simplify(buffer_map, curbuf);
- }
+ setup_buffers(&maxaddr);
/*
* Allocate a submap for exec arguments. This map effectively
* limits the number of processes exec'ing at any time.
*/
- exec_map = kmem_suballoc(kernel_map, &minaddr, &maxaddr,
- 16*NCARGS, TRUE);
+ exec_map = kmem_suballoc(kernel_map, &minaddr, &maxaddr, 16*NCARGS,
+ TRUE);
/*
* Allocate a submap for physio
*/
- phys_map = kmem_suballoc(kernel_map, &minaddr, &maxaddr,
- VM_PHYS_SIZE, TRUE);
+ phys_map = kmem_suballoc(kernel_map, &minaddr, &maxaddr, VM_PHYS_SIZE,
+ TRUE);
/*
* Finally, allocate mbuf pool. Since mclrefcnt is an off-size
* we use the more space efficient malloc in place of kmem_alloc.
*/
- mclrefcnt = (char *)malloc(NMBCLUSTERS+CLBYTES/MCLBYTES,
- M_MBUF, M_NOWAIT);
+ mclrefcnt = (char *)malloc(NMBCLUSTERS+CLBYTES/MCLBYTES, M_MBUF,
+ M_NOWAIT);
bzero(mclrefcnt, NMBCLUSTERS+CLBYTES/MCLBYTES);
mb_map = kmem_suballoc(kernel_map, (vm_offset_t *)&mbutl, &maxaddr,
VM_MBUF_SIZE, FALSE);
@@ -491,6 +459,92 @@ allocsys(v)
return v;
}
+void
+setup_buffers(maxaddr)
+ vm_offset_t *maxaddr;
+{
+ vm_size_t size;
+ vm_offset_t addr;
+ int base, residual, left, i;
+ struct pglist pgs, freepgs;
+ vm_page_t pg, *last, *last2;
+
+ size = MAXBSIZE * nbuf;
+ buffer_map = kmem_suballoc(kernel_map, (vm_offset_t *)&buffers,
+ maxaddr, size, TRUE);
+ addr = (vm_offset_t)buffers;
+ if (vm_map_find(buffer_map, vm_object_allocate(size), (vm_offset_t)0,
+ &addr, size, FALSE) != KERN_SUCCESS)
+ panic("startup: cannot allocate buffers");
+
+ base = bufpages / nbuf;
+ residual = bufpages % nbuf;
+ if (base >= MAXBSIZE / CLBYTES) {
+ /* don't want to alloc more physical mem than needed */
+ base = MAXBSIZE / CLBYTES;
+ residual = 0;
+ }
+
+ /*
+ * In case we might need DMA bouncing we have to make sure there
+ * is some memory below 16MB available. On machines with many
+ * pages reserved for the buffer cache we risk filling all of that
+ * area with buffer pages. We still want much of the buffers
+ * reside there as that lowers the probability of them needing to
+ * bounce, but we have to set aside some space for DMA buffers too.
+ *
+ * The current strategy is to grab hold of 2MB chunks at a time as long
+ * as they fit below 16MB, and as soon as that fail, give back the
+ * last one and allocate the rest above 16MB. That should guarantee
+ * at least 2MB below 16MB left for DMA buffers.
+ */
+ TAILQ_INIT(&pgs);
+ last = last2 = 0;
+ addr = 0;
+ for (left = bufpages; left > 2 * 1024 * 1024 / CLBYTES;
+ left -= 2 * 1024 * 1024 / CLBYTES) {
+ if (vm_page_alloc_memory(2 * 1024 * 1024, 0, 16 * 1024 * 1024,
+ CLBYTES, 0, &pgs, 1, 0)) {
+ if (last2) {
+ TAILQ_INIT(&freepgs);
+ freepgs.tqh_first = *last2;
+ freepgs.tqh_last = pgs.tqh_last;
+ (*last2)->pageq.tqe_prev = &freepgs.tqh_first;
+ pgs.tqh_last = last2;
+ *last2 = NULL;
+ vm_page_free_memory(&freepgs);
+ left += 2 * 1024 * 1024 / CLBYTES;
+ addr = 16 * 1024 * 1024;
+ }
+ break;
+ }
+ last2 = last ? last : &pgs.tqh_first;
+ last = pgs.tqh_last;
+ }
+ if (left > 0)
+ if (vm_page_alloc_memory(left * CLBYTES, addr, avail_end,
+ CLBYTES, 0, &pgs, 1, 0))
+ panic("cannot get physical memory for buffer cache");
+
+ pg = pgs.tqh_first;
+ for (i = 0; i < nbuf; i++) {
+ /*
+ * First <residual> buffers get (base+1) physical pages
+ * allocated for them. The rest get (base) physical pages.
+ *
+ * The rest of each buffer occupies virtual space,
+ * but has no physical memory allocated for it.
+ */
+ addr = (vm_offset_t)buffers + i * MAXBSIZE;
+ for (size = CLBYTES * (i < residual ? base + 1 : base);
+ size > 0; size -= NBPG, addr += NBPG) {
+ pmap_enter(pmap_kernel(), addr, pg->phys_addr,
+ VM_PROT_READ|VM_PROT_WRITE, TRUE);
+ pg = pg->pageq.tqe_next;
+ }
+ }
+}
+
/*
* Info for CTL_HW
*/