/* Copyright (c) 2013 Microsoft Corporation. All rights reserved. Released under Apache 2.0 license as described in the file LICENSE. Author: Leonardo de Moura */ #include #include #include #include #if !defined(LEAN_USE_SPLIT_STACK) #if defined(LEAN_WINDOWS) // no extra included needed so far #elif defined(__APPLE__) #include // NOLINT #else #include // NOLINT #include // NOLINT #endif namespace lean { void throw_get_stack_size_failed() { throw exception("failed to retrieve thread stack size"); } #if defined(LEAN_WINDOWS) size_t get_stack_size(int main) { if (main) { return LEAN_WIN_STACK_SIZE; } else { return lthread::get_thread_stack_size(); } } #elif defined (__APPLE__) size_t get_stack_size(int main) { if (main) { // Retrieve stack size of the main thread. struct rlimit curr; if (getrlimit(RLIMIT_STACK, &curr) != 0) { throw_get_stack_size_failed(); } return curr.rlim_cur; } else { return lthread::get_thread_stack_size(); } } #else size_t get_stack_size(int main) { if (main) { // Retrieve stack size of the main thread. struct rlimit curr; if (getrlimit(RLIMIT_STACK, &curr) != 0) { throw_get_stack_size_failed(); } return curr.rlim_cur; } else { return lthread::get_thread_stack_size(); } } #endif LEAN_THREAD_VALUE(bool, g_stack_info_init, false); LEAN_THREAD_VALUE(size_t, g_stack_size, 0); LEAN_THREAD_VALUE(size_t, g_stack_base, 0); LEAN_THREAD_VALUE(size_t, g_stack_threshold, 0); void save_stack_info(bool main) { g_stack_info_init = true; g_stack_size = get_stack_size(main); char x; g_stack_base = reinterpret_cast(&x); /* g_stack_threshold is a redundant value used to optimize check_stack */ g_stack_threshold = g_stack_base + LEAN_STACK_BUFFER_SPACE - g_stack_size; if (g_stack_threshold > g_stack_base + LEAN_STACK_BUFFER_SPACE) { // negative overflow g_stack_threshold = 0; } } size_t get_used_stack_size() { char y; size_t curr_stack = reinterpret_cast(&y); return g_stack_base - curr_stack; } size_t get_available_stack_size() { size_t sz = get_used_stack_size(); if (sz > g_stack_size) return 0; else return g_stack_size - sz; } void check_stack(char const * component_name) { if (!g_stack_info_init) save_stack_info(false); char y; size_t curr_stack = reinterpret_cast(&y); if (curr_stack < g_stack_threshold) throw stack_space_exception(component_name); } } #endif