/* * Copyright (c) 2021 Huawei Device 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. */ #ifndef PANDA_LIBPANDABASE_OS_THREAD_H_ #define PANDA_LIBPANDABASE_OS_THREAD_H_ #include "os/error.h" #include "utils/expected.h" #include #include #include namespace panda::os::thread { using ThreadId = uint32_t; using native_handle_type = std::thread::native_handle_type; ThreadId GetCurrentThreadId(); int SetPriority(int thread_id, int prio); int GetPriority(int thread_id); int SetThreadName(native_handle_type pthread_id, const char *name); native_handle_type GetNativeHandle(); void Yield(); void NativeSleep(unsigned int ms); void ThreadDetach(native_handle_type pthread_id); void ThreadExit(void *retval); void ThreadJoin(native_handle_type pthread_id, void **retval); // Templated functions need to be defined here to be accessible everywhere namespace internal { template struct SharedPtrStruct; template // CODECHECK-NOLINTNEXTLINE(CPP_RULE_ID_NO_USE_SHAREDPTR) using SharedPtrToSharedPtrStruct = std::shared_ptr>; template struct SharedPtrStruct { SharedPtrToSharedPtrStruct this_ptr; // NOLINT(misc-non-private-member-variables-in-classes) T data; // NOLINT(misc-non-private-member-variables-in-classes) SharedPtrStruct(SharedPtrToSharedPtrStruct ptr_in, T data_in) : this_ptr(std::move(ptr_in)), data(std::move(data_in)) { } }; template struct Seq { }; template struct GenArgSeq : GenArgSeq { }; template struct GenArgSeq<1, Is...> : Seq { }; template static void CallFunc(Func &func, Tuple &args, Seq /* unused */) { func(std::get(args)...); } template static void CallFunc(Func &func, Tuple &args) { CallFunc(func, args, GenArgSeq()); } template static void *ProxyFunc(void *args) { // Parse pointer and move args to local tuple. // We need this pointer to be destroyed by the time function starts to avoid memleak on thread termination. Tuple args_tuple; { auto args_ptr = static_cast *>(args); SharedPtrToSharedPtrStruct local; // This breaks shared pointer loop local.swap(args_ptr->this_ptr); // This moves tuple data to local variable args_tuple = args_ptr->data; } Func *func = std::get<0>(args_tuple); CallFunc(*func, args_tuple); return nullptr; } } // namespace internal template native_handle_type ThreadStart(Func *func, Args... args) { native_handle_type tid; auto args_tuple = std::make_tuple(func, std::move(args)...); internal::SharedPtrStruct *ptr = nullptr; { // CODECHECK-NOLINTNEXTLINE(CPP_RULE_ID_NO_USE_SHAREDPTR) auto shared_ptr = std::make_shared>(nullptr, args_tuple); // CODECHECK-NOLINTNEXTLINE(CPP_RULE_ID_NO_USE_SHAREDPTR) ptr = shared_ptr.get(); // Make recursive ref to prevent shared pointer from being destroyed until child thread acquires it. // CODECHECK-NOLINTNEXTLINE(CPP_RULE_ID_NO_USE_SHAREDPTR) ptr->this_ptr = shared_ptr; // Leave scope to make sure that local shared_ptr is destroyed before thread creation. } pthread_create(&tid, nullptr, &internal::ProxyFunc::value>, static_cast(ptr)); return tid; } } // namespace panda::os::thread #endif // PANDA_LIBPANDABASE_OS_THREAD_H_