| version 1.111, 2005/05/08 18:44:39 |
version 1.111.2.1, 2005/07/07 12:07:38 |
| Line 265 sokvafree(vaddr_t sva, vsize_t len) |
|
| Line 265 sokvafree(vaddr_t sva, vsize_t len) |
|
| } |
} |
| |
|
| static void |
static void |
| sodoloanfree(struct vm_page **pgs, caddr_t buf, size_t size) |
sodoloanfree(struct vm_page **pgs, caddr_t buf, size_t size, boolean_t mapped) |
| { |
{ |
| vaddr_t va, sva, eva; |
vaddr_t sva, eva; |
| vsize_t len; |
vsize_t len; |
| paddr_t pa; |
int npgs; |
| int i, npgs; |
|
| |
KASSERT(pgs != NULL); |
| |
|
| eva = round_page((vaddr_t) buf + size); |
eva = round_page((vaddr_t) buf + size); |
| sva = trunc_page((vaddr_t) buf); |
sva = trunc_page((vaddr_t) buf); |
| len = eva - sva; |
len = eva - sva; |
| npgs = len >> PAGE_SHIFT; |
npgs = len >> PAGE_SHIFT; |
| |
|
| if (__predict_false(pgs == NULL)) { |
if (mapped) { |
| pgs = alloca(npgs * sizeof(*pgs)); |
pmap_kremove(sva, len); |
| |
pmap_update(pmap_kernel()); |
| for (i = 0, va = sva; va < eva; i++, va += PAGE_SIZE) { |
|
| if (pmap_extract(pmap_kernel(), va, &pa) == FALSE) |
|
| panic("sodoloanfree: va 0x%lx not mapped", va); |
|
| pgs[i] = PHYS_TO_VM_PAGE(pa); |
|
| } |
|
| } |
} |
| |
|
| pmap_kremove(sva, len); |
|
| pmap_update(pmap_kernel()); |
|
| uvm_unloan(pgs, npgs, UVM_LOAN_TOPAGE); |
uvm_unloan(pgs, npgs, UVM_LOAN_TOPAGE); |
| sokvafree(sva, len); |
sokvafree(sva, len); |
| } |
} |
| Line 336 sodopendfreel(struct socket *so) |
|
| Line 329 sodopendfreel(struct socket *so) |
|
| |
|
| for (; m != NULL; m = next) { |
for (; m != NULL; m = next) { |
| next = m->m_next; |
next = m->m_next; |
| |
KASSERT((~m->m_flags & (M_EXT|M_EXT_PAGES)) == 0); |
| |
KASSERT(!MCLISREFERENCED(m)); |
| |
|
| rv += m->m_ext.ext_size; |
rv += m->m_ext.ext_size; |
| sodoloanfree((m->m_flags & M_EXT_PAGES) ? |
sodoloanfree(m->m_ext.ext_pgs, m->m_ext.ext_buf, |
| m->m_ext.ext_pgs : NULL, m->m_ext.ext_buf, |
m->m_ext.ext_size, |
| m->m_ext.ext_size); |
(m->m_ext.ext_flags & M_EXT_LAZY) == 0); |
| pool_cache_put(&mbpool_cache, m); |
pool_cache_put(&mbpool_cache, m); |
| } |
} |
| |
|
| Line 356 soloanfree(struct mbuf *m, caddr_t buf, |
|
| Line 351 soloanfree(struct mbuf *m, caddr_t buf, |
|
| { |
{ |
| int s; |
int s; |
| |
|
| if (m == NULL) { |
KASSERT(m != NULL); |
| |
|
| /* |
|
| * called from MEXTREMOVE. |
|
| */ |
|
| |
|
| sodoloanfree(NULL, buf, size); |
|
| return; |
|
| } |
|
| |
|
| /* |
/* |
| * postpone freeing mbuf. |
* postpone freeing mbuf. |
| Line 389 sosend_loan(struct socket *so, struct ui |
|
| Line 376 sosend_loan(struct socket *so, struct ui |
|
| struct iovec *iov = uio->uio_iov; |
struct iovec *iov = uio->uio_iov; |
| vaddr_t sva, eva; |
vaddr_t sva, eva; |
| vsize_t len; |
vsize_t len; |
| vaddr_t lva, va; |
vaddr_t lva; |
| int npgs, i, error; |
int npgs, error; |
| |
#if 0 |
| |
vaddr_t va; |
| |
int i; |
| |
#endif |
| |
|
| if (uio->uio_segflg != UIO_USERSPACE) |
if (uio->uio_segflg != UIO_USERSPACE) |
| return (0); |
return (0); |
| Line 420 sosend_loan(struct socket *so, struct ui |
|
| Line 411 sosend_loan(struct socket *so, struct ui |
|
| return (0); |
return (0); |
| } |
} |
| |
|
| |
#if 0 |
| for (i = 0, va = lva; i < npgs; i++, va += PAGE_SIZE) |
for (i = 0, va = lva; i < npgs; i++, va += PAGE_SIZE) |
| pmap_kenter_pa(va, VM_PAGE_TO_PHYS(m->m_ext.ext_pgs[i]), |
pmap_kenter_pa(va, VM_PAGE_TO_PHYS(m->m_ext.ext_pgs[i]), |
| VM_PROT_READ); |
VM_PROT_READ); |
| pmap_update(pmap_kernel()); |
pmap_update(pmap_kernel()); |
| |
#endif |
| |
|
| lva += (vaddr_t) iov->iov_base & PAGE_MASK; |
lva += (vaddr_t) iov->iov_base & PAGE_MASK; |
| |
|
| MEXTADD(m, (caddr_t) lva, space, M_MBUF, soloanfree, so); |
MEXTADD(m, (caddr_t) lva, space, M_MBUF, soloanfree, so); |
| m->m_flags |= M_EXT_PAGES | M_EXT_ROMAP; |
m->m_flags |= M_EXT_PAGES | M_EXT_ROMAP; |
| |
|
| |
m->m_flags |= M_EXT_LAZY; |
| |
m->m_ext.ext_flags |= M_EXT_LAZY; |
| |
|
| uio->uio_resid -= space; |
uio->uio_resid -= space; |
| /* uio_offset not updated, not set/used for write(2) */ |
/* uio_offset not updated, not set/used for write(2) */ |
| uio->uio_iov->iov_base = (caddr_t) uio->uio_iov->iov_base + space; |
uio->uio_iov->iov_base = (caddr_t) uio->uio_iov->iov_base + space; |