Date: Mon, 22 May 2006 08:24:14 GMT From: Kip Macy <kmacy@FreeBSD.org> To: Perforce Change Reviews <perforce@freebsd.org> Subject: PERFORCE change 97607 for review Message-ID: <200605220824.k4M8OEjJ020316@repoman.freebsd.org>
next in thread | raw e-mail | index | archive | help
http://perforce.freebsd.org/chv.cgi?CH=97607 Change 97607 by kmacy@kmacy_storage:sun4v_work on 2006/05/22 08:23:21 fix alignment fault handling by setting %g1 correctly to point to trap implement pmap_ts_referenced clear TSB when doing pmap_invalidate on exit from pmap_enter remove raciness in resetting tlbactive remove cruft from pmap_ipi GIANT should no longer be needed in swap_pager Affected files ... .. //depot/projects/kmacy_sun4v/src/sys/sun4v/sun4v/exception.S#56 edit .. //depot/projects/kmacy_sun4v/src/sys/sun4v/sun4v/pmap.c#54 edit .. //depot/projects/kmacy_sun4v/src/sys/vm/swap_pager.c#5 edit Differences ... ==== //depot/projects/kmacy_sun4v/src/sys/sun4v/sun4v/exception.S#56 (text+ko) ==== @@ -322,13 +322,18 @@ ! mov MMFSA_D_CTX, %g7 ldxa [%g1 + %g3]ASI_REAL, %g3 ! ldxa [%g1 + %g7]ASI_REAL, %g4 + ba,a,pt %xcc, align_fault + .align 32 + .endm + +ENTRY(align_fault) +! or %g4, %g3, %g3 sub %g0, 1, %g4 - or %g4, %g3, %g3 + set trap, %g1 ba,pt %xcc, tl0_trap mov T_MEM_ADDRESS_NOT_ALIGNED, %g2 - .align 32 - .endm - +END(align_fault) + .macro cpu_mondo ba,a,pt %xcc, cpu_mondo .align 32 ==== //depot/projects/kmacy_sun4v/src/sys/sun4v/sun4v/pmap.c#54 (text+ko) ==== @@ -995,7 +995,7 @@ } if (invlva) - pmap_invalidate_page(pmap, va, FALSE); + pmap_invalidate_page(pmap, va, TRUE); sched_unpin(); PMAP_UNLOCK(pmap); @@ -1150,20 +1150,20 @@ } #ifdef SMP -static void +static cpumask_t pmap_ipi(pmap_t pmap, char *func, uint64_t arg1, uint64_t arg2) { - int i, active, cpu_count; + int i, cpu_count; u_int cpus; - cpumask_t cpumask; + cpumask_t cpumask, active; + cpumask_t active_total, ackmask; uint16_t *cpulist; - uint32_t ackmask, ackexpect; - int inext; + if (!smp_started) - return; - + return (0); + cpumask = PCPU_GET(cpumask); cpulist = PCPU_GET(cpulist); @@ -1178,68 +1178,60 @@ #endif - if (cpumask == pmap->pm_tlbactive) - return; + if ((active_total = (pmap->pm_tlbactive & ~cpumask)) == 0) + goto done; if (pmap->pm_context != 0) - active = (pmap->pm_tlbactive & ~cpumask); + active_total = active = (pmap->pm_tlbactive & ~cpumask); else - active = PCPU_GET(other_cpus); + active_total = active = PCPU_GET(other_cpus); + if (active_total == 0) + goto done; + retry: - for (cpu_count = 0, i = 0, ackexpect = 0, cpus = active; i < 32 && cpus;) { - - if (!(cpus & 0x1) /*|| (i & ~0x3) == (curcpu & ~0x3) */ ) - { - cpus = cpus >> 1; - i++; + for (cpu_count = 0, i = 0, cpus = active; i < 32 && cpus; i++, cpus >>= 1) { + if (!(cpus & 0x1)) continue; - } cpulist[cpu_count] = (uint16_t)i; cpu_count++; - ackexpect |= (1 << i); - inext = i++; - cpus = cpus >> 1; - - } - - if (cpu_count == 0) - return; - ackmask = 0; cpu_ipi_selected(cpu_count, cpulist, (uint64_t)func, (uint64_t)arg1, (uint64_t)arg2, (uint64_t *)&ackmask); - while (ackmask != ackexpect) { + while (ackmask != active) { DELAY(1); i++; if (i > 1000000) - panic(" ackmask=0x%x active=0x%x\n", ackmask, ackexpect); + panic(" ackmask=0x%x active=0x%x\n", ackmask, active); } + + active_total |= active; + if ((active = (pmap->pm_tlbactive & ~(active_total|cpumask))) != 0) + goto retry; + + done: + return (active_total); } #endif void pmap_invalidate_page(pmap_t pmap, vm_offset_t va, int cleartsb) { - spinlock_enter(); if (cleartsb == TRUE) tsb_clear_tte(&pmap->pm_tsb, va); DPRINTF("pmap_invalidate_page(va=0x%lx)\n", va); + spinlock_enter(); invlpg(va, pmap->pm_context); - #ifdef SMP pmap_ipi(pmap, (void *)tl_invlpg, (uint64_t)va, (uint64_t)pmap->pm_context); #endif spinlock_exit(); - - - } void @@ -1248,6 +1240,7 @@ vm_offset_t tva; #ifdef SMP char *func; + cpumask_t active; #endif @@ -1259,13 +1252,11 @@ if (sva >= eva) panic("invalidating negative or zero range sva=0x%lx eva=0x%lx", sva, eva); - - spinlock_enter(); - if (cleartsb == TRUE) tsb_clear_range(&pmap->pm_tsb, sva, eva); - if ((sva - eva) < PAGE_SIZE*32 ) { + spinlock_enter(); + if ((sva - eva) < PAGE_SIZE*64) { for (tva = sva; tva < eva; tva += PAGE_SIZE_8K) invlpg(tva, pmap->pm_context); } else if (pmap->pm_context) @@ -1279,46 +1270,35 @@ else func = tl_invlctx; - pmap_ipi(pmap, (void *)func, pmap->pm_context, 0); - if (pmap != kernel_pmap) - pmap->pm_tlbactive = pmap->pm_active; + active = pmap_ipi(pmap, (void *)func, pmap->pm_context, 0); + active &= ~pmap->pm_active; + atomic_clear_int(&pmap->pm_tlbactive, active); #endif - spinlock_exit(); } void pmap_invalidate_all(pmap_t pmap) { - char *func; - spinlock_enter(); - if (pmap == kernel_pmap) panic("invalidate_all called on kernel_pmap"); tsb_clear(&pmap->pm_tsb); - if (pmap->pm_context) { - invlctx(pmap->pm_context); - func = tl_invlctx; - } else { - invltlb(); - func = tl_invltlb; - } - + spinlock_enter(); + invlctx(pmap->pm_context); #ifdef SMP - pmap_ipi(pmap, func, pmap->pm_context, 0); + pmap_ipi(pmap, tl_invlctx, pmap->pm_context, 0); pmap->pm_tlbactive = pmap->pm_active; #endif - spinlock_exit(); } boolean_t pmap_is_modified(vm_page_t m) { - return tte_get_phys_bit(m, VTD_W); + return (tte_get_phys_bit(m, VTD_W)); } @@ -1847,8 +1827,46 @@ int pmap_ts_referenced(vm_page_t m) { - UNIMPLEMENTED; - return (0); + + int rv; + pv_entry_t pv, pvf, pvn; + pmap_t pmap; + + rv = 0; + if (m->flags & PG_FICTITIOUS) + return (rv); + + sched_pin(); + mtx_assert(&vm_page_queue_mtx, MA_OWNED); + if ((pv = TAILQ_FIRST(&m->md.pv_list)) != NULL) { + + pvf = pv; + + do { + pvn = TAILQ_NEXT(pv, pv_list); + + TAILQ_REMOVE(&m->md.pv_list, pv, pv_list); + + TAILQ_INSERT_TAIL(&m->md.pv_list, pv, pv_list); + + pmap = pv->pv_pmap; + PMAP_LOCK(pmap); + if ((tte_hash_clear_bits(pmap->pm_hash, pv->pv_va, VTD_REF) & VTD_REF) != 0) { + pmap_invalidate_page(pmap, pv->pv_va, TRUE); + + rv++; + if (rv > 4) { + PMAP_UNLOCK(pmap); + break; + } + } + + PMAP_UNLOCK(pmap); + } while ((pv = pvn) != NULL && pv != pvf); + } + sched_unpin(); + + return (rv); } void ==== //depot/projects/kmacy_sun4v/src/sys/vm/swap_pager.c#5 (text+ko) ==== @@ -1158,7 +1158,6 @@ int i; int n = 0; - GIANT_REQUIRED; if (count && m[0]->object != object) { panic("swap_pager_getpages: object mismatch %p/%p", object,
Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?200605220824.k4M8OEjJ020316>