Received: by 2002:a25:8b91:0:0:0:0:0 with SMTP id j17csp2298197ybl; Thu, 19 Dec 2019 11:11:37 -0800 (PST) X-Google-Smtp-Source: APXvYqxkCWHDQ6fUsxOl5VPepNZfwzk9O+CzKdVR2oP1PI9ScssQhlmeqEmRDYei1PlomsNmpqs8 X-Received: by 2002:a05:6808:b38:: with SMTP id t24mr2984867oij.110.1576782697306; Thu, 19 Dec 2019 11:11:37 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1576782697; cv=none; d=google.com; s=arc-20160816; b=XVhrMpeRo2bVfNg7ptVLQOXEYUdQosNcg8LS5sDTdF6w09TVK2cP4F1lXzglTmqQT/ BCKZttpM+jETvOgVmpA8IzoCPvTXzhwtN1vxcTgIUNEiZZMc0sn/hMsUOXqfzhHmqhwU VmC/39gVPzK3cCZHk0JgeL3Rl4gkiS8UqmUeg+4HYLkoMZJkDMyTlbyfcq4/a/5tTsNw s3DTgZHlgYmxbID+wDW3NXEXUCHe+M6L49zY4ZwK4/Dm1SlTUdtpnAlvqxyfbwrxOY8/ tpQqqqq4M9vLDR2M1qMf5kvbB0JRYkZZh+4LKJgCxdkbgVczM1UNhcjzRsXyGZkalTqO AF+w== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :user-agent:references:in-reply-to:message-id:date:subject:cc:to :from:dkim-signature; bh=+Fd4qPBmrz5BbDpBv/DOMRfCs2CEHS5Kimh56AlkB0A=; b=JA5DGbwORNFwAha99+I23Q9kTUrWkUY10ur0A8ieGL4yt/JVkvOGiX9iYIdg/ah8r/ vcm7Q5vpI29T5pAR8XjqoafWr0DdgqgScWbCx6KRvcMoyosBxW7MryuGk9HYxZXTqfqP FCPu9WxjBTbrtTYi2brQ68pgNrVL2neLl0qKR4tlISW2DbiekGu9m5RFAEX/sMc5uAaz qv5o/dOR3svEI7SCjhi2Tebr81Sot1ItCNduazAQZ10BOV87C0QdXzwcanKM23SakxTZ lD/ixYh6IOTj5og9NI/FOoKpeuaVfe8T7o0cVuVHedAmQavrHF2EN+5e15nytCEY+bvI 01JA== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@kernel.org header.s=default header.b=hKd4LV5I; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id i6si3001048otj.24.2019.12.19.11.11.24; Thu, 19 Dec 2019 11:11:37 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@kernel.org header.s=default header.b=hKd4LV5I; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1727821AbfLSSjH (ORCPT + 99 others); Thu, 19 Dec 2019 13:39:07 -0500 Received: from mail.kernel.org ([198.145.29.99]:56980 "EHLO mail.kernel.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1728110AbfLSSi6 (ORCPT ); Thu, 19 Dec 2019 13:38:58 -0500 Received: from localhost (83-86-89-107.cable.dynamic.v4.ziggo.nl [83.86.89.107]) (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 5E22424672; Thu, 19 Dec 2019 18:38:56 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=kernel.org; s=default; t=1576780736; bh=n0R84Wlbi1EN+zgAoGETcxrfBsDxb76mo6+qqDnhY10=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=hKd4LV5I0K6jVQdekYS2yewHjVsx+Cww2oe5KNRyJFLeFvbyXHxvXrUh2r6J7WKMX zEFcBa3YWHf9c6qmctKjKAp15Y/GO43EK9m3rANTHe/3gS+dLgIwLg6AKoQ3HutGpv bJiPe5X2BWk/WlByI8IWws/6Ob6fJ1QybKpJifMw= From: Greg Kroah-Hartman To: linux-kernel@vger.kernel.org Cc: Greg Kroah-Hartman , stable@vger.kernel.org, Pete Zaitcev , syzbot+56f9673bb4cdcbeb0e92@syzkaller.appspotmail.com, Alan Stern Subject: [PATCH 4.4 096/162] usb: mon: Fix a deadlock in usbmon between mmap and read Date: Thu, 19 Dec 2019 19:33:24 +0100 Message-Id: <20191219183213.631341595@linuxfoundation.org> X-Mailer: git-send-email 2.24.1 In-Reply-To: <20191219183150.477687052@linuxfoundation.org> References: <20191219183150.477687052@linuxfoundation.org> User-Agent: quilt/0.66 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: Pete Zaitcev commit 19e6317d24c25ee737c65d1ffb7483bdda4bb54a upstream. The problem arises because our read() function grabs a lock of the circular buffer, finds something of interest, then invokes copy_to_user() straight from the buffer, which in turn takes mm->mmap_sem. In the same time, the callback mon_bin_vma_fault() is invoked under mm->mmap_sem. It attempts to take the fetch lock and deadlocks. This patch does away with protecting of our page list with any semaphores, and instead relies on the kernel not close the device while mmap is active in a process. In addition, we prohibit re-sizing of a buffer while mmap is active. This way, when (now unlocked) fault is processed, it works with the page that is intended to be mapped-in, and not some other random page. Note that this may have an ABI impact, but hopefully no legitimate program is this wrong. Signed-off-by: Pete Zaitcev Reported-by: syzbot+56f9673bb4cdcbeb0e92@syzkaller.appspotmail.com Reviewed-by: Alan Stern Fixes: 46eb14a6e158 ("USB: fix usbmon BUG trigger") Cc: Link: https://lore.kernel.org/r/20191204203941.3503452b@suzdal.zaitcev.lan Signed-off-by: Greg Kroah-Hartman --- drivers/usb/mon/mon_bin.c | 32 +++++++++++++++++++++----------- 1 file changed, 21 insertions(+), 11 deletions(-) --- a/drivers/usb/mon/mon_bin.c +++ b/drivers/usb/mon/mon_bin.c @@ -1035,12 +1035,18 @@ static long mon_bin_ioctl(struct file *f mutex_lock(&rp->fetch_lock); spin_lock_irqsave(&rp->b_lock, flags); - mon_free_buff(rp->b_vec, rp->b_size/CHUNK_SIZE); - kfree(rp->b_vec); - rp->b_vec = vec; - rp->b_size = size; - rp->b_read = rp->b_in = rp->b_out = rp->b_cnt = 0; - rp->cnt_lost = 0; + if (rp->mmap_active) { + mon_free_buff(vec, size/CHUNK_SIZE); + kfree(vec); + ret = -EBUSY; + } else { + mon_free_buff(rp->b_vec, rp->b_size/CHUNK_SIZE); + kfree(rp->b_vec); + rp->b_vec = vec; + rp->b_size = size; + rp->b_read = rp->b_in = rp->b_out = rp->b_cnt = 0; + rp->cnt_lost = 0; + } spin_unlock_irqrestore(&rp->b_lock, flags); mutex_unlock(&rp->fetch_lock); } @@ -1212,13 +1218,21 @@ mon_bin_poll(struct file *file, struct p static void mon_bin_vma_open(struct vm_area_struct *vma) { struct mon_reader_bin *rp = vma->vm_private_data; + unsigned long flags; + + spin_lock_irqsave(&rp->b_lock, flags); rp->mmap_active++; + spin_unlock_irqrestore(&rp->b_lock, flags); } static void mon_bin_vma_close(struct vm_area_struct *vma) { + unsigned long flags; + struct mon_reader_bin *rp = vma->vm_private_data; + spin_lock_irqsave(&rp->b_lock, flags); rp->mmap_active--; + spin_unlock_irqrestore(&rp->b_lock, flags); } /* @@ -1230,16 +1244,12 @@ static int mon_bin_vma_fault(struct vm_a unsigned long offset, chunk_idx; struct page *pageptr; - mutex_lock(&rp->fetch_lock); offset = vmf->pgoff << PAGE_SHIFT; - if (offset >= rp->b_size) { - mutex_unlock(&rp->fetch_lock); + if (offset >= rp->b_size) return VM_FAULT_SIGBUS; - } chunk_idx = offset / CHUNK_SIZE; pageptr = rp->b_vec[chunk_idx].pg; get_page(pageptr); - mutex_unlock(&rp->fetch_lock); vmf->page = pageptr; return 0; }