/** * Copyright - See the COPYRIGHT that is included with this distribution. * pvxs is distributed subject to a Software License Agreement found * in file LICENSE that is included with this distribution. */ #include #include #include "clientimpl.h" namespace pvxs { namespace client { DEFINE_LOGGER(setup, "pvxs.client.setup"); DEFINE_LOGGER(io, "pvxs.client.io"); namespace { struct InfoOp : public OperationBase { std::function done; Value result; enum state_t { Connecting, // waiting for an active Channel Waiting, // waiting for reply to GET_INFO Done, } state = Connecting; INST_COUNTER(InfoOp); explicit InfoOp(const std::shared_ptr& chan) :OperationBase(Info, chan) {} virtual ~InfoOp() { chan->context->tcp_loop.assertInLoop(); _cancel(true); } virtual bool cancel() override final { auto context = chan->context; decltype (done) junk; bool ret = false; context->tcp_loop.call([this, &junk, &ret](){ ret = _cancel(false); junk = std::move(done); // leave opByIOID for GC }); return ret; } bool _cancel(bool implicit) { if(implicit && state!=Done) { log_info_printf(setup, "implied cancel of INFO on channel '%s'\n", chan ? chan->name.c_str() : ""); } if(state==Waiting) { chan->conn->sendDestroyRequest(chan->sid, ioid); // This opens up a race with an in-flight reply. chan->conn->opByIOID.erase(ioid); chan->opByIOID.erase(ioid); } bool ret = state!=Done; state = Done; return ret; } void reExec(const Value& arg, std::function&& resultcb) override final {} virtual void createOp() override final { if(state!=Connecting) { return; } auto& conn = chan->conn; { (void)evbuffer_drain(conn->txBody.get(), evbuffer_get_length(conn->txBody.get())); EvOutBuf R(hostBE, conn->txBody.get()); to_wire(R, chan->sid); to_wire(R, ioid); // sub-field, which no one knows how to use... to_wire(R, ""); } conn->enqueueTxBody(CMD_GET_FIELD); log_debug_printf(io, "Server %s channel '%s' GET_INFO\n", conn->peerName.c_str(), chan->name.c_str()); state = Waiting; } virtual void disconnected(const std::shared_ptr& self) override final { // Do nothing when Connecting or Done if(state==Waiting) { // return to pending chan->pending.push_back(self); state = Connecting; } } }; } // namespace void Connection::handle_GET_FIELD() { EvInBuf M(peerBE, segBuf.get(), 16); uint32_t ioid=0u; Status sts{Status::Fatal}; Value prototype; from_wire(M, ioid); from_wire(M, sts); if(sts.isSuccess()) from_wire_type(M, rxRegistry, prototype); if(!M.good()) { log_crit_printf(io, "%s:%d Server %s sends invalid GET_FIELD. Disconnecting...\n", M.file(), M.line(), peerName.c_str()); bev.reset(); return; } std::shared_ptr op; InfoOp* info; { auto it = opByIOID.find(ioid); if(it==opByIOID.end() || !(op = it->second.handle.lock()) || op->op!=Operation::Info) { log_warn_printf(io, "Server %s sends stale GET_FIELD\n", peerName.c_str()); return; } info = static_cast(op.get()); opByIOID.erase(it); info->chan->opByIOID.erase(ioid); } if(info->state!=InfoOp::Waiting) { log_warn_printf(io, "Server %s ignore second reply to GET_FIELD\n", peerName.c_str()); return; } log_debug_printf(io, "Server %s completes GET_FIELD.\n", peerName.c_str()); info->state = InfoOp::Done; if(info->done) { auto done = std::move(info->done); Result res; if(sts.isSuccess()) { res = Result(std::move(prototype), peerName); } else { res = Result(std::make_exception_ptr(RemoteError(sts.msg))); } try { done(std::move(res)); }catch(std::exception& e){ log_exc_printf(setup, "Unhandled exception %s in Info result() callback: %s\n", typeid (e).name(), e.what()); } } else { info->result = prototype; } } std::shared_ptr GetBuilder::_exec_info() { if(!ctx) throw std::logic_error("NULL Builder"); std::shared_ptr ret; assert(!_get); ctx->tcp_loop.call([&ret, this]() { auto chan = Channel::build(ctx->shared_from_this(), _name); auto op = std::make_shared(chan); if(_result) { op->done = std::move(_result); } else { auto waiter = op->waiter = std::make_shared(); op->done = [waiter](Result&& result) { waiter->complete(std::move(result), false); }; } chan->pending.push_back(op); chan->createOperations(); auto loop(op->chan->context->tcp_loop); ret.reset(op.get(), [op, loop](Operation*) mutable { // on user thread auto temp(std::move(op)); auto L(std::move(loop)); L.call([&temp]() { // on worker try { temp->_cancel(true); }catch(std::exception& e){ log_exc_printf(setup, "Channel %s error in info cancel(): %s", temp->chan->name.c_str(), e.what()); } // ensure dtor on worker temp.reset(); }); }); }); return ret; } } // namespace client } // namespace pvxs