hal_nordic/drivers/nrf_802154/serialization/src/nrf_802154_spinel_net.c

388 lines
12 KiB
C

/*
* Copyright (c) 2020 - 2021, Nordic Semiconductor ASA
* All rights reserved.
*
* SPDX-License-Identifier: BSD-3-Clause
*
* 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 Nordic Semiconductor ASA 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 COPYRIGHT HOLDER 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.
*
*/
/**
* @file nrf_802154_spinel_net.c
*
* @brief Client of nRF 802.15.4 Radio Driver for the network core
* responsible for serializing callouts through spinel to application core.
*
* This file is responsible for providing implementation callouts defined by
* defined by nrf_802154.h as required by nRF 802.15.4 API. The implementation
* of callouts send commands through spinel to application core effectively
* causing calling of corresponding callouts in the application core.
*/
#include <stdint.h>
#include "../spinel_base/spinel.h"
#include "nrf_802154_spinel.h"
#include "nrf_802154_spinel_datatypes.h"
#include "nrf_802154_spinel_enc_net.h"
#include "nrf_802154_spinel_log.h"
#include "nrf_802154_spinel_response_notifier.h"
#include "nrf_802154_serialization_error.h"
#include "nrf_802154_serialization_error_helper.h"
#include "nrf_802154_buffer_mgr_dst.h"
#include "nrf_802154_buffer_mgr_src.h"
#include "nrf_802154.h"
/**@brief A pointer to the last transmitted ACK frame. */
static const uint8_t * volatile mp_last_tx_ack;
static void local_transmitted_frame_ptr_free(void * p_frame)
{
SERIALIZATION_ERROR_INIT(error);
bool frame_found = nrf_802154_buffer_mgr_dst_remove_by_local_pointer(
nrf_802154_spinel_dst_buffer_mgr_get(),
p_frame);
SERIALIZATION_ERROR_IF(!frame_found,
NRF_802154_SERIALIZATION_ERROR_INVALID_BUFFER,
error,
bail);
bail:
SERIALIZATION_ERROR_RAISE_IF_FAILED(error);
return;
}
void nrf_802154_cca_done(bool channel_free)
{
nrf_802154_ser_err_t res;
SERIALIZATION_ERROR_INIT(error);
NRF_802154_SPINEL_LOG_BANNER_CALLING();
NRF_802154_SPINEL_LOG_VAR_NAMED("%s", channel_free ? "true" : "false", "channel_free");
res = nrf_802154_spinel_send_cmd_prop_value_is(
SPINEL_PROP_VENDOR_NORDIC_NRF_802154_CCA_DONE,
SPINEL_DATATYPE_NRF_802154_CCA_DONE,
channel_free);
SERIALIZATION_ERROR_CHECK(res, error, bail);
bail:
SERIALIZATION_ERROR_RAISE_IF_FAILED(error);
return;
}
void nrf_802154_cca_failed(nrf_802154_cca_error_t err)
{
nrf_802154_ser_err_t res;
SERIALIZATION_ERROR_INIT(error);
NRF_802154_SPINEL_LOG_BANNER_CALLING();
NRF_802154_SPINEL_LOG_VAR("%u", err);
res = nrf_802154_spinel_send_cmd_prop_value_is(
SPINEL_PROP_VENDOR_NORDIC_NRF_802154_CCA_FAILED,
SPINEL_DATATYPE_NRF_802154_CCA_FAILED,
err);
SERIALIZATION_ERROR_CHECK(res, error, bail);
bail:
SERIALIZATION_ERROR_RAISE_IF_FAILED(error);
return;
}
void nrf_802154_energy_detected(uint8_t result)
{
nrf_802154_ser_err_t res;
SERIALIZATION_ERROR_INIT(error);
NRF_802154_SPINEL_LOG_BANNER_CALLING();
NRF_802154_SPINEL_LOG_VAR("%u", result);
res = nrf_802154_spinel_send_cmd_prop_value_is(
SPINEL_PROP_VENDOR_NORDIC_NRF_802154_ENERGY_DETECTED,
SPINEL_DATATYPE_NRF_802154_ENERGY_DETECTED,
result);
SERIALIZATION_ERROR_CHECK(res, error, bail);
bail:
SERIALIZATION_ERROR_RAISE_IF_FAILED(error);
return;
}
void nrf_802154_energy_detection_failed(nrf_802154_ed_error_t err)
{
nrf_802154_ser_err_t res;
SERIALIZATION_ERROR_INIT(error);
NRF_802154_SPINEL_LOG_BANNER_CALLING();
NRF_802154_SPINEL_LOG_VAR("%u", err);
res = nrf_802154_spinel_send_cmd_prop_value_is(
SPINEL_PROP_VENDOR_NORDIC_NRF_802154_ENERGY_DETECTION_FAILED,
SPINEL_DATATYPE_NRF_802154_ENERGY_DETECTION_FAILED,
err);
SERIALIZATION_ERROR_CHECK(res, error, bail);
bail:
SERIALIZATION_ERROR_RAISE_IF_FAILED(error);
return;
}
void nrf_802154_tx_ack_started(const uint8_t * p_data)
{
/* Due to timing restrictions this function cannot be serialized directly.
* Instead parameter is memorized and serialization is triggered by
* later call to last_tx_ack_started_send
*/
mp_last_tx_ack = p_data;
}
static nrf_802154_ser_err_t last_tx_ack_started_send(void)
{
nrf_802154_ser_err_t res = NRF_802154_SERIALIZATION_ERROR_OK;
const uint8_t * p_last_tx_ack = mp_last_tx_ack;
NRF_802154_SPINEL_LOG_BANNER_CALLING();
if (p_last_tx_ack != NULL)
{
mp_last_tx_ack = NULL;
res = nrf_802154_spinel_send_cmd_prop_value_is(
SPINEL_PROP_VENDOR_NORDIC_NRF_802154_TX_ACK_STARTED,
SPINEL_DATATYPE_NRF_802154_TX_ACK_STARTED,
p_last_tx_ack,
(size_t)(p_last_tx_ack[0]));
}
return res;
}
void nrf_802154_received_timestamp_raw(uint8_t * p_data,
int8_t power,
uint8_t lqi,
uint32_t time)
{
nrf_802154_ser_err_t res;
uint32_t local_data_handle;
SERIALIZATION_ERROR_INIT(error);
res = last_tx_ack_started_send();
SERIALIZATION_ERROR_CHECK(res, error, bail);
NRF_802154_SPINEL_LOG_BANNER_CALLING();
NRF_802154_SPINEL_LOG_BUFF(p_data, p_data[0]);
// Create a handle to the original frame buffer
bool handle_added = nrf_802154_buffer_mgr_src_add(nrf_802154_spinel_src_buffer_mgr_get(),
(void *)p_data,
&local_data_handle);
if (!handle_added)
{
// Handle could not be created. Drop the frame and throw an error
nrf_802154_buffer_free_raw(p_data);
SERIALIZATION_ERROR(NRF_802154_SERIALIZATION_ERROR_NO_MEMORY, error, bail);
}
// Serialize the call
res = nrf_802154_spinel_send_cmd_prop_value_is(
SPINEL_PROP_VENDOR_NORDIC_NRF_802154_RECEIVED_TIMESTAMP_RAW,
SPINEL_DATATYPE_NRF_802154_RECEIVED_TIMESTAMP_RAW,
NRF_802154_HDATA_ENCODE(local_data_handle, p_data, p_data[0]),
power,
lqi,
time);
if (res < 0)
{
// Serialization failed. Drop the frame, clean up and throw an error
nrf_802154_buffer_mgr_src_remove_by_buffer_handle(nrf_802154_spinel_src_buffer_mgr_get(),
local_data_handle);
nrf_802154_buffer_free_raw(p_data);
SERIALIZATION_ERROR(res, error, bail);
}
bail:
SERIALIZATION_ERROR_RAISE_IF_FAILED(error);
return;
}
void nrf_802154_receive_failed(nrf_802154_rx_error_t error, uint32_t id)
{
nrf_802154_ser_err_t res;
SERIALIZATION_ERROR_INIT(ser_error);
res = last_tx_ack_started_send();
SERIALIZATION_ERROR_CHECK(res, ser_error, bail);
NRF_802154_SPINEL_LOG_BANNER_CALLING();
NRF_802154_SPINEL_LOG_VAR("%u", error);
// Serialize the call
res = nrf_802154_spinel_send_cmd_prop_value_is(
SPINEL_PROP_VENDOR_NORDIC_NRF_802154_RECEIVE_FAILED,
SPINEL_DATATYPE_NRF_802154_RECEIVE_FAILED,
error,
id);
SERIALIZATION_ERROR_CHECK(res, ser_error, bail);
bail:
SERIALIZATION_ERROR_RAISE_IF_FAILED(ser_error);
}
void nrf_802154_transmitted_raw(uint8_t * p_frame,
const nrf_802154_transmit_done_metadata_t * p_metadata)
{
uint32_t remote_frame_handle;
uint32_t ack_handle = 0;
uint8_t * p_ack = p_metadata->data.transmitted.p_ack;
SERIALIZATION_ERROR_INIT(error);
NRF_802154_SPINEL_LOG_BANNER_CALLING();
NRF_802154_SPINEL_LOG_BUFF(p_frame, p_frame[0]);
// Search for the handle to the original frame buffer based on the local pointer
bool frame_found = nrf_802154_buffer_mgr_dst_search_by_local_pointer(
nrf_802154_spinel_dst_buffer_mgr_get(),
(void *)p_frame,
&remote_frame_handle);
// The handle is expected to be found, throw an error if it was not found
SERIALIZATION_ERROR_IF(!frame_found,
NRF_802154_SERIALIZATION_ERROR_INVALID_BUFFER,
error,
bail);
if (p_ack != NULL)
{
// Create a handle to the original Ack buffer
bool ack_handle_added = nrf_802154_buffer_mgr_src_add(
nrf_802154_spinel_src_buffer_mgr_get(),
(void *)p_ack,
&ack_handle);
if (!ack_handle_added)
{
// Drop the transmitted frame and throw an error if Ack could not be stored
local_transmitted_frame_ptr_free((void *)p_frame);
SERIALIZATION_ERROR(NRF_802154_SERIALIZATION_ERROR_NO_MEMORY, error, bail);
}
}
// Serialize the call
nrf_802154_ser_err_t res = nrf_802154_spinel_send_cmd_prop_value_is(
SPINEL_PROP_VENDOR_NORDIC_NRF_802154_TRANSMITTED_RAW,
SPINEL_DATATYPE_NRF_802154_TRANSMITTED_RAW,
NRF_802154_TRANSMITTED_RAW_ENCODE(remote_frame_handle, p_frame, *p_metadata, ack_handle));
// Free the local frame pointer no matter the result of serialization
local_transmitted_frame_ptr_free((void *)p_frame);
// Throw an error if serialization failed
SERIALIZATION_ERROR_CHECK(res, error, bail);
bail:
SERIALIZATION_ERROR_RAISE_IF_FAILED(error);
return;
}
void nrf_802154_transmit_failed(uint8_t * p_frame,
nrf_802154_tx_error_t tx_error,
const nrf_802154_transmit_done_metadata_t * p_metadata)
{
uint32_t remote_frame_handle;
SERIALIZATION_ERROR_INIT(error);
NRF_802154_SPINEL_LOG_BANNER_CALLING();
NRF_802154_SPINEL_LOG_BUFF(p_frame, p_frame[0]);
// Search for the handle to the original frame buffer based on the local pointer
bool frame_found = nrf_802154_buffer_mgr_dst_search_by_local_pointer(
nrf_802154_spinel_dst_buffer_mgr_get(),
(void *)p_frame,
&remote_frame_handle);
// The handle is expected to be found, throw an error if it was not found
SERIALIZATION_ERROR_IF(!frame_found,
NRF_802154_SERIALIZATION_ERROR_INVALID_BUFFER,
error,
bail);
// Serialize the call
nrf_802154_ser_err_t res = nrf_802154_spinel_send_cmd_prop_value_is(
SPINEL_PROP_VENDOR_NORDIC_NRF_802154_TRANSMIT_FAILED,
SPINEL_DATATYPE_NRF_802154_TRANSMIT_FAILED,
NRF_802154_TRANSMIT_FAILED_ENCODE(remote_frame_handle, p_frame, tx_error, *p_metadata));
// Free the local frame pointer no matter the result of serialization
local_transmitted_frame_ptr_free((void *)p_frame);
// Throw an error if serialization failed
SERIALIZATION_ERROR_CHECK(res, error, bail);
bail:
SERIALIZATION_ERROR_RAISE_IF_FAILED(error);
return;
}
#ifdef TEST
/**@brief Unit test facility */
void nrf_802154_spinel_net_module_reset(void)
{
mp_last_tx_ack = NULL;
}
#endif