Re: KASAN: use-after-free Write in page_counter_uncharge
From: Michal Hocko
Date: Wed Aug 19 2020 - 02:34:25 EST
On Tue 18-08-20 16:18:56, Andrew Morton wrote:
> On Tue, 18 Aug 2020 07:50:28 -0700 syzbot <syzbot+b305848212deec86eabe@xxxxxxxxxxxxxxxxxxxxxxxxx> wrote:
>
> > Hello,
> >
> > syzbot found the following issue on:
> >
> > HEAD commit: a1d21081 Merge git://git.kernel.org/pub/scm/linux/kernel/g..
> > git tree: upstream
> > console output: https://syzkaller.appspot.com/x/log.txt?x=17ceb0ce900000
> > kernel config: https://syzkaller.appspot.com/x/.config?x=21f0d1d2df6d5fc
> > dashboard link: https://syzkaller.appspot.com/bug?extid=b305848212deec86eabe
> > compiler: clang version 10.0.0 (https://github.com/llvm/llvm-project/ c2443155a0fb245c8f17f2c1c72b6ea391e86e81)
>
> Presumably this is the same as
> http://lkml.kernel.org/r/00000000000011710f05ad27fe8a@xxxxxxxxxx.
Very likely.
> > Unfortunately, I don't have any reproducer for this issue yet.
> >
> > IMPORTANT: if you fix the issue, please add the following tag to the commit:
> > Reported-by: syzbot+b305848212deec86eabe@xxxxxxxxxxxxxxxxxxxxxxxxx
> >
> > ==================================================================
> > BUG: KASAN: use-after-free in instrument_atomic_write include/linux/instrumented.h:71 [inline]
> > BUG: KASAN: use-after-free in atomic64_sub_return include/asm-generic/atomic-instrumented.h:970 [inline]
> > BUG: KASAN: use-after-free in atomic_long_sub_return include/asm-generic/atomic-long.h:113 [inline]
> > BUG: KASAN: use-after-free in page_counter_cancel mm/page_counter.c:54 [inline]
> > BUG: KASAN: use-after-free in page_counter_uncharge+0x3d/0xc0 mm/page_counter.c:155
> > Write of size 8 at addr ffff8880371c0148 by task syz-executor.0/9304
> >
> > CPU: 0 PID: 9304 Comm: syz-executor.0 Not tainted 5.8.0-syzkaller #0
> > Hardware name: Google Google Compute Engine/Google Compute Engine, BIOS Google 01/01/2011
> > Call Trace:
> > __dump_stack lib/dump_stack.c:77 [inline]
> > dump_stack+0x1f0/0x31e lib/dump_stack.c:118
> > print_address_description+0x66/0x620 mm/kasan/report.c:383
> > __kasan_report mm/kasan/report.c:513 [inline]
> > kasan_report+0x132/0x1d0 mm/kasan/report.c:530
> > check_memory_region_inline mm/kasan/generic.c:183 [inline]
> > check_memory_region+0x2b5/0x2f0 mm/kasan/generic.c:192
> > instrument_atomic_write include/linux/instrumented.h:71 [inline]
> > atomic64_sub_return include/asm-generic/atomic-instrumented.h:970 [inline]
> > atomic_long_sub_return include/asm-generic/atomic-long.h:113 [inline]
> > page_counter_cancel mm/page_counter.c:54 [inline]
> > page_counter_uncharge+0x3d/0xc0 mm/page_counter.c:155
> > uncharge_batch+0x6c/0x350 mm/memcontrol.c:6764
> > uncharge_page+0x115/0x430 mm/memcontrol.c:6796
> > uncharge_list mm/memcontrol.c:6835 [inline]
> > mem_cgroup_uncharge_list+0x70/0xe0 mm/memcontrol.c:6877
> > release_pages+0x13a2/0x1550 mm/swap.c:911
This looks like a reference count unbalance when memcg is released
early. My first guess would be 1a3e1f40962c ("mm: memcontrol: decouple
reference counting from page accounting").
Unless I am missing something nothing really prevents the memcg for the
current batch to go away. uncharge_page collects all the charges for the
same memcg but it later drops the reference for the current page. Later
on when the memcg changes or when the final clean up is done in uncharge_list
uncharge_batch needs to access memcg but this might be after the last
page dropped the reference and memcg went away. The whole process of
tear down is quite complex and takes some time with all the RCU/WQ
involvement so this is quite unlikely to hit.
That being said the below should cure the reference count but I am not
sure this is a complete fix. If this looks reasonable I will post the
full patch. Johannes?
---
diff --git a/mm/memcontrol.c b/mm/memcontrol.c
index b807952b4d43..11b6dd1c4f64 100644
--- a/mm/memcontrol.c
+++ b/mm/memcontrol.c
@@ -6753,6 +6753,7 @@ struct uncharge_gather {
static inline void uncharge_gather_clear(struct uncharge_gather *ug)
{
+ css_put(&ug->memcg->css);
memset(ug, 0, sizeof(*ug));
}
@@ -6797,6 +6798,7 @@ static void uncharge_page(struct page *page, struct uncharge_gather *ug)
uncharge_gather_clear(ug);
}
ug->memcg = page->mem_cgroup;
+ css_get(&ug->memcg->css);
}
nr_pages = compound_nr(page);
--
Michal Hocko
SUSE Labs