Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[NXP] Switch to using BLE mgr impl files from common NXP folder #34349

Merged
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
138 changes: 138 additions & 0 deletions src/platform/nxp/common/ble_zephyr/BLEAdvertisingArbiter.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,138 @@
/*
*
* Copyright (c) 2023-2024 Project CHIP Authors
*
* 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 "BLEAdvertisingArbiter.h"

#include <lib/support/CodeUtils.h>
#include <lib/support/logging/CHIPLogging.h>
#include <system/SystemError.h>

namespace chip {
namespace DeviceLayer {
namespace BLEAdvertisingArbiter {
namespace {

// List of advertising requests ordered by priority
sys_slist_t sRequests;

// Cast an intrusive list node to the containing request object
const BLEAdvertisingArbiter::Request & ToRequest(const sys_snode_t * node)
{
return *static_cast<const BLEAdvertisingArbiter::Request *>(node);
}

// Notify application about stopped advertising if the callback has been provided
void NotifyAdvertisingStopped(const sys_snode_t * node)
{
VerifyOrReturn(node);

const Request & request = ToRequest(node);

if (request.onStopped != nullptr)
{
request.onStopped();
}
}

// Restart advertising using the top-priority request
CHIP_ERROR RestartAdvertising()
{
// Note: bt_le_adv_stop() returns success when the advertising was not started
ReturnErrorOnFailure(MapErrorZephyr(bt_le_adv_stop()));
ReturnErrorCodeIf(sys_slist_is_empty(&sRequests), CHIP_NO_ERROR);

const Request & top = ToRequest(sys_slist_peek_head(&sRequests));
const bt_le_adv_param params = BT_LE_ADV_PARAM_INIT(top.options, top.minInterval, top.maxInterval, nullptr);
const int result = bt_le_adv_start(&params, top.advertisingData.data(), top.advertisingData.size(), top.scanResponseData.data(),
top.scanResponseData.size());

if (top.onStarted != nullptr)
{
top.onStarted(result);
}

return MapErrorZephyr(result);
}

} // namespace

CHIP_ERROR InsertRequest(Request & request)
{
CancelRequest(request);

sys_snode_t * prev = nullptr;
sys_snode_t * node = nullptr;

// Find position of the request in the list that preserves ordering by priority
SYS_SLIST_FOR_EACH_NODE(&sRequests, node)
{
if (request.priority < ToRequest(node).priority)
{
break;
}

prev = node;
}

if (prev == nullptr)
{
NotifyAdvertisingStopped(sys_slist_peek_head(&sRequests));
sys_slist_prepend(&sRequests, &request);
}
else
{
sys_slist_insert(&sRequests, prev, &request);
}

// If the request is top-priority, restart the advertising
if (sys_slist_peek_head(&sRequests) == &request)
{
return RestartAdvertising();
}

return CHIP_NO_ERROR;
}

void CancelRequest(Request & request)
{
const bool isTopPriority = (sys_slist_peek_head(&sRequests) == &request);
VerifyOrReturn(sys_slist_find_and_remove(&sRequests, &request));

// If cancelled request was top-priority, restart the advertising.
if (isTopPriority)
{
RestartAdvertising();
}
}

} // namespace BLEAdvertisingArbiter

/**
* This implements a mapping function for CHIP System Layer errors that allows mapping integers in the number space of the
* Zephyr OS user API stack errors into the POSIX range.
*
* @param[in] aError The native Zephyr API error to map.
*
* @return The mapped POSIX error.
*/
DLL_EXPORT CHIP_ERROR MapErrorZephyr(int aError)
{
return chip::System::Internal::MapErrorPOSIX(-aError);
}

} // namespace DeviceLayer
} // namespace chip
107 changes: 107 additions & 0 deletions src/platform/nxp/common/ble_zephyr/BLEAdvertisingArbiter.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,107 @@
/*
*
* Copyright (c) 2023-2024 Project CHIP Authors
*
* 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.
*/

#pragma once

#include <lib/support/Span.h>

#include <zephyr/bluetooth/bluetooth.h>
#include <zephyr/sys/slist.h>

#include <cstdint>

/**
* @file
* Bluetooth LE advertising arbiter.
*
* The purpose for this module is to coordinate BLE advertising between
* different application components.
*
* An application component that wants to advertise BLE services is expected to
* define a request with a desired priority, and pass it to the BLE advertising
* arbiter. If there are multiple components that request BLE advertising at the
* same time, the arbiter selects the one with the highest priority (represented
* by the lowest numeric value) and starts the BLE advertising using parameters
* defined in the winning request.
*
* The BLE arbiter does not take ownership of a submitted request, so the
* request object must be sustained until it is cancelled by the application.
*/

namespace chip {
namespace DeviceLayer {
namespace BLEAdvertisingArbiter {

using OnAdvertisingStarted = void (*)(int result);
using OnAdvertisingStopped = void (*)();

struct Request : public sys_snode_t
{
uint8_t priority; ///< Advertising request priority. Lower value means higher priority
uint32_t options; ///< Advertising options: bitmask of BT_LE_ADV_OPT_XXX constants from Zephyr
uint16_t minInterval; ///< Minimum advertising interval in 0.625 ms units
uint16_t maxInterval; ///< Maximum advertising interval in 0.625 ms units
Span<const bt_data> advertisingData; ///< Advertising data fields
Span<const bt_data> scanResponseData; ///< Scan response data fields
OnAdvertisingStarted onStarted; ///< (Optional) Callback invoked when the request becomes top-priority.
OnAdvertisingStopped onStopped; ///< (Optional) Callback invoked when the request stops being top-priority.
};

/**
* @brief Request BLE advertising
*
* Add the request to the internal list of competing requests. If the request
* has higher priority than other requests in the list, restart the BLE
* advertising immediately using parameters defined in the new request.
*
* Inserting a request object that is already registered at the advertising
* arbiter automatically cancels the previous request.
*
* @note This method does not take ownership of the request object so the object
* must not get destroyed before it is cancelled.
*
* @param request Reference to advertising request that contains priority and
* other advertising parameters.
* @return error If the request is top-priority and failed to restart the
* advertising.
* @return success Otherwise.
*/
CHIP_ERROR InsertRequest(Request & request);

/**
* @brief Cancel BLE advertising request
*
* Remove the request from the internal list of competing requests. If the
* request is the winning (top-priority) one at the time of calling this
* function, restart the BLE advertising using parameters defined in the 2nd
* top-priority request in the list, or stop the BLE advertising completely if
* this is the last request in the list.
*
* An attempt to cancel a request that has not been registered at the
* advertising arbiter is a no-op. That is, it returns immediately.
*
* @param request Reference to advertising request that contains priority and
* other advertising parameters.
*/
void CancelRequest(Request & request);

} // namespace BLEAdvertisingArbiter

extern CHIP_ERROR MapErrorZephyr(int code);

} // namespace DeviceLayer
} // namespace chip
Loading
Loading