Re: [PATCH 09/10] memory-hotplug: enable memory hotplug to handlehugepage

From: Naoya Horiguchi
Date: Wed Mar 27 2013 - 17:30:25 EST


On Wed, Mar 27, 2013 at 03:19:21PM +0100, Michal Hocko wrote:
> On Tue 26-03-13 14:23:24, Naoya Horiguchi wrote:
> > On Mon, Mar 25, 2013 at 04:09:52PM +0100, Michal Hocko wrote:
> > > On Fri 22-03-13 16:23:54, Naoya Horiguchi wrote:
> > ...
> > > > index d9d3dd7..ef79871 100644
> > > > --- v3.9-rc3.orig/mm/hugetlb.c
> > > > +++ v3.9-rc3/mm/hugetlb.c
> > > > @@ -844,6 +844,36 @@ static int free_pool_huge_page(struct hstate *h, nodemask_t *nodes_allowed,
> > > > return ret;
> > > > }
> > > >
> > > > +/* Dissolve a given free hugepage into free pages. */
> > > > +static void dissolve_free_huge_page(struct page *page)
> > > > +{
> > > > + spin_lock(&hugetlb_lock);
> > > > + if (PageHuge(page) && !page_count(page)) {
> > > > + struct hstate *h = page_hstate(page);
> > > > + int nid = page_to_nid(page);
> > > > + list_del(&page->lru);
> > > > + h->free_huge_pages--;
> > > > + h->free_huge_pages_node[nid]--;
> > > > + update_and_free_page(h, page);
> > > > + }
> > >
> > > What about surplus pages?
> >
> > This function is only for free hugepage, not for surplus hugepages
> > (which are considered as in-use hugepages.)
>
> How do you want to get rid of those then? You cannot offline the node if
> there are any pages...

My intention is that offlining should fail if there remains some active
hugepages in the memblock you try to offline.

> > dissolve_free_huge_pages() can be called only when all source hugepages
> > are free (all in-use hugepages are successfully migrated.)
> >
...
> > > > +/* Returns true for head pages of in-use hugepages, otherwise returns false. */
> > > > +bool is_hugepage_movable(struct page *hpage)
> > > > +{
> > > > + struct page *page;
> > > > + struct hstate *h;
> > > > + bool ret = false;
> > > > +
> > > > + VM_BUG_ON(!PageHuge(hpage));
> > > > + /*
> > > > + * This function can be called for a tail page because memory hotplug
> > > > + * scans movability of pages by pfn range of a memory block.
> > > > + * Larger hugepages (1GB for x86_64) are larger than memory block, so
> > > > + * the scan can start at the tail page of larger hugepages.
> > > > + * 1GB hugepage is not movable now, so we return with false for now.
> > > > + */
> > > > + if (PageTail(hpage))
> > > > + return false;
> > > > + h = page_hstate(hpage);
> > > > + spin_lock(&hugetlb_lock);
> > > > + list_for_each_entry(page, &h->hugepage_activelist, lru)
> > > > + if (page == hpage) {
> > > > + ret = true;
> > > > + break;
> > > > + }
> > >
> > > Why are you checking that the page is active?
> >
> > This is the counterpart to doing PageLRU check for normal pages.
> >
> > > It doesn't make much sense
> > > to me because nothing prevents it from being freed/allocated right after
> > > you release hugetlb_lock.
> >
> > Such a race can also happen for normal pages because scan_movable_pages()
> > just check PageLRU flags without holding any lock.
> > But the caller, __offline_pages(), repeats to call scan_movable_pages()
> > until no page in the memblock are judged as movable, and in the repeat loop
> > do_migrate_range() does nothing for free (unmovable) pages.
> > So there is no behavioral problem even if the movable page is freed just
> > after the if(PageLRU) check in scan_movable_page().
>
> yes
>
> > Note that in this loop, allocating pages in the memblock is forbidden
> > because we already do set_migratetype_isolate() for them, so we don't have
> > to worry about being allocated just after scan_movable_pages().
>
> yes
>
> > I want the same thing to be the case for hugepage. As you pointed out,
> > is_hugepage_movable() is not safe from such a race, but in "being freed
> > just after is_hugepage_movable() returns true" case we have no problem
> > for the same reason described above.
>
> yes, this was my point, sorry for not being clear about that. I meant
> the costly test is pointless because it doesn't prevent any races and
> doesn't tell us much.

Yes, running over hugepage_activelist is not preferable when this list
become longer.

> If we made sure that all page on the hugepage_freelists have reference
> 0 (which is now not the case and it is yet another source of confusion)
> then the whole loop could be replaced by page_count check.

I think that free hugepages have refcount 0, but hwpoisoned hugepages
have also refcount 0. But hwpoison can happen only on limited hardware
and we consider them as exceptional, so replacing page_count check and
checking PG_hwpoisoned flag looks more reasonable to me.

> > However, in "being allocated just after is_hugepage_movable() returns false"
> > case, it seems to be possible to hot-remove an active hugepage.
>
> check_pages_isolated should catch this but it is still racy.

Right.

> > I think we can avoid this by adding migratetype check in
> > alloc_huge_page().
>
> I think dequeue_huge_page_vma should be sufficient, because we are going
> through page allocator otherwise and that one is aware of migrate types.

OK.

Thanks,
Naoya
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/