/** * Copyright (C) 2018-present MongoDB, Inc. * * This program is free software: you can redistribute it and/or modify * it under the terms of the Server Side Public License, version 1, * as published by MongoDB, Inc. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * Server Side Public License for more details. * * You should have received a copy of the Server Side Public License * along with this program. If not, see * . * * As a special exception, the copyright holders give permission to link the * code of portions of this program with the OpenSSL library under certain * conditions as described in each individual source file and distribute * linked combinations including the program with the OpenSSL library. You * must comply with the Server Side Public License in all respects for * all of the code used other than as permitted herein. If you modify file(s) * with this exception, you may extend this exception to your version of the * file(s), but you are not obligated to do so. If you do not wish to do so, * delete this exception statement from your version. If you delete this * exception statement from all source files in the program, then also delete * it in the license file. */ #include "mongo/platform/basic.h" #include "mongo/platform/stack_locator.h" #include "mongo/stdx/thread.h" #include "mongo/unittest/unittest.h" namespace mongo { namespace { TEST(StackLocator, StacLocatorFindsStackOfTestExecutorThread) { const StackLocator locator; const auto begin = locator.begin(); ASSERT_TRUE(nullptr != begin); const auto end = locator.end(); ASSERT_TRUE(nullptr != end); ASSERT_TRUE(begin != end); const auto available = locator.available(); ASSERT_TRUE(available); ASSERT_TRUE(available.value() > 0); const auto size = locator.size(); ASSERT_TRUE(size); ASSERT_TRUE(size.value() > 0); ASSERT_TRUE(size.value() > available.value()); } TEST(StackLocator, StacksGrowsDown) { // The current implementation assumes a downward growing stack. Write a test // that confirms that the current platform is downward growing, so that if the // system is ever ported to an upward growing stack, we get a test failure. const StackLocator locator; ASSERT_TRUE(nullptr != locator.begin()); ASSERT_TRUE(nullptr != locator.end()); // NOTE: Technically, comparing pointers for ordering is UB if // they aren't in the same aggregate, but we are already out // with the dragons at the edge of the map. ASSERT_TRUE(locator.begin() > locator.end()); } TEST(StackLocator, StackLocatorFindsStackOfStdThread) { bool foundBounds = false; stdx::thread thr([&] { const StackLocator locator; auto avail = locator.available(); foundBounds = static_cast(avail); }); thr.join(); ASSERT_TRUE(foundBounds); } struct LocatorThreadHelper { #ifdef _WIN32 static DWORD WINAPI run(LPVOID arg) { static_cast(arg)->_run(); return 0; } #else static void* run(void* arg) { static_cast(arg)->_run(); return nullptr; } #endif void _run() { const StackLocator locator; located = static_cast(locator.available()); if (located) size = locator.size().value(); } bool located = false; size_t size = 0; }; TEST(StackLocator, StackLocatorFindsStackOfNativeThreadWithDefaultStack) { LocatorThreadHelper helper; #ifdef _WIN32 HANDLE thread = CreateThread(nullptr, 0, &LocatorThreadHelper::run, &helper, 0, nullptr); ASSERT_NE(WAIT_FAILED, WaitForSingleObject(thread, INFINITE)); #else pthread_attr_t attrs; ASSERT_EQ(0, pthread_attr_init(&attrs)); pthread_t thread; ASSERT_EQ(0, pthread_create(&thread, &attrs, &LocatorThreadHelper::run, &helper)); ASSERT_EQ(0, pthread_join(thread, nullptr)); #endif ASSERT_TRUE(helper.located); } TEST(StackLocator, StackLocatorFindStackOfNativeThreadWithCustomStack) { const size_t kThreadStackSize = 64 * 1024 * 1024; #ifdef _WIN32 LocatorThreadHelper helperNoCommit; HANDLE thread = CreateThread(nullptr, kThreadStackSize, &LocatorThreadHelper::run, &helperNoCommit, STACK_SIZE_PARAM_IS_A_RESERVATION, nullptr); ASSERT_NE(WAIT_FAILED, WaitForSingleObject(thread, INFINITE)); ASSERT_TRUE(helperNoCommit.located); ASSERT_EQ(kThreadStackSize, helperNoCommit.size); LocatorThreadHelper helperCommit; thread = CreateThread( nullptr, kThreadStackSize, &LocatorThreadHelper::run, &helperCommit, 0, nullptr); ASSERT_NE(WAIT_FAILED, WaitForSingleObject(thread, INFINITE)); ASSERT_TRUE(helperCommit.located); ASSERT_TRUE(kThreadStackSize <= helperCommit.size); #else LocatorThreadHelper helper; pthread_attr_t attrs; ASSERT_EQ(0, pthread_attr_init(&attrs)); ASSERT_EQ(0, pthread_attr_setstacksize(&attrs, kThreadStackSize)); pthread_t thread; ASSERT_EQ(0, pthread_create(&thread, &attrs, &LocatorThreadHelper::run, &helper)); ASSERT_EQ(0, pthread_join(thread, nullptr)); ASSERT_TRUE(helper.located); ASSERT_TRUE(kThreadStackSize <= helper.size); #endif } } // namespace } // namespace mongo