implement proper Abort() for composite lambdas
This commit is contained in:
parent
fa6870fa74
commit
5fd3c9ae5b
@ -286,7 +286,12 @@ class Node::Lambda final : public nf7::Node::Lambda,
|
||||
auto self = d.self.lock();
|
||||
if (!self) return luaL_error(L, "context expired");
|
||||
|
||||
d.caller->Handle(name, nf7::luajit::CheckValue(L, 3), self);
|
||||
auto v = nf7::luajit::CheckValue(L, 3);
|
||||
|
||||
auto caller = d.caller;
|
||||
caller->env().ExecSub(self, [self, caller, v = std::move(v)]() mutable {
|
||||
caller->Handle(name, std::move(value), self);
|
||||
});
|
||||
return 0;
|
||||
});
|
||||
lua_setfield(L, -2, "__call");
|
||||
|
@ -421,6 +421,9 @@ class Network::Lambda : public Node::Lambda,
|
||||
}
|
||||
void Abort() noexcept override {
|
||||
abort_ = true;
|
||||
for (auto& p : lamap_) {
|
||||
p.second->Abort();
|
||||
}
|
||||
}
|
||||
size_t GetMemoryUsage() const noexcept override {
|
||||
return 0;
|
||||
@ -438,7 +441,7 @@ class Network::Lambda : public Node::Lambda,
|
||||
std::unordered_map<ItemId, std::shared_ptr<Node::Lambda>> lamap_;
|
||||
std::unordered_map<Node::Lambda*, ItemId> idmap_;
|
||||
|
||||
std::atomic<bool> abort_ = false;
|
||||
bool abort_ = false;
|
||||
};
|
||||
void Network::DetachLambda() noexcept {
|
||||
if (lambda_ && lambda_->isRoot()) {
|
||||
|
@ -164,6 +164,8 @@ class Ref final : public nf7::File, public nf7::Node {
|
||||
class Ref::Lambda final : public Node::Lambda,
|
||||
public std::enable_shared_from_this<Ref::Lambda> {
|
||||
public:
|
||||
static constexpr size_t kMaxDepth = 1024;
|
||||
|
||||
Lambda(Ref& f, const std::shared_ptr<Node::Lambda>& parent) :
|
||||
Node::Lambda(f, parent), ref_(&f), log_(f.log_) {
|
||||
}
|
||||
@ -180,12 +182,22 @@ class Ref::Lambda final : public Node::Lambda,
|
||||
}
|
||||
if (caller == parent) {
|
||||
if (!base_) {
|
||||
if (depth() > kMaxDepth) {
|
||||
log_->Error("stack overflow");
|
||||
return;
|
||||
}
|
||||
base_ = ref_->target().CreateLambda(shared_from_this());
|
||||
}
|
||||
base_->Handle(name, v, shared_from_this());
|
||||
}
|
||||
}
|
||||
|
||||
void Abort() noexcept override {
|
||||
if (base_) {
|
||||
base_->Abort();
|
||||
}
|
||||
}
|
||||
|
||||
private:
|
||||
Ref* const ref_;
|
||||
std::shared_ptr<nf7::LoggerRef> log_;
|
||||
|
@ -121,7 +121,8 @@ class Call::Lambda final : public nf7::Sequencer::Lambda,
|
||||
Sequencer::Lambda(f, ctx), file_(f.life_) {
|
||||
}
|
||||
|
||||
void Run(const std::shared_ptr<Sequencer::Session>& ss) noexcept;
|
||||
void Run(const std::shared_ptr<Sequencer::Session>& ss) noexcept override;
|
||||
void Abort() noexcept override;
|
||||
|
||||
private:
|
||||
nf7::Life<Call>::Ref file_;
|
||||
@ -130,17 +131,14 @@ class Call::Lambda final : public nf7::Sequencer::Lambda,
|
||||
|
||||
nf7::Node* cached_node_ = nullptr;
|
||||
std::shared_ptr<Node::Lambda> la_;
|
||||
|
||||
bool abort_ = false;
|
||||
};
|
||||
class Call::SessionLambda final : public nf7::Node::Lambda {
|
||||
public:
|
||||
SessionLambda(Call& f, const std::shared_ptr<Call::Lambda>& parent) noexcept :
|
||||
nf7::Node::Lambda(f, parent) {
|
||||
}
|
||||
~SessionLambda() noexcept {
|
||||
if (ss_ && expects_.size() > 0) {
|
||||
ss_->Finish();
|
||||
}
|
||||
}
|
||||
|
||||
void Listen(Call& f, const std::shared_ptr<Sequencer::Session>& ss) noexcept {
|
||||
assert(!ss_);
|
||||
@ -167,6 +165,13 @@ class Call::SessionLambda final : public nf7::Node::Lambda {
|
||||
expects_.erase(std::string {name});
|
||||
FinishIf();
|
||||
}
|
||||
void Abort() noexcept override {
|
||||
if (ss_) {
|
||||
ss_->Finish();
|
||||
ss_ = nullptr;
|
||||
expects_.clear();
|
||||
}
|
||||
}
|
||||
|
||||
private:
|
||||
std::shared_ptr<Sequencer::Session> ss_;
|
||||
@ -186,6 +191,7 @@ std::shared_ptr<Sequencer::Lambda> Call::CreateLambda(
|
||||
}
|
||||
void Call::Lambda::Run(const std::shared_ptr<Sequencer::Session>& ss) noexcept
|
||||
try {
|
||||
if (abort_) return;
|
||||
file_.EnforceAlive();
|
||||
|
||||
auto& data = file_->data();
|
||||
@ -219,6 +225,16 @@ try {
|
||||
} catch (nf7::File::NotImplementedException&) {
|
||||
ss->Finish();
|
||||
}
|
||||
void Call::Lambda::Abort() noexcept {
|
||||
if (ssla_) {
|
||||
ssla_->Abort();
|
||||
ssla_ = nullptr;
|
||||
}
|
||||
if (la_) {
|
||||
la_->Abort();
|
||||
la_ = nullptr;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void Call::UpdateItem(Sequencer::Editor&) noexcept {
|
||||
|
Loading…
x
Reference in New Issue
Block a user