tty: n_gsm: fix wrong command retry handling
commitd0bcdffcadupstream. n_gsm is based on the 3GPP 07.010 and its newer version is the 3GPP 27.010. See https://portal.3gpp.org/desktopmodules/Specifications/SpecificationDetails.aspx?specificationId=1516 The changes from 07.010 to 27.010 are non-functional. Therefore, I refer to the newer 27.010 here. Chapter 5.7.3 states that the valid range for the maximum number of retransmissions (N2) is from 0 to 255 (both including). gsm_config() fails to limit this range correctly. Furthermore, gsm_control_retransmit() handles this number incorrectly by performing N2 - 1 retransmission attempts. Setting N2 to zero results in more than 255 retransmission attempts. Fix the range check in gsm_config() and the value handling in gsm_control_send() and gsm_control_retransmit() to comply with 3GPP 27.010. Fixes:e1eaea46bb("tty: n_gsm line discipline") Cc: stable@vger.kernel.org Signed-off-by: Daniel Starke <daniel.starke@siemens.com> Link: https://lore.kernel.org/r/20220414094225.4527-11-daniel.starke@siemens.com Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
parent
17b86db43c
commit
935f314b6f
1 changed files with 3 additions and 3 deletions
|
|
@ -1324,7 +1324,6 @@ static void gsm_control_retransmit(struct timer_list *t)
|
||||||
spin_lock_irqsave(&gsm->control_lock, flags);
|
spin_lock_irqsave(&gsm->control_lock, flags);
|
||||||
ctrl = gsm->pending_cmd;
|
ctrl = gsm->pending_cmd;
|
||||||
if (ctrl) {
|
if (ctrl) {
|
||||||
gsm->cretries--;
|
|
||||||
if (gsm->cretries == 0) {
|
if (gsm->cretries == 0) {
|
||||||
gsm->pending_cmd = NULL;
|
gsm->pending_cmd = NULL;
|
||||||
ctrl->error = -ETIMEDOUT;
|
ctrl->error = -ETIMEDOUT;
|
||||||
|
|
@ -1333,6 +1332,7 @@ static void gsm_control_retransmit(struct timer_list *t)
|
||||||
wake_up(&gsm->event);
|
wake_up(&gsm->event);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
gsm->cretries--;
|
||||||
gsm_control_transmit(gsm, ctrl);
|
gsm_control_transmit(gsm, ctrl);
|
||||||
mod_timer(&gsm->t2_timer, jiffies + gsm->t2 * HZ / 100);
|
mod_timer(&gsm->t2_timer, jiffies + gsm->t2 * HZ / 100);
|
||||||
}
|
}
|
||||||
|
|
@ -1373,7 +1373,7 @@ retry:
|
||||||
|
|
||||||
/* If DLCI0 is in ADM mode skip retries, it won't respond */
|
/* If DLCI0 is in ADM mode skip retries, it won't respond */
|
||||||
if (gsm->dlci[0]->mode == DLCI_MODE_ADM)
|
if (gsm->dlci[0]->mode == DLCI_MODE_ADM)
|
||||||
gsm->cretries = 1;
|
gsm->cretries = 0;
|
||||||
else
|
else
|
||||||
gsm->cretries = gsm->n2;
|
gsm->cretries = gsm->n2;
|
||||||
|
|
||||||
|
|
@ -2270,7 +2270,7 @@ static int gsm_config(struct gsm_mux *gsm, struct gsm_config *c)
|
||||||
/* Check the MRU/MTU range looks sane */
|
/* Check the MRU/MTU range looks sane */
|
||||||
if (c->mru > MAX_MRU || c->mtu > MAX_MTU || c->mru < 8 || c->mtu < 8)
|
if (c->mru > MAX_MRU || c->mtu > MAX_MTU || c->mru < 8 || c->mtu < 8)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
if (c->n2 < 3)
|
if (c->n2 > 255)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
if (c->encapsulation > 1) /* Basic, advanced, no I */
|
if (c->encapsulation > 1) /* Basic, advanced, no I */
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue