2017-02-15 23:43:41 +00:00
|
|
|
/* mbed Microcontroller Library
|
|
|
|
* Copyright (c) 2006-2017 ARM Limited
|
2018-11-09 11:22:52 +00:00
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
2017-02-15 23:43:41 +00:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
2018-10-21 13:26:03 +00:00
|
|
|
#include "drivers/UARTSerial.h"
|
2017-02-15 23:43:41 +00:00
|
|
|
|
2017-06-13 12:25:48 +00:00
|
|
|
#if (DEVICE_SERIAL && DEVICE_INTERRUPTIN)
|
2017-02-15 23:43:41 +00:00
|
|
|
|
|
|
|
#include "platform/mbed_poll.h"
|
2017-09-28 07:44:43 +00:00
|
|
|
|
|
|
|
#if MBED_CONF_RTOS_PRESENT
|
2018-10-09 11:41:36 +00:00
|
|
|
#include "rtos/ThisThread.h"
|
2017-09-28 07:44:43 +00:00
|
|
|
#else
|
2017-02-15 23:43:41 +00:00
|
|
|
#include "platform/mbed_wait_api.h"
|
2017-09-28 07:44:43 +00:00
|
|
|
#endif
|
2017-02-15 23:43:41 +00:00
|
|
|
|
|
|
|
namespace mbed {
|
|
|
|
|
2017-05-23 17:07:24 +00:00
|
|
|
UARTSerial::UARTSerial(PinName tx, PinName rx, int baud) :
|
2018-05-24 15:58:14 +00:00
|
|
|
SerialBase(tx, rx, baud),
|
|
|
|
_blocking(true),
|
|
|
|
_tx_irq_enabled(false),
|
2019-02-21 14:44:54 +00:00
|
|
|
_rx_irq_enabled(false),
|
|
|
|
_tx_enabled(true),
|
|
|
|
_rx_enabled(true),
|
2018-05-24 15:58:14 +00:00
|
|
|
_dcd_irq(NULL)
|
2017-02-15 23:43:41 +00:00
|
|
|
{
|
|
|
|
/* Attatch IRQ routines to the serial device. */
|
2019-02-21 14:44:54 +00:00
|
|
|
enable_rx_irq();
|
2017-02-15 23:43:41 +00:00
|
|
|
}
|
|
|
|
|
2017-05-23 17:07:24 +00:00
|
|
|
UARTSerial::~UARTSerial()
|
2017-02-22 11:10:28 +00:00
|
|
|
{
|
2017-05-23 17:07:24 +00:00
|
|
|
delete _dcd_irq;
|
2017-02-22 11:10:28 +00:00
|
|
|
}
|
|
|
|
|
2017-05-23 17:07:24 +00:00
|
|
|
void UARTSerial::dcd_irq()
|
2017-02-22 11:10:28 +00:00
|
|
|
{
|
2017-05-08 10:11:12 +00:00
|
|
|
wake();
|
2017-02-22 11:10:28 +00:00
|
|
|
}
|
|
|
|
|
2017-06-22 11:26:48 +00:00
|
|
|
void UARTSerial::set_baud(int baud)
|
|
|
|
{
|
|
|
|
SerialBase::baud(baud);
|
|
|
|
}
|
|
|
|
|
2017-05-23 17:07:24 +00:00
|
|
|
void UARTSerial::set_data_carrier_detect(PinName dcd_pin, bool active_high)
|
2017-02-22 11:10:28 +00:00
|
|
|
{
|
2018-05-24 15:58:14 +00:00
|
|
|
delete _dcd_irq;
|
2017-05-23 17:07:24 +00:00
|
|
|
_dcd_irq = NULL;
|
2017-02-22 11:10:28 +00:00
|
|
|
|
2017-05-23 17:07:24 +00:00
|
|
|
if (dcd_pin != NC) {
|
|
|
|
_dcd_irq = new InterruptIn(dcd_pin);
|
2017-02-22 11:10:28 +00:00
|
|
|
if (active_high) {
|
2017-05-23 17:07:24 +00:00
|
|
|
_dcd_irq->fall(callback(this, &UARTSerial::dcd_irq));
|
2017-02-22 11:10:28 +00:00
|
|
|
} else {
|
2017-05-23 17:07:24 +00:00
|
|
|
_dcd_irq->rise(callback(this, &UARTSerial::dcd_irq));
|
2017-02-22 11:10:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-06 10:27:01 +00:00
|
|
|
void UARTSerial::set_format(int bits, Parity parity, int stop_bits)
|
|
|
|
{
|
|
|
|
api_lock();
|
|
|
|
SerialBase::format(bits, parity, stop_bits);
|
|
|
|
api_unlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
#if DEVICE_SERIAL_FC
|
|
|
|
void UARTSerial::set_flow_control(Flow type, PinName flow1, PinName flow2)
|
|
|
|
{
|
|
|
|
api_lock();
|
|
|
|
SerialBase::set_flow_control(type, flow1, flow2);
|
|
|
|
api_unlock();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2017-05-23 17:07:24 +00:00
|
|
|
int UARTSerial::close()
|
2017-02-15 23:43:41 +00:00
|
|
|
{
|
|
|
|
/* Does not let us pass a file descriptor. So how to close ?
|
|
|
|
* Also, does it make sense to close a device type file descriptor*/
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-05-23 17:07:24 +00:00
|
|
|
int UARTSerial::isatty()
|
2017-02-15 23:43:41 +00:00
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2017-05-23 17:07:24 +00:00
|
|
|
off_t UARTSerial::seek(off_t offset, int whence)
|
2017-02-15 23:43:41 +00:00
|
|
|
{
|
|
|
|
/*XXX lseek can be done theoratically, but is it sane to mark positions on a dynamically growing/shrinking
|
|
|
|
* buffer system (from an interrupt context) */
|
|
|
|
return -ESPIPE;
|
|
|
|
}
|
|
|
|
|
2017-05-23 17:07:24 +00:00
|
|
|
int UARTSerial::sync()
|
2017-02-15 23:43:41 +00:00
|
|
|
{
|
2017-06-13 12:25:48 +00:00
|
|
|
api_lock();
|
2017-02-15 23:43:41 +00:00
|
|
|
|
|
|
|
while (!_txbuf.empty()) {
|
2017-06-13 12:25:48 +00:00
|
|
|
api_unlock();
|
2017-02-15 23:43:41 +00:00
|
|
|
// Doing better than wait would require TxIRQ to also do wake() when becoming empty. Worth it?
|
|
|
|
wait_ms(1);
|
2017-06-13 12:25:48 +00:00
|
|
|
api_lock();
|
2017-02-15 23:43:41 +00:00
|
|
|
}
|
|
|
|
|
2017-06-13 12:25:48 +00:00
|
|
|
api_unlock();
|
2017-02-15 23:43:41 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-05-24 15:58:14 +00:00
|
|
|
void UARTSerial::sigio(Callback<void()> func)
|
|
|
|
{
|
2017-05-08 10:11:12 +00:00
|
|
|
core_util_critical_section_enter();
|
|
|
|
_sigio_cb = func;
|
|
|
|
if (_sigio_cb) {
|
|
|
|
short current_events = poll(0x7FFF);
|
|
|
|
if (current_events) {
|
|
|
|
_sigio_cb();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
core_util_critical_section_exit();
|
|
|
|
}
|
|
|
|
|
2018-09-10 08:15:08 +00:00
|
|
|
/* Special synchronous write designed to work from critical section, such
|
2018-09-11 08:18:32 +00:00
|
|
|
* as in mbed_error_vprintf.
|
2018-09-10 08:15:08 +00:00
|
|
|
*/
|
|
|
|
ssize_t UARTSerial::write_unbuffered(const char *buf_ptr, size_t length)
|
|
|
|
{
|
|
|
|
while (!_txbuf.empty()) {
|
|
|
|
tx_irq();
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t data_written = 0; data_written < length; data_written++) {
|
|
|
|
SerialBase::_base_putc(*buf_ptr++);
|
|
|
|
}
|
|
|
|
|
|
|
|
return length;
|
|
|
|
}
|
|
|
|
|
2018-05-24 15:58:14 +00:00
|
|
|
ssize_t UARTSerial::write(const void *buffer, size_t length)
|
2017-02-15 23:43:41 +00:00
|
|
|
{
|
|
|
|
size_t data_written = 0;
|
|
|
|
const char *buf_ptr = static_cast<const char *>(buffer);
|
|
|
|
|
Make UARTSerial send all data when blocking
Previously, write() was somewhat soft - it only ever made one attempt to
wait for buffer space, so it would take as much data as would fit in the
buffer in one call.
This is not the intent of a POSIX filehandle write. It should try to
send everything if blocking, and only send less if interrupted by a
signal:
- If the O_NONBLOCK flag is clear, write() shall block the calling
thread until the data can be accepted.
- If the O_NONBLOCK flag is set, write() shall not block the thread.
If some data can be written without blocking the thread, write()
shall write what it can and return the number of bytes written.
Otherwise, it shall return -1 and set errno to [EAGAIN].
This "send all" behaviour is of slightly limited usefulness in POSIX, as
you still usually have to worry about the interruption possibility:
- If write() is interrupted by a signal before it writes any data, it
shall return -1 with errno set to [EINTR].
- If write() is interrupted by a signal after it successfully writes
some data, it shall return the number of bytes written.
But as mbed OS does not have the possibility of signal interruption, if we
strengthen write to write everything, we can make applications' lives
easier - they can just do "write(large amount)" confident that it will
all go in one call (if no errors).
So, rework to make multiple writes to the buffer, blocking as necessary,
until all data is written.
This change does not apply to read(), which is correct in only blocking until
some data is available:
- If O_NONBLOCK is set, read() shall return -1 and set errno to [EAGAIN].
- If O_NONBLOCK is clear, read() shall block the calling thread until some
data becomes available.
- The use of the O_NONBLOCK flag has no effect if there is some data
available.
2017-11-09 12:30:55 +00:00
|
|
|
if (length == 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-09-10 08:15:08 +00:00
|
|
|
if (core_util_in_critical_section()) {
|
|
|
|
return write_unbuffered(buf_ptr, length);
|
|
|
|
}
|
|
|
|
|
2017-06-13 12:25:48 +00:00
|
|
|
api_lock();
|
2017-02-15 23:43:41 +00:00
|
|
|
|
Make UARTSerial send all data when blocking
Previously, write() was somewhat soft - it only ever made one attempt to
wait for buffer space, so it would take as much data as would fit in the
buffer in one call.
This is not the intent of a POSIX filehandle write. It should try to
send everything if blocking, and only send less if interrupted by a
signal:
- If the O_NONBLOCK flag is clear, write() shall block the calling
thread until the data can be accepted.
- If the O_NONBLOCK flag is set, write() shall not block the thread.
If some data can be written without blocking the thread, write()
shall write what it can and return the number of bytes written.
Otherwise, it shall return -1 and set errno to [EAGAIN].
This "send all" behaviour is of slightly limited usefulness in POSIX, as
you still usually have to worry about the interruption possibility:
- If write() is interrupted by a signal before it writes any data, it
shall return -1 with errno set to [EINTR].
- If write() is interrupted by a signal after it successfully writes
some data, it shall return the number of bytes written.
But as mbed OS does not have the possibility of signal interruption, if we
strengthen write to write everything, we can make applications' lives
easier - they can just do "write(large amount)" confident that it will
all go in one call (if no errors).
So, rework to make multiple writes to the buffer, blocking as necessary,
until all data is written.
This change does not apply to read(), which is correct in only blocking until
some data is available:
- If O_NONBLOCK is set, read() shall return -1 and set errno to [EAGAIN].
- If O_NONBLOCK is clear, read() shall block the calling thread until some
data becomes available.
- The use of the O_NONBLOCK flag has no effect if there is some data
available.
2017-11-09 12:30:55 +00:00
|
|
|
// Unlike read, we should write the whole thing if blocking. POSIX only
|
|
|
|
// allows partial as a side-effect of signal handling; it normally tries to
|
|
|
|
// write everything if blocking. Without signals we can always write all.
|
|
|
|
while (data_written < length) {
|
|
|
|
|
|
|
|
if (_txbuf.full()) {
|
|
|
|
if (!_blocking) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
do {
|
|
|
|
api_unlock();
|
|
|
|
wait_ms(1); // XXX todo - proper wait, WFE for non-rtos ?
|
|
|
|
api_lock();
|
|
|
|
} while (_txbuf.full());
|
2017-02-15 23:43:41 +00:00
|
|
|
}
|
|
|
|
|
Make UARTSerial send all data when blocking
Previously, write() was somewhat soft - it only ever made one attempt to
wait for buffer space, so it would take as much data as would fit in the
buffer in one call.
This is not the intent of a POSIX filehandle write. It should try to
send everything if blocking, and only send less if interrupted by a
signal:
- If the O_NONBLOCK flag is clear, write() shall block the calling
thread until the data can be accepted.
- If the O_NONBLOCK flag is set, write() shall not block the thread.
If some data can be written without blocking the thread, write()
shall write what it can and return the number of bytes written.
Otherwise, it shall return -1 and set errno to [EAGAIN].
This "send all" behaviour is of slightly limited usefulness in POSIX, as
you still usually have to worry about the interruption possibility:
- If write() is interrupted by a signal before it writes any data, it
shall return -1 with errno set to [EINTR].
- If write() is interrupted by a signal after it successfully writes
some data, it shall return the number of bytes written.
But as mbed OS does not have the possibility of signal interruption, if we
strengthen write to write everything, we can make applications' lives
easier - they can just do "write(large amount)" confident that it will
all go in one call (if no errors).
So, rework to make multiple writes to the buffer, blocking as necessary,
until all data is written.
This change does not apply to read(), which is correct in only blocking until
some data is available:
- If O_NONBLOCK is set, read() shall return -1 and set errno to [EAGAIN].
- If O_NONBLOCK is clear, read() shall block the calling thread until some
data becomes available.
- The use of the O_NONBLOCK flag has no effect if there is some data
available.
2017-11-09 12:30:55 +00:00
|
|
|
while (data_written < length && !_txbuf.full()) {
|
|
|
|
_txbuf.push(*buf_ptr++);
|
|
|
|
data_written++;
|
|
|
|
}
|
2017-02-15 23:43:41 +00:00
|
|
|
|
Make UARTSerial send all data when blocking
Previously, write() was somewhat soft - it only ever made one attempt to
wait for buffer space, so it would take as much data as would fit in the
buffer in one call.
This is not the intent of a POSIX filehandle write. It should try to
send everything if blocking, and only send less if interrupted by a
signal:
- If the O_NONBLOCK flag is clear, write() shall block the calling
thread until the data can be accepted.
- If the O_NONBLOCK flag is set, write() shall not block the thread.
If some data can be written without blocking the thread, write()
shall write what it can and return the number of bytes written.
Otherwise, it shall return -1 and set errno to [EAGAIN].
This "send all" behaviour is of slightly limited usefulness in POSIX, as
you still usually have to worry about the interruption possibility:
- If write() is interrupted by a signal before it writes any data, it
shall return -1 with errno set to [EINTR].
- If write() is interrupted by a signal after it successfully writes
some data, it shall return the number of bytes written.
But as mbed OS does not have the possibility of signal interruption, if we
strengthen write to write everything, we can make applications' lives
easier - they can just do "write(large amount)" confident that it will
all go in one call (if no errors).
So, rework to make multiple writes to the buffer, blocking as necessary,
until all data is written.
This change does not apply to read(), which is correct in only blocking until
some data is available:
- If O_NONBLOCK is set, read() shall return -1 and set errno to [EAGAIN].
- If O_NONBLOCK is clear, read() shall block the calling thread until some
data becomes available.
- The use of the O_NONBLOCK flag has no effect if there is some data
available.
2017-11-09 12:30:55 +00:00
|
|
|
core_util_critical_section_enter();
|
2019-02-21 14:44:54 +00:00
|
|
|
if (_tx_enabled && !_tx_irq_enabled) {
|
Make UARTSerial send all data when blocking
Previously, write() was somewhat soft - it only ever made one attempt to
wait for buffer space, so it would take as much data as would fit in the
buffer in one call.
This is not the intent of a POSIX filehandle write. It should try to
send everything if blocking, and only send less if interrupted by a
signal:
- If the O_NONBLOCK flag is clear, write() shall block the calling
thread until the data can be accepted.
- If the O_NONBLOCK flag is set, write() shall not block the thread.
If some data can be written without blocking the thread, write()
shall write what it can and return the number of bytes written.
Otherwise, it shall return -1 and set errno to [EAGAIN].
This "send all" behaviour is of slightly limited usefulness in POSIX, as
you still usually have to worry about the interruption possibility:
- If write() is interrupted by a signal before it writes any data, it
shall return -1 with errno set to [EINTR].
- If write() is interrupted by a signal after it successfully writes
some data, it shall return the number of bytes written.
But as mbed OS does not have the possibility of signal interruption, if we
strengthen write to write everything, we can make applications' lives
easier - they can just do "write(large amount)" confident that it will
all go in one call (if no errors).
So, rework to make multiple writes to the buffer, blocking as necessary,
until all data is written.
This change does not apply to read(), which is correct in only blocking until
some data is available:
- If O_NONBLOCK is set, read() shall return -1 and set errno to [EAGAIN].
- If O_NONBLOCK is clear, read() shall block the calling thread until some
data becomes available.
- The use of the O_NONBLOCK flag has no effect if there is some data
available.
2017-11-09 12:30:55 +00:00
|
|
|
UARTSerial::tx_irq(); // only write to hardware in one place
|
|
|
|
if (!_txbuf.empty()) {
|
2019-02-21 14:44:54 +00:00
|
|
|
enable_tx_irq();
|
Make UARTSerial send all data when blocking
Previously, write() was somewhat soft - it only ever made one attempt to
wait for buffer space, so it would take as much data as would fit in the
buffer in one call.
This is not the intent of a POSIX filehandle write. It should try to
send everything if blocking, and only send less if interrupted by a
signal:
- If the O_NONBLOCK flag is clear, write() shall block the calling
thread until the data can be accepted.
- If the O_NONBLOCK flag is set, write() shall not block the thread.
If some data can be written without blocking the thread, write()
shall write what it can and return the number of bytes written.
Otherwise, it shall return -1 and set errno to [EAGAIN].
This "send all" behaviour is of slightly limited usefulness in POSIX, as
you still usually have to worry about the interruption possibility:
- If write() is interrupted by a signal before it writes any data, it
shall return -1 with errno set to [EINTR].
- If write() is interrupted by a signal after it successfully writes
some data, it shall return the number of bytes written.
But as mbed OS does not have the possibility of signal interruption, if we
strengthen write to write everything, we can make applications' lives
easier - they can just do "write(large amount)" confident that it will
all go in one call (if no errors).
So, rework to make multiple writes to the buffer, blocking as necessary,
until all data is written.
This change does not apply to read(), which is correct in only blocking until
some data is available:
- If O_NONBLOCK is set, read() shall return -1 and set errno to [EAGAIN].
- If O_NONBLOCK is clear, read() shall block the calling thread until some
data becomes available.
- The use of the O_NONBLOCK flag has no effect if there is some data
available.
2017-11-09 12:30:55 +00:00
|
|
|
}
|
2017-02-15 23:43:41 +00:00
|
|
|
}
|
Make UARTSerial send all data when blocking
Previously, write() was somewhat soft - it only ever made one attempt to
wait for buffer space, so it would take as much data as would fit in the
buffer in one call.
This is not the intent of a POSIX filehandle write. It should try to
send everything if blocking, and only send less if interrupted by a
signal:
- If the O_NONBLOCK flag is clear, write() shall block the calling
thread until the data can be accepted.
- If the O_NONBLOCK flag is set, write() shall not block the thread.
If some data can be written without blocking the thread, write()
shall write what it can and return the number of bytes written.
Otherwise, it shall return -1 and set errno to [EAGAIN].
This "send all" behaviour is of slightly limited usefulness in POSIX, as
you still usually have to worry about the interruption possibility:
- If write() is interrupted by a signal before it writes any data, it
shall return -1 with errno set to [EINTR].
- If write() is interrupted by a signal after it successfully writes
some data, it shall return the number of bytes written.
But as mbed OS does not have the possibility of signal interruption, if we
strengthen write to write everything, we can make applications' lives
easier - they can just do "write(large amount)" confident that it will
all go in one call (if no errors).
So, rework to make multiple writes to the buffer, blocking as necessary,
until all data is written.
This change does not apply to read(), which is correct in only blocking until
some data is available:
- If O_NONBLOCK is set, read() shall return -1 and set errno to [EAGAIN].
- If O_NONBLOCK is clear, read() shall block the calling thread until some
data becomes available.
- The use of the O_NONBLOCK flag has no effect if there is some data
available.
2017-11-09 12:30:55 +00:00
|
|
|
core_util_critical_section_exit();
|
2017-02-15 23:43:41 +00:00
|
|
|
}
|
|
|
|
|
2017-06-13 12:25:48 +00:00
|
|
|
api_unlock();
|
2017-02-15 23:43:41 +00:00
|
|
|
|
2018-05-24 15:58:14 +00:00
|
|
|
return data_written != 0 ? (ssize_t) data_written : (ssize_t) - EAGAIN;
|
2017-02-15 23:43:41 +00:00
|
|
|
}
|
|
|
|
|
2018-05-24 15:58:14 +00:00
|
|
|
ssize_t UARTSerial::read(void *buffer, size_t length)
|
2017-02-15 23:43:41 +00:00
|
|
|
{
|
|
|
|
size_t data_read = 0;
|
|
|
|
|
|
|
|
char *ptr = static_cast<char *>(buffer);
|
|
|
|
|
Make UARTSerial send all data when blocking
Previously, write() was somewhat soft - it only ever made one attempt to
wait for buffer space, so it would take as much data as would fit in the
buffer in one call.
This is not the intent of a POSIX filehandle write. It should try to
send everything if blocking, and only send less if interrupted by a
signal:
- If the O_NONBLOCK flag is clear, write() shall block the calling
thread until the data can be accepted.
- If the O_NONBLOCK flag is set, write() shall not block the thread.
If some data can be written without blocking the thread, write()
shall write what it can and return the number of bytes written.
Otherwise, it shall return -1 and set errno to [EAGAIN].
This "send all" behaviour is of slightly limited usefulness in POSIX, as
you still usually have to worry about the interruption possibility:
- If write() is interrupted by a signal before it writes any data, it
shall return -1 with errno set to [EINTR].
- If write() is interrupted by a signal after it successfully writes
some data, it shall return the number of bytes written.
But as mbed OS does not have the possibility of signal interruption, if we
strengthen write to write everything, we can make applications' lives
easier - they can just do "write(large amount)" confident that it will
all go in one call (if no errors).
So, rework to make multiple writes to the buffer, blocking as necessary,
until all data is written.
This change does not apply to read(), which is correct in only blocking until
some data is available:
- If O_NONBLOCK is set, read() shall return -1 and set errno to [EAGAIN].
- If O_NONBLOCK is clear, read() shall block the calling thread until some
data becomes available.
- The use of the O_NONBLOCK flag has no effect if there is some data
available.
2017-11-09 12:30:55 +00:00
|
|
|
if (length == 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-06-13 12:25:48 +00:00
|
|
|
api_lock();
|
2017-02-15 23:43:41 +00:00
|
|
|
|
|
|
|
while (_rxbuf.empty()) {
|
|
|
|
if (!_blocking) {
|
2017-06-13 12:25:48 +00:00
|
|
|
api_unlock();
|
2017-02-15 23:43:41 +00:00
|
|
|
return -EAGAIN;
|
|
|
|
}
|
2017-06-13 12:25:48 +00:00
|
|
|
api_unlock();
|
2017-02-15 23:43:41 +00:00
|
|
|
wait_ms(1); // XXX todo - proper wait, WFE for non-rtos ?
|
2017-06-13 12:25:48 +00:00
|
|
|
api_lock();
|
2017-02-15 23:43:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
while (data_read < length && !_rxbuf.empty()) {
|
|
|
|
_rxbuf.pop(*ptr++);
|
|
|
|
data_read++;
|
|
|
|
}
|
|
|
|
|
2017-07-06 10:27:01 +00:00
|
|
|
core_util_critical_section_enter();
|
2019-02-21 14:44:54 +00:00
|
|
|
if (_rx_enabled && !_rx_irq_enabled) {
|
2017-07-06 10:27:01 +00:00
|
|
|
UARTSerial::rx_irq(); // only read from hardware in one place
|
|
|
|
if (!_rxbuf.full()) {
|
2019-02-21 14:44:54 +00:00
|
|
|
enable_rx_irq();
|
2017-07-06 10:27:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
core_util_critical_section_exit();
|
|
|
|
|
2017-06-13 12:25:48 +00:00
|
|
|
api_unlock();
|
2017-02-15 23:43:41 +00:00
|
|
|
|
|
|
|
return data_read;
|
|
|
|
}
|
|
|
|
|
2017-05-23 17:07:24 +00:00
|
|
|
bool UARTSerial::hup() const
|
2017-02-22 11:10:28 +00:00
|
|
|
{
|
2017-05-23 17:07:24 +00:00
|
|
|
return _dcd_irq && _dcd_irq->read() != 0;
|
2017-02-22 11:10:28 +00:00
|
|
|
}
|
|
|
|
|
2017-05-23 17:07:24 +00:00
|
|
|
void UARTSerial::wake()
|
2017-05-08 10:11:12 +00:00
|
|
|
{
|
|
|
|
if (_sigio_cb) {
|
|
|
|
_sigio_cb();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-24 15:58:14 +00:00
|
|
|
short UARTSerial::poll(short events) const
|
|
|
|
{
|
2017-02-15 23:43:41 +00:00
|
|
|
|
|
|
|
short revents = 0;
|
|
|
|
/* Check the Circular Buffer if space available for writing out */
|
2017-02-22 11:10:28 +00:00
|
|
|
|
2017-02-15 23:43:41 +00:00
|
|
|
|
|
|
|
if (!_rxbuf.empty()) {
|
|
|
|
revents |= POLLIN;
|
|
|
|
}
|
|
|
|
|
2017-02-22 11:10:28 +00:00
|
|
|
/* POLLHUP and POLLOUT are mutually exclusive */
|
|
|
|
if (hup()) {
|
|
|
|
revents |= POLLHUP;
|
|
|
|
} else if (!_txbuf.full()) {
|
|
|
|
revents |= POLLOUT;
|
|
|
|
}
|
|
|
|
|
2017-02-15 23:43:41 +00:00
|
|
|
/*TODO Handle other event types */
|
|
|
|
|
|
|
|
return revents;
|
|
|
|
}
|
|
|
|
|
2017-06-13 12:25:48 +00:00
|
|
|
void UARTSerial::lock()
|
|
|
|
{
|
|
|
|
// This is the override for SerialBase.
|
|
|
|
// No lock required as we only use SerialBase from interrupt or from
|
|
|
|
// inside our own critical section.
|
|
|
|
}
|
|
|
|
|
|
|
|
void UARTSerial::unlock()
|
|
|
|
{
|
|
|
|
// This is the override for SerialBase.
|
|
|
|
}
|
|
|
|
|
|
|
|
void UARTSerial::api_lock(void)
|
2017-02-15 23:43:41 +00:00
|
|
|
{
|
|
|
|
_mutex.lock();
|
|
|
|
}
|
|
|
|
|
2017-06-13 12:25:48 +00:00
|
|
|
void UARTSerial::api_unlock(void)
|
2017-02-15 23:43:41 +00:00
|
|
|
{
|
|
|
|
_mutex.unlock();
|
|
|
|
}
|
|
|
|
|
2017-05-23 17:07:24 +00:00
|
|
|
void UARTSerial::rx_irq(void)
|
2017-02-15 23:43:41 +00:00
|
|
|
{
|
|
|
|
bool was_empty = _rxbuf.empty();
|
|
|
|
|
|
|
|
/* Fill in the receive buffer if the peripheral is readable
|
|
|
|
* and receive buffer is not full. */
|
2017-07-06 10:27:01 +00:00
|
|
|
while (!_rxbuf.full() && SerialBase::readable()) {
|
2017-02-15 23:43:41 +00:00
|
|
|
char data = SerialBase::_base_getc();
|
2017-07-06 10:27:01 +00:00
|
|
|
_rxbuf.push(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_rx_irq_enabled && _rxbuf.full()) {
|
2019-02-21 14:44:54 +00:00
|
|
|
disable_rx_irq();
|
2017-02-15 23:43:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Report the File handler that data is ready to be read from the buffer. */
|
|
|
|
if (was_empty && !_rxbuf.empty()) {
|
2017-05-08 10:11:12 +00:00
|
|
|
wake();
|
2017-02-15 23:43:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Also called from write to start transfer
|
2017-05-23 17:07:24 +00:00
|
|
|
void UARTSerial::tx_irq(void)
|
2017-02-15 23:43:41 +00:00
|
|
|
{
|
|
|
|
bool was_full = _txbuf.full();
|
2018-04-16 14:23:47 +00:00
|
|
|
char data;
|
2017-02-15 23:43:41 +00:00
|
|
|
|
|
|
|
/* Write to the peripheral if there is something to write
|
|
|
|
* and if the peripheral is available to write. */
|
2018-04-16 14:23:47 +00:00
|
|
|
while (SerialBase::writeable() && _txbuf.pop(data)) {
|
2017-02-15 23:43:41 +00:00
|
|
|
SerialBase::_base_putc(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_tx_irq_enabled && _txbuf.empty()) {
|
2019-02-21 14:44:54 +00:00
|
|
|
disable_tx_irq();
|
2017-02-15 23:43:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Report the File handler that data can be written to peripheral. */
|
2017-02-22 11:10:28 +00:00
|
|
|
if (was_full && !_txbuf.full() && !hup()) {
|
2017-05-08 10:11:12 +00:00
|
|
|
wake();
|
2017-02-15 23:43:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-21 14:44:54 +00:00
|
|
|
/* These are all called from critical section */
|
|
|
|
void UARTSerial::enable_rx_irq()
|
|
|
|
{
|
|
|
|
SerialBase::attach(callback(this, &UARTSerial::rx_irq), RxIrq);
|
|
|
|
_rx_irq_enabled = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void UARTSerial::disable_rx_irq()
|
|
|
|
{
|
|
|
|
SerialBase::attach(NULL, RxIrq);
|
|
|
|
_rx_irq_enabled = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void UARTSerial::enable_tx_irq()
|
|
|
|
{
|
|
|
|
SerialBase::attach(callback(this, &UARTSerial::tx_irq), TxIrq);
|
|
|
|
_tx_irq_enabled = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void UARTSerial::disable_tx_irq()
|
|
|
|
{
|
|
|
|
SerialBase::attach(NULL, TxIrq);
|
|
|
|
_tx_irq_enabled = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
int UARTSerial::enable_input(bool enabled)
|
|
|
|
{
|
|
|
|
core_util_critical_section_enter();
|
|
|
|
if (_rx_enabled != enabled) {
|
|
|
|
if (enabled) {
|
|
|
|
UARTSerial::rx_irq();
|
|
|
|
if (!_rxbuf.full()) {
|
|
|
|
enable_rx_irq();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
disable_rx_irq();
|
|
|
|
}
|
|
|
|
_rx_enabled = enabled;
|
|
|
|
}
|
|
|
|
core_util_critical_section_exit();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int UARTSerial::enable_output(bool enabled)
|
|
|
|
{
|
|
|
|
core_util_critical_section_enter();
|
|
|
|
if (_tx_enabled != enabled) {
|
|
|
|
if (enabled) {
|
|
|
|
UARTSerial::tx_irq();
|
|
|
|
if (!_txbuf.empty()) {
|
|
|
|
enable_tx_irq();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
disable_tx_irq();
|
|
|
|
}
|
|
|
|
_tx_enabled = enabled;
|
|
|
|
}
|
|
|
|
core_util_critical_section_exit();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-09-28 07:44:43 +00:00
|
|
|
void UARTSerial::wait_ms(uint32_t millisec)
|
|
|
|
{
|
|
|
|
/* wait_ms implementation for RTOS spins until exact microseconds - we
|
|
|
|
* want to just sleep until next tick.
|
|
|
|
*/
|
|
|
|
#if MBED_CONF_RTOS_PRESENT
|
2018-10-09 11:41:36 +00:00
|
|
|
rtos::ThisThread::sleep_for(millisec);
|
2017-09-28 07:44:43 +00:00
|
|
|
#else
|
|
|
|
::wait_ms(millisec);
|
|
|
|
#endif
|
|
|
|
}
|
2017-02-15 23:43:41 +00:00
|
|
|
} //namespace mbed
|
|
|
|
|
2017-06-13 12:25:48 +00:00
|
|
|
#endif //(DEVICE_SERIAL && DEVICE_INTERRUPTIN)
|