/* * * 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. */ /** * @file * Provides an implementation of the ThreadStackManager object * for Telink platform. * */ /* this file behaves like a config.h, comes first */ #include #include #include #include #include namespace chip { namespace DeviceLayer { using namespace ::chip::DeviceLayer::Internal; ThreadStackManagerImpl ThreadStackManagerImpl::sInstance; CHIP_ERROR ThreadStackManagerImpl::_InitThreadStack() { mRadioBlocked = false; mReadyToAttach = false; otInstance * const instance = openthread_get_default_instance(); ReturnErrorOnFailure(GenericThreadStackManagerImpl_OpenThread::DoInit(instance)); #if CHIP_DEVICE_CONFIG_ENABLE_THREAD_SRP_CLIENT k_sem_init(&mSrpClearAllSemaphore, 0, 1); #endif // CHIP_DEVICE_CONFIG_ENABLE_THREAD_SRP_CLIENT return CHIP_NO_ERROR; } void ThreadStackManagerImpl::_LockThreadStack() { openthread_api_mutex_lock(openthread_get_default_context()); } bool ThreadStackManagerImpl::_TryLockThreadStack() { // There's no openthread_api_mutex_try_lock() in Zephyr, so until it's contributed we must use the low-level API return k_mutex_lock(&openthread_get_default_context()->api_lock, K_NO_WAIT) == 0; } void ThreadStackManagerImpl::_UnlockThreadStack() { openthread_api_mutex_unlock(openthread_get_default_context()); } #if CHIP_DEVICE_CONFIG_ENABLE_THREAD_SRP_CLIENT void ThreadStackManagerImpl::_WaitOnSrpClearAllComplete() { k_sem_take(&mSrpClearAllSemaphore, K_SECONDS(2)); } void ThreadStackManagerImpl::_NotifySrpClearAllComplete() { k_sem_give(&mSrpClearAllSemaphore); } #endif // CHIP_DEVICE_CONFIG_ENABLE_THREAD_SRP_CLIENT CHIP_ERROR ThreadStackManagerImpl::_AttachToThreadNetwork(const Thread::OperationalDataset & dataset, NetworkCommissioning::Internal::WirelessDriver::ConnectCallback * callback) { CHIP_ERROR result = CHIP_NO_ERROR; if (mRadioBlocked) { /* On Telink platform it's not possible to rise Thread network when its used by BLE, so just mark that it's provisioned and rise Thread after BLE disconnect */ result = SetThreadProvision(dataset.AsByteSpan()); if (result == CHIP_NO_ERROR) { mReadyToAttach = true; callback->OnResult(NetworkCommissioning::Status::kSuccess, CharSpan(), 0); } } else { result = Internal::GenericThreadStackManagerImpl_OpenThread::_AttachToThreadNetwork(dataset, callback); } return result; } CHIP_ERROR ThreadStackManagerImpl::_StartThreadScan(NetworkCommissioning::ThreadDriver::ScanCallback * callback) { mpScanCallback = callback; /* On Telink platform it's not possible to rise Thread network when its used by BLE, so Thread networks scanning performed before start BLE and also available after switch into Thread */ if (mRadioBlocked) { if (mpScanCallback != nullptr) { DeviceLayer::SystemLayer().ScheduleLambda([this]() { mpScanCallback->OnFinished(NetworkCommissioning::Status::kSuccess, CharSpan(), &mScanResponseIter); mpScanCallback = nullptr; }); } } else { return Internal::GenericThreadStackManagerImpl_OpenThread::_StartThreadScan(mpScanCallback); } return CHIP_NO_ERROR; } void ThreadStackManagerImpl::Finalize(void) { otInstanceFinalize(openthread_get_default_instance()); } } // namespace DeviceLayer } // namespace chip