mirror of
https://github.com/RT-Thread/rt-thread.git
synced 2025-01-25 22:17:27 +08:00
719 lines
20 KiB
C
719 lines
20 KiB
C
|
/* ****************************************************************************
|
||
|
* Copyright (C) 2014-2018 Maxim Integrated Products, Inc., All Rights Reserved.
|
||
|
*
|
||
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
||
|
* copy of this software and associated documentation files (the "Software"),
|
||
|
* to deal in the Software without restriction, including without limitation
|
||
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||
|
* and/or sell copies of the Software, and to permit persons to whom the
|
||
|
* Software is furnished to do so, subject to the following conditions:
|
||
|
*
|
||
|
* The above copyright notice and this permission notice shall be included
|
||
|
* in all copies or substantial portions of the Software.
|
||
|
*
|
||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
||
|
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
||
|
* IN NO EVENT SHALL MAXIM INTEGRATED BE LIABLE FOR ANY CLAIM, DAMAGES
|
||
|
* OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
||
|
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
||
|
* OTHER DEALINGS IN THE SOFTWARE.
|
||
|
*
|
||
|
* Except as contained in this notice, the name of Maxim Integrated
|
||
|
* Products, Inc. shall not be used except as stated in the Maxim Integrated
|
||
|
* Products, Inc. Branding Policy.
|
||
|
*
|
||
|
* The mere transfer of this software does not imply any licenses
|
||
|
* of trade secrets, proprietary technology, copyrights, patents,
|
||
|
* trademarks, maskwork rights, or any other form of intellectual
|
||
|
* property whatsoever. Maxim Integrated Products, Inc. retains all
|
||
|
* ownership rights.
|
||
|
*
|
||
|
* $Date: 2020-09-08 13:28:39 -0500 (Tue, 08 Sep 2020) $
|
||
|
* $Revision: 55611 $
|
||
|
*
|
||
|
*************************************************************************** */
|
||
|
|
||
|
/* **** Includes **** */
|
||
|
#include <stdint.h>
|
||
|
#include <string.h>
|
||
|
#include "mxc_config.h"
|
||
|
#include "mxc_assert.h"
|
||
|
#include "uart_regs.h"
|
||
|
#include "uart.h"
|
||
|
#include "mxc_lock.h"
|
||
|
#include "mxc_sys.h"
|
||
|
|
||
|
/* **** Definitions **** */
|
||
|
|
||
|
#define UART_ER_IF (MXC_F_UART_INT_FL_RX_FRAME_ERROR | \
|
||
|
MXC_F_UART_INT_FL_RX_PARITY_ERROR | \
|
||
|
MXC_F_UART_INT_FL_RX_OVERRUN)
|
||
|
|
||
|
#define UART_ER_IE (MXC_F_UART_INT_EN_RX_FRAME_ERROR | \
|
||
|
MXC_F_UART_INT_EN_RX_PARITY_ERROR | \
|
||
|
MXC_F_UART_INT_EN_RX_OVERRUN )
|
||
|
|
||
|
#define UART_RX_IF (MXC_F_UART_INT_FL_RX_FIFO_THRESH)
|
||
|
|
||
|
#define UART_RX_IE (MXC_F_UART_INT_EN_RX_FIFO_THRESH)
|
||
|
|
||
|
#define UART_TX_IF (MXC_F_UART_INT_FL_TX_FIFO_ALMOST_EMPTY | \
|
||
|
MXC_F_UART_INT_FL_TX_FIFO_THRESH)
|
||
|
|
||
|
#define UART_TX_IE (MXC_F_UART_INT_EN_TX_FIFO_ALMOST_EMPTY | \
|
||
|
MXC_F_UART_INT_EN_TX_FIFO_THRESH)
|
||
|
|
||
|
#if (TARGET == 32660) || (TARGET == 32665)
|
||
|
#define MAX_FACTOR 3
|
||
|
#else
|
||
|
#define MAX_FACTOR 7
|
||
|
#endif
|
||
|
|
||
|
/* **** File Scope Data **** */
|
||
|
|
||
|
// Saves the state of the non-blocking read requests.
|
||
|
static uart_req_t *rx_states[MXC_UART_INSTANCES];
|
||
|
|
||
|
// Saves the state of the non-blocking write requests.
|
||
|
static uart_req_t *tx_states[MXC_UART_INSTANCES];
|
||
|
|
||
|
|
||
|
/* **** Functions **** */
|
||
|
static void UART_WriteHandler(mxc_uart_regs_t *uart, uart_req_t *req, int uart_num);
|
||
|
static void UART_ReadHandler(mxc_uart_regs_t *uart, uart_req_t *req, int uart_num,
|
||
|
uint32_t flags);
|
||
|
static uint32_t uart_error_check(mxc_uart_regs_t *uart);
|
||
|
static void uart_error_clear(mxc_uart_regs_t *uart);
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
uint32_t uart_error_check(mxc_uart_regs_t *uart)
|
||
|
{
|
||
|
return (uart->int_fl & UART_ER_IF);
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
void uart_error_clear(mxc_uart_regs_t *uart)
|
||
|
{
|
||
|
UART_ClearFlags(uart,UART_ER_IF);
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
int UART_Init(mxc_uart_regs_t *uart, const uart_cfg_t *cfg, const sys_cfg_uart_t* sys_cfg)
|
||
|
{
|
||
|
int err;
|
||
|
int uart_num;
|
||
|
|
||
|
uint32_t baud0 = 0, baud1 = 0,div;
|
||
|
int32_t factor = -1;
|
||
|
|
||
|
// Get the state array index
|
||
|
uart_num = MXC_UART_GET_IDX(uart);
|
||
|
if (uart_num == -1) {
|
||
|
return E_BAD_PARAM;
|
||
|
}
|
||
|
|
||
|
if ((err = SYS_UART_Init(uart, sys_cfg)) != E_NO_ERROR) {
|
||
|
return err;
|
||
|
}
|
||
|
|
||
|
// Initialize state pointers
|
||
|
rx_states[uart_num] = NULL;
|
||
|
tx_states[uart_num] = NULL;
|
||
|
|
||
|
// Drain FIFOs, enable UART, and set configuration
|
||
|
uart->ctrl = (MXC_F_UART_CTRL_ENABLE | cfg->parity | cfg->size | cfg->stop | cfg->flow | cfg->pol);
|
||
|
|
||
|
// Set the baud rate
|
||
|
// Calculate divisor
|
||
|
#if (TARGET != 32660)
|
||
|
uart->ctrl |= cfg->clksel;
|
||
|
if (cfg->clksel == UART_CLKSEL_ALTERNATE) {
|
||
|
div = UART_ALTERNATE_CLOCK_HZ / ((cfg->baud));
|
||
|
} else {
|
||
|
div = PeripheralClock / ((cfg->baud));
|
||
|
}
|
||
|
#else
|
||
|
div = PeripheralClock / ((cfg->baud));
|
||
|
#endif
|
||
|
// Search for integer and fractional baud rate registers based on divisor
|
||
|
do {
|
||
|
factor += 1;
|
||
|
baud0 = div >> (7-factor); // divide by 128,64,32,16 to extract integer part
|
||
|
baud1 = ((div << factor) - (baud0 << 7)); //subtract factor corrected div - integer parts
|
||
|
} while ((baud0 == 0) && (factor < MAX_FACTOR));
|
||
|
|
||
|
uart->baud0 = ((factor << MXC_F_UART_BAUD0_FACTOR_POS) | baud0);
|
||
|
#if (TARGET == 32660) || (TARGET == 32665) || (TARGET == 32650)
|
||
|
/* Erratum:
|
||
|
* Hardware bug causes exact baud rates to generate framing error. Slightly mis-adjust timing
|
||
|
* to help avoid this bug.
|
||
|
*/
|
||
|
if (baud1 > 3) {
|
||
|
uart->baud1 = baud1 - 3;
|
||
|
} else {
|
||
|
uart->baud1 = baud1 + 3;
|
||
|
}
|
||
|
#else
|
||
|
uart->baud1 = baud1;
|
||
|
#endif
|
||
|
|
||
|
// Clear pending requests
|
||
|
rx_states[uart_num] = NULL;
|
||
|
tx_states[uart_num] = NULL;
|
||
|
|
||
|
return E_NO_ERROR;
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
int UART_Shutdown(mxc_uart_regs_t *uart)
|
||
|
{
|
||
|
int uart_num;
|
||
|
uart_req_t *temp_req;
|
||
|
|
||
|
// Get the state array index
|
||
|
uart_num = MXC_UART_GET_IDX(uart);
|
||
|
if (uart_num < 0) {
|
||
|
return E_BAD_PARAM;
|
||
|
}
|
||
|
|
||
|
// Disable interrupts
|
||
|
uart->int_en = 0;
|
||
|
|
||
|
// Flush RX and TX FIFOS
|
||
|
uart->ctrl |= (MXC_F_UART_CTRL_TX_FLUSH | MXC_F_UART_CTRL_RX_FLUSH);
|
||
|
|
||
|
// Call all of the pending callbacks for this UART
|
||
|
if(rx_states[uart_num] != NULL) {
|
||
|
|
||
|
// Save the request
|
||
|
temp_req = rx_states[uart_num];
|
||
|
|
||
|
// Unlock this UART to read
|
||
|
mxc_free_lock((uint32_t*)&rx_states[uart_num]);
|
||
|
|
||
|
// Callback if not NULL
|
||
|
if (temp_req->callback != NULL) {
|
||
|
temp_req->callback(temp_req, E_SHUTDOWN);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (tx_states[uart_num] != NULL) {
|
||
|
|
||
|
// Save the request
|
||
|
temp_req = tx_states[uart_num];
|
||
|
|
||
|
// Unlock this UART to write
|
||
|
mxc_free_lock((uint32_t*)&tx_states[uart_num]);
|
||
|
|
||
|
// Callback if not NULL
|
||
|
if (temp_req->callback != NULL) {
|
||
|
temp_req->callback(temp_req, E_SHUTDOWN);
|
||
|
}
|
||
|
}
|
||
|
// Wait for not busy
|
||
|
while (uart->status & (MXC_F_UART_STATUS_TX_BUSY | MXC_F_UART_STATUS_RX_BUSY)) {
|
||
|
|
||
|
}
|
||
|
|
||
|
// Shutdown the UART
|
||
|
uart->ctrl = 0;
|
||
|
|
||
|
// Shutdown any system level setup
|
||
|
SYS_UART_Shutdown(uart);
|
||
|
|
||
|
// Clear pending requests
|
||
|
rx_states[uart_num] = NULL;
|
||
|
tx_states[uart_num] = NULL;
|
||
|
|
||
|
return E_NO_ERROR;
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
void UART_Handler(mxc_uart_regs_t *uart)
|
||
|
{
|
||
|
int uart_num; // Holds the current index of rx_states or tx_states
|
||
|
uint32_t intst;
|
||
|
|
||
|
// Get the state array index
|
||
|
uart_num = MXC_UART_GET_IDX(uart);
|
||
|
if (uart_num == -1) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Read and clear interrupts
|
||
|
intst = uart->int_fl;
|
||
|
uart->int_fl = intst;
|
||
|
|
||
|
// Read interrupt
|
||
|
if (intst & (UART_RX_IF | UART_ER_IF)) {
|
||
|
UART_ReadHandler(uart, rx_states[uart_num], uart_num, intst);
|
||
|
}
|
||
|
|
||
|
// Write Interrupt
|
||
|
if (intst & (UART_TX_IF | UART_ER_IF)) {
|
||
|
UART_WriteHandler(uart, tx_states[uart_num], uart_num);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
static void UART_WriteHandler(mxc_uart_regs_t *uart, uart_req_t *req, int uart_num)
|
||
|
{
|
||
|
int remain, avail;
|
||
|
req = tx_states[uart_num];
|
||
|
|
||
|
if (req == NULL) {
|
||
|
// Nothing to do
|
||
|
uart->int_en &= ~MXC_F_UART_INT_EN_TX_FIFO_ALMOST_EMPTY; // disable interrupt
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Refill the TX FIFO
|
||
|
avail = UART_NumWriteAvail(uart);
|
||
|
remain = req->len - req->num;
|
||
|
|
||
|
while (avail && remain) {
|
||
|
uart->fifo = req->data[req->num++];
|
||
|
remain--;
|
||
|
avail--;
|
||
|
}
|
||
|
|
||
|
// See if we've sent all of the characters
|
||
|
if (req->len == req->num) {
|
||
|
// Disable interrupts
|
||
|
uart->int_en &= ~MXC_F_UART_INT_EN_TX_FIFO_ALMOST_EMPTY;
|
||
|
|
||
|
// Deinit state before callback in case another is requested
|
||
|
tx_states[uart_num] = NULL;
|
||
|
mxc_free_lock((uint32_t*)&tx_states[uart_num]);
|
||
|
|
||
|
// Callback when we've written all the characters
|
||
|
if (req->callback != NULL) {
|
||
|
req->callback(req, E_NO_ERROR);
|
||
|
}
|
||
|
}
|
||
|
// Enable the interrupts
|
||
|
uart->int_en |= UART_TX_IE | UART_ER_IE;
|
||
|
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
static void UART_ReadHandler(mxc_uart_regs_t *uart, uart_req_t *req, int uart_num,
|
||
|
uint32_t flags)
|
||
|
{
|
||
|
int remain, avail;
|
||
|
|
||
|
if (req == NULL) {
|
||
|
// Nothing to do
|
||
|
uart->int_en &= ~(UART_RX_IE | UART_ER_IE); // disable interrupts
|
||
|
return;
|
||
|
}
|
||
|
// Save the data in the FIFO while we still need data
|
||
|
avail = UART_NumReadAvail(uart);
|
||
|
remain = req->len - req->num;
|
||
|
while (avail && remain) {
|
||
|
req->data[req->num++] = uart->fifo;
|
||
|
remain--;
|
||
|
avail--;
|
||
|
}
|
||
|
// Check for errors
|
||
|
if (flags & MXC_F_UART_INT_FL_RX_OVERRUN) {
|
||
|
|
||
|
// Unlock this UART to read
|
||
|
mxc_free_lock((uint32_t*)&rx_states[uart_num]);
|
||
|
|
||
|
if (req->callback != NULL) {
|
||
|
req->callback(req, E_OVERFLOW);
|
||
|
}
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
if (flags & (MXC_F_UART_INT_FL_RX_FRAME_ERROR |
|
||
|
MXC_F_UART_INT_FL_RX_PARITY_ERROR)) {
|
||
|
|
||
|
// Unlock this UART to read
|
||
|
mxc_free_lock((uint32_t*)&rx_states[uart_num]);
|
||
|
|
||
|
if (req->callback != NULL) {
|
||
|
req->callback(req, E_COMM_ERR);
|
||
|
}
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
// Check to see if we've received all of the characters.
|
||
|
if (req->num == req->len) {
|
||
|
// Disable interrupts
|
||
|
uart->int_en &= ~(UART_RX_IE | UART_ER_IE);
|
||
|
|
||
|
// Deinit state before callback in case another is requested
|
||
|
rx_states[uart_num] = NULL;
|
||
|
|
||
|
// Call the callback function
|
||
|
if (req->callback != NULL) {
|
||
|
req->callback(req, E_NO_ERROR);
|
||
|
}
|
||
|
|
||
|
return;
|
||
|
} else if (req->num > (req->len - MXC_UART_FIFO_DEPTH)) {
|
||
|
// Set RX threshold less than FIFO_DEPTH characters if needed
|
||
|
uart->thresh_ctrl = ((req->len - req->num)<<
|
||
|
MXC_F_UART_THRESH_CTRL_RX_FIFO_THRESH_POS);
|
||
|
} else {
|
||
|
uart->thresh_ctrl = MXC_UART_FIFO_DEPTH<<
|
||
|
MXC_F_UART_THRESH_CTRL_RX_FIFO_THRESH_POS;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
int UART_Read(mxc_uart_regs_t *uart, uint8_t *data, int len, int *num)
|
||
|
{
|
||
|
int uart_num; // Holds the current index of rx_states
|
||
|
int char_read = 0; // Holds the number of characters successfully read
|
||
|
int error_code =0; // Holds the error to return while reading
|
||
|
|
||
|
// Get the state array index
|
||
|
uart_num = MXC_UART_GET_IDX(uart);
|
||
|
if (uart_num < 0) {
|
||
|
return E_BAD_PARAM;
|
||
|
}
|
||
|
|
||
|
// Check to make sure baud rate has been set
|
||
|
if (uart->baud0 == 0) {
|
||
|
return E_UNINITIALIZED;
|
||
|
}
|
||
|
|
||
|
// Check data pointer
|
||
|
if (data == NULL) {
|
||
|
return E_BAD_PARAM;
|
||
|
}
|
||
|
|
||
|
// Check if there is already a request in progress
|
||
|
if (rx_states[uart_num] != NULL) {
|
||
|
return E_BUSY;
|
||
|
}
|
||
|
|
||
|
// Lock this UART from reading
|
||
|
while (mxc_get_lock((uint32_t*)&rx_states[uart_num], 1) != E_NO_ERROR) {
|
||
|
|
||
|
}
|
||
|
|
||
|
// Get bytes FIFO
|
||
|
while (char_read < len) {
|
||
|
// Wait for RXFIFO to not be empty
|
||
|
while (uart->status & MXC_F_UART_STATUS_RX_EMPTY) {
|
||
|
// Check for error
|
||
|
if (uart_error_check(uart) != E_NO_ERROR) {
|
||
|
if (uart->int_fl & MXC_F_UART_INT_FL_RX_OVERRUN) {
|
||
|
error_code = E_OVERFLOW;
|
||
|
} else {
|
||
|
error_code = E_COMM_ERR;
|
||
|
}
|
||
|
|
||
|
uart_error_clear(uart);
|
||
|
mxc_free_lock((uint32_t*)&rx_states[uart_num]);
|
||
|
return error_code;
|
||
|
}
|
||
|
}
|
||
|
data[char_read] = uart->fifo;
|
||
|
char_read++;
|
||
|
}
|
||
|
if (num != NULL) {
|
||
|
*num = char_read;
|
||
|
}
|
||
|
// Unlock this UART to read
|
||
|
mxc_free_lock((uint32_t*)&rx_states[uart_num]);
|
||
|
|
||
|
return char_read;
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
uint8_t UART_ReadByte(mxc_uart_regs_t *uart)
|
||
|
{
|
||
|
|
||
|
while (uart->status & MXC_F_UART_STATUS_RX_EMPTY) {}
|
||
|
|
||
|
return uart->fifo;
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
int UART_Write(mxc_uart_regs_t *uart, const uint8_t *data, int len)
|
||
|
{
|
||
|
int uart_num; // Holds the current index of tx_states
|
||
|
int char_written = 0; // Holds the number of characters successfully written
|
||
|
|
||
|
// Get the state array index
|
||
|
uart_num = MXC_UART_GET_IDX(uart);
|
||
|
if (uart_num < 0) {
|
||
|
return E_BAD_PARAM;
|
||
|
}
|
||
|
|
||
|
// Check to make sure baud rate has been set
|
||
|
if (uart->baud0 == 0) {
|
||
|
return E_UNINITIALIZED;
|
||
|
}
|
||
|
|
||
|
// Check data pointer
|
||
|
if (data == NULL) {
|
||
|
return E_BAD_PARAM;
|
||
|
}
|
||
|
|
||
|
// Check if there is already a request in progress
|
||
|
if (tx_states[uart_num] != NULL) {
|
||
|
return E_BUSY;
|
||
|
}
|
||
|
|
||
|
// Lock this UART from writing
|
||
|
while (mxc_get_lock((uint32_t*)&tx_states[uart_num], 1) != E_NO_ERROR) {
|
||
|
|
||
|
}
|
||
|
|
||
|
// Clear errors
|
||
|
uart_error_clear(uart);
|
||
|
|
||
|
// Put bytes into FIFO
|
||
|
while (char_written < len) {
|
||
|
UART_WriteByte(uart,data[char_written]);
|
||
|
char_written++;
|
||
|
}
|
||
|
|
||
|
// Unlock this UART to write
|
||
|
mxc_free_lock((uint32_t*)&tx_states[uart_num]);
|
||
|
|
||
|
return char_written;
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
void UART_WriteByte(mxc_uart_regs_t *uart, uint8_t data)
|
||
|
{
|
||
|
|
||
|
// Wait for TXFIFO if full
|
||
|
while (uart->status & MXC_F_UART_STATUS_TX_FULL) {
|
||
|
|
||
|
}
|
||
|
|
||
|
// Put data into fifo
|
||
|
uart->fifo = data;
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
int UART_ReadAsync(mxc_uart_regs_t *uart, uart_req_t *req)
|
||
|
{
|
||
|
int uart_num; // Holds the current index of tx_states
|
||
|
uint32_t flags; // Holds the Interrupt flags
|
||
|
|
||
|
// Check data pointer
|
||
|
if (req == NULL) {
|
||
|
return E_BAD_PARAM;
|
||
|
}
|
||
|
|
||
|
// Get the state array index
|
||
|
uart_num = MXC_UART_GET_IDX(uart);
|
||
|
if (uart_num < 0) {
|
||
|
return E_BAD_PARAM;
|
||
|
}
|
||
|
|
||
|
if (req->data == NULL) {
|
||
|
return E_NULL_PTR;
|
||
|
}
|
||
|
// Check to make sure baud rate has been set
|
||
|
if (uart->baud0 == 0) {
|
||
|
return E_UNINITIALIZED;
|
||
|
}
|
||
|
|
||
|
// Check if there is already a request in progress
|
||
|
if (rx_states[uart_num] != NULL) {
|
||
|
return E_BUSY;
|
||
|
}
|
||
|
|
||
|
if (!(req->len > 0)) {
|
||
|
return E_NO_ERROR;
|
||
|
}
|
||
|
|
||
|
// Attempt to register this write request
|
||
|
if (mxc_get_lock((uint32_t*)&rx_states[uart_num], (uint32_t)req) != E_NO_ERROR) {
|
||
|
return E_BUSY;
|
||
|
}
|
||
|
|
||
|
// Clear the data counter
|
||
|
req->num = 0;
|
||
|
|
||
|
// Clear Interrupt Flags
|
||
|
flags = uart->int_fl;
|
||
|
uart->int_fl = flags;
|
||
|
UART_ReadHandler(uart,req,uart_num,flags);
|
||
|
|
||
|
// Enable the interrupts
|
||
|
uart->int_en |= UART_RX_IE | UART_ER_IE;
|
||
|
|
||
|
return E_NO_ERROR;
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
int UART_WriteAsync(mxc_uart_regs_t *uart, uart_req_t *req)
|
||
|
{
|
||
|
int uart_num; // Holds the current index of tx_states
|
||
|
|
||
|
// Check data pointer
|
||
|
if (req == NULL) {
|
||
|
return E_BAD_PARAM;
|
||
|
}
|
||
|
|
||
|
// Get the state array index
|
||
|
uart_num = MXC_UART_GET_IDX(uart);
|
||
|
if (uart_num < 0) {
|
||
|
return E_BAD_PARAM;
|
||
|
}
|
||
|
|
||
|
if (req->data == NULL) {
|
||
|
return E_NULL_PTR;
|
||
|
}
|
||
|
// Check to make sure baud rate has been set
|
||
|
if (uart->baud0 == 0) {
|
||
|
return E_UNINITIALIZED;
|
||
|
}
|
||
|
|
||
|
// Check if there is already a request in progress
|
||
|
if (tx_states[uart_num] != NULL) {
|
||
|
return E_BUSY;
|
||
|
}
|
||
|
if (!(req->len > 0)) {
|
||
|
return E_NO_ERROR;
|
||
|
}
|
||
|
// Attempt to register this write request
|
||
|
if (mxc_get_lock((uint32_t*)&tx_states[uart_num], (uint32_t)req) != E_NO_ERROR) {
|
||
|
return E_BUSY;
|
||
|
}
|
||
|
|
||
|
// Clear the data counter
|
||
|
req->num = 0;
|
||
|
UART_WriteHandler(uart, req, uart_num);
|
||
|
|
||
|
return E_NO_ERROR;
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
int UART_Busy(mxc_uart_regs_t *uart)
|
||
|
{
|
||
|
int uart_num = MXC_UART_GET_IDX(uart); // Holds the current index of tx_states
|
||
|
MXC_ASSERT(uart_num >= 0);
|
||
|
if ((uart->status & MXC_F_UART_STATUS_TX_BUSY) || (uart->status & MXC_F_UART_STATUS_RX_BUSY)) {
|
||
|
return E_BUSY;
|
||
|
}
|
||
|
// Check to see if there are any ongoing transactions and the UART has room in its FIFO
|
||
|
if ((tx_states[uart_num] == NULL) &&
|
||
|
!(uart->status & MXC_F_UART_STATUS_TX_FULL)) {
|
||
|
|
||
|
return E_NO_ERROR;
|
||
|
}
|
||
|
|
||
|
return E_BUSY;
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
int UART_PrepForSleep(mxc_uart_regs_t *uart)
|
||
|
{
|
||
|
if (UART_Busy(uart) != E_NO_ERROR) {
|
||
|
return E_BUSY;
|
||
|
}
|
||
|
|
||
|
// Leave read interrupts enabled, if already enabled
|
||
|
uart->int_en &= (UART_RX_IE | UART_ER_IE);
|
||
|
|
||
|
return E_NO_ERROR;
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
int UART_AbortAsync(uart_req_t *req)
|
||
|
{
|
||
|
int uart_num;
|
||
|
|
||
|
// Figure out if this was a read or write request, find the request, set to NULL
|
||
|
for (uart_num = 0; uart_num < MXC_UART_INSTANCES; uart_num++) {
|
||
|
if (req == rx_states[uart_num]) {
|
||
|
|
||
|
// Disable read interrupts, clear flags.
|
||
|
MXC_UART_GET_UART(uart_num)->int_en &= ~(UART_RX_IE | UART_ER_IE);
|
||
|
MXC_UART_GET_UART(uart_num)->int_fl = (UART_RX_IF | UART_ER_IF);
|
||
|
|
||
|
// Unlock this UART to read
|
||
|
mxc_free_lock((uint32_t*)&rx_states[uart_num]);
|
||
|
|
||
|
// Callback if not NULL
|
||
|
if (req->callback != NULL) {
|
||
|
req->callback(req, E_ABORT);
|
||
|
}
|
||
|
|
||
|
return E_NO_ERROR;
|
||
|
}
|
||
|
|
||
|
if (req == tx_states[uart_num]) {
|
||
|
|
||
|
// Disable write interrupts, clear flags.
|
||
|
MXC_UART_GET_UART(uart_num)->int_en &= ~(UART_TX_IE | UART_ER_IE);
|
||
|
MXC_UART_GET_UART(uart_num)->int_fl = (UART_TX_IF | UART_ER_IF);
|
||
|
|
||
|
// Unlock this UART to write
|
||
|
mxc_free_lock((uint32_t*)&tx_states[uart_num]);
|
||
|
|
||
|
// Callback if not NULL
|
||
|
if (req->callback != NULL) {
|
||
|
req->callback(req, E_ABORT);
|
||
|
}
|
||
|
|
||
|
return E_NO_ERROR;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return E_BAD_PARAM;
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
unsigned UART_NumWriteAvail(mxc_uart_regs_t *uart)
|
||
|
{
|
||
|
return MXC_UART_FIFO_DEPTH - ((uart->status & MXC_F_UART_STATUS_TX_FIFO_CNT) >>
|
||
|
MXC_F_UART_STATUS_TX_FIFO_CNT_POS);
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
unsigned UART_NumReadAvail(mxc_uart_regs_t *uart)
|
||
|
{
|
||
|
return ((uart->status & MXC_F_UART_STATUS_RX_FIFO_CNT) >>
|
||
|
MXC_F_UART_STATUS_RX_FIFO_CNT_POS);
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
unsigned UART_GetFlags(mxc_uart_regs_t *uart)
|
||
|
{
|
||
|
return (uart->int_fl);
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
void UART_ClearFlags(mxc_uart_regs_t *uart, uint32_t mask)
|
||
|
{
|
||
|
uart->int_fl = mask;
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
void UART_Enable(mxc_uart_regs_t *uart)
|
||
|
{
|
||
|
uart->ctrl |= MXC_F_UART_CTRL_ENABLE;
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
void UART_Disable(mxc_uart_regs_t *uart)
|
||
|
{
|
||
|
uart->ctrl &= ~MXC_F_UART_CTRL_ENABLE;
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
void UART_DrainRX(mxc_uart_regs_t *uart)
|
||
|
{
|
||
|
uart->ctrl |= MXC_F_UART_CTRL_RX_FLUSH;
|
||
|
}
|
||
|
|
||
|
/* ************************************************************************* */
|
||
|
void UART_DrainTX(mxc_uart_regs_t *uart)
|
||
|
{
|
||
|
uart->ctrl |= MXC_F_UART_CTRL_TX_FLUSH;
|
||
|
}
|