Re: Strict page reservation for hugepage inodes

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



David Gibson <[email protected]> wrote:
>
> These days, hugepages are demand-allocated at first fault time.
> There's a somewhat dubious (and racy) heuristic when making a new
> mmap() to check if there are enough available hugepages to fully
> satisfy that mapping.
> 
> A particularly obvious case where the heuristic breaks down is where a
> process maps its hugepages not as a single chunk, but as a bunch of
> individually mmap()ed (or shmat()ed) blocks without touching and
> instantiating the pages in between allocations.  In this case the size
> of each block is compared against the total number of available
> hugepages.  It's thus easy for the process to become overcommitted,
> because each block mapping will succeed, although the total number of
> hugepages required by all blocks exceeds the number available.  In
> particular, this defeats such a program which will detect a mapping
> failure and adjust its hugepage usage downward accordingly.
> 
> The patch below is a draft attempt to address this problem, by
> strictly reserving a number of physical hugepages for hugepages inodes
> which have been mapped, but not instatiated.  MAP_SHARED mappings are
> thus "safe" - they will fail on mmap(), not later with a SIGBUS.
> MAP_PRIVATE mappings can still SIGBUS.  (Actually SHARED mappings can
> technically still SIGBUS, but only if the sysadmin explicitly reduces
> the hugepage pool between mapping and instantiation)
> 
> This patch appears to address the problem at hand - it allows DB2 to
> start correctly, for instance, which previously suffered the failure
> described above.
> 
> I'm not terribly convinced that I don't need some more locking, but if
> so it's far from obvious what.  VFS synchronization baffles me.
> 
> Signed-off-by: David Gibson <[email protected]>
> 
> Index: working-2.6/mm/hugetlb.c
> ===================================================================
> --- working-2.6.orig/mm/hugetlb.c	2006-02-17 14:44:04.000000000 +1100
> +++ working-2.6/mm/hugetlb.c	2006-02-20 14:10:24.000000000 +1100
> @@ -20,7 +20,7 @@
>  #include <linux/hugetlb.h>
>  
>  const unsigned long hugetlb_zero = 0, hugetlb_infinity = ~0UL;
> -static unsigned long nr_huge_pages, free_huge_pages;
> +static unsigned long nr_huge_pages, free_huge_pages, reserved_huge_pages;
>  unsigned long max_huge_pages;
>  static struct list_head hugepage_freelists[MAX_NUMNODES];
>  static unsigned int nr_huge_pages_node[MAX_NUMNODES];
> @@ -94,21 +94,87 @@ void free_huge_page(struct page *page)
>  
>  struct page *alloc_huge_page(struct vm_area_struct *vma, unsigned long addr)
>  {
> +	struct inode *inode = vma->vm_file->f_dentry->d_inode;
>  	struct page *page;
>  	int i;
> +	int use_reserve = 0;
> +
> +	if (vma->vm_flags & VM_MAYSHARE) {
> +		unsigned long idx;
> +
> +		idx = ((addr - vma->vm_start) >> HPAGE_SHIFT)
> +			+ (vma->vm_pgoff >> (HPAGE_SHIFT - PAGE_SHIFT));

That hurt my brain.  It's the offset into the radix tree, isn't it?  Index
into the file in HPAGE_SIZE units?

> +		if (idx < inode->i_blocks)
> +			use_reserve = 1;

i_blocks is being used for something, not clear what.  Needs big comment,
please.

> +	if (! use_reserve) {
> +	if (!page)

no-space-after-! is preferred, please.

> +}
> +
> +int hugetlb_reserve_for_inode(struct inode *inode, unsigned long npages)

Nice comment needed here, please.  Not only for posterity - they should be
in there up-front to aid in patch review.

> +{
> +	struct address_space *mapping = inode->i_mapping;
> +	unsigned long pg;
> +	long change_in_reserve = 0;
> +	struct page *page;
> +	int ret = -ENOMEM;
> +
> +	read_lock_irq(&inode->i_mapping->tree_lock);
> +	for (pg = inode->i_blocks; pg < npages; pg++) {

So `npages' is in fact the highest-desired pagecache index?  Or is it
highest-desired+1?  I think it needs a better name (npages sorta implies a
delta, not an absolute value) and it needs a comment-borne description of
whether it's inclusive or exclusive, so we can check for offs-by-one.

> +		page = radix_tree_lookup(&mapping->page_tree, pg);
> +		if (! page)
> +			change_in_reserve++;
> +	}
> +
> +	for (pg = npages; pg < inode->i_blocks; pg++) {
> +		page = radix_tree_lookup(&mapping->page_tree, pg);
> +		if (! page)
> +			change_in_reserve--;

Should that be "if (page)"?  Am all confused now.

> +	}
> +	spin_lock(&hugetlb_lock);
> +	if ((change_in_reserve > 0)
> +	    && (change_in_reserve > (free_huge_pages-reserved_huge_pages))) {
> +		printk("Failing:  change=%ld   free=%lu\n",
> +		       change_in_reserve, free_huge_pages - reserved_huge_pages);
> +		goto out;
> +	}
> +
> +	reserved_huge_pages += change_in_reserve;
> +	inode->i_blocks = npages;
> +
> +	ret = 0;
> +
> + out:
> +	spin_unlock(&hugetlb_lock);
> +	read_unlock_irq(&inode->i_mapping->tree_lock);

Yes, hugetlb_lock protects free_huge_pages.  And as Nick says, this lock
coupling is undesirable.

The code does a bunch of calculations based upon a radix-tree probe.  Once
we've dropped tree_lock, some other process can come in and make those
calculations no-longer-true.  So, assuming that all other places which
modify the radix-tree are also updating free_huge_pages/reserved_huge_pages
under hugetlb_lock (are they?) then yeah, we need to hold both locks
throughout.

hugetlb_lock is almost a tight, innermost lock.  Unfortunately we also call
__free_pages in one spot while holding it.  The code could be reworked so
we don't do that.

(Be aware that there are a coupld of hugetlb.c patches queued in -mm).

>  static int hugetlbfs_file_mmap(struct file *file, struct vm_area_struct *vma)
>  {
>  	struct inode *inode = file->f_dentry->d_inode;
> -	struct address_space *mapping = inode->i_mapping;
> -	unsigned long bytes;
>  	loff_t len, vma_len;
>  	int ret;
>  
> @@ -113,13 +74,8 @@ static int hugetlbfs_file_mmap(struct fi
>  	if (vma->vm_end - vma->vm_start < HPAGE_SIZE)
>  		return -EINVAL;
>  
> -	bytes = huge_pages_needed(mapping, vma);
> -	if (!is_hugepage_mem_enough(bytes))
> -		return -ENOMEM;
> -
>  	vma_len = (loff_t)(vma->vm_end - vma->vm_start);
>  
> -	mutex_lock(&inode->i_mutex);

What was that protecting?

>  	file_accessed(file);
>  	vma->vm_flags |= VM_HUGETLB | VM_RESERVED;
>  	vma->vm_ops = &hugetlb_vm_ops;
> @@ -129,13 +85,22 @@ static int hugetlbfs_file_mmap(struct fi
>  	if (!(vma->vm_flags & VM_WRITE) && len > inode->i_size)
>  		goto out;
>  
> -	ret = 0;
> +	if (inode->i_blocks < (len >> HPAGE_SHIFT)) {

What locking does i_blocks use?   tree_lock, I thought?

>  static void hugetlbfs_delete_inode(struct inode *inode)
>  {
> -	if (inode->i_data.nrpages)
> -		truncate_hugepages(&inode->i_data, 0);
> +	truncate_hugepages(inode, 0);

This optimisation was removed because there might be a reservation in the
inode (yes?)


-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to [email protected]
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

[Index of Archives]     [Kernel Newbies]     [Netfilter]     [Bugtraq]     [Photo]     [Stuff]     [Gimp]     [Yosemite News]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Video 4 Linux]     [Linux for the blind]     [Linux Resources]
  Powered by Linux