--- fs/hugetlbfs/inode.c | 8 +++++--- include/linux/hugetlb.h | 3 ++- mm/hugetlb.c | 6 ++++-- 3 files changed, 11 insertions(+), 6 deletions(-) Index: linux-2.6/fs/hugetlbfs/inode.c =================================================================== --- linux-2.6.orig/fs/hugetlbfs/inode.c 2008-01-25 19:16:24.000000000 -0800 +++ linux-2.6/fs/hugetlbfs/inode.c 2008-01-25 19:18:28.000000000 -0800 @@ -421,7 +421,8 @@ static void hugetlbfs_drop_inode(struct } static inline void -hugetlb_vmtruncate_list(struct prio_tree_root *root, pgoff_t pgoff) +hugetlb_vmtruncate_list(struct prio_tree_root *root, pgoff_t pgoff, + spinlock_t *lockp) { struct vm_area_struct *vma; struct prio_tree_iter iter; @@ -441,7 +442,7 @@ hugetlb_vmtruncate_list(struct prio_tree v_offset = 0; __unmap_hugepage_range(vma, - vma->vm_start + v_offset, vma->vm_end); + vma->vm_start + v_offset, vma->vm_end, &lockp); } } @@ -456,7 +457,8 @@ static int hugetlb_vmtruncate(struct ino i_size_write(inode, offset); spin_lock(&mapping->i_mmap_lock); if (!prio_tree_empty(&mapping->i_mmap)) - hugetlb_vmtruncate_list(&mapping->i_mmap, pgoff); + hugetlb_vmtruncate_list(&mapping->i_mmap, pgoff, + &mapping->i_mmap_lock); spin_unlock(&mapping->i_mmap_lock); truncate_hugepages(inode, offset); return 0; Index: linux-2.6/include/linux/hugetlb.h =================================================================== --- linux-2.6.orig/include/linux/hugetlb.h 2008-01-25 19:16:24.000000000 -0800 +++ linux-2.6/include/linux/hugetlb.h 2008-01-25 19:18:28.000000000 -0800 @@ -21,7 +21,8 @@ int hugetlb_treat_movable_handler(struct int copy_hugetlb_page_range(struct mm_struct *, struct mm_struct *, struct vm_area_struct *); int follow_hugetlb_page(struct mm_struct *, struct vm_area_struct *, struct page **, struct vm_area_struct **, unsigned long *, int *, int, int); void unmap_hugepage_range(struct vm_area_struct *, unsigned long, unsigned long); -void __unmap_hugepage_range(struct vm_area_struct *, unsigned long, unsigned long); +void __unmap_hugepage_range(struct vm_area_struct *, unsigned long, + unsigned long, spinlock_t *); int hugetlb_prefault(struct address_space *, struct vm_area_struct *); int hugetlb_report_meminfo(char *); int hugetlb_report_node_meminfo(int, char *); Index: linux-2.6/mm/hugetlb.c =================================================================== --- linux-2.6.orig/mm/hugetlb.c 2008-01-25 19:16:24.000000000 -0800 +++ linux-2.6/mm/hugetlb.c 2008-01-25 19:18:28.000000000 -0800 @@ -14,6 +14,7 @@ #include #include #include +#include #include #include @@ -724,7 +725,7 @@ nomem: } void __unmap_hugepage_range(struct vm_area_struct *vma, unsigned long start, - unsigned long end) + unsigned long end, spinlock_t *lock) { struct mm_struct *mm = vma->vm_mm; unsigned long address; @@ -763,6 +764,7 @@ void __unmap_hugepage_range(struct vm_ar } spin_unlock(&mm->page_table_lock); flush_tlb_range(vma, start, end); + mmu_notifier(invalidate_range, mm, start, end, &lockp); list_for_each_entry_safe(page, tmp, &page_list, lru) { list_del(&page->lru); put_page(page); @@ -782,7 +784,7 @@ void unmap_hugepage_range(struct vm_area */ if (vma->vm_file) { spin_lock(&vma->vm_file->f_mapping->i_mmap_lock); - __unmap_hugepage_range(vma, start, end); + __unmap_hugepage_range(vma, start, end, &vma->vm_file->f_mapping->i_mmap_lock); spin_unlock(&vma->vm_file->f_mapping->i_mmap_lock); } }