Received: by 2002:a05:6a10:f347:0:0:0:0 with SMTP id d7csp343862pxu; Thu, 7 Jan 2021 06:33:59 -0800 (PST) X-Google-Smtp-Source: ABdhPJzRM+/K+osblfMYb9kGGUNXBIRm5kTRu2kmJSS1+nbydFkjt+sbSh1K8ahPSa/m4mq09YUv X-Received: by 2002:a17:906:c00c:: with SMTP id e12mr6369371ejz.103.1610030039763; Thu, 07 Jan 2021 06:33:59 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1610030039; cv=none; d=google.com; s=arc-20160816; b=MwEMNZglfYLWfgcbd8PJ6hoBeZDpF/+NX9xff2t4/lJ8qr9v3/Adu0HvtCiCqMQGIV KsoufCoEAHmofudR1ILC1AyeZ9eJoCrrE4payud820A0OhMDl2uQhw0i8LjbWgJqePTO GL4nf//vNK835Y2MqGGLDcLmu4mhQDGHT0PvgpoSGIQt/7rnGXNh9Gp+cZeBa47bbpQg pEEI6p709fOGnEgjwgpdQe5Ly6m3t9BujK04xTW6CJ4msq/GHMe0HmIkHFlW4OLNaNnh V1WXQi94476hUzWTQAd7bhLoEyFR9e59P6Zu8TUuMew0ePTcRCSVNTheOMOQeNejubUn ES0w== 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=u78eTtENhKCkeS/Az6OHTXLiOcbMO/vTnvaBxIHhjvQ=; b=ulq9sJUNLbXA1/nM6b4e2M6pTzJqH0fOc4J7NyKvefivuLc7qHav4vKSjz8wX6+Koo /mAX85iJMA7FLF+/gpVQ7MsBAAq9n5PvhjS3ZsQex0n72BXI1ouPQwQ0L9AMAoq90IMA MaSFOy6dapmCS4zeDP7JRK3ZjFXBknlckxoB/b/FY09mbXGSGF+TF2yhZWLU3fDihAp2 ErDPsv9AusO8R/uJSAVtNYad+7RcFabwT34+vj4tLn4AgraiDNiUNbmN8PyIlW0wtifV QELpPvrQHSc6WMJ1DWkQbIk8fxgwXQuImFoE2M0ydONOfBP7BAEj41UjTjOTRH+YeF24 00/w== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@linuxfoundation.org header.s=korg header.b=YqUJqvtU; 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=pass (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 a10si375590ejv.36.2021.01.07.06.33.36; Thu, 07 Jan 2021 06:33:59 -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=@linuxfoundation.org header.s=korg header.b=YqUJqvtU; 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=pass (p=NONE sp=NONE dis=NONE) header.from=linuxfoundation.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1728670AbhAGObX (ORCPT + 99 others); Thu, 7 Jan 2021 09:31:23 -0500 Received: from mail.kernel.org ([198.145.29.99]:45346 "EHLO mail.kernel.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1726319AbhAGObX (ORCPT ); Thu, 7 Jan 2021 09:31:23 -0500 Received: by mail.kernel.org (Postfix) with ESMTPSA id E5C3F23340; Thu, 7 Jan 2021 14:30:35 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=linuxfoundation.org; s=korg; t=1610029836; bh=jl339vriLGZLpgkDzivFjQWIKXWJVzq+jjPTlcZH6jA=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=YqUJqvtUWUNjRB66/82lzwPq+l8xHTZz1hfJ+DmGuVKJqj4YPHbc2uTMaoOQB70tn AGHgUQW0AxzfK+hCOk51MOCXDwCsvEHwvZPWwQYVQZYmj+vIaOawlomw70Zyx+sP9a j/4mKd4D2LRDsakkW/YidCmuYscWHa6xhGByPKso= From: Greg Kroah-Hartman To: linux-kernel@vger.kernel.org Cc: Greg Kroah-Hartman , stable@vger.kernel.org, Johan Hovold , Sasha Levin Subject: [PATCH 4.14 11/29] USB: serial: digi_acceleport: fix write-wakeup deadlocks Date: Thu, 7 Jan 2021 15:31:26 +0100 Message-Id: <20210107143054.549193648@linuxfoundation.org> X-Mailer: git-send-email 2.30.0 In-Reply-To: <20210107143052.973437064@linuxfoundation.org> References: <20210107143052.973437064@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: Johan Hovold [ Upstream commit 5098e77962e7c8947f87bd8c5869c83e000a522a ] The driver must not call tty_wakeup() while holding its private lock as line disciplines are allowed to call back into write() from write_wakeup(), leading to a deadlock. Also remove the unneeded work struct that was used to defer wakeup in order to work around a possible race in ancient times (see comment about n_tty write_chan() in commit 14b54e39b412 ("USB: serial: remove changelogs and old todo entries")). Fixes: 1da177e4c3f4 ("Linux-2.6.12-rc2") Cc: stable@vger.kernel.org Reviewed-by: Greg Kroah-Hartman Signed-off-by: Johan Hovold Signed-off-by: Sasha Levin --- drivers/usb/serial/digi_acceleport.c | 45 ++++++++-------------------- 1 file changed, 13 insertions(+), 32 deletions(-) diff --git a/drivers/usb/serial/digi_acceleport.c b/drivers/usb/serial/digi_acceleport.c index 2ce39af32cfa6..e494ffdc06bc9 100644 --- a/drivers/usb/serial/digi_acceleport.c +++ b/drivers/usb/serial/digi_acceleport.c @@ -23,7 +23,6 @@ #include #include #include -#include #include #include #include @@ -202,14 +201,12 @@ struct digi_port { int dp_throttle_restart; wait_queue_head_t dp_flush_wait; wait_queue_head_t dp_close_wait; /* wait queue for close */ - struct work_struct dp_wakeup_work; struct usb_serial_port *dp_port; }; /* Local Function Declarations */ -static void digi_wakeup_write_lock(struct work_struct *work); static int digi_write_oob_command(struct usb_serial_port *port, unsigned char *buf, int count, int interruptible); static int digi_write_inb_command(struct usb_serial_port *port, @@ -360,26 +357,6 @@ __releases(lock) return timeout; } - -/* - * Digi Wakeup Write - * - * Wake up port, line discipline, and tty processes sleeping - * on writes. - */ - -static void digi_wakeup_write_lock(struct work_struct *work) -{ - struct digi_port *priv = - container_of(work, struct digi_port, dp_wakeup_work); - struct usb_serial_port *port = priv->dp_port; - unsigned long flags; - - spin_lock_irqsave(&priv->dp_port_lock, flags); - tty_port_tty_wakeup(&port->port); - spin_unlock_irqrestore(&priv->dp_port_lock, flags); -} - /* * Digi Write OOB Command * @@ -990,6 +967,7 @@ static void digi_write_bulk_callback(struct urb *urb) struct digi_serial *serial_priv; int ret = 0; int status = urb->status; + bool wakeup; /* port and serial sanity check */ if (port == NULL || (priv = usb_get_serial_port_data(port)) == NULL) { @@ -1016,6 +994,7 @@ static void digi_write_bulk_callback(struct urb *urb) } /* try to send any buffered data on this port */ + wakeup = true; spin_lock(&priv->dp_port_lock); priv->dp_write_urb_in_use = 0; if (priv->dp_out_buf_len > 0) { @@ -1031,19 +1010,18 @@ static void digi_write_bulk_callback(struct urb *urb) if (ret == 0) { priv->dp_write_urb_in_use = 1; priv->dp_out_buf_len = 0; + wakeup = false; } } - /* wake up processes sleeping on writes immediately */ - tty_port_tty_wakeup(&port->port); - /* also queue up a wakeup at scheduler time, in case we */ - /* lost the race in write_chan(). */ - schedule_work(&priv->dp_wakeup_work); - spin_unlock(&priv->dp_port_lock); + if (ret && ret != -EPERM) dev_err_console(port, "%s: usb_submit_urb failed, ret=%d, port=%d\n", __func__, ret, priv->dp_port_num); + + if (wakeup) + tty_port_tty_wakeup(&port->port); } static int digi_write_room(struct tty_struct *tty) @@ -1243,7 +1221,6 @@ static int digi_port_init(struct usb_serial_port *port, unsigned port_num) init_waitqueue_head(&priv->dp_transmit_idle_wait); init_waitqueue_head(&priv->dp_flush_wait); init_waitqueue_head(&priv->dp_close_wait); - INIT_WORK(&priv->dp_wakeup_work, digi_wakeup_write_lock); priv->dp_port = port; init_waitqueue_head(&port->write_wait); @@ -1510,13 +1487,14 @@ static int digi_read_oob_callback(struct urb *urb) rts = C_CRTSCTS(tty); if (tty && opcode == DIGI_CMD_READ_INPUT_SIGNALS) { + bool wakeup = false; + spin_lock(&priv->dp_port_lock); /* convert from digi flags to termiox flags */ if (val & DIGI_READ_INPUT_SIGNALS_CTS) { priv->dp_modem_signals |= TIOCM_CTS; - /* port must be open to use tty struct */ if (rts) - tty_port_tty_wakeup(&port->port); + wakeup = true; } else { priv->dp_modem_signals &= ~TIOCM_CTS; /* port must be open to use tty struct */ @@ -1535,6 +1513,9 @@ static int digi_read_oob_callback(struct urb *urb) priv->dp_modem_signals &= ~TIOCM_CD; spin_unlock(&priv->dp_port_lock); + + if (wakeup) + tty_port_tty_wakeup(&port->port); } else if (opcode == DIGI_CMD_TRANSMIT_IDLE) { spin_lock(&priv->dp_port_lock); priv->dp_transmit_idle = 1; -- 2.27.0