Received: by 2002:a05:6a10:206:0:0:0:0 with SMTP id 6csp4680480pxj; Wed, 12 May 2021 10:44:16 -0700 (PDT) X-Google-Smtp-Source: ABdhPJyj/KaRHjJaFWJiDMK2ZWXyPX7nIoil93QegrWIzaef7GS/4Xrl5C6g0ZI/rXyikt3YiaPh X-Received: by 2002:a05:6402:14c1:: with SMTP id f1mr2176885edx.334.1620841327431; Wed, 12 May 2021 10:42:07 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1620841327; cv=none; d=google.com; s=arc-20160816; b=V9tzi4xwd/qhO+bXuH12+/0NuoeNtVWgZyLKhkQ7hbor+AW4bJ9Ab1NEqHpoyYJ39g YwK5LVmMD5uHhhh8cY8AIlhes6GRmKwNfOLxzWBWPb/ji5+WCgV/GD5rCLAQgCvtwvlr 1evva57O98fTwMn/L05cWNE8E9Ajc1gwIaR+duGy+l/8MAma91XXKKA4ZqU5bxlLUqJL NP/V4565WWc5RpCvKnmnk2jxkmrDaRHnnzBtcjv+KdslhHIhEyxU+f74QfihJ/t0QWOQ MeXS88lj0B2yRXBudfWFAXmJWYI9cyOME6Clj69fyDaHq1CpiKF1VscvaUkt9WaEZ0x0 0wwQ== 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=k+hNOgmLUgpTEPiH8yrZFgtHm4pojIT/cxkpKF4SWrs=; b=vjoDl53LkbGE9JJLAp+sWK72HPjOJoCRk24xBOkWpMyH053DyJU9ycnqbOUb6mKq0Z RrZBfCstDrEg8F+NAdzhGmJy3WPTjskYz5X/V7Gp68LJ4omc7Uu2zFQhgmIfK+kG13xj DblUyd2fBHa6MYv4Zonzy89Du9tiSs84G5e9O1y00qsBkDby7/ch71RN94Uw2hubRFY4 hILr/dYiJ0GyEj47TVHnfKsTLNYZy0A08XiPpwK+7HYuQ6bM5KrLYCRMmiVoxENd8azk pQ211D5aRT2S53GgU8ohFbdpHcXbf9wIAvhze1FA4fbjPM/Jj9KCLlcHxqnPHyHfLHZo i/mg== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@linuxfoundation.org header.s=korg header.b=a4z4rtXe; 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 gz4si558728ejc.241.2021.05.12.10.41.43; Wed, 12 May 2021 10:42:07 -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; dkim=pass header.i=@linuxfoundation.org header.s=korg header.b=a4z4rtXe; 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 S1347294AbhELR1y (ORCPT + 99 others); Wed, 12 May 2021 13:27:54 -0400 Received: from mail.kernel.org ([198.145.29.99]:36284 "EHLO mail.kernel.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S239271AbhELQHj (ORCPT ); Wed, 12 May 2021 12:07:39 -0400 Received: by mail.kernel.org (Postfix) with ESMTPSA id C1C8661C53; Wed, 12 May 2021 15:37:04 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=linuxfoundation.org; s=korg; t=1620833825; bh=kX150s1Ymg4Ua5g2q+5MLrgZHq/tJpM78ypxZUsZkf0=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=a4z4rtXeUG+bO/Ko35Exc+Etz63sWbss48Y6zSLXN6JUd7IeRDrCs/mqRKJodF0DS 2dUZf6FX8ZmhlrQFrGnY1F0kTreZeUVs+Up2C5hYJbADoR7YgTgC/4MS0H649Yvzfn DXUgoHq1T1FaSkU+DUiAk6MOwbTb1Hl9388CTOhQ= From: Greg Kroah-Hartman To: linux-kernel@vger.kernel.org Cc: Greg Kroah-Hartman , stable@vger.kernel.org, Dario Binacchi , Sasha Levin Subject: [PATCH 5.11 304/601] serial: omap: fix rs485 half-duplex filtering Date: Wed, 12 May 2021 16:46:21 +0200 Message-Id: <20210512144837.826683833@linuxfoundation.org> X-Mailer: git-send-email 2.31.1 In-Reply-To: <20210512144827.811958675@linuxfoundation.org> References: <20210512144827.811958675@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: Dario Binacchi [ Upstream commit e2a5e8448e7393e96ccde346c68764b40a52cc10 ] Data received during half-duplex transmission must be filtered. If the target device responds quickly, emptying the FIFO at the end of the transmission can erase not only the echo characters but also part of the response message. By keeping the receive interrupt enabled even during transmission, it allows you to filter each echo character and only in a number equal to those transmitted. The issue was generated by a target device that started responding 240us later having received a request in communication at 115200bps. Sometimes, some messages received by the target were missing some of the first bytes. Fixes: 3a13884abea0 ("tty/serial: omap: empty the RX FIFO at the end of half-duplex TX") Signed-off-by: Dario Binacchi Link: https://lore.kernel.org/r/20210418094705.27014-1-dariobin@libero.it Signed-off-by: Greg Kroah-Hartman Signed-off-by: Sasha Levin --- drivers/tty/serial/omap-serial.c | 39 ++++++++++++++++++++------------ 1 file changed, 24 insertions(+), 15 deletions(-) diff --git a/drivers/tty/serial/omap-serial.c b/drivers/tty/serial/omap-serial.c index 1583e93b2202..84e8158088cd 100644 --- a/drivers/tty/serial/omap-serial.c +++ b/drivers/tty/serial/omap-serial.c @@ -159,6 +159,8 @@ struct uart_omap_port { u32 calc_latency; struct work_struct qos_work; bool is_suspending; + + unsigned int rs485_tx_filter_count; }; #define to_uart_omap_port(p) ((container_of((p), struct uart_omap_port, port))) @@ -329,19 +331,6 @@ static void serial_omap_stop_tx(struct uart_port *port) serial_out(up, UART_IER, up->ier); } - if ((port->rs485.flags & SER_RS485_ENABLED) && - !(port->rs485.flags & SER_RS485_RX_DURING_TX)) { - /* - * Empty the RX FIFO, we are not interested in anything - * received during the half-duplex transmission. - */ - serial_out(up, UART_FCR, up->fcr | UART_FCR_CLEAR_RCVR); - /* Re-enable RX interrupts */ - up->ier |= UART_IER_RLSI | UART_IER_RDI; - up->port.read_status_mask |= UART_LSR_DR; - serial_out(up, UART_IER, up->ier); - } - pm_runtime_mark_last_busy(up->dev); pm_runtime_put_autosuspend(up->dev); } @@ -367,6 +356,10 @@ static void transmit_chars(struct uart_omap_port *up, unsigned int lsr) serial_out(up, UART_TX, up->port.x_char); up->port.icount.tx++; up->port.x_char = 0; + if ((up->port.rs485.flags & SER_RS485_ENABLED) && + !(up->port.rs485.flags & SER_RS485_RX_DURING_TX)) + up->rs485_tx_filter_count++; + return; } if (uart_circ_empty(xmit) || uart_tx_stopped(&up->port)) { @@ -378,6 +371,10 @@ static void transmit_chars(struct uart_omap_port *up, unsigned int lsr) serial_out(up, UART_TX, xmit->buf[xmit->tail]); xmit->tail = (xmit->tail + 1) & (UART_XMIT_SIZE - 1); up->port.icount.tx++; + if ((up->port.rs485.flags & SER_RS485_ENABLED) && + !(up->port.rs485.flags & SER_RS485_RX_DURING_TX)) + up->rs485_tx_filter_count++; + if (uart_circ_empty(xmit)) break; } while (--count > 0); @@ -421,7 +418,7 @@ static void serial_omap_start_tx(struct uart_port *port) if ((port->rs485.flags & SER_RS485_ENABLED) && !(port->rs485.flags & SER_RS485_RX_DURING_TX)) - serial_omap_stop_rx(port); + up->rs485_tx_filter_count = 0; serial_omap_enable_ier_thri(up); pm_runtime_mark_last_busy(up->dev); @@ -492,8 +489,13 @@ static void serial_omap_rlsi(struct uart_omap_port *up, unsigned int lsr) * Read one data character out to avoid stalling the receiver according * to the table 23-246 of the omap4 TRM. */ - if (likely(lsr & UART_LSR_DR)) + if (likely(lsr & UART_LSR_DR)) { serial_in(up, UART_RX); + if ((up->port.rs485.flags & SER_RS485_ENABLED) && + !(up->port.rs485.flags & SER_RS485_RX_DURING_TX) && + up->rs485_tx_filter_count) + up->rs485_tx_filter_count--; + } up->port.icount.rx++; flag = TTY_NORMAL; @@ -544,6 +546,13 @@ static void serial_omap_rdi(struct uart_omap_port *up, unsigned int lsr) return; ch = serial_in(up, UART_RX); + if ((up->port.rs485.flags & SER_RS485_ENABLED) && + !(up->port.rs485.flags & SER_RS485_RX_DURING_TX) && + up->rs485_tx_filter_count) { + up->rs485_tx_filter_count--; + return; + } + flag = TTY_NORMAL; up->port.icount.rx++; -- 2.30.2