annotate vamp-client/client.cpp @ 83:154e94ea84d4

Toward QProcess pipe comms take on it
author Chris Cannam <c.cannam@qmul.ac.uk>
date Tue, 11 Oct 2016 17:08:31 +0100
parents fa2c0358c2b6
children db9a6ab618bc
rev   line source
c@78 1
c@78 2 #include "stub.h"
c@78 3
c@80 4 #include "vamp-capnp/VampnProto.h"
c@80 5
c@80 6 #include "vamp-support/AssignedPluginHandleMapper.h"
c@80 7
c@83 8 #include <QProcess>
c@83 9
c@83 10 #include <stdexcept>
c@83 11
c@83 12 using std::cerr;
c@83 13 using std::endl;
c@83 14
c@82 15 // First cut plan: this is to be client-qt.cpp, using a QProcess, so
c@82 16 // we're using pipes and the server is completely synchronous,
c@82 17 // handling only one call at once. Our PiperClient will fire off a
c@82 18 // QProcess and refer to its io device. Each request message is
c@82 19 // serialised into memory using capnp::MallocMessageBuilder and
c@82 20 // shunted into the process pipe; we then wait for some bytes to come
c@82 21 // back and use capnp::expectedSizeInWordsFromPrefix to work out when
c@82 22 // a whole message is available, reading only that amount from the
c@82 23 // device and using FlatArrayMessageReader to convert to a response
c@82 24 // message. If the response message's id does not match the request
c@82 25 // message's, then the server has gone wrong (it should never be
c@82 26 // servicing more than one request at a time).
c@82 27
c@82 28 // Next level: Capnp RPC, but I want to get the first level to work
c@83 29 // first, not least because the server already exists.
c@82 30
c@81 31 namespace piper { //!!! probably something different
c@80 32
c@80 33 class PiperClient : public PiperClientBase
c@80 34 {
c@81 35 // unsigned to avoid undefined behaviour on possible wrap
c@81 36 typedef uint32_t ReqId;
c@81 37
c@80 38 public:
c@83 39 PiperClient() {
c@83 40 m_process = new QProcess();
c@83 41 m_process->setReadChannel(QProcess::StandardOutput);
c@83 42 m_process->setProcessChannelMode(QProcess::ForwardedErrorChannel);
c@83 43 m_process->start("../bin/piper-vamp-server"); //!!!
c@83 44 if (!m_process->waitForStarted()) {
c@83 45 cerr << "server failed to start" << endl;
c@83 46 delete m_process;
c@83 47 m_process = 0;
c@83 48 }
c@83 49 }
c@81 50
c@83 51 ~PiperClient() {
c@83 52 if (m_process) {
c@83 53 if (m_process->state() != QProcess::NotRunning) {
c@83 54 m_process->close();
c@83 55 m_process->waitForFinished();
c@83 56 }
c@83 57 delete m_process;
c@83 58 }
c@83 59 }
c@81 60
c@83 61 //!!! obviously, factor out all repetitive guff
c@83 62
c@81 63 Vamp::Plugin *
c@81 64 load(std::string key, float inputSampleRate, int adapterFlags) {
c@81 65
c@83 66 if (!m_process) {
c@83 67 throw std::runtime_error("Piper server failed to start");
c@83 68 }
c@83 69
c@81 70 Vamp::HostExt::LoadRequest request;
c@81 71 request.pluginKey = key;
c@81 72 request.inputSampleRate = inputSampleRate;
c@81 73 request.adapterFlags = adapterFlags;
c@81 74
c@81 75 ::capnp::MallocMessageBuilder message;
c@81 76 RpcRequest::Builder builder = message.initRoot<RpcRequest>();
c@81 77
c@81 78 VampnProto::buildRpcRequest_Load(builder, request);
c@81 79 ReqId id = getId();
c@81 80 builder.getId().setNumber(id);
c@83 81
c@83 82 auto arr = messageToFlatArray(message);
c@83 83 m_process->write(arr.asChars().begin(), arr.asChars().size());
c@83 84
c@83 85 ///.... read...
c@81 86 };
c@80 87
c@80 88 virtual
c@80 89 Vamp::Plugin::OutputList
c@80 90 configure(PiperStubPlugin *plugin,
c@80 91 Vamp::HostExt::PluginConfiguration config) {
c@80 92
c@83 93 if (!m_process) {
c@83 94 throw std::runtime_error("Piper server failed to start");
c@83 95 }
c@83 96
c@80 97 Vamp::HostExt::ConfigurationRequest request;
c@80 98 request.plugin = plugin;
c@80 99 request.configuration = config;
c@80 100
c@80 101 ::capnp::MallocMessageBuilder message;
c@80 102 RpcRequest::Builder builder = message.initRoot<RpcRequest>();
c@80 103
c@80 104 VampnProto::buildRpcRequest_Configure(builder, request, m_mapper);
c@81 105 ReqId id = getId();
c@81 106 builder.getId().setNumber(id);
c@80 107
c@80 108 //!!! now what?
c@81 109 };
c@80 110
c@80 111
c@80 112 virtual
c@80 113 Vamp::Plugin::FeatureSet
c@80 114 process(PiperStubPlugin *plugin,
c@80 115 const float *const *inputBuffers,
c@80 116 Vamp::RealTime timestamp) = 0;
c@80 117
c@80 118 virtual Vamp::Plugin::FeatureSet
c@80 119 finish(PiperStubPlugin *plugin) = 0;
c@80 120
c@80 121 private:
c@83 122 QProcess *m_process;
c@80 123 AssignedPluginHandleMapper m_mapper;
c@81 124 int getId() {
c@81 125 //!!! todo: mutex
c@81 126 static ReqId m_nextId = 0;
c@81 127 return m_nextId++;
c@81 128 }
c@80 129 };
c@80 130
c@80 131 }
c@80 132