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