Received: by 2002:a6b:500f:0:0:0:0:0 with SMTP id e15csp1187207iob; Thu, 19 May 2022 00:53:34 -0700 (PDT) X-Google-Smtp-Source: ABdhPJzhbkNY5m6mkxXtDjHaAMuEgCKi2wtItVDCnPGvvCEoQsWAKc0mIph/F0Hb5bbLMpuLf2zJ X-Received: by 2002:a63:2117:0:b0:3c2:85f9:1b6f with SMTP id h23-20020a632117000000b003c285f91b6fmr2968317pgh.66.1652946814767; Thu, 19 May 2022 00:53:34 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1652946814; cv=none; d=google.com; s=arc-20160816; b=UYpGfrO2XehDFKDBpROQGGIO2fFsDDPrweEOAhBhX58tScxAtlMCG8osuXCHvYB08A fPoyYrYe3B6cwEd1yojvJyrqqLpKOkC4DPWKLiIuVuYhoaQBoRrw72XwyEVRicYxfnWr x4meZI2/xeNeGYd5OkhkSAOxfDrb836IW9Ev7bBdbd6QZzOrT1SyjHurRlkG74Aplbdu 9DzDiYqkXj6QSyfb/57kQHDgECr4CMB8ZKYbbdxlHVQTl2w4lwND465jF6Uuruy5CJEk 4NTB8kzHhBfdKiMSe7VKGtPaionWP/ewWzpt/60Vt80cwnRRVJvJMOGaUaA6Tqk1UqP6 x+gA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:feedback-id:content-transfer-encoding :mime-version:references:in-reply-to:message-id:date:subject:cc:to :from:dkim-signature; bh=ETzLW+MBUZNXPayW3Chm6h6U8ObCVBXcowJbowiynow=; b=GPR94n8NPpC4DpSqjnH1ssyOjlcYwRRWPC/Ajs3PWu9bk/RTTXi9vso+vHbELM0yyL HItjG7nsSgF+HyaExPRNS4zyPosXCBBkizrjav7Ijt4h/kd25HIaxYjyRSeVTgkOj/0i ihi5XDRslBXlLFLHWoJzPXZd6DLTQ5mN/Nejq5D3mpb4JiyLBFYJYviVGU046I8M3Bfk hJGsifOweXOk69y6H4jKo/wu3NnpHoaSGJeW2V2BuxtSOMz5WwsEzwOyaFArNOPXzIVP FNXpTTWywMEBRHeJsDCeoS+04uqmpSsLlQH9oaNsgb09CSQ7ol1OrDYxenBYdaqvVerW e2xg== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@siemens.com header.s=fm1 header.b=GuG7GX9B; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=siemens.com Return-Path: Received: from out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id v8-20020aa78088000000b0051844a9a436si524348pff.220.2022.05.19.00.53.23; Thu, 19 May 2022 00:53:34 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) client-ip=2620:137:e000::1:20; Authentication-Results: mx.google.com; dkim=pass header.i=@siemens.com header.s=fm1 header.b=GuG7GX9B; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=siemens.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S233674AbiESHKA (ORCPT + 99 others); Thu, 19 May 2022 03:10:00 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:39248 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S234040AbiESHJZ (ORCPT ); Thu, 19 May 2022 03:09:25 -0400 Received: from mta-64-228.siemens.flowmailer.net (mta-64-228.siemens.flowmailer.net [185.136.64.228]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 5EC30B8BD9 for ; Thu, 19 May 2022 00:09:22 -0700 (PDT) Received: by mta-64-228.siemens.flowmailer.net with ESMTPSA id 202205190709203e46717e6d796fd0c6 for ; Thu, 19 May 2022 09:09:20 +0200 DKIM-Signature: v=1; a=rsa-sha256; q=dns/txt; c=relaxed/relaxed; s=fm1; d=siemens.com; i=daniel.starke@siemens.com; h=Date:From:Subject:To:Message-ID:MIME-Version:Content-Type:Content-Transfer-Encoding:Cc:References:In-Reply-To; bh=ETzLW+MBUZNXPayW3Chm6h6U8ObCVBXcowJbowiynow=; b=GuG7GX9BKym6JkvBwfUy3u2KvAjh8nkmqAg6ZOg3d3pFsBn5V4R4+DhgGVtVDXxJB/V4Pa ktJK4Mmh6dGUZ5eGALapmo+st8dCB/UMA1IFNH1qf+nK9wUcFWsxUcEeRs6NKuqUpN/KoBUO 9b7SNWUws17TdGlxtf/mPSz5rls2M=; From: "D. Starke" To: linux-serial@vger.kernel.org, gregkh@linuxfoundation.org, jirislaby@kernel.org Cc: linux-kernel@vger.kernel.org, Daniel Starke Subject: [PATCH v2 3/9] tty: n_gsm: fix wrong queuing behavior in gsm_dlci_data_output() Date: Thu, 19 May 2022 09:07:51 +0200 Message-Id: <20220519070757.2096-3-daniel.starke@siemens.com> In-Reply-To: <20220519070757.2096-1-daniel.starke@siemens.com> References: <20220519070757.2096-1-daniel.starke@siemens.com> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit X-Flowmailer-Platform: Siemens Feedback-ID: 519:519-314044:519-21489:flowmailer X-Spam-Status: No, score=-2.1 required=5.0 tests=BAYES_00,DKIM_SIGNED, DKIM_VALID,DKIM_VALID_AU,DKIM_VALID_EF,SPF_HELO_NONE,SPF_PASS, T_SCC_BODY_TEXT_LINE autolearn=unavailable autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: Daniel Starke 1) The function drains the fifo for the given user tty/DLCI without considering 'TX_THRESH_HI' and different to gsm_dlci_data_output_framed(), which moves only one packet from the user side to the internal transmission queue. We can only handle one packet at a time here if we want to allow DLCI priority handling in gsm_dlci_data_sweep() to avoid link starvation. 2) Furthermore, the additional header octet from convergence layer type 2 is not counted against MTU. It is part of the UI/UIH frame message which needs to be limited to MTU. Hence, it is wrong not to consider this octet. 3) Finally, the waiting user tty is not informed about freed space in its send queue. Take at most one packet worth of data out of the DLCI fifo to fix 1). Limit the max user data size per packet to MTU - 1 in case of convergence layer type 2 to leave space for the control signal octet which is added in the later part of the function. This fixes 2). Add tty_port_tty_wakeup() to wake up the user tty if new write space has been made available to fix 3). Fixes: 268e526b935e ("tty/n_gsm: avoid fifo overflow in gsm_dlci_data_output") Cc: stable@vger.kernel.org Signed-off-by: Daniel Starke --- drivers/tty/n_gsm.c | 74 +++++++++++++++++++++++++-------------------- 1 file changed, 42 insertions(+), 32 deletions(-) See patch 6 regarding changes since to v1. diff --git a/drivers/tty/n_gsm.c b/drivers/tty/n_gsm.c index e695956fc96c..0a9924445968 100644 --- a/drivers/tty/n_gsm.c +++ b/drivers/tty/n_gsm.c @@ -869,41 +869,51 @@ static int gsm_dlci_data_output(struct gsm_mux *gsm, struct gsm_dlci *dlci) { struct gsm_msg *msg; u8 *dp; - int len, total_size, size; - int h = dlci->adaption - 1; + int h, len, size; - total_size = 0; - while (1) { - len = kfifo_len(&dlci->fifo); - if (len == 0) - return total_size; - - /* MTU/MRU count only the data bits */ - if (len > gsm->mtu) - len = gsm->mtu; - - size = len + h; - - msg = gsm_data_alloc(gsm, dlci->addr, size, gsm->ftype); - /* FIXME: need a timer or something to kick this so it can't - get stuck with no work outstanding and no buffer free */ - if (msg == NULL) - return -ENOMEM; - dp = msg->data; - switch (dlci->adaption) { - case 1: /* Unstructured */ - break; - case 2: /* Unstructed with modem bits. - Always one byte as we never send inline break data */ - *dp++ = (gsm_encode_modem(dlci) << 1) | EA; - break; - } - WARN_ON(kfifo_out_locked(&dlci->fifo, dp , len, &dlci->lock) != len); - __gsm_data_queue(dlci, msg); - total_size += size; + /* for modem bits without break data */ + h = ((dlci->adaption == 1) ? 0 : 1); + + len = kfifo_len(&dlci->fifo); + if (len == 0) + return 0; + + /* MTU/MRU count only the data bits but watch adaption mode */ + if ((len + h) > gsm->mtu) + len = gsm->mtu - h; + + size = len + h; + + msg = gsm_data_alloc(gsm, dlci->addr, size, gsm->ftype); + /* FIXME: need a timer or something to kick this so it can't + * get stuck with no work outstanding and no buffer free + */ + if (!msg) + return -ENOMEM; + dp = msg->data; + switch (dlci->adaption) { + case 1: /* Unstructured */ + break; + case 2: /* Unstructured with modem bits. + * Always one byte as we never send inline break data + */ + *dp++ = (gsm_encode_modem(dlci) << 1) | EA; + break; + default: + pr_err("%s: unsupported adaption %d\n", __func__, + dlci->adaption); + break; } + + WARN_ON(len != kfifo_out_locked(&dlci->fifo, dp, len, + &dlci->lock)); + + /* Notify upper layer about available send space. */ + tty_port_tty_wakeup(&dlci->port); + + __gsm_data_queue(dlci, msg); /* Bytes of data we used up */ - return total_size; + return size; } /** -- 2.34.1