mirror of
https://github.com/apple/swift.git
synced 2025-12-21 12:14:44 +01:00
153 lines
4.5 KiB
C++
153 lines
4.5 KiB
C++
//===--- FormalEvaluation.h -------------------------------------*- C++ -*-===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2017 Apple Inc. and the Swift project authors
|
|
// Licensed under Apache License v2.0 with Runtime Library Exception
|
|
//
|
|
// See https://swift.org/LICENSE.txt for license information
|
|
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef SWIFT_SILGEN_FORMALEVALUATION_H
|
|
#define SWIFT_SILGEN_FORMALEVALUATION_H
|
|
|
|
#include "Cleanup.h"
|
|
#include "swift/Basic/DiverseStack.h"
|
|
#include "swift/SIL/SILValue.h"
|
|
#include "llvm/ADT/Optional.h"
|
|
|
|
namespace swift {
|
|
namespace Lowering {
|
|
|
|
class SILGenFunction;
|
|
class LogicalPathComponent;
|
|
|
|
class FormalEvaluation {
|
|
public:
|
|
enum Kind { Shared, Exclusive };
|
|
|
|
private:
|
|
unsigned allocatedSize;
|
|
Kind kind;
|
|
|
|
protected:
|
|
SILLocation loc;
|
|
CleanupHandle cleanup;
|
|
|
|
FormalEvaluation(unsigned allocatedSize, Kind kind, SILLocation loc,
|
|
CleanupHandle cleanup)
|
|
: allocatedSize(allocatedSize), kind(kind), loc(loc), cleanup(cleanup) {}
|
|
|
|
public:
|
|
virtual ~FormalEvaluation() {}
|
|
|
|
// This anchor method serves three purposes: it aligns the class to
|
|
// a pointer boundary, it makes the class a primary base so that
|
|
// subclasses will be at offset zero, and it anchors the v-table
|
|
// to a specific file.
|
|
virtual void _anchor();
|
|
|
|
/// Return the allocated size of this object. This is required by
|
|
/// DiverseStack for iteration.
|
|
size_t allocated_size() const { return allocatedSize; }
|
|
|
|
CleanupHandle getCleanup() const { return cleanup; }
|
|
|
|
Kind getKind() const { return kind; }
|
|
|
|
virtual void finish(SILGenFunction &gen) = 0;
|
|
};
|
|
|
|
class SharedBorrowFormalEvaluation : public FormalEvaluation {
|
|
SILValue originalValue;
|
|
SILValue borrowedValue;
|
|
|
|
public:
|
|
SharedBorrowFormalEvaluation(SILLocation loc, CleanupHandle cleanup,
|
|
SILValue originalValue, SILValue borrowedValue)
|
|
: FormalEvaluation(sizeof(*this), FormalEvaluation::Shared, loc, cleanup),
|
|
originalValue(originalValue), borrowedValue(borrowedValue) {}
|
|
void finish(SILGenFunction &gen) override;
|
|
|
|
SILValue getBorrowedValue() const { return borrowedValue; }
|
|
SILValue getOriginalValue() const { return originalValue; }
|
|
};
|
|
|
|
class FormalEvaluationContext {
|
|
DiverseStack<FormalEvaluation, 128> stack;
|
|
|
|
public:
|
|
using stable_iterator = decltype(stack)::stable_iterator;
|
|
using iterator = decltype(stack)::iterator;
|
|
|
|
FormalEvaluationContext() : stack() {}
|
|
|
|
// This is a type that can only be embedded in other types, it can not be
|
|
// moved or copied.
|
|
FormalEvaluationContext(const FormalEvaluationContext &) = delete;
|
|
FormalEvaluationContext(FormalEvaluationContext &&) = delete;
|
|
FormalEvaluationContext &operator=(const FormalEvaluationContext &) = delete;
|
|
FormalEvaluationContext &operator=(FormalEvaluationContext &&) = delete;
|
|
|
|
~FormalEvaluationContext() {
|
|
assert(stack.empty() &&
|
|
"entries remaining on writeback stack at end of function!");
|
|
}
|
|
|
|
iterator begin() { return stack.begin(); }
|
|
iterator end() { return stack.end(); }
|
|
stable_iterator stabilize(iterator iter) const {
|
|
return stack.stabilize(iter);
|
|
}
|
|
stable_iterator stable_begin() { return stabilize(begin()); }
|
|
iterator find(stable_iterator iter) { return stack.find(iter); }
|
|
|
|
template <class U, class... ArgTypes> void push(ArgTypes &&... args) {
|
|
stack.push<U>(std::forward<ArgTypes>(args)...);
|
|
}
|
|
|
|
void pop() { stack.pop(); }
|
|
|
|
/// Pop objects off of the stack until \p the object pointed to by stable_iter
|
|
/// is the top element of the stack.
|
|
void pop(stable_iterator stable_iter) { stack.pop(stable_iter); }
|
|
};
|
|
|
|
class FormalEvaluationScope {
|
|
SILGenFunction &gen;
|
|
llvm::Optional<FormalEvaluationContext::stable_iterator> savedDepth;
|
|
bool wasInWritebackScope;
|
|
|
|
public:
|
|
FormalEvaluationScope(SILGenFunction &gen);
|
|
~FormalEvaluationScope() {
|
|
if (!savedDepth.hasValue())
|
|
return;
|
|
popImpl();
|
|
}
|
|
|
|
bool isPopped() const { return !savedDepth.hasValue(); }
|
|
|
|
void pop() {
|
|
assert(!isPopped() && "popping an already-popped writeback scope!");
|
|
popImpl();
|
|
savedDepth.reset();
|
|
}
|
|
|
|
FormalEvaluationScope(const FormalEvaluationScope &) = delete;
|
|
FormalEvaluationScope &operator=(const FormalEvaluationScope &) = delete;
|
|
|
|
FormalEvaluationScope(FormalEvaluationScope &&o);
|
|
FormalEvaluationScope &operator=(FormalEvaluationScope &&o) = delete;
|
|
|
|
private:
|
|
void popImpl();
|
|
};
|
|
|
|
} // namespace Lowering
|
|
} // namespace swift
|
|
|
|
#endif
|