/* * * Copyright (c) 2020-2021 Project CHIP Authors * Copyright (c) 2018 Nest Labs, Inc. * * 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 PlatformManager object. */ #pragma once #include #include #include #include #if CHIP_DEVICE_CONFIG_WITH_GLIB_MAIN_LOOP #include #endif namespace chip { namespace DeviceLayer { /** * Concrete implementation of the PlatformManager singleton object for Linux platforms. */ class PlatformManagerImpl final : public PlatformManager, public Internal::GenericPlatformManagerImpl_POSIX { // Allow the PlatformManager interface class to delegate method calls to // the implementation methods provided by this class. friend PlatformManager; // Allow the generic implementation base class to call helper methods on // this class. #ifndef DOXYGEN_SHOULD_SKIP_THIS friend Internal::GenericPlatformManagerImpl_POSIX; #endif public: // ===== Platform-specific members that may be accessed directly by the application. #if CHIP_DEVICE_CONFIG_WITH_GLIB_MAIN_LOOP /** * @brief Invoke a function on the Matter GLib context. * * If execution of the function will have to be scheduled on other thread, * this call will block the current thread until the function is executed. * * @param[in] function The function to call. * @param[in] userData User data to pass to the function. * @returns The result of the function. */ template CHIP_ERROR GLibMatterContextInvokeSync(CHIP_ERROR (*func)(T *), T * userData) { struct { CHIP_ERROR returnValue = CHIP_NO_ERROR; CHIP_ERROR (*functionToCall)(T *); T * userData; } context; context.functionToCall = func; context.userData = userData; LambdaBridge bridge; bridge.Initialize([&context]() { context.returnValue = context.functionToCall(context.userData); }); _GLibMatterContextInvokeSync(std::move(bridge)); return context.returnValue; } unsigned int GLibMatterContextAttachSource(GSource * source) { VerifyOrDie(mGLibMainLoop != nullptr); return g_source_attach(source, g_main_loop_get_context(mGLibMainLoop)); } #endif System::Clock::Timestamp GetStartTime() { return mStartTime; } private: // ===== Methods that implement the PlatformManager abstract interface. CHIP_ERROR _InitChipStack(); void _Shutdown(); // ===== Members for internal use by the following friends. friend PlatformManager & PlatformMgr(); friend PlatformManagerImpl & PlatformMgrImpl(); friend class Internal::BLEManagerImpl; System::Clock::Timestamp mStartTime = System::Clock::kZero; static PlatformManagerImpl sInstance; #if CHIP_DEVICE_CONFIG_WITH_GLIB_MAIN_LOOP struct GLibMatterContextInvokeData { LambdaBridge bridge; // Sync primitives to wait for the function to be executed std::condition_variable mDoneCond; bool mDone = false; }; /** * @brief Invoke a function on the Matter GLib context. * * @param[in] bridge a LambdaBridge object that holds the lambda to be invoked within the GLib context. * * @note This function moves the LambdaBridge into the GLib context for invocation. * The LambdaBridge is created and initialised in GLibMatterContextInvokeSync(). * use the GLibMatterContextInvokeSync() template function instead of this one. */ void _GLibMatterContextInvokeSync(LambdaBridge && bridge); // XXX: Mutex for guarding access to glib main event loop callback indirection // synchronization primitives. This is a workaround to suppress TSAN warnings. // TSAN does not know that from the thread synchronization perspective the // g_source_attach() function should be treated as pthread_create(). Memory // access to shared data before the call to g_source_attach() without mutex // is not a race condition - the callback will not be executed on glib main // event loop thread before the call to g_source_attach(). std::mutex mGLibMainLoopCallbackIndirectionMutex; GMainLoop * mGLibMainLoop = nullptr; GThread * mGLibMainLoopThread = nullptr; #endif // CHIP_DEVICE_CONFIG_WITH_GLIB_MAIN_LOOP }; /** * Returns the public interface of the PlatformManager singleton object. * * chip applications should use this to access features of the PlatformManager object * that are common to all platforms. */ inline PlatformManager & PlatformMgr() { return PlatformManagerImpl::sInstance; } /** * Returns the platform-specific implementation of the PlatformManager singleton object. * * chip applications can use this to gain access to features of the PlatformManager * that are specific to the platform. */ inline PlatformManagerImpl & PlatformMgrImpl() { return PlatformManagerImpl::sInstance; } } // namespace DeviceLayer } // namespace chip