xref: /llvm-project/clang/lib/Basic/Stack.cpp (revision a996cc217cefb9071888de38c6f05e5742d0106f)
1ac1d5986SRichard Smith //===--- Stack.cpp - Utilities for dealing with stack space ---------------===//
226a92d58SRichard Smith //
326a92d58SRichard Smith // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
426a92d58SRichard Smith // See https://llvm.org/LICENSE.txt for license information.
526a92d58SRichard Smith // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
626a92d58SRichard Smith //
726a92d58SRichard Smith //===----------------------------------------------------------------------===//
826a92d58SRichard Smith ///
926a92d58SRichard Smith /// \file
1026a92d58SRichard Smith /// Defines utilities for dealing with stack allocation and stack space.
1126a92d58SRichard Smith ///
1226a92d58SRichard Smith //===----------------------------------------------------------------------===//
1326a92d58SRichard Smith 
1426a92d58SRichard Smith #include "clang/Basic/Stack.h"
1526a92d58SRichard Smith #include "llvm/Support/CrashRecoveryContext.h"
1626a92d58SRichard Smith 
17*4d3a3366SRichard Smith #ifdef _MSC_VER
18*4d3a3366SRichard Smith #include <intrin.h>  // for _AddressOfReturnAddress
19*4d3a3366SRichard Smith #endif
20*4d3a3366SRichard Smith 
2126a92d58SRichard Smith static LLVM_THREAD_LOCAL void *BottomOfStack = nullptr;
2226a92d58SRichard Smith 
getStackPointer()2326a92d58SRichard Smith static void *getStackPointer() {
2426a92d58SRichard Smith #if __GNUC__ || __has_builtin(__builtin_frame_address)
2526a92d58SRichard Smith   return __builtin_frame_address(0);
2626a92d58SRichard Smith #elif defined(_MSC_VER)
2726a92d58SRichard Smith   return _AddressOfReturnAddress();
2826a92d58SRichard Smith #else
2926a92d58SRichard Smith   char CharOnStack = 0;
3026a92d58SRichard Smith   // The volatile store here is intended to escape the local variable, to
3126a92d58SRichard Smith   // prevent the compiler from optimizing CharOnStack into anything other
3226a92d58SRichard Smith   // than a char on the stack.
3326a92d58SRichard Smith   //
3426a92d58SRichard Smith   // Tested on: MSVC 2015 - 2019, GCC 4.9 - 9, Clang 3.2 - 9, ICC 13 - 19.
3526a92d58SRichard Smith   char *volatile Ptr = &CharOnStack;
3626a92d58SRichard Smith   return Ptr;
3726a92d58SRichard Smith #endif
3826a92d58SRichard Smith }
3926a92d58SRichard Smith 
noteBottomOfStack()4026a92d58SRichard Smith void clang::noteBottomOfStack() {
4126a92d58SRichard Smith   if (!BottomOfStack)
4226a92d58SRichard Smith     BottomOfStack = getStackPointer();
4326a92d58SRichard Smith }
4426a92d58SRichard Smith 
isStackNearlyExhausted()4526a92d58SRichard Smith bool clang::isStackNearlyExhausted() {
4626a92d58SRichard Smith   // We consider 256 KiB to be sufficient for any code that runs between checks
4726a92d58SRichard Smith   // for stack size.
4826a92d58SRichard Smith   constexpr size_t SufficientStack = 256 << 10;
4926a92d58SRichard Smith 
5026a92d58SRichard Smith   // If we don't know where the bottom of the stack is, hope for the best.
5126a92d58SRichard Smith   if (!BottomOfStack)
5226a92d58SRichard Smith     return false;
5326a92d58SRichard Smith 
5426a92d58SRichard Smith   intptr_t StackDiff = (intptr_t)getStackPointer() - (intptr_t)BottomOfStack;
5526a92d58SRichard Smith   size_t StackUsage = (size_t)std::abs(StackDiff);
5626a92d58SRichard Smith 
5726a92d58SRichard Smith   // If the stack pointer has a surprising value, we do not understand this
5826a92d58SRichard Smith   // stack usage scheme. (Perhaps the target allocates new stack regions on
5926a92d58SRichard Smith   // demand for us.) Don't try to guess what's going on.
6026a92d58SRichard Smith   if (StackUsage > DesiredStackSize)
6126a92d58SRichard Smith     return false;
6226a92d58SRichard Smith 
6326a92d58SRichard Smith   return StackUsage >= DesiredStackSize - SufficientStack;
6426a92d58SRichard Smith }
6526a92d58SRichard Smith 
runWithSufficientStackSpaceSlow(llvm::function_ref<void ()> Diag,llvm::function_ref<void ()> Fn)6626a92d58SRichard Smith void clang::runWithSufficientStackSpaceSlow(llvm::function_ref<void()> Diag,
6726a92d58SRichard Smith                                             llvm::function_ref<void()> Fn) {
6826a92d58SRichard Smith   llvm::CrashRecoveryContext CRC;
6926a92d58SRichard Smith   CRC.RunSafelyOnThread([&] {
7026a92d58SRichard Smith     noteBottomOfStack();
7126a92d58SRichard Smith     Diag();
7226a92d58SRichard Smith     Fn();
7326a92d58SRichard Smith   }, DesiredStackSize);
7426a92d58SRichard Smith }
75