Cancellation.cpp 1.31 KB
//===--- Cancellation.cpp -----------------------------------------*-C++-*-===//
//
// 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
//
//===----------------------------------------------------------------------===//

#include "support/Cancellation.h"
#include <atomic>

namespace clang {
namespace clangd {

char CancelledError::ID = 0;

// We don't want a cancelable scope to "shadow" an enclosing one.
struct CancelState {
  std::shared_ptr<std::atomic<int>> Cancelled;
  const CancelState *Parent;
};
static Key<CancelState> StateKey;

std::pair<Context, Canceler> cancelableTask(int Reason) {
  assert(Reason != 0 && "Can't detect cancellation if Reason is zero");
  CancelState State;
  State.Cancelled = std::make_shared<std::atomic<int>>();
  State.Parent = Context::current().get(StateKey);
  return {
      Context::current().derive(StateKey, State),
      [Reason, Flag(State.Cancelled)] { *Flag = Reason; },
  };
}

int isCancelled(const Context &Ctx) {
  for (const CancelState *State = Ctx.get(StateKey); State != nullptr;
       State = State->Parent)
    if (int Reason = State->Cancelled->load())
      return Reason;
  return 0;
}

} // namespace clangd
} // namespace clang