blob: d2a349bfc51193b80b5bcc91265ce2b9679693f7 (
plain) (
tree)
|
|
#pragma once
#include "align.h"
#include "defaults.h"
#include "compiler.h"
#include "sanitizers.h"
#include <util/generic/array_ref.h>
#include <util/generic/utility.h>
#include <util/generic/yexception.h>
#define STACK_ALIGN (8 * PLATFORM_DATA_ALIGN)
#if defined(_x86_64_) || defined(_i386_) || defined(_arm_) || defined(_ppc64_)
#define STACK_GROW_DOWN 1
#else
#error todo
#endif
/*
* switch method
*/
#if defined(_bionic_) || defined(__IOS__)
#define USE_GENERIC_CONT
#elif defined(_cygwin_)
#define USE_UCONTEXT_CONT
#elif defined(_win_)
#define USE_FIBER_CONT
#elif (defined(_i386_) || defined(_x86_64_) || defined(_arm64_)) && !defined(_k1om_)
#define USE_JUMP_CONT
#else
#define USE_UCONTEXT_CONT
#endif
#if defined(USE_JUMP_CONT)
#if defined(_arm64_)
#include "context_aarch64.h"
#else
#include "context_x86.h"
#endif
#endif
#if defined(USE_UCONTEXT_CONT)
#include <ucontext.h>
#endif
struct ITrampoLine {
virtual ~ITrampoLine() = default;
virtual void DoRun();
virtual void DoRunNaked();
};
struct TContClosure {
ITrampoLine* TrampoLine;
TArrayRef<char> Stack;
const char* ContName = nullptr;
};
#if defined(USE_UCONTEXT_CONT)
class TContMachineContext {
typedef void (*ucontext_func_t)(void);
public:
inline TContMachineContext() {
getcontext(&Ctx_);
}
inline TContMachineContext(const TContClosure& c) {
getcontext(&Ctx_);
Ctx_.uc_link = 0;
Ctx_.uc_stack.ss_sp = (void*)c.Stack.data();
Ctx_.uc_stack.ss_size = c.Stack.size();
Ctx_.uc_stack.ss_flags = 0;
extern void ContextTrampoLine(void* arg);
makecontext(&Ctx_, (ucontext_func_t)ContextTrampoLine, 1, c.TrampoLine);
}
inline ~TContMachineContext() {
}
inline void SwitchTo(TContMachineContext* next) noexcept {
swapcontext(&Ctx_, &next->Ctx_);
}
private:
ucontext_t Ctx_;
};
#endif
#if defined(USE_GENERIC_CONT)
class TContMachineContext {
struct TImpl;
public:
TContMachineContext();
TContMachineContext(const TContClosure& c);
~TContMachineContext();
void SwitchTo(TContMachineContext* next) noexcept;
private:
THolder<TImpl> Impl_;
};
#endif
#if defined(USE_FIBER_CONT)
class TContMachineContext {
public:
TContMachineContext();
TContMachineContext(const TContClosure& c);
~TContMachineContext();
void SwitchTo(TContMachineContext* next) noexcept;
private:
void* Fiber_;
bool MainFiber_;
};
#endif
#if defined(USE_JUMP_CONT)
class TContMachineContext {
public:
inline TContMachineContext() {
Zero(Buf_);
}
TContMachineContext(const TContClosure& c);
inline ~TContMachineContext() = default;
void SwitchTo(TContMachineContext* next) noexcept;
private:
__myjmp_buf Buf_;
struct TSan: public ITrampoLine, public ::NSan::TFiberContext {
TSan() noexcept;
TSan(const TContClosure& c) noexcept;
void DoRunNaked() override;
ITrampoLine* TL;
};
#if defined(_asan_enabled_) || defined(_tsan_enabled_)
TSan San_;
#endif
};
#endif
static inline size_t MachineContextSize() noexcept {
return sizeof(TContMachineContext);
}
/*
* be polite
*/
#if !defined(FROM_CONTEXT_IMPL)
#undef USE_JUMP_CONT
#undef USE_FIBER_CONT
#undef USE_GENERIC_CONT
#undef USE_UCONTEXT_CONT
#undef PROGR_CNT
#undef STACK_CNT
#undef EXTRA_PUSH_ARGS
#endif
struct TExceptionSafeContext: public TContMachineContext {
using TContMachineContext::TContMachineContext;
void SwitchTo(TExceptionSafeContext* to) noexcept;
#if defined(_unix_)
void* Buf_[2] = {nullptr, nullptr};
#endif
};
|