2020-03-02 04:45:21 +08:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0 */
|
2016-09-16 00:42:35 +08:00
|
|
|
/*
|
|
|
|
* Copyright (C) Maxime Coquelin 2015
|
2017-07-13 23:08:26 +08:00
|
|
|
* Copyright (C) STMicroelectronics SA 2017
|
2016-09-16 00:42:35 +08:00
|
|
|
* Authors: Maxime Coquelin <mcoquelin.stm32@gmail.com>
|
|
|
|
* Gerald Baeza <gerald_baeza@yahoo.fr>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define DRIVER_NAME "stm32-usart"
|
|
|
|
|
|
|
|
struct stm32_usart_offsets {
|
|
|
|
u8 cr1;
|
|
|
|
u8 cr2;
|
|
|
|
u8 cr3;
|
|
|
|
u8 brr;
|
|
|
|
u8 gtpr;
|
|
|
|
u8 rtor;
|
|
|
|
u8 rqr;
|
|
|
|
u8 isr;
|
|
|
|
u8 icr;
|
|
|
|
u8 rdr;
|
|
|
|
u8 tdr;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct stm32_usart_config {
|
|
|
|
u8 uart_enable_bit; /* USART_CR1_UE */
|
|
|
|
bool has_7bits_data;
|
2021-03-28 23:43:06 +08:00
|
|
|
bool has_swap;
|
2017-07-13 23:08:30 +08:00
|
|
|
bool has_wakeup;
|
2017-07-13 23:08:30 +08:00
|
|
|
bool has_fifo;
|
2019-06-18 18:02:24 +08:00
|
|
|
int fifosize;
|
2016-09-16 00:42:35 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
struct stm32_usart_info {
|
|
|
|
struct stm32_usart_offsets ofs;
|
|
|
|
struct stm32_usart_config cfg;
|
|
|
|
};
|
|
|
|
|
2016-10-06 21:42:54 +08:00
|
|
|
#define UNDEF_REG 0xff
|
2016-09-16 00:42:35 +08:00
|
|
|
|
|
|
|
/* USART_SR (F4) / USART_ISR (F7) */
|
|
|
|
#define USART_SR_PE BIT(0)
|
|
|
|
#define USART_SR_FE BIT(1)
|
serial: stm32: rework RX over DMA
This patch reworks RX support over DMA to improve reliability:
- change dma buffer cyclic configuration by using 2 periods. DMA buffer
data are handled by a flip-flop between the 2 periods in order to avoid
risk of data loss/corruption
- change the size of dma buffer to 4096 to limit overruns
- add rx errors management (breaks, parity, framing and overrun).
When an error occurs on the uart line, the dma request line is masked at
HW level. The SW must 1st clear DMAR (dma request line enable), to
handle the error, then re-enable DMAR to recover. So, any correct data
is taken from the DMA buffer, before handling the error itself. Then
errors are handled from RDR/ISR/FIFO (e.g. in PIO mode). Last, DMA
reception is resumed.
- add a condition on DMA request line in DMA RX routines in order to
switch to PIO mode when no DMA request line is disabled, even if the DMA
channel is still enabled.
When the UART is wakeup source and is configured to use DMA for RX, any
incoming data that wakes up the system isn't correctly received.
At data reception, the irq_handler handles the WUF irq, and then the
data reception over DMA.
As the DMA transfer has been terminated at suspend, and will be restored
by resume callback (which has no yet been called by system), the data
can't be received.
The wake-up data has to be handled in PIO mode while suspend callback
has not been called.
Signed-off-by: Valentin Caron <valentin.caron@foss.st.com>
Signed-off-by: Erwan Le Ray <erwan.leray@foss.st.com>
Link: https://lore.kernel.org/r/20211020150332.10214-3-erwan.leray@foss.st.com
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2021-10-20 23:03:31 +08:00
|
|
|
#define USART_SR_NE BIT(2) /* F7 (NF for F4) */
|
2016-09-16 00:42:35 +08:00
|
|
|
#define USART_SR_ORE BIT(3)
|
|
|
|
#define USART_SR_IDLE BIT(4)
|
|
|
|
#define USART_SR_RXNE BIT(5)
|
|
|
|
#define USART_SR_TC BIT(6)
|
|
|
|
#define USART_SR_TXE BIT(7)
|
|
|
|
#define USART_SR_CTSIF BIT(9)
|
|
|
|
#define USART_SR_CTS BIT(10) /* F7 */
|
|
|
|
#define USART_SR_RTOF BIT(11) /* F7 */
|
|
|
|
#define USART_SR_EOBF BIT(12) /* F7 */
|
|
|
|
#define USART_SR_ABRE BIT(14) /* F7 */
|
|
|
|
#define USART_SR_ABRF BIT(15) /* F7 */
|
|
|
|
#define USART_SR_BUSY BIT(16) /* F7 */
|
|
|
|
#define USART_SR_CMF BIT(17) /* F7 */
|
|
|
|
#define USART_SR_SBKF BIT(18) /* F7 */
|
2017-07-13 23:08:30 +08:00
|
|
|
#define USART_SR_WUF BIT(20) /* H7 */
|
2016-09-16 00:42:35 +08:00
|
|
|
#define USART_SR_TEACK BIT(21) /* F7 */
|
serial: stm32: rework RX over DMA
This patch reworks RX support over DMA to improve reliability:
- change dma buffer cyclic configuration by using 2 periods. DMA buffer
data are handled by a flip-flop between the 2 periods in order to avoid
risk of data loss/corruption
- change the size of dma buffer to 4096 to limit overruns
- add rx errors management (breaks, parity, framing and overrun).
When an error occurs on the uart line, the dma request line is masked at
HW level. The SW must 1st clear DMAR (dma request line enable), to
handle the error, then re-enable DMAR to recover. So, any correct data
is taken from the DMA buffer, before handling the error itself. Then
errors are handled from RDR/ISR/FIFO (e.g. in PIO mode). Last, DMA
reception is resumed.
- add a condition on DMA request line in DMA RX routines in order to
switch to PIO mode when no DMA request line is disabled, even if the DMA
channel is still enabled.
When the UART is wakeup source and is configured to use DMA for RX, any
incoming data that wakes up the system isn't correctly received.
At data reception, the irq_handler handles the WUF irq, and then the
data reception over DMA.
As the DMA transfer has been terminated at suspend, and will be restored
by resume callback (which has no yet been called by system), the data
can't be received.
The wake-up data has to be handled in PIO mode while suspend callback
has not been called.
Signed-off-by: Valentin Caron <valentin.caron@foss.st.com>
Signed-off-by: Erwan Le Ray <erwan.leray@foss.st.com>
Link: https://lore.kernel.org/r/20211020150332.10214-3-erwan.leray@foss.st.com
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2021-10-20 23:03:31 +08:00
|
|
|
#define USART_SR_ERR_MASK (USART_SR_ORE | USART_SR_NE | USART_SR_FE |\
|
|
|
|
USART_SR_PE)
|
2016-09-16 00:42:35 +08:00
|
|
|
/* Dummy bits */
|
|
|
|
#define USART_SR_DUMMY_RX BIT(16)
|
|
|
|
|
|
|
|
/* USART_DR */
|
|
|
|
#define USART_DR_MASK GENMASK(8, 0)
|
|
|
|
|
|
|
|
/* USART_BRR */
|
|
|
|
#define USART_BRR_DIV_F_MASK GENMASK(3, 0)
|
|
|
|
#define USART_BRR_DIV_M_MASK GENMASK(15, 4)
|
|
|
|
#define USART_BRR_DIV_M_SHIFT 4
|
2018-03-12 17:50:05 +08:00
|
|
|
#define USART_BRR_04_R_SHIFT 1
|
2016-09-16 00:42:35 +08:00
|
|
|
|
|
|
|
/* USART_CR1 */
|
|
|
|
#define USART_CR1_SBK BIT(0)
|
|
|
|
#define USART_CR1_RWU BIT(1) /* F4 */
|
2017-07-13 23:08:30 +08:00
|
|
|
#define USART_CR1_UESM BIT(1) /* H7 */
|
2016-09-16 00:42:35 +08:00
|
|
|
#define USART_CR1_RE BIT(2)
|
|
|
|
#define USART_CR1_TE BIT(3)
|
|
|
|
#define USART_CR1_IDLEIE BIT(4)
|
|
|
|
#define USART_CR1_RXNEIE BIT(5)
|
|
|
|
#define USART_CR1_TCIE BIT(6)
|
|
|
|
#define USART_CR1_TXEIE BIT(7)
|
|
|
|
#define USART_CR1_PEIE BIT(8)
|
|
|
|
#define USART_CR1_PS BIT(9)
|
|
|
|
#define USART_CR1_PCE BIT(10)
|
|
|
|
#define USART_CR1_WAKE BIT(11)
|
2019-05-21 23:45:41 +08:00
|
|
|
#define USART_CR1_M0 BIT(12) /* F7 (CR1_M for F4) */
|
2016-09-16 00:42:35 +08:00
|
|
|
#define USART_CR1_MME BIT(13) /* F7 */
|
|
|
|
#define USART_CR1_CMIE BIT(14) /* F7 */
|
|
|
|
#define USART_CR1_OVER8 BIT(15)
|
|
|
|
#define USART_CR1_DEDT_MASK GENMASK(20, 16) /* F7 */
|
|
|
|
#define USART_CR1_DEAT_MASK GENMASK(25, 21) /* F7 */
|
|
|
|
#define USART_CR1_RTOIE BIT(26) /* F7 */
|
|
|
|
#define USART_CR1_EOBIE BIT(27) /* F7 */
|
|
|
|
#define USART_CR1_M1 BIT(28) /* F7 */
|
|
|
|
#define USART_CR1_IE_MASK (GENMASK(8, 4) | BIT(14) | BIT(26) | BIT(27))
|
2017-07-13 23:08:30 +08:00
|
|
|
#define USART_CR1_FIFOEN BIT(29) /* H7 */
|
2018-03-12 17:50:05 +08:00
|
|
|
#define USART_CR1_DEAT_SHIFT 21
|
|
|
|
#define USART_CR1_DEDT_SHIFT 16
|
2016-09-16 00:42:35 +08:00
|
|
|
|
|
|
|
/* USART_CR2 */
|
|
|
|
#define USART_CR2_ADD_MASK GENMASK(3, 0) /* F4 */
|
|
|
|
#define USART_CR2_ADDM7 BIT(4) /* F7 */
|
|
|
|
#define USART_CR2_LBCL BIT(8)
|
|
|
|
#define USART_CR2_CPHA BIT(9)
|
|
|
|
#define USART_CR2_CPOL BIT(10)
|
|
|
|
#define USART_CR2_CLKEN BIT(11)
|
|
|
|
#define USART_CR2_STOP_2B BIT(13)
|
|
|
|
#define USART_CR2_STOP_MASK GENMASK(13, 12)
|
|
|
|
#define USART_CR2_LINEN BIT(14)
|
|
|
|
#define USART_CR2_SWAP BIT(15) /* F7 */
|
|
|
|
#define USART_CR2_RXINV BIT(16) /* F7 */
|
|
|
|
#define USART_CR2_TXINV BIT(17) /* F7 */
|
|
|
|
#define USART_CR2_DATAINV BIT(18) /* F7 */
|
|
|
|
#define USART_CR2_MSBFIRST BIT(19) /* F7 */
|
|
|
|
#define USART_CR2_ABREN BIT(20) /* F7 */
|
|
|
|
#define USART_CR2_ABRMOD_MASK GENMASK(22, 21) /* F7 */
|
|
|
|
#define USART_CR2_RTOEN BIT(23) /* F7 */
|
|
|
|
#define USART_CR2_ADD_F7_MASK GENMASK(31, 24) /* F7 */
|
|
|
|
|
|
|
|
/* USART_CR3 */
|
|
|
|
#define USART_CR3_EIE BIT(0)
|
|
|
|
#define USART_CR3_IREN BIT(1)
|
|
|
|
#define USART_CR3_IRLP BIT(2)
|
|
|
|
#define USART_CR3_HDSEL BIT(3)
|
|
|
|
#define USART_CR3_NACK BIT(4)
|
|
|
|
#define USART_CR3_SCEN BIT(5)
|
|
|
|
#define USART_CR3_DMAR BIT(6)
|
|
|
|
#define USART_CR3_DMAT BIT(7)
|
|
|
|
#define USART_CR3_RTSE BIT(8)
|
|
|
|
#define USART_CR3_CTSE BIT(9)
|
|
|
|
#define USART_CR3_CTSIE BIT(10)
|
|
|
|
#define USART_CR3_ONEBIT BIT(11)
|
|
|
|
#define USART_CR3_OVRDIS BIT(12) /* F7 */
|
|
|
|
#define USART_CR3_DDRE BIT(13) /* F7 */
|
|
|
|
#define USART_CR3_DEM BIT(14) /* F7 */
|
|
|
|
#define USART_CR3_DEP BIT(15) /* F7 */
|
|
|
|
#define USART_CR3_SCARCNT_MASK GENMASK(19, 17) /* F7 */
|
2017-07-13 23:08:30 +08:00
|
|
|
#define USART_CR3_WUS_MASK GENMASK(21, 20) /* H7 */
|
|
|
|
#define USART_CR3_WUS_START_BIT BIT(21) /* H7 */
|
|
|
|
#define USART_CR3_WUFIE BIT(22) /* H7 */
|
2019-06-18 18:02:24 +08:00
|
|
|
#define USART_CR3_TXFTIE BIT(23) /* H7 */
|
|
|
|
#define USART_CR3_TCBGTIE BIT(24) /* H7 */
|
2019-06-18 18:02:25 +08:00
|
|
|
#define USART_CR3_RXFTCFG_MASK GENMASK(27, 25) /* H7 */
|
|
|
|
#define USART_CR3_RXFTCFG_SHIFT 25 /* H7 */
|
2019-06-18 18:02:24 +08:00
|
|
|
#define USART_CR3_RXFTIE BIT(28) /* H7 */
|
|
|
|
#define USART_CR3_TXFTCFG_MASK GENMASK(31, 29) /* H7 */
|
|
|
|
#define USART_CR3_TXFTCFG_SHIFT 29 /* H7 */
|
|
|
|
|
2016-09-16 00:42:35 +08:00
|
|
|
/* USART_GTPR */
|
|
|
|
#define USART_GTPR_PSC_MASK GENMASK(7, 0)
|
|
|
|
#define USART_GTPR_GT_MASK GENMASK(15, 8)
|
|
|
|
|
|
|
|
/* USART_RTOR */
|
|
|
|
#define USART_RTOR_RTO_MASK GENMASK(23, 0) /* F7 */
|
|
|
|
#define USART_RTOR_BLEN_MASK GENMASK(31, 24) /* F7 */
|
|
|
|
|
|
|
|
/* USART_RQR */
|
|
|
|
#define USART_RQR_ABRRQ BIT(0) /* F7 */
|
|
|
|
#define USART_RQR_SBKRQ BIT(1) /* F7 */
|
|
|
|
#define USART_RQR_MMRQ BIT(2) /* F7 */
|
|
|
|
#define USART_RQR_RXFRQ BIT(3) /* F7 */
|
|
|
|
#define USART_RQR_TXFRQ BIT(4) /* F7 */
|
|
|
|
|
|
|
|
/* USART_ICR */
|
|
|
|
#define USART_ICR_PECF BIT(0) /* F7 */
|
2019-05-21 23:45:42 +08:00
|
|
|
#define USART_ICR_FECF BIT(1) /* F7 */
|
2016-09-16 00:42:35 +08:00
|
|
|
#define USART_ICR_ORECF BIT(3) /* F7 */
|
|
|
|
#define USART_ICR_IDLECF BIT(4) /* F7 */
|
|
|
|
#define USART_ICR_TCCF BIT(6) /* F7 */
|
|
|
|
#define USART_ICR_CTSCF BIT(9) /* F7 */
|
|
|
|
#define USART_ICR_RTOCF BIT(11) /* F7 */
|
|
|
|
#define USART_ICR_EOBCF BIT(12) /* F7 */
|
|
|
|
#define USART_ICR_CMCF BIT(17) /* F7 */
|
2017-07-13 23:08:30 +08:00
|
|
|
#define USART_ICR_WUCF BIT(20) /* H7 */
|
2016-09-16 00:42:35 +08:00
|
|
|
|
2017-12-02 00:36:51 +08:00
|
|
|
#define STM32_SERIAL_NAME "ttySTM"
|
2017-07-13 23:08:27 +08:00
|
|
|
#define STM32_MAX_PORTS 8
|
2016-09-16 00:42:35 +08:00
|
|
|
|
serial: stm32: rework RX over DMA
This patch reworks RX support over DMA to improve reliability:
- change dma buffer cyclic configuration by using 2 periods. DMA buffer
data are handled by a flip-flop between the 2 periods in order to avoid
risk of data loss/corruption
- change the size of dma buffer to 4096 to limit overruns
- add rx errors management (breaks, parity, framing and overrun).
When an error occurs on the uart line, the dma request line is masked at
HW level. The SW must 1st clear DMAR (dma request line enable), to
handle the error, then re-enable DMAR to recover. So, any correct data
is taken from the DMA buffer, before handling the error itself. Then
errors are handled from RDR/ISR/FIFO (e.g. in PIO mode). Last, DMA
reception is resumed.
- add a condition on DMA request line in DMA RX routines in order to
switch to PIO mode when no DMA request line is disabled, even if the DMA
channel is still enabled.
When the UART is wakeup source and is configured to use DMA for RX, any
incoming data that wakes up the system isn't correctly received.
At data reception, the irq_handler handles the WUF irq, and then the
data reception over DMA.
As the DMA transfer has been terminated at suspend, and will be restored
by resume callback (which has no yet been called by system), the data
can't be received.
The wake-up data has to be handled in PIO mode while suspend callback
has not been called.
Signed-off-by: Valentin Caron <valentin.caron@foss.st.com>
Signed-off-by: Erwan Le Ray <erwan.leray@foss.st.com>
Link: https://lore.kernel.org/r/20211020150332.10214-3-erwan.leray@foss.st.com
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2021-10-20 23:03:31 +08:00
|
|
|
#define RX_BUF_L 4096 /* dma rx buffer length */
|
|
|
|
#define RX_BUF_P (RX_BUF_L / 2) /* dma rx buffer period */
|
|
|
|
#define TX_BUF_L RX_BUF_L /* dma tx buffer length */
|
2016-09-16 00:42:40 +08:00
|
|
|
|
2022-04-19 16:53:28 +08:00
|
|
|
#define STM32_USART_TIMEOUT_USEC USEC_PER_SEC /* 1s timeout in µs */
|
|
|
|
|
2016-09-16 00:42:35 +08:00
|
|
|
struct stm32_port {
|
|
|
|
struct uart_port port;
|
|
|
|
struct clk *clk;
|
2021-01-23 11:44:25 +08:00
|
|
|
const struct stm32_usart_info *info;
|
2016-09-16 00:42:40 +08:00
|
|
|
struct dma_chan *rx_ch; /* dma rx channel */
|
|
|
|
dma_addr_t rx_dma_buf; /* dma rx buffer bus address */
|
|
|
|
unsigned char *rx_buf; /* dma rx buffer cpu address */
|
|
|
|
struct dma_chan *tx_ch; /* dma tx channel */
|
|
|
|
dma_addr_t tx_dma_buf; /* dma tx buffer bus address */
|
|
|
|
unsigned char *tx_buf; /* dma tx buffer cpu address */
|
2019-06-18 18:02:22 +08:00
|
|
|
u32 cr1_irq; /* USART_CR1_RXNEIE or RTOIE */
|
2019-06-18 18:02:25 +08:00
|
|
|
u32 cr3_irq; /* USART_CR3_RXFTIE */
|
2017-07-13 23:08:27 +08:00
|
|
|
int last_res;
|
2022-01-05 02:24:43 +08:00
|
|
|
bool tx_dma_busy; /* dma tx transaction in progress */
|
2021-10-20 23:03:32 +08:00
|
|
|
bool throttled; /* port throttled */
|
2016-09-16 00:42:35 +08:00
|
|
|
bool hw_flow_control;
|
2021-03-28 23:43:06 +08:00
|
|
|
bool swap; /* swap RX & TX pins */
|
2017-07-13 23:08:30 +08:00
|
|
|
bool fifoen;
|
2022-05-01 00:28:45 +08:00
|
|
|
bool txdone;
|
2021-04-14 01:40:15 +08:00
|
|
|
int rxftcfg; /* RX FIFO threshold CFG */
|
|
|
|
int txftcfg; /* TX FIFO threshold CFG */
|
2021-03-20 02:42:52 +08:00
|
|
|
bool wakeup_src;
|
2019-05-21 23:45:43 +08:00
|
|
|
int rdr_mask; /* receive data register mask */
|
2020-04-21 01:02:04 +08:00
|
|
|
struct mctrl_gpios *gpios; /* modem control gpios */
|
serial: stm32: rework RX over DMA
This patch reworks RX support over DMA to improve reliability:
- change dma buffer cyclic configuration by using 2 periods. DMA buffer
data are handled by a flip-flop between the 2 periods in order to avoid
risk of data loss/corruption
- change the size of dma buffer to 4096 to limit overruns
- add rx errors management (breaks, parity, framing and overrun).
When an error occurs on the uart line, the dma request line is masked at
HW level. The SW must 1st clear DMAR (dma request line enable), to
handle the error, then re-enable DMAR to recover. So, any correct data
is taken from the DMA buffer, before handling the error itself. Then
errors are handled from RDR/ISR/FIFO (e.g. in PIO mode). Last, DMA
reception is resumed.
- add a condition on DMA request line in DMA RX routines in order to
switch to PIO mode when no DMA request line is disabled, even if the DMA
channel is still enabled.
When the UART is wakeup source and is configured to use DMA for RX, any
incoming data that wakes up the system isn't correctly received.
At data reception, the irq_handler handles the WUF irq, and then the
data reception over DMA.
As the DMA transfer has been terminated at suspend, and will be restored
by resume callback (which has no yet been called by system), the data
can't be received.
The wake-up data has to be handled in PIO mode while suspend callback
has not been called.
Signed-off-by: Valentin Caron <valentin.caron@foss.st.com>
Signed-off-by: Erwan Le Ray <erwan.leray@foss.st.com>
Link: https://lore.kernel.org/r/20211020150332.10214-3-erwan.leray@foss.st.com
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2021-10-20 23:03:31 +08:00
|
|
|
struct dma_tx_state rx_dma_state;
|
2016-09-16 00:42:35 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
static struct stm32_port stm32_ports[STM32_MAX_PORTS];
|
|
|
|
static struct uart_driver stm32_usart_driver;
|