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
|