| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295 |
- // Copyright 2021 Espressif Systems (Shanghai) CO LTD
- //
- // 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 "esp_uart_spinel_interface.hpp"
- #include <errno.h>
- #include <fcntl.h>
- #include <sys/select.h>
- #include <sys/unistd.h>
- #include "esp_check.h"
- #include "esp_err.h"
- #include "esp_log.h"
- #include "esp_openthread_common_macro.h"
- #include "esp_openthread_types.h"
- #include "esp_openthread_uart.h"
- #include "esp_vfs_dev.h"
- #include "core/common/code_utils.hpp"
- #include "core/common/logging.hpp"
- #include "driver/uart.h"
- #include "lib/platform/exit_code.h"
- #include "openthread/platform/time.h"
- namespace esp {
- namespace openthread {
- UartSpinelInterface::UartSpinelInterface(
- ot::Spinel::SpinelInterface::ReceiveFrameCallback callback,
- void *callback_context,
- ot::Spinel::SpinelInterface::RxFrameBuffer &frame_buffer)
- : m_receiver_frame_callback(callback)
- , m_receiver_frame_context(callback_context)
- , m_receive_frame_buffer(frame_buffer)
- , m_hdlc_decoder(frame_buffer, HandleHdlcFrame, this)
- , m_uart_fd(-1)
- {
- }
- UartSpinelInterface::~UartSpinelInterface(void)
- {
- }
- esp_err_t UartSpinelInterface::Init(const esp_openthread_uart_config_t &radio_uart_config)
- {
- m_uart_rx_buffer = static_cast<uint8_t *>(heap_caps_malloc(kMaxFrameSize, MALLOC_CAP_8BIT));
- if (m_uart_rx_buffer == NULL) {
- return ESP_ERR_NO_MEM;
- }
- return InitUart(radio_uart_config);
- }
- esp_err_t UartSpinelInterface::Deinit(void)
- {
- if (m_uart_rx_buffer) {
- heap_caps_free(m_uart_rx_buffer);
- }
- m_uart_rx_buffer = NULL;
- return DeinitUart();
- }
- otError UartSpinelInterface::SendFrame(const uint8_t *frame, uint16_t length)
- {
- otError error = OT_ERROR_NONE;
- ot::Hdlc::FrameBuffer<kMaxFrameSize> encoder_buffer;
- ot::Hdlc::Encoder hdlc_encoder(encoder_buffer);
- SuccessOrExit(error = hdlc_encoder.BeginFrame());
- SuccessOrExit(error = hdlc_encoder.Encode(frame, length));
- SuccessOrExit(error = hdlc_encoder.EndFrame());
- SuccessOrExit(error = Write(encoder_buffer.GetFrame(), encoder_buffer.GetLength()));
- exit:
- if (error != OT_ERROR_NONE) {
- otLogCritPlat("send radio frame failed");
- } else {
- otLogDebgPlat("sent radio frame");
- }
- return error;
- }
- void UartSpinelInterface::Process(const esp_openthread_mainloop_context_t &mainloop)
- {
- if (FD_ISSET(m_uart_fd, &mainloop.read_fds)) {
- otLogDebgPlat("radio uart read event");
- TryReadAndDecode();
- }
- }
- void UartSpinelInterface::Update(esp_openthread_mainloop_context_t &mainloop)
- {
- // Register only READ events for radio UART and always wait
- // for a radio WRITE to complete.
- FD_SET(m_uart_fd, &mainloop.read_fds);
- if (m_uart_fd > mainloop.max_fd) {
- mainloop.max_fd = m_uart_fd;
- }
- }
- int UartSpinelInterface::TryReadAndDecode(void)
- {
- uint8_t buffer[UART_FIFO_LEN];
- ssize_t rval;
- do {
- rval = read(m_uart_fd, buffer, sizeof(buffer));
- if (rval > 0) {
- m_hdlc_decoder.Decode(buffer, static_cast<uint16_t>(rval));
- }
- } while (rval > 0);
- if ((rval < 0) && (errno != EAGAIN) && (errno != EWOULDBLOCK)) {
- ESP_ERROR_CHECK(TryRecoverUart());
- }
- return rval;
- }
- otError UartSpinelInterface::WaitForWritable(void)
- {
- otError error = OT_ERROR_NONE;
- struct timeval timeout = {kMaxWaitTime / MS_PER_S, (kMaxWaitTime % MS_PER_S) *US_PER_MS};
- uint64_t now = otPlatTimeGet();
- uint64_t end = now + kMaxWaitTime * US_PER_MS;
- fd_set write_fds;
- fd_set error_fds;
- int rval;
- while (true) {
- FD_ZERO(&write_fds);
- FD_ZERO(&error_fds);
- FD_SET(m_uart_fd, &write_fds);
- FD_SET(m_uart_fd, &error_fds);
- rval = select(m_uart_fd + 1, NULL, &write_fds, &error_fds, &timeout);
- if (rval > 0) {
- if (FD_ISSET(m_uart_fd, &write_fds)) {
- ExitNow();
- } else if (FD_ISSET(m_uart_fd, &error_fds)) {
- ExitNow(error = OT_ERROR_FAILED);
- }
- } else if ((rval < 0) && (errno != EINTR)) {
- ESP_ERROR_CHECK(TryRecoverUart());
- ExitNow(error = OT_ERROR_FAILED);
- }
- now = otPlatTimeGet();
- if (end > now) {
- uint64_t remain = end - now;
- timeout.tv_sec = static_cast<time_t>(remain / 1000000);
- timeout.tv_usec = static_cast<suseconds_t>(remain % 1000000);
- } else {
- break;
- }
- }
- error = OT_ERROR_FAILED;
- exit:
- return error;
- }
- otError UartSpinelInterface::Write(const uint8_t *aFrame, uint16_t length)
- {
- otError error = OT_ERROR_NONE;
- while (length) {
- ssize_t rval;
- rval = write(m_uart_fd, aFrame, length);
- if (rval > 0) {
- assert(rval <= length);
- length -= static_cast<uint16_t>(rval);
- aFrame += static_cast<uint16_t>(rval);
- continue;
- } else if (rval < 0) {
- ESP_ERROR_CHECK(TryRecoverUart());
- ExitNow(error = OT_ERROR_FAILED);
- }
- SuccessOrExit(error = WaitForWritable());
- }
- exit:
- return error;
- }
- otError UartSpinelInterface::WaitForFrame(uint64_t timeout_us)
- {
- otError error = OT_ERROR_NONE;
- struct timeval timeout;
- fd_set read_fds;
- fd_set error_fds;
- int rval;
- FD_ZERO(&read_fds);
- FD_ZERO(&error_fds);
- FD_SET(m_uart_fd, &read_fds);
- FD_SET(m_uart_fd, &error_fds);
- timeout.tv_sec = static_cast<time_t>(timeout_us / US_PER_S);
- timeout.tv_usec = static_cast<suseconds_t>(timeout_us % US_PER_S);
- rval = select(m_uart_fd + 1, &read_fds, NULL, &error_fds, &timeout);
- if (rval > 0) {
- if (FD_ISSET(m_uart_fd, &read_fds)) {
- TryReadAndDecode();
- } else if (FD_ISSET(m_uart_fd, &error_fds)) {
- ESP_ERROR_CHECK(TryRecoverUart());
- ExitNow(error = OT_ERROR_FAILED);
- }
- } else if (rval == 0) {
- ExitNow(error = OT_ERROR_RESPONSE_TIMEOUT);
- } else {
- ESP_ERROR_CHECK(TryRecoverUart());
- ExitNow(error = OT_ERROR_FAILED);
- }
- exit:
- return error;
- }
- void UartSpinelInterface::HandleHdlcFrame(void *context, otError error)
- {
- static_cast<UartSpinelInterface *>(context)->HandleHdlcFrame(error);
- }
- void UartSpinelInterface::HandleHdlcFrame(otError error)
- {
- if (error == OT_ERROR_NONE) {
- otLogDebgPlat("received hdlc radio frame");
- m_receiver_frame_callback(m_receiver_frame_context);
- } else {
- otLogCritPlat("dropping radio frame: %s", otThreadErrorToString(error));
- m_receive_frame_buffer.DiscardFrame();
- }
- }
- esp_err_t UartSpinelInterface::InitUart(const esp_openthread_uart_config_t &radio_uart_config)
- {
- char uart_path[16];
- m_uart_config = radio_uart_config;
- ESP_RETURN_ON_ERROR(esp_openthread_uart_init_port(&radio_uart_config), OT_PLAT_LOG_TAG,
- "esp_openthread_uart_init_port failed");
- // We have a driver now installed so set up the read/write functions to use driver also.
- esp_vfs_dev_uart_port_set_tx_line_endings(m_uart_config.port, ESP_LINE_ENDINGS_LF);
- esp_vfs_dev_uart_port_set_rx_line_endings(m_uart_config.port, ESP_LINE_ENDINGS_LF);
- snprintf(uart_path, sizeof(uart_path), "/dev/uart/%d", radio_uart_config.port);
- m_uart_fd = open(uart_path, O_RDWR | O_NONBLOCK);
- return m_uart_fd >= 0 ? ESP_OK : ESP_FAIL;
- }
- esp_err_t UartSpinelInterface::DeinitUart(void)
- {
- if (m_uart_fd != -1) {
- close(m_uart_fd);
- m_uart_fd = -1;
- return uart_driver_delete(m_uart_config.port);
- } else {
- return ESP_ERR_INVALID_STATE;
- }
- }
- esp_err_t UartSpinelInterface::TryRecoverUart(void)
- {
- ESP_RETURN_ON_ERROR(DeinitUart(), OT_PLAT_LOG_TAG, "DeInitUart failed");
- ESP_RETURN_ON_ERROR(InitUart(m_uart_config), OT_PLAT_LOG_TAG, "InitUart failed");
- return ESP_OK;
- }
- } // namespace openthread
- } // namespace esp
|