/* * Copyright (c) 2016, Intel Corporation * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * 3. Neither the name of the Intel Corporation nor the names of its * contributors may be used to endorse or promote products derived from this * software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE INTEL CORPORATION OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #include #include "qm_uart.h" #ifndef UNIT_TEST qm_uart_reg_t *qm_uart[QM_UART_NUM] = {(qm_uart_reg_t *)QM_UART_0_BASE, (qm_uart_reg_t *)QM_UART_1_BASE}; #endif typedef void (*uart_client_callback_t)(void *data, int error, qm_uart_status_t status, uint32_t len); /** * DMA transfer information, relevant on callback invocations from the DMA * driver. */ typedef struct { qm_dma_channel_id_t dma_channel_id; /**< DMA channel. */ const qm_uart_transfer_t *xfer; /**< User transfer structure. */ } dma_context_t; /** * Parameters returned by DMA driver on DMA transfer complete callback. */ typedef volatile struct { void *context; /**< Pointer to dma_context_t struct. */ uint32_t len; /**< Amount of data successfully transferred. */ int error_code; /**< Error code of failed transfer. */ } dma_callback_par_t; /* Buffer pointers to store transmit / receive data for UART */ static uint32_t write_pos[QM_UART_NUM]; static uint32_t read_pos[QM_UART_NUM]; static const qm_uart_transfer_t *uart_read_transfer[QM_UART_NUM]; static const qm_uart_transfer_t *uart_write_transfer[QM_UART_NUM]; /* DMA (memory to UART) callback information. */ static dma_context_t dma_context_tx[QM_UART_NUM]; /* DMA (UART to memory) callback information. */ static dma_context_t dma_context_rx[QM_UART_NUM]; /* DMA core being used by each UART. */ static qm_dma_t dma_core[QM_UART_NUM]; /* DMA callback parameters returned by DMA driver (TX transfers). */ static dma_callback_par_t dma_delayed_callback_par[QM_UART_NUM]; static bool is_read_xfer_complete(const qm_uart_t uart) { const qm_uart_transfer_t *const transfer = uart_read_transfer[uart]; return read_pos[uart] >= transfer->data_len; } static bool is_write_xfer_complete(const qm_uart_t uart) { const qm_uart_transfer_t *const transfer = uart_write_transfer[uart]; return write_pos[uart] >= transfer->data_len; } static void uart_client_callback(qm_uart_t uart, void *data, int error, uint32_t len); static void qm_uart_isr_handler(const qm_uart_t uart) { qm_uart_reg_t *const regs = QM_UART[uart]; uint8_t interrupt_id = regs->iir_fcr & QM_UART_IIR_IID_MASK; const qm_uart_transfer_t *const read_transfer = uart_read_transfer[uart]; const qm_uart_transfer_t *const write_transfer = uart_write_transfer[uart]; /* * Interrupt ID priority levels (from highest to lowest): * 1: QM_UART_IIR_RECV_LINE_STATUS * 2: QM_UART_IIR_RECV_DATA_AVAIL and QM_UART_IIR_CHAR_TIMEOUT * 3: QM_UART_IIR_THR_EMPTY */ switch (interrupt_id) { case QM_UART_IIR_THR_EMPTY: if (dma_delayed_callback_par[uart].context) { /* * A DMA TX transfer just completed, disable interrupt * and invoke client callback. */ regs->ier_dlh &= ~QM_UART_IER_ETBEI; uart_client_callback( uart, dma_delayed_callback_par[uart].context, dma_delayed_callback_par[uart].error_code, dma_delayed_callback_par[uart].len); dma_delayed_callback_par[uart].context = NULL; return; } if (is_write_xfer_complete(uart)) { regs->ier_dlh &= ~QM_UART_IER_ETBEI; /* * At this point the FIFOs are empty, but the shift * register still is transmitting the last 8 bits. So if * we were to read LSR, it would say the device is still * busy. Use the SCR Bit 0 to indicate an irq tx is * complete. */ regs->scr |= BIT(0); if (write_transfer->callback) { write_transfer->callback( write_transfer->callback_data, 0, QM_UART_IDLE, write_pos[uart]); } return; } /* * If we are starting the transfer then the TX FIFO is empty. * In that case we set 'count' variable to QM_UART_FIFO_DEPTH * in order to take advantage of the whole FIFO capacity. */ int count = (write_pos[uart] == 0) ? QM_UART_FIFO_DEPTH : QM_UART_FIFO_HALF_DEPTH; while (count-- && !is_write_xfer_complete(uart)) { regs->rbr_thr_dll = write_transfer->data[write_pos[uart]++]; } /* * Change the threshold level to trigger an interrupt when the * TX buffer is empty. */ if (is_write_xfer_complete(uart)) { regs->iir_fcr = QM_UART_FCR_TX_0_RX_1_2_THRESHOLD | QM_UART_FCR_FIFOE; } break; case QM_UART_IIR_CHAR_TIMEOUT: case QM_UART_IIR_RECV_DATA_AVAIL: /* * Copy data from RX FIFO to xfer buffer as long as the xfer * has not completed and we have data in the RX FIFO. */ while (!is_read_xfer_complete(uart)) { uint32_t lsr = regs->lsr; /* * A break condition may cause a line status interrupt * to follow very closely after a char timeout * interrupt, but reading the lsr effectively clears the * pending interrupts so we issue here the callback * instead, otherwise we would miss it. * NOTE: Returned len is 0 for now, this might change * in the future. */ if (lsr & QM_UART_LSR_ERROR_BITS) { if (read_transfer->callback) { read_transfer->callback( read_transfer->callback_data, -EIO, lsr & QM_UART_LSR_ERROR_BITS, 0); } } if (lsr & QM_UART_LSR_DR) { read_transfer->data[read_pos[uart]++] = regs->rbr_thr_dll; } else { /* No more data in the RX FIFO */ break; } } if (is_read_xfer_complete(uart)) { /* * Disable both 'Receiver Data Available' and * 'Receiver Line Status' interrupts. */ regs->ier_dlh &= ~(QM_UART_IER_ERBFI | QM_UART_IER_ELSI); if (read_transfer->callback) { read_transfer->callback( read_transfer->callback_data, 0, QM_UART_IDLE, read_pos[uart]); } } break; case QM_UART_IIR_RECV_LINE_STATUS: if (read_transfer->callback) { /* * NOTE: Returned len is 0 for now, this might change * in the future. */ read_transfer->callback( read_transfer->callback_data, -EIO, regs->lsr & QM_UART_LSR_ERROR_BITS, 0); } break; } } QM_ISR_DECLARE(qm_uart_0_isr) { qm_uart_isr_handler(QM_UART_0); QM_ISR_EOI(QM_IRQ_UART_0_INT_VECTOR); } QM_ISR_DECLARE(qm_uart_1_isr) { qm_uart_isr_handler(QM_UART_1); QM_ISR_EOI(QM_IRQ_UART_1_INT_VECTOR); } int qm_uart_set_config(const qm_uart_t uart, const qm_uart_config_t *cfg) { QM_CHECK(uart < QM_UART_NUM, -EINVAL); QM_CHECK(cfg != NULL, -EINVAL); qm_uart_reg_t *const regs = QM_UART[uart]; volatile uint32_t unused_lsr __attribute__((unused)); /* Clear DLAB by unsetting line parameters */ regs->lcr = 0; /* Set divisor latch registers (integer + fractional part) */ regs->lcr = QM_UART_LCR_DLAB; regs->ier_dlh = QM_UART_CFG_BAUD_DLH_UNPACK(cfg->baud_divisor); regs->rbr_thr_dll = QM_UART_CFG_BAUD_DLL_UNPACK(cfg->baud_divisor); regs->dlf = QM_UART_CFG_BAUD_DLF_UNPACK(cfg->baud_divisor); /* Set line parameters. This also unsets the DLAB */ regs->lcr = cfg->line_control; /* Hardware automatic flow control */ regs->mcr = 0; if (true == cfg->hw_fc) { regs->mcr |= QM_UART_MCR_AFCE | QM_UART_MCR_RTS; } /* FIFO's enable and reset, set interrupt threshold */ regs->iir_fcr = (QM_UART_FCR_FIFOE | QM_UART_FCR_RFIFOR | QM_UART_FCR_XFIFOR | QM_UART_FCR_DEFAULT_TX_RX_THRESHOLD); regs->ier_dlh |= QM_UART_IER_PTIME; /* Clear LSR */ unused_lsr = regs->lsr; return 0; } int qm_uart_get_status(const qm_uart_t uart, qm_uart_status_t *const status) { QM_CHECK(uart < QM_UART_NUM, -EINVAL); QM_CHECK(status != NULL, -EINVAL); qm_uart_reg_t *const regs = QM_UART[uart]; uint32_t lsr = regs->lsr; *status = (lsr & (QM_UART_LSR_OE | QM_UART_LSR_PE | QM_UART_LSR_FE | QM_UART_LSR_BI)); /* * Check as an IRQ TX completed, if so, the Shift register may still be * busy. */ if (regs->scr & BIT(0)) { regs->scr &= ~BIT(0); } else if (!(lsr & (QM_UART_LSR_TEMT))) { *status |= QM_UART_TX_BUSY; } if (lsr & QM_UART_LSR_DR) { *status |= QM_UART_RX_BUSY; } return 0; } int qm_uart_write(const qm_uart_t uart, const uint8_t data) { QM_CHECK(uart < QM_UART_NUM, -EINVAL); qm_uart_reg_t *const regs = QM_UART[uart]; while (regs->lsr & QM_UART_LSR_THRE) { } regs->rbr_thr_dll = data; /* Wait for transaction to complete. */ while (!(regs->lsr & QM_UART_LSR_TEMT)) { } return 0; } int qm_uart_read(const qm_uart_t uart, uint8_t *const data, qm_uart_status_t *status) { QM_CHECK(uart < QM_UART_NUM, -EINVAL); QM_CHECK(data != NULL, -EINVAL); qm_uart_reg_t *const regs = QM_UART[uart]; uint32_t lsr = regs->lsr; while (!(lsr & QM_UART_LSR_DR)) { lsr = regs->lsr; } /* Check if there are any errors on the line. */ if (lsr & QM_UART_LSR_ERROR_BITS) { if (status) { *status = (lsr & QM_UART_LSR_ERROR_BITS); } return -EIO; } *data = regs->rbr_thr_dll; return 0; } int qm_uart_write_non_block(const qm_uart_t uart, const uint8_t data) { QM_CHECK(uart < QM_UART_NUM, -EINVAL); qm_uart_reg_t *const regs = QM_UART[uart]; regs->rbr_thr_dll = data; return 0; } int qm_uart_read_non_block(const qm_uart_t uart, uint8_t *const data) { QM_CHECK(uart < QM_UART_NUM, -EINVAL); QM_CHECK(data != NULL, -EINVAL); qm_uart_reg_t *const regs = QM_UART[uart]; *data = regs->rbr_thr_dll; return 0; } int qm_uart_write_buffer(const qm_uart_t uart, const uint8_t *const data, uint32_t len) { QM_CHECK(uart < QM_UART_NUM, -EINVAL); QM_CHECK(data != NULL, -EINVAL); qm_uart_reg_t *const regs = QM_UART[uart]; uint8_t *d = (uint8_t *)data; while (len--) { /* * Because FCR_FIFOE and IER_PTIME are enabled, LSR_THRE * behaves as a TX FIFO full indicator. */ while (regs->lsr & QM_UART_LSR_THRE) { } regs->rbr_thr_dll = *d; d++; } /* Wait for transaction to complete. */ while (!(regs->lsr & QM_UART_LSR_TEMT)) { } return 0; } int qm_uart_irq_write(const qm_uart_t uart, const qm_uart_transfer_t *const xfer) { QM_CHECK(uart < QM_UART_NUM, -EINVAL); QM_CHECK(xfer != NULL, -EINVAL); qm_uart_reg_t *const regs = QM_UART[uart]; write_pos[uart] = 0; uart_write_transfer[uart] = xfer; /* Set threshold */ regs->iir_fcr = (QM_UART_FCR_FIFOE | QM_UART_FCR_DEFAULT_TX_RX_THRESHOLD); /* Enable TX holding reg empty interrupt. */ regs->ier_dlh |= QM_UART_IER_ETBEI; return 0; } int qm_uart_irq_read(const qm_uart_t uart, const qm_uart_transfer_t *const xfer) { QM_CHECK(uart < QM_UART_NUM, -EINVAL); QM_CHECK(xfer != NULL, -EINVAL); qm_uart_reg_t *const regs = QM_UART[uart]; read_pos[uart] = 0; uart_read_transfer[uart] = xfer; /* Set threshold */ regs->iir_fcr = (QM_UART_FCR_FIFOE | QM_UART_FCR_DEFAULT_TX_RX_THRESHOLD); /* * Enable both 'Receiver Data Available' and 'Receiver * Line Status' interrupts. */ regs->ier_dlh |= QM_UART_IER_ERBFI | QM_UART_IER_ELSI; return 0; } int qm_uart_irq_write_terminate(const qm_uart_t uart) { QM_CHECK(uart < QM_UART_NUM, -EINVAL); qm_uart_reg_t *const regs = QM_UART[uart]; const qm_uart_transfer_t *const transfer = uart_write_transfer[uart]; /* Disable TX holding reg empty interrupt. */ regs->ier_dlh &= ~QM_UART_IER_ETBEI; if (transfer->callback) { transfer->callback(transfer->callback_data, -ECANCELED, QM_UART_IDLE, write_pos[uart]); } return 0; } int qm_uart_irq_read_terminate(const qm_uart_t uart) { QM_CHECK(uart < QM_UART_NUM, -EINVAL); qm_uart_reg_t *const regs = QM_UART[uart]; const qm_uart_transfer_t *const transfer = uart_read_transfer[uart]; /* * Disable both 'Receiver Data Available' and 'Receiver Line Status' * interrupts. */ regs->ier_dlh &= ~(QM_UART_IER_ERBFI | QM_UART_IER_ELSI); if (transfer->callback) { transfer->callback(transfer->callback_data, -ECANCELED, QM_UART_IDLE, read_pos[uart]); } return 0; } /* * Called by the DMA driver when the whole TX buffer has been written to the TX * FIFO (write transfers) or the expected amount of data has been read from the * RX FIFO (read transfers). */ static void uart_dma_callback(void *callback_context, uint32_t len, int error_code) { qm_uart_t uart; QM_ASSERT(callback_context); /* * On TX transfers, the DMA driver invokes this function as soon as all * data has been written to the TX FIFO, but we still need to wait until * everything has been written to the shift register (TX FIFO empty * interrupt) before the client callback is invoked. */ if (callback_context >= (void *)&dma_context_tx[0] && callback_context <= (void *)&dma_context_tx[QM_UART_NUM - 1]) { /* * callback_context is within dma_context_tx array so this is a * TX transfer, we extract the uart index from the position in * the array. */ uart = (dma_context_t *)callback_context - dma_context_tx; QM_ASSERT(callback_context == (void *)&dma_context_tx[uart]); qm_uart_reg_t *const regs = QM_UART[uart]; dma_delayed_callback_par[uart].context = callback_context; dma_delayed_callback_par[uart].len = len; dma_delayed_callback_par[uart].error_code = error_code; /* * Change the threshold level to trigger an interrupt when the * TX FIFO is empty and enable the TX FIFO empty interrupt. */ regs->iir_fcr = (QM_UART_FCR_FIFOE | QM_UART_FCR_TX_0_RX_1_2_THRESHOLD); regs->ier_dlh |= QM_UART_IER_ETBEI; } else { /* RX transfer. */ uart = (dma_context_t *)callback_context - dma_context_rx; QM_ASSERT(callback_context == (void *)&dma_context_rx[uart]); uart_client_callback(uart, callback_context, error_code, len); } } /* Invoke the UART client callback. */ static void uart_client_callback(qm_uart_t uart, void *data, int error, uint32_t len) { const qm_uart_transfer_t *const xfer = ((dma_context_t *)data)->xfer; QM_ASSERT(xfer); const uart_client_callback_t client_callback = xfer->callback; void *const client_data = xfer->callback_data; const uint32_t client_expected_len = xfer->data_len; qm_uart_reg_t *const regs = QM_UART[uart]; qm_uart_status_t status; if (!client_callback) { return; } status = regs->lsr & QM_UART_LSR_ERROR_BITS; status |= (regs->lsr & QM_UART_LSR_DR) ? QM_UART_RX_BUSY : 0; if (error) { /* * Transfer failed, pass to client the error code returned by * the DMA driver. */ client_callback(client_data, error, status, 0); } else if (len == client_expected_len) { /* Transfer completed successfully. */ client_callback(client_data, 0, status, len); } else { QM_ASSERT(len < client_expected_len); /* Transfer cancelled. */ client_callback(client_data, -ECANCELED, status, len); } } int qm_uart_dma_channel_config( const qm_uart_t uart, const qm_dma_t dma_ctrl_id, const qm_dma_channel_id_t dma_channel_id, const qm_dma_channel_direction_t dma_channel_direction) { QM_CHECK(uart < QM_UART_NUM, -EINVAL); QM_CHECK(dma_ctrl_id < QM_DMA_NUM, -EINVAL); QM_CHECK(dma_channel_id < QM_DMA_CHANNEL_NUM, -EINVAL); int ret = -EINVAL; qm_dma_channel_config_t dma_chan_cfg = {0}; /* UART has inverted handshake polarity. */ dma_chan_cfg.handshake_polarity = QM_DMA_HANDSHAKE_POLARITY_LOW; dma_chan_cfg.channel_direction = dma_channel_direction; dma_chan_cfg.source_transfer_width = QM_DMA_TRANS_WIDTH_8; dma_chan_cfg.destination_transfer_width = QM_DMA_TRANS_WIDTH_8; /* Default FIFO threshold is 1/2 full (8 bytes). */ dma_chan_cfg.source_burst_length = QM_DMA_BURST_TRANS_LENGTH_8; dma_chan_cfg.destination_burst_length = QM_DMA_BURST_TRANS_LENGTH_8; dma_chan_cfg.client_callback = uart_dma_callback; dma_chan_cfg.transfer_type = QM_DMA_TYPE_SINGLE; switch (dma_channel_direction) { case QM_DMA_MEMORY_TO_PERIPHERAL: dma_chan_cfg.handshake_interface = (QM_UART_0 == uart) ? DMA_HW_IF_UART_A_TX : DMA_HW_IF_UART_B_TX; /* * The DMA driver needs a pointer to the DMA context structure * used on DMA callback invocation. */ dma_context_tx[uart].dma_channel_id = dma_channel_id; dma_chan_cfg.callback_context = &dma_context_tx[uart]; break; case QM_DMA_PERIPHERAL_TO_MEMORY: dma_chan_cfg.handshake_interface = (QM_UART_0 == uart) ? DMA_HW_IF_UART_A_RX : DMA_HW_IF_UART_B_RX; /* * The DMA driver needs a pointer to the DMA context structure * used on DMA callback invocation. */ dma_context_rx[uart].dma_channel_id = dma_channel_id; dma_chan_cfg.callback_context = &dma_context_rx[uart]; break; default: /* Direction not allowed on UART transfers. */ return -EINVAL; } dma_core[uart] = dma_ctrl_id; ret = qm_dma_channel_set_config(dma_ctrl_id, dma_channel_id, &dma_chan_cfg); return ret; } int qm_uart_dma_write(const qm_uart_t uart, const qm_uart_transfer_t *const xfer) { QM_CHECK(uart < QM_UART_NUM, -EINVAL); QM_CHECK(xfer, -EINVAL); QM_CHECK(xfer->data, -EINVAL); QM_CHECK(xfer->data_len, -EINVAL); int ret = -EINVAL; qm_uart_reg_t *regs = QM_UART[uart]; qm_dma_transfer_t dma_trans = {0}; dma_trans.block_size = xfer->data_len; dma_trans.source_address = (uint32_t *)xfer->data; dma_trans.destination_address = (uint32_t *)®s->rbr_thr_dll; ret = qm_dma_transfer_set_config( dma_core[uart], dma_context_tx[uart].dma_channel_id, &dma_trans); if (ret) { return ret; } /* Store the user transfer pointer that we will need on DMA callback. */ dma_context_tx[uart].xfer = xfer; /* Set the FCR register FIFO thresholds. */ regs->iir_fcr = (QM_UART_FCR_FIFOE | QM_UART_FCR_DEFAULT_TX_RX_THRESHOLD); ret = qm_dma_transfer_start(dma_core[uart], dma_context_tx[uart].dma_channel_id); return ret; } int qm_uart_dma_read(const qm_uart_t uart, const qm_uart_transfer_t *const xfer) { QM_CHECK(uart < QM_UART_NUM, -EINVAL); QM_CHECK(xfer, -EINVAL); QM_CHECK(xfer->data, -EINVAL); QM_CHECK(xfer->data_len, -EINVAL); int ret = -EINVAL; qm_uart_reg_t *regs = QM_UART[uart]; qm_dma_transfer_t dma_trans = {0}; dma_trans.block_size = xfer->data_len; dma_trans.source_address = (uint32_t *)®s->rbr_thr_dll; dma_trans.destination_address = (uint32_t *)xfer->data; ret = qm_dma_transfer_set_config( dma_core[uart], dma_context_rx[uart].dma_channel_id, &dma_trans); if (ret) { return ret; } /* Store the user transfer pointer that we will need on DMA callback. */ dma_context_rx[uart].xfer = xfer; /* Set the FCR register FIFO thresholds. */ regs->iir_fcr = (QM_UART_FCR_FIFOE | QM_UART_FCR_DEFAULT_TX_RX_THRESHOLD); ret = qm_dma_transfer_start(dma_core[uart], dma_context_rx[uart].dma_channel_id); return ret; } int qm_uart_dma_write_terminate(const qm_uart_t uart) { QM_CHECK(uart < QM_UART_NUM, -EINVAL); int ret = qm_dma_transfer_terminate( dma_core[uart], dma_context_tx[uart].dma_channel_id); return ret; } int qm_uart_dma_read_terminate(const qm_uart_t uart) { QM_CHECK(uart < QM_UART_NUM, -EINVAL); int ret = qm_dma_transfer_terminate( dma_core[uart], dma_context_rx[uart].dma_channel_id); return ret; } #if (ENABLE_RESTORE_CONTEXT) int qm_uart_save_context(const qm_uart_t uart, qm_uart_context_t *const ctx) { QM_CHECK(uart < QM_UART_NUM, -EINVAL); QM_CHECK(ctx != NULL, -EINVAL); qm_uart_reg_t *const regs = QM_UART[uart]; ctx->ier = regs->ier_dlh; ctx->lcr = regs->lcr; ctx->mcr = regs->mcr; ctx->scr = regs->scr; ctx->htx = regs->htx; ctx->dlf = regs->dlf; regs->lcr |= QM_UART_LCR_DLAB; ctx->dlh = regs->ier_dlh; ctx->dll = regs->rbr_thr_dll; regs->lcr &= ~QM_UART_LCR_DLAB; return 0; } int qm_uart_restore_context(const qm_uart_t uart, const qm_uart_context_t *const ctx) { QM_CHECK(uart < QM_UART_NUM, -EINVAL); QM_CHECK(ctx != NULL, -EINVAL); qm_uart_reg_t *const regs = QM_UART[uart]; /* When DLAB is set, DLL and DLH registers can be accessed. */ regs->lcr |= QM_UART_LCR_DLAB; regs->ier_dlh = ctx->dlh; regs->rbr_thr_dll = ctx->dll; regs->lcr &= ~QM_UART_LCR_DLAB; regs->ier_dlh = ctx->ier; regs->lcr = ctx->lcr; regs->mcr = ctx->mcr; regs->scr = ctx->scr; regs->htx = ctx->htx; regs->dlf = ctx->dlf; /* * FIFO control register cannot be read back, * default config is applied for this register. * Application will need to restore its own parameters. */ regs->iir_fcr = (QM_UART_FCR_FIFOE | QM_UART_FCR_RFIFOR | QM_UART_FCR_XFIFOR | QM_UART_FCR_DEFAULT_TX_RX_THRESHOLD); return 0; } #endif /* ENABLE_RESTORE_CONTEXT */