245 lines
7.2 KiB
C
245 lines
7.2 KiB
C
/***************************************************************************//**
|
|
* \file cy_retarget_io.c
|
|
*
|
|
* \brief
|
|
* Provides APIs for retargeting stdio to UART hardware contained on the Cypress
|
|
* kits.
|
|
*
|
|
********************************************************************************
|
|
* \copyright
|
|
* Copyright 2018-2019 Cypress Semiconductor Corporation
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*******************************************************************************/
|
|
|
|
#include "cy_retarget_io_pdl.h"
|
|
|
|
#include "cycfg_peripherals.h"
|
|
|
|
#include "cy_sysint.h"
|
|
#include "cy_scb_uart.h"
|
|
|
|
#if defined(__cplusplus)
|
|
extern "C" {
|
|
#endif
|
|
|
|
/* Tracks the previous character sent to output stream */
|
|
#ifdef CY_RETARGET_IO_CONVERT_LF_TO_CRLF
|
|
static char cy_retarget_io_stdout_prev_char = 0;
|
|
#endif /* CY_RETARGET_IO_CONVERT_LF_TO_CRLF */
|
|
|
|
cy_stc_scb_uart_context_t CYBSP_UART_context;
|
|
|
|
static uint8_t cy_retarget_io_getchar(void);
|
|
static void cy_retarget_io_putchar(char c);
|
|
|
|
#if defined(__ARMCC_VERSION) /* ARM-MDK */
|
|
/***************************************************************************
|
|
* Function Name: fputc
|
|
***************************************************************************/
|
|
__attribute__((weak)) int fputc(int ch, FILE *f)
|
|
{
|
|
(void)f;
|
|
#ifdef CY_RETARGET_IO_CONVERT_LF_TO_CRLF
|
|
if ((char)ch == '\n' && cy_retarget_io_stdout_prev_char != '\r')
|
|
{
|
|
cy_retarget_io_putchar('\r');
|
|
}
|
|
|
|
cy_retarget_io_stdout_prev_char = (char)ch;
|
|
#endif /* CY_RETARGET_IO_CONVERT_LF_TO_CRLF */
|
|
cy_retarget_io_putchar(ch);
|
|
return (ch);
|
|
}
|
|
#elif defined (__ICCARM__) /* IAR */
|
|
#include <yfuns.h>
|
|
|
|
/***************************************************************************
|
|
* Function Name: __write
|
|
***************************************************************************/
|
|
__weak size_t __write(int handle, const unsigned char * buffer, size_t size)
|
|
{
|
|
size_t nChars = 0;
|
|
/* This template only writes to "standard out", for all other file
|
|
* handles it returns failure. */
|
|
if (handle != _LLIO_STDOUT)
|
|
{
|
|
return (_LLIO_ERROR);
|
|
}
|
|
if (buffer != NULL)
|
|
{
|
|
for (/* Empty */; nChars < size; ++nChars)
|
|
{
|
|
#ifdef CY_RETARGET_IO_CONVERT_LF_TO_CRLF
|
|
if (*buffer == '\n' && cy_retarget_io_stdout_prev_char != '\r')
|
|
{
|
|
cy_retarget_io_putchar('\r');
|
|
}
|
|
|
|
cy_retarget_io_stdout_prev_char = *buffer;
|
|
#endif /* CY_RETARGET_IO_CONVERT_LF_TO_CRLF */
|
|
cy_retarget_io_putchar(*buffer);
|
|
++buffer;
|
|
}
|
|
}
|
|
return (nChars);
|
|
}
|
|
#else /* (__GNUC__) GCC */
|
|
/* Add an explicit reference to the floating point printf library to allow
|
|
the usage of floating point conversion specifier. */
|
|
__asm (".global _printf_float");
|
|
/***************************************************************************
|
|
* Function Name: _write
|
|
***************************************************************************/
|
|
__attribute__((weak)) int _write (int fd, const char *ptr, int len)
|
|
{
|
|
int nChars = 0;
|
|
(void)fd;
|
|
if (ptr != NULL)
|
|
{
|
|
for (/* Empty */; nChars < len; ++nChars)
|
|
{
|
|
#ifdef CY_RETARGET_IO_CONVERT_LF_TO_CRLF
|
|
if (*ptr == '\n' && cy_retarget_io_stdout_prev_char != '\r')
|
|
{
|
|
cy_retarget_io_putchar('\r');
|
|
}
|
|
|
|
cy_retarget_io_stdout_prev_char = *ptr;
|
|
#endif /* CY_RETARGET_IO_CONVERT_LF_TO_CRLF */
|
|
cy_retarget_io_putchar((uint32_t)*ptr);
|
|
++ptr;
|
|
}
|
|
}
|
|
return (nChars);
|
|
}
|
|
#endif
|
|
|
|
|
|
#if defined(__ARMCC_VERSION) /* ARM-MDK */
|
|
/***************************************************************************
|
|
* Function Name: fgetc
|
|
***************************************************************************/
|
|
__attribute__((weak)) int fgetc(FILE *f)
|
|
{
|
|
(void)f;
|
|
return (cy_retarget_io_getchar());
|
|
}
|
|
#elif defined (__ICCARM__) /* IAR */
|
|
__weak size_t __read(int handle, unsigned char * buffer, size_t size)
|
|
{
|
|
/* This template only reads from "standard in", for all other file
|
|
handles it returns failure. */
|
|
if ((handle != _LLIO_STDIN) || (buffer == NULL))
|
|
{
|
|
return (_LLIO_ERROR);
|
|
}
|
|
else
|
|
{
|
|
*buffer = cy_retarget_io_getchar();
|
|
return (1);
|
|
}
|
|
}
|
|
#else /* (__GNUC__) GCC */
|
|
/* Add an explicit reference to the floating point scanf library to allow
|
|
the usage of floating point conversion specifier. */
|
|
__asm (".global _scanf_float");
|
|
__attribute__((weak)) int _read (int fd, char *ptr, int len)
|
|
{
|
|
int nChars = 0;
|
|
(void)fd;
|
|
if (ptr != NULL)
|
|
{
|
|
for(/* Empty */;nChars < len;++ptr)
|
|
{
|
|
*ptr = (char)cy_retarget_io_getchar();
|
|
++nChars;
|
|
if((*ptr == '\n') || (*ptr == '\r'))
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return (nChars);
|
|
}
|
|
#endif
|
|
|
|
static uint8_t cy_retarget_io_getchar(void)
|
|
{
|
|
uint32_t read_value = Cy_SCB_UART_Get(CYBSP_UART_HW);
|
|
while (read_value == CY_SCB_UART_RX_NO_DATA)
|
|
{
|
|
read_value = Cy_SCB_UART_Get(CYBSP_UART_HW);
|
|
}
|
|
|
|
return (uint8_t)read_value;
|
|
}
|
|
|
|
static void cy_retarget_io_putchar(char c)
|
|
{
|
|
uint32_t count = 0;
|
|
while (count == 0)
|
|
{
|
|
count = Cy_SCB_UART_Put(CYBSP_UART_HW, c);
|
|
}
|
|
}
|
|
|
|
static cy_rslt_t cy_retarget_io_pdl_setbaud(CySCB_Type *base, uint32_t baudrate)
|
|
{
|
|
cy_rslt_t status;
|
|
|
|
uint8_t oversample_value = 8u;
|
|
uint8_t frac_bits = 0u;
|
|
uint32_t divider;
|
|
|
|
Cy_SCB_UART_Disable(base, NULL);
|
|
|
|
Cy_SysClk_PeriphDisableDivider(CY_SYSCLK_DIV_16_BIT, 0);
|
|
|
|
divider = ((Cy_SysClk_ClkPeriGetFrequency() * (1 << frac_bits)) + ((baudrate * oversample_value) / 2)) / (baudrate * oversample_value) - 1;
|
|
|
|
status = (cy_rslt_t) Cy_SysClk_PeriphSetDivider(CY_SYSCLK_DIV_16_BIT, 0u, divider);
|
|
|
|
Cy_SysClk_PeriphEnableDivider(CY_SYSCLK_DIV_16_BIT, 0u);
|
|
|
|
Cy_SCB_UART_Enable(base);
|
|
|
|
return status;
|
|
}
|
|
|
|
cy_rslt_t cy_retarget_io_pdl_init(uint32_t baudrate)
|
|
{
|
|
cy_rslt_t result;
|
|
|
|
/* Configure and enable UART */
|
|
(void)Cy_SCB_UART_Init(CYBSP_UART_HW, &CYBSP_UART_config, &CYBSP_UART_context);
|
|
|
|
cy_retarget_io_pdl_setbaud(CYBSP_UART_HW, baudrate);
|
|
|
|
Cy_SCB_UART_Enable(CYBSP_UART_HW);
|
|
|
|
result = CY_RSLT_SUCCESS;
|
|
|
|
return result;
|
|
}
|
|
|
|
void cy_retarget_io_pdl_deinit()
|
|
{
|
|
Cy_SCB_UART_DeInit(CYBSP_UART_HW);
|
|
}
|
|
|
|
#if defined(__cplusplus)
|
|
}
|
|
#endif
|