Received: by 2002:a05:6a10:17d3:0:0:0:0 with SMTP id hz19csp175728pxb; Fri, 16 Apr 2021 02:47:32 -0700 (PDT) X-Google-Smtp-Source: ABdhPJwmZgGclbgfqqPrWenalziE+wM4HTzJ0AwHVeV04sfAQllI2g2OBL1FJ7UG6GFKq52YPbOz X-Received: by 2002:a62:7c05:0:b029:25a:3f7c:2fe2 with SMTP id x5-20020a627c050000b029025a3f7c2fe2mr1461978pfc.59.1618566451872; Fri, 16 Apr 2021 02:47:31 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1618566451; cv=none; d=google.com; s=arc-20160816; b=sy/BX7ZQF4p8VxGb40O4KgwGIM8zeXDK8sH1AEbG7It7FL7Ds+dXwbosIJZnQHAO9X tmzQF67f3Zf5U+P6m1ki5Qvm6U2uBaBeNoh9ZtJQJAl2gGFqcRemXMvePNICSu19I0Za i0I3cs6Q/G5WUxYucQSUQpXRu6226e77WSflJKRyvYWmB0HvOGXEGTjCHSy+X2Kgex8N fuL7dwvu16v7XVSdFrFP/n8W7t6gtD0J92c0oXN7Iujoe/g6ur3NxMFfQsP+9cZCvrCn jPBCQT57w7474mo99zMJNcnBGWX65pT6S/Er8FB/2WSdb9IlyBF+/lb6+RsA+vmKqJLT s5EA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:mime-version:message-id:date:subject:cc:to:from; bh=G+k0LvztAmWD/ih1Ia3fz1p9KlB1lgMKQT/GFIdmSR8=; b=q01XLiOufoaYTow+ENn7FqVZeUDeLHhpJ1s6EYEp1ONw/e6RNu2QpNW/LyB08CKLz0 MRMO9IOTHD9qTeY+VlmBxW3kVOmV0W8uuVIeYtNylmuWGwEllZR0KVKN+45x7xMJO4vb /XFBziCCY94ZfdA/YQ8B4/vT1u2/+7ny94PPnmlCmxOPm6OoPd8IEwHI0zG2QrjZS4MW zALx93gFWan/28iK6SX/oZkL9eopQocvf/IGQ4Ko5DXyCNiV1OnH6fmBFqKYWVml5oml rWJ5G70JASlo/wAspeen809tTHtiFkInl4x6Ye+EhxiXSXesWZ10zKt1XDgy4EdBG/fS oq1A== ARC-Authentication-Results: i=1; mx.google.com; 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=huawei.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id v3si6207994pff.73.2021.04.16.02.47.19; Fri, 16 Apr 2021 02:47:31 -0700 (PDT) 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; 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=huawei.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S240290AbhDPI1z (ORCPT + 99 others); Fri, 16 Apr 2021 04:27:55 -0400 Received: from szxga06-in.huawei.com ([45.249.212.32]:17367 "EHLO szxga06-in.huawei.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S237554AbhDPI1y (ORCPT ); Fri, 16 Apr 2021 04:27:54 -0400 Received: from DGGEMS411-HUB.china.huawei.com (unknown [172.30.72.59]) by szxga06-in.huawei.com (SkyGuard) with ESMTP id 4FM8RM2QhBzlYDD; Fri, 16 Apr 2021 16:25:35 +0800 (CST) Received: from huawei.com (10.67.174.53) by DGGEMS411-HUB.china.huawei.com (10.3.19.211) with Microsoft SMTP Server id 14.3.498.0; Fri, 16 Apr 2021 16:27:21 +0800 From: Liao Chang To: , , , , , , , , , CC: , , Subject: [PATCH v2] riscv/kprobe: Restore local irqflag if kprobe is cancelled Date: Fri, 16 Apr 2021 16:27:31 +0800 Message-ID: <20210416082731.121494-1-liaochang1@huawei.com> X-Mailer: git-send-email 2.17.1 MIME-Version: 1.0 Content-Type: text/plain X-Originating-IP: [10.67.174.53] X-CFilter-Loop: Reflected Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org The execution of sys_read end up hitting a BUG_ON() in __find_get_block after installing probe at sys_read via kprobe, the BUG message like the following: [ 65.708663] ------------[ cut here ]------------ [ 65.709987] kernel BUG at fs/buffer.c:1251! [ 65.711283] Kernel BUG [#1] [ 65.712032] Modules linked in: [ 65.712925] CPU: 0 PID: 51 Comm: sh Not tainted 5.12.0-rc4 #1 [ 65.714407] Hardware name: riscv-virtio,qemu (DT) [ 65.715696] epc : __find_get_block+0x218/0x2c8 [ 65.716835] ra : __getblk_gfp+0x1c/0x4a [ 65.717831] epc : ffffffe00019f11e ra : ffffffe00019f56a sp : ffffffe002437930 [ 65.719553] gp : ffffffe000f06030 tp : ffffffe0015abc00 t0 : ffffffe00191e038 [ 65.721290] t1 : ffffffe00191e038 t2 : 000000000000000a s0 : ffffffe002437960 [ 65.723051] s1 : ffffffe00160ad00 a0 : ffffffe00160ad00 a1 : 000000000000012a [ 65.724772] a2 : 0000000000000400 a3 : 0000000000000008 a4 : 0000000000000040 [ 65.726545] a5 : 0000000000000000 a6 : ffffffe00191e000 a7 : 0000000000000000 [ 65.728308] s2 : 000000000000012a s3 : 0000000000000400 s4 : 0000000000000008 [ 65.730049] s5 : 000000000000006c s6 : ffffffe00240f800 s7 : ffffffe000f080a8 [ 65.731802] s8 : 0000000000000001 s9 : 000000000000012a s10: 0000000000000008 [ 65.733516] s11: 0000000000000008 t3 : 00000000000003ff t4 : 000000000000000f [ 65.734434] t5 : 00000000000003ff t6 : 0000000000040000 [ 65.734613] status: 0000000000000100 badaddr: 0000000000000000 cause: 0000000000000003 [ 65.734901] Call Trace: [ 65.735076] [] __find_get_block+0x218/0x2c8 [ 65.735417] [] __ext4_get_inode_loc+0xb2/0x2f6 [ 65.735618] [] ext4_get_inode_loc+0x3a/0x8a [ 65.735802] [] ext4_reserve_inode_write+0x2e/0x8c [ 65.735999] [] __ext4_mark_inode_dirty+0x4c/0x18e [ 65.736208] [] ext4_dirty_inode+0x46/0x66 [ 65.736387] [] __mark_inode_dirty+0x12c/0x3da [ 65.736576] [] touch_atime+0x146/0x150 [ 65.736748] [] filemap_read+0x234/0x246 [ 65.736920] [] generic_file_read_iter+0xc0/0x114 [ 65.737114] [] ext4_file_read_iter+0x42/0xea [ 65.737310] [] new_sync_read+0xe2/0x15a [ 65.737483] [] vfs_read+0xca/0xf2 [ 65.737641] [] ksys_read+0x5e/0xc8 [ 65.737816] [] sys_read+0xe/0x16 [ 65.737973] [] ret_from_syscall+0x0/0x2 [ 65.738858] ---[ end trace fe93f985456c935d ]--- A simple reproducer looks like: echo 'p:myprobe sys_read fd=%a0 buf=%a1 count=%a2' > /sys/kernel/debug/tracing/kprobe_events echo 1 > /sys/kernel/debug/tracing/events/kprobes/myprobe/enable cat trace Here's what happens to hit that BUG_ON(): If instruction being single stepped caused page fault, the kprobe is cancelled to let the page fault handler continues as normal page fault. But the local irqflags are disabled, so CPU will restore 'sstatus' with 'SIE' masked. After page fault is serviced, the kprobe is triggered again, we overwrite the saved irqflag by calling kprobe_save_local_irqflag(). Note, 'SIE' is masked in this new saved irqflag. After kprobe is serviced, the CPU 'sstatus' is restored with 'SIE' masked. This overwritten 'sstatus' cause BUG_ON() in __find_get_block. This bug is already fixed on arm64 by Jisheng Zhang. Fixes: c22b0bcb1dd02 ("riscv: Add kprobes supported") Signed-off-by: Liao Chang --- Changes in v2: - Reorganize commit message. arch/riscv/kernel/probes/kprobes.c | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/arch/riscv/kernel/probes/kprobes.c b/arch/riscv/kernel/probes/kprobes.c index 7e2c78e2ca6b..d71f7c49a721 100644 --- a/arch/riscv/kernel/probes/kprobes.c +++ b/arch/riscv/kernel/probes/kprobes.c @@ -260,8 +260,10 @@ int __kprobes kprobe_fault_handler(struct pt_regs *regs, unsigned int trapnr) if (kcb->kprobe_status == KPROBE_REENTER) restore_previous_kprobe(kcb); - else + else { + kprobes_restore_local_irqflag(kcb, regs); reset_current_kprobe(); + } break; case KPROBE_HIT_ACTIVE: -- 2.17.1