Index: sys/arm/arm/busdma_machdep-v6.c =================================================================== --- sys/arm/arm/busdma_machdep-v6.c +++ sys/arm/arm/busdma_machdep-v6.c @@ -117,6 +117,7 @@ struct sync_list { vm_offset_t vaddr; /* kva of client data */ + bus_addr_t paddr; /* physical address */ vm_page_t pages; /* starting page of client data */ vm_offset_t dataoffs; /* page offset of client data */ bus_size_t datacount; /* client data count */ @@ -1084,8 +1085,12 @@ break; sl++; sl->vaddr = 0; + sl->paddr = 0; sl->datacount = sgsize; sl->pages = PHYS_TO_VM_PAGE(curaddr); + KASSERT(sl->pages != NULL, + ("%s: page at PA:0x%08lx is not in " + "vm_page_array", __func__, curaddr)); sl->dataoffs = curaddr & PAGE_MASK; } else sl->datacount += sgsize; @@ -1118,6 +1123,20 @@ segs, segp)); } + +static __inline bus_addr_t +sl_get_paddr(struct sync_list *sl) +{ + bus_addr_t paddr; + + /* Is segment already mapped to KVA ? */ + if (sl->vaddr != 0) + paddr = sl->paddr; + else + paddr = VM_PAGE_TO_PHYS(sl->pages) + sl->dataoffs; + return (paddr); +} + /* * Utility function to load a linear buffer. segp contains * the starting segment on entrance, and the ending segment on exit. @@ -1188,8 +1207,7 @@ sgsize); } else { if (map->sync_count > 0) { - sl_pend = VM_PAGE_TO_PHYS(sl->pages) + - sl->dataoffs + sl->datacount; + sl_pend = sl_get_paddr(sl) + sl->datacount; sl_vend = sl->vaddr + sl->datacount; } @@ -1201,9 +1219,20 @@ goto cleanup; sl++; sl->vaddr = kvaddr; + if (kvaddr != 0) { + sl->paddr = curaddr; + sl->pages = NULL; + sl->dataoffs = 0; + } else { + sl->paddr = 0; + sl->pages = PHYS_TO_VM_PAGE(curaddr); + sl->dataoffs = curaddr & PAGE_MASK; + KASSERT(sl->pages != NULL, + ("%s: page at PA:0x%08lx is not " + "in vm_page_array", __func__, + curaddr)); + } sl->datacount = sgsize; - sl->pages = PHYS_TO_VM_PAGE(curaddr); - sl->dataoffs = curaddr & PAGE_MASK; } else sl->datacount += sgsize; } @@ -1302,7 +1331,7 @@ offset = sl->dataoffs; m = sl->pages; size = sl->datacount; - pa = VM_PAGE_TO_PHYS(m) | offset; + pa = sl_get_paddr(sl); for ( ; size != 0; size -= len, pa += len, offset = 0, ++m) { tempva = 0; @@ -1310,13 +1339,13 @@ len = min(PAGE_SIZE - offset, size); tempva = pmap_quick_enter_page(m); va = tempva | offset; + KASSERT(pa == (VM_PAGE_TO_PHYS(m) | offset), + ("unexpected vm_page_t phys: 0x%08x != 0x%08x", + VM_PAGE_TO_PHYS(m) | offset, pa)); } else { len = sl->datacount; va = sl->vaddr; } - KASSERT(pa == (VM_PAGE_TO_PHYS(m) | offset), - ("unexpected vm_page_t phys: 0x%08x != 0x%08x", - VM_PAGE_TO_PHYS(m) | offset, pa)); switch (op) { case BUS_DMASYNC_PREWRITE: