1 //===--- Stack.cpp - Utilities for dealing with stack space ---------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
10 /// Defines utilities for dealing with stack allocation and stack space.
12 //===----------------------------------------------------------------------===//
14 #include "clang/Basic/Stack.h"
15 #include "llvm/Support/CrashRecoveryContext.h"
18 #include <intrin.h> // for _AddressOfReturnAddress
21 static LLVM_THREAD_LOCAL
void *BottomOfStack
= nullptr;
23 static void *getStackPointer() {
24 #if __GNUC__ || __has_builtin(__builtin_frame_address)
25 return __builtin_frame_address(0);
26 #elif defined(_MSC_VER)
27 return _AddressOfReturnAddress();
30 // The volatile store here is intended to escape the local variable, to
31 // prevent the compiler from optimizing CharOnStack into anything other
32 // than a char on the stack.
34 // Tested on: MSVC 2015 - 2019, GCC 4.9 - 9, Clang 3.2 - 9, ICC 13 - 19.
35 char *volatile Ptr
= &CharOnStack
;
40 void clang::noteBottomOfStack() {
42 BottomOfStack
= getStackPointer();
45 bool clang::isStackNearlyExhausted() {
46 // We consider 256 KiB to be sufficient for any code that runs between checks
48 constexpr size_t SufficientStack
= 256 << 10;
50 // If we don't know where the bottom of the stack is, hope for the best.
54 intptr_t StackDiff
= (intptr_t)getStackPointer() - (intptr_t)BottomOfStack
;
55 size_t StackUsage
= (size_t)std::abs(StackDiff
);
57 // If the stack pointer has a surprising value, we do not understand this
58 // stack usage scheme. (Perhaps the target allocates new stack regions on
59 // demand for us.) Don't try to guess what's going on.
60 if (StackUsage
> DesiredStackSize
)
63 return StackUsage
>= DesiredStackSize
- SufficientStack
;
66 void clang::runWithSufficientStackSpaceSlow(llvm::function_ref
<void()> Diag
,
67 llvm::function_ref
<void()> Fn
) {
68 llvm::CrashRecoveryContext CRC
;
69 CRC
.RunSafelyOnThread([&] {