Received: by 2002:a05:6a10:16a7:0:0:0:0 with SMTP id gp39csp3851283pxb; Tue, 17 Nov 2020 05:23:26 -0800 (PST) X-Google-Smtp-Source: ABdhPJzs/PcLKDE3QW8S2+ZIV1MhesK3c29O+G2xDYN7mrOjDVrg9hD5zeRPYHnNJs8BQ1pFKSpG X-Received: by 2002:a17:906:b150:: with SMTP id bt16mr19726789ejb.257.1605619405963; Tue, 17 Nov 2020 05:23:25 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1605619405; cv=none; d=google.com; s=arc-20160816; b=YrzZoBzEHU6IxucoWEx4bHNe9H2oyOJU7/c14gTv31ZKtvx9kX1ieFCmOA3E8yRISw 84THayJt3loKyxtJuX6PyPevnOuU5S/XH0SUgERWRd1d9gS6YZMnytnb2qNl8ZIY2fe5 mdI/EC7cWJq1VOVCqg1aWBmFj9cHpb+9/ak26a3T36wo2Bwvkq6NG1/RznbWuSMwsql7 2sgm+htyTqD2+PNDPCRQWHB26yMvluoVZz7GAKVqDyTdDRrZLN05hP65dQz8d+r3Mp64 3COz3LqF3tp/hoyrj+rfymylNQi1xeKElung2LNEv5RWv5SR6PgfKKpWrpzeXYEM+E14 OCsA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:mime-version :user-agent:references:in-reply-to:message-id:date:subject:cc:to :from:dkim-signature; bh=XX/HMjAy4h3GyqV86hwpVS9KjumpDFsSTRP9pl8c/GY=; b=MpvTrHKXTYWmSyhPNkdPy0wWwKlsRKk+LttMrT1uW+PEMfU78GULGz+C2VwS3uhZhn +lvtz9WUoKTWwOjw0vQ3gjVS/sMyWKPtFnDqHpn8yD4UMNybfcTYtDGhVC/z2YWVs6Xf ZTfqeinHWpIXglVzrA+92eDw06Mk47CsWlQVIprBAnbZoFiyQ5S0dq5A8bir9YArc/tq dWbMAyZ2Ha10xxZIpeO0bIf058FQl1Om1TKOIh3vSymmbTGmYiYepXkbQMx1uVWJEGe5 G8J/Oub4RuH1ORRHgyNPgecOUy4wydF/pcLRJCfN8hPlnCIZ+Trdh8OjxEdyzUdyCo5N kX0A== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@kernel.org header.s=default header.b="u/UA9g2C"; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=linuxfoundation.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id f15si11532691edj.239.2020.11.17.05.23.03; Tue, 17 Nov 2020 05:23:25 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; dkim=pass header.i=@kernel.org header.s=default header.b="u/UA9g2C"; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=linuxfoundation.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1730716AbgKQNSp (ORCPT + 99 others); Tue, 17 Nov 2020 08:18:45 -0500 Received: from mail.kernel.org ([198.145.29.99]:48798 "EHLO mail.kernel.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1730403AbgKQNQo (ORCPT ); Tue, 17 Nov 2020 08:16:44 -0500 Received: from localhost (83-86-74-64.cable.dynamic.v4.ziggo.nl [83.86.74.64]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by mail.kernel.org (Postfix) with ESMTPSA id 7D954221EB; Tue, 17 Nov 2020 13:16:43 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=kernel.org; s=default; t=1605619004; bh=z3Or68PiewAElmm4wbmq3UlPvEMzqrO8teh/bEWY3tA=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=u/UA9g2C03eTHivHZjizY2dUeLytp7q9a1G4F8mJgAvAtcQWfj17pY06DYRNArgwJ m5rRzYf5kx4eJz8YOuXh/CTAn3RXGQTp92XjosCLE/ucUQwfHAcyrZJO/2JG83lihl I7G1gobM1K/XfyuoOrl43Fa4aqDE9fX8QDC57CW8= From: Greg Kroah-Hartman To: linux-kernel@vger.kernel.org Cc: Greg Kroah-Hartman , stable@vger.kernel.org, Michael Petlan , Jiri Olsa , Ingo Molnar , Peter Zijlstra , Namhyung Kim , Wade Mealing , Sudip Mukherjee Subject: [PATCH 4.14 82/85] perf/core: Fix race in the perf_mmap_close() function Date: Tue, 17 Nov 2020 14:05:51 +0100 Message-Id: <20201117122115.077694082@linuxfoundation.org> X-Mailer: git-send-email 2.29.2 In-Reply-To: <20201117122111.018425544@linuxfoundation.org> References: <20201117122111.018425544@linuxfoundation.org> User-Agent: quilt/0.66 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: Jiri Olsa commit f91072ed1b7283b13ca57fcfbece5a3b92726143 upstream. There's a possible race in perf_mmap_close() when checking ring buffer's mmap_count refcount value. The problem is that the mmap_count check is not atomic because we call atomic_dec() and atomic_read() separately. perf_mmap_close: ... atomic_dec(&rb->mmap_count); ... if (atomic_read(&rb->mmap_count)) goto out_put; free_uid out_put: ring_buffer_put(rb); /* could be last */ The race can happen when we have two (or more) events sharing same ring buffer and they go through atomic_dec() and then they both see 0 as refcount value later in atomic_read(). Then both will go on and execute code which is meant to be run just once. The code that detaches ring buffer is probably fine to be executed more than once, but the problem is in calling free_uid(), which will later on demonstrate in related crashes and refcount warnings, like: refcount_t: addition on 0; use-after-free. ... RIP: 0010:refcount_warn_saturate+0x6d/0xf ... Call Trace: prepare_creds+0x190/0x1e0 copy_creds+0x35/0x172 copy_process+0x471/0x1a80 _do_fork+0x83/0x3a0 __do_sys_wait4+0x83/0x90 __do_sys_clone+0x85/0xa0 do_syscall_64+0x5b/0x1e0 entry_SYSCALL_64_after_hwframe+0x44/0xa9 Using atomic decrease and check instead of separated calls. Tested-by: Michael Petlan Signed-off-by: Jiri Olsa Signed-off-by: Ingo Molnar Acked-by: Peter Zijlstra Acked-by: Namhyung Kim Acked-by: Wade Mealing Fixes: 9bb5d40cd93c ("perf: Fix mmap() accounting hole"); Link: https://lore.kernel.org/r/20200916115311.GE2301783@krava [sudip: used ring_buffer] Signed-off-by: Sudip Mukherjee Signed-off-by: Greg Kroah-Hartman --- kernel/events/core.c | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) --- a/kernel/events/core.c +++ b/kernel/events/core.c @@ -5207,11 +5207,11 @@ static void perf_pmu_output_stop(struct static void perf_mmap_close(struct vm_area_struct *vma) { struct perf_event *event = vma->vm_file->private_data; - struct ring_buffer *rb = ring_buffer_get(event); struct user_struct *mmap_user = rb->mmap_user; int mmap_locked = rb->mmap_locked; unsigned long size = perf_data_size(rb); + bool detach_rest = false; if (event->pmu->event_unmapped) event->pmu->event_unmapped(event, vma->vm_mm); @@ -5242,7 +5242,8 @@ static void perf_mmap_close(struct vm_ar mutex_unlock(&event->mmap_mutex); } - atomic_dec(&rb->mmap_count); + if (atomic_dec_and_test(&rb->mmap_count)) + detach_rest = true; if (!atomic_dec_and_mutex_lock(&event->mmap_count, &event->mmap_mutex)) goto out_put; @@ -5251,7 +5252,7 @@ static void perf_mmap_close(struct vm_ar mutex_unlock(&event->mmap_mutex); /* If there's still other mmap()s of this buffer, we're done. */ - if (atomic_read(&rb->mmap_count)) + if (!detach_rest) goto out_put; /*