//===----------------------- Queue.h - RPC Queue ------------------*-c++-*-===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #ifndef LLVM_UNITTESTS_EXECUTIONENGINE_ORC_QUEUECHANNEL_H #define LLVM_UNITTESTS_EXECUTIONENGINE_ORC_QUEUECHANNEL_H #include "llvm/ExecutionEngine/Orc/RawByteChannel.h" #include "llvm/Support/Error.h" #include #include namespace llvm { class QueueChannelError : public ErrorInfo { public: static char ID; }; class QueueChannelClosedError : public ErrorInfo { public: static char ID; std::error_code convertToErrorCode() const override { return inconvertibleErrorCode(); } void log(raw_ostream &OS) const override { OS << "Queue closed"; } }; class Queue : public std::queue { public: using ErrorInjector = std::function; Queue() : ReadError([]() { return Error::success(); }), WriteError([]() { return Error::success(); }) {} Queue(const Queue&) = delete; Queue& operator=(const Queue&) = delete; Queue(Queue&&) = delete; Queue& operator=(Queue&&) = delete; std::mutex &getMutex() { return M; } std::condition_variable &getCondVar() { return CV; } Error checkReadError() { return ReadError(); } Error checkWriteError() { return WriteError(); } void setReadError(ErrorInjector NewReadError) { { std::lock_guard Lock(M); ReadError = std::move(NewReadError); } CV.notify_one(); } void setWriteError(ErrorInjector NewWriteError) { std::lock_guard Lock(M); WriteError = std::move(NewWriteError); } private: std::mutex M; std::condition_variable CV; std::function ReadError, WriteError; }; class QueueChannel : public orc::rpc::RawByteChannel { public: QueueChannel(std::shared_ptr InQueue, std::shared_ptr OutQueue) : InQueue(InQueue), OutQueue(OutQueue) {} QueueChannel(const QueueChannel&) = delete; QueueChannel& operator=(const QueueChannel&) = delete; QueueChannel(QueueChannel&&) = delete; QueueChannel& operator=(QueueChannel&&) = delete; Error readBytes(char *Dst, unsigned Size) override { std::unique_lock Lock(InQueue->getMutex()); while (Size) { { Error Err = InQueue->checkReadError(); while (!Err && InQueue->empty()) { InQueue->getCondVar().wait(Lock); Err = InQueue->checkReadError(); } if (Err) return Err; } *Dst++ = InQueue->front(); --Size; ++NumRead; InQueue->pop(); } return Error::success(); } Error appendBytes(const char *Src, unsigned Size) override { std::unique_lock Lock(OutQueue->getMutex()); while (Size--) { if (Error Err = OutQueue->checkWriteError()) return Err; OutQueue->push(*Src++); ++NumWritten; } OutQueue->getCondVar().notify_one(); return Error::success(); } Error send() override { return Error::success(); } void close() { auto ChannelClosed = []() { return make_error(); }; InQueue->setReadError(ChannelClosed); InQueue->setWriteError(ChannelClosed); OutQueue->setReadError(ChannelClosed); OutQueue->setWriteError(ChannelClosed); } uint64_t NumWritten = 0; uint64_t NumRead = 0; private: std::shared_ptr InQueue; std::shared_ptr OutQueue; }; inline std::pair, std::unique_ptr> createPairedQueueChannels() { auto Q1 = std::make_shared(); auto Q2 = std::make_shared(); auto C1 = llvm::make_unique(Q1, Q2); auto C2 = llvm::make_unique(Q2, Q1); return std::make_pair(std::move(C1), std::move(C2)); } } #endif