Hello Muchun, On Fri, Mar 12, 2021 at 03:14:07PM +0800, Muchun Song wrote: > On Thu, Mar 11, 2021 at 9:12 PM Johannes Weiner <hannes@xxxxxxxxxxx> wrote: > > > @@ -358,14 +358,26 @@ enum page_memcg_data_flags { > > > > > > #define MEMCG_DATA_FLAGS_MASK (__NR_MEMCG_DATA_FLAGS - 1) > > > > > > +/* Return true for charged page, otherwise false. */ > > > +static inline bool page_memcg_charged(struct page *page) > > > +{ > > > + unsigned long memcg_data = page->memcg_data; > > > + > > > + VM_BUG_ON_PAGE(PageSlab(page), page); > > > + VM_BUG_ON_PAGE(memcg_data & MEMCG_DATA_OBJCGS, page); > > > + > > > + return !!memcg_data; > > > +} > > > > This is mosntly used right before a page_memcg_check(), which makes it > > somewhat redundant except for the VM_BUG_ON_PAGE() for slab pages. > > Should I rename page_memcg_charged to page_memcg_raw? > And use page_memcg_raw to check whether the page is charged. > > static inline bool page_memcg_charged(struct page *page) > { > return page->memcg_data; > } You can just directly access page->memcg_data in places where you'd use this helper. I think it's only the two places in mm/page_alloc.c, and they already have CONFIG_MEMCG in place, so raw access works. > > But it's also a bit of a confusing name: slab pages are charged too, > > but this function would crash if you called it on one. > > > > In light of this, and in light of what I wrote above about hopefully > > converting more and more allocations from raw memcg pins to > > reparentable objcg, it would be bettor to have > > > > page_memcg() for 1:1 page-memcg mappings, i.e. LRU & kmem > > Sorry. I do not get the point. Because in the next patch, the kmem > page will use objcg to charge memory. So the page_memcg() > should not be suitable for the kmem pages. So I add a VM_BUG_ON > in the page_memcg() to catch invalid usage. > > So I changed some page_memcg() calling to page_memcg_check() > in this patch, but you suggest using page_memcg(). It would be better if page_memcg() worked on LRU and kmem pages. I'm proposing to change its implementation. The reason is that page_memcg_check() allows everything and does no sanity checking. You need page_memcg_charged() for the sanity checks that it's LRU or kmem, but that's a bit difficult to understand, and it's possible people will add more callsites to page_memcg_check() without the page_memcg_charged() checks. It makes the code less safe. We should discourage page_memcg_check() and make page_memcg() more useful instead. > I am very confused. Are you worried about the extra overhead brought > by calling page_memcg_rcu()? In the next patch, I will remove > page_memcg_check() calling and use objcg APIs. I'm just worried about the usability of the interface. It should be easy to use, and make it obvious if there is a user bug. For example, in your next patch, mod_lruvec_page_state does this: if (PageMemcgKmem(head)) { rcu_read_lock(); memcg = obj_cgroup_memcg(page_objcg(page)); } else { memcg = page_memcg(head); /* * Untracked pages have no memcg, no lruvec. Update only the * node. */ if (!memcg) { __mod_node_page_state(pgdat, idx, val); return; } } lruvec = mem_cgroup_lruvec(memcg, pgdat); __mod_lruvec_state(lruvec, idx, val); if (PageMemcgKmem(head)) rcu_read_unlock(); I'm proposing to implement page_memcg() in a way where you can do this: rcu_read_lock(); memcg = page_memcg(page); if (!memcg) { rcu_read_unlock(); __mod_node_page_state(); return; } lruvec = mem_cgroup_lruvec(memcg, pgdat); __mod_lruvec_state(lruvec, idx, val); rcu_read_unlock(); [ page_memcg() is: if (PageMemcgKmem(page)) return obj_cgroup_memcg(__page_objcg(page)); else return __page_memcg(page); and __page_objcg() and __page_memcg() do the raw page->memcg_data translation and the VM_BUG_ON_PAGE() checks for MEMCG_DATA_* ] This is a lot simpler and less error prone. It does take rcu_read_lock() for LRU pages too, which strictly it doesn't need to right now. But it's cheap enough (and actually saves a branch). Longer term we most likely need it there anyway. The issue you are describing in the cover letter - allocations pinning memcgs for a long time - it exists at a larger scale and is causing recurring problems in the real world: page cache doesn't get reclaimed for a long time, or is used by the second, third, fourth, ... instance of the same job that was restarted into a new cgroup every time. Unreclaimable dying cgroups pile up, waste memory, and make page reclaim very inefficient. We likely have to convert LRU pages and most other raw memcg pins to the objcg direction to fix this problem, and then the page->memcg lookup will always require the rcu read lock anyway. Finally, a universal page_memcg() should also make uncharge_page() simpler. Instead of obj_cgroup_memcg_get(), you could use the new page_memcg() to implement a universal get_mem_cgroup_from_page(): rcu_read_lock(); retry: memcg = page_memcg(page); if (unlikely(!css_tryget(&memcg->css))) goto retry; rcu_read_unlock(); return memcg; and then uncharge_page() becomes something like this: /* Look up page's memcg & prepare the batch */ memcg = get_mem_cgroup_from_page(page); if (!memcg) return; if (ug->memcg != memcg) { ... css_get(&memcg->css); /* batch ref, put in uncharge_batch() */ } mem_cgroup_put(memcg); /* Add page to batch */ nr_pages = compound_nr(page); ... /* Clear the page->memcg link */ if (PageMemcgKmem(page)) obj_cgroup_put(__page_objcg(page)); else css_put(__page_memcg(&memcg->css)); page->memcg_data = 0; Does that sound reasonable? PS: We have several page_memcg() callsites that could use the raw __page_memcg() directly for now. Is it worth switching them and saving the branch? I think probably not, because these paths aren't hot, and as per above, we should switch them to objcg sooner or later anyway.