76 lines
2.4 KiB
C++
76 lines
2.4 KiB
C++
|
//===--- Stack.cpp - Utilities for dealing with stack space ---------------===//
|
||
|
//
|
||
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
||
|
// See https://llvm.org/LICENSE.txt for license information.
|
||
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
||
|
//
|
||
|
//===----------------------------------------------------------------------===//
|
||
|
///
|
||
|
/// \file
|
||
|
/// Defines utilities for dealing with stack allocation and stack space.
|
||
|
///
|
||
|
//===----------------------------------------------------------------------===//
|
||
|
|
||
|
#include "clang/Basic/Stack.h"
|
||
|
#include "llvm/ADT/Optional.h"
|
||
|
#include "llvm/Support/CrashRecoveryContext.h"
|
||
|
|
||
|
#ifdef _MSC_VER
|
||
|
#include <intrin.h> // for _AddressOfReturnAddress
|
||
|
#endif
|
||
|
|
||
|
static LLVM_THREAD_LOCAL void *BottomOfStack = nullptr;
|
||
|
|
||
|
static void *getStackPointer() {
|
||
|
#if __GNUC__ || __has_builtin(__builtin_frame_address)
|
||
|
return __builtin_frame_address(0);
|
||
|
#elif defined(_MSC_VER)
|
||
|
return _AddressOfReturnAddress();
|
||
|
#else
|
||
|
char CharOnStack = 0;
|
||
|
// The volatile store here is intended to escape the local variable, to
|
||
|
// prevent the compiler from optimizing CharOnStack into anything other
|
||
|
// than a char on the stack.
|
||
|
//
|
||
|
// Tested on: MSVC 2015 - 2019, GCC 4.9 - 9, Clang 3.2 - 9, ICC 13 - 19.
|
||
|
char *volatile Ptr = &CharOnStack;
|
||
|
return Ptr;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
void clang::noteBottomOfStack() {
|
||
|
if (!BottomOfStack)
|
||
|
BottomOfStack = getStackPointer();
|
||
|
}
|
||
|
|
||
|
bool clang::isStackNearlyExhausted() {
|
||
|
// We consider 256 KiB to be sufficient for any code that runs between checks
|
||
|
// for stack size.
|
||
|
constexpr size_t SufficientStack = 256 << 10;
|
||
|
|
||
|
// If we don't know where the bottom of the stack is, hope for the best.
|
||
|
if (!BottomOfStack)
|
||
|
return false;
|
||
|
|
||
|
intptr_t StackDiff = (intptr_t)getStackPointer() - (intptr_t)BottomOfStack;
|
||
|
size_t StackUsage = (size_t)std::abs(StackDiff);
|
||
|
|
||
|
// If the stack pointer has a surprising value, we do not understand this
|
||
|
// stack usage scheme. (Perhaps the target allocates new stack regions on
|
||
|
// demand for us.) Don't try to guess what's going on.
|
||
|
if (StackUsage > DesiredStackSize)
|
||
|
return false;
|
||
|
|
||
|
return StackUsage >= DesiredStackSize - SufficientStack;
|
||
|
}
|
||
|
|
||
|
void clang::runWithSufficientStackSpaceSlow(llvm::function_ref<void()> Diag,
|
||
|
llvm::function_ref<void()> Fn) {
|
||
|
llvm::CrashRecoveryContext CRC;
|
||
|
CRC.RunSafelyOnThread([&] {
|
||
|
noteBottomOfStack();
|
||
|
Diag();
|
||
|
Fn();
|
||
|
}, DesiredStackSize);
|
||
|
}
|