4#if (!(defined(__APPLE__) || defined(_WIN32))) && __has_include(<xcb/xcb.h>)
5#include <Vst3/UI/WindowContainer.hpp>
7#include <ossia/detail/algorithms.hpp>
8#include <ossia/detail/hash_map.hpp>
11#include <QSocketNotifier>
15#include <pluginterfaces/gui/iplugview.h>
26namespace Linux = Steinberg::Linux;
31 : read{fd, QSocketNotifier::Read}
32 , write{fd, QSocketNotifier::Write}
33 , error{fd, QSocketNotifier::Exception}
37 void connect(QObject* sink, F f)
39 QObject::connect(&read, &QSocketNotifier::activated, sink, f);
40 QObject::connect(&write, &QSocketNotifier::activated, sink, f);
41 QObject::connect(&error, &QSocketNotifier::activated, sink, f);
43 read.setEnabled(
true);
44 write.setEnabled(
true);
45 error.setEnabled(
true);
48 QSocketNotifier write;
49 QSocketNotifier error;
55 xcb_connection_t* connection{};
58 std::unique_ptr<SocketPair> notifiers;
60 : xcb{dlopen(
"libxcb.so.1", RTLD_LOCAL)}
66 =
reinterpret_cast<decltype(&::xcb_connect)
>(dlsym(xcb,
"xcb_connect"));
69 auto xcb_get_file_descriptor
70 =
reinterpret_cast<decltype(&::xcb_get_file_descriptor)
>(
71 dlsym(xcb,
"xcb_get_file_descriptor"));
72 if(!xcb_get_file_descriptor)
75 connection = xcb_connect(
nullptr,
nullptr);
79 fd = xcb_get_file_descriptor(connection);
83 notifiers = std::make_unique<SocketPair>(fd);
94 =
reinterpret_cast<decltype(&::xcb_disconnect)
>(dlsym(xcb,
"xcb_disconnect"));
97 xcb_disconnect(connection);
103 Linux::IEventHandler* handler{};
104 QSocketNotifier* r_notifier{};
105 QSocketNotifier* w_notifier{};
106 QSocketNotifier* e_notifier{};
110class GlobalSocketHandlers :
public QObject
115 ossia::hash_map<int, std::vector<Linux::IEventHandler*>> handlers;
116 GlobalSocketHandlers()
119 xcb.notifiers->connect(
this, [
this] { on_xcb(); });
124 if(
auto it = handlers.find(xcb.fd); it != handlers.end())
126 for(
auto& hdl : it->second)
127 hdl->onFDIsSet(xcb.fd);
131 void registerHandler(Linux::IEventHandler* handler,
int fd)
133 handlers[fd].push_back(handler);
135 void unregisterHandler(Linux::IEventHandler* handler,
int fd)
137 if(
auto it = handlers.find(fd); it != handlers.end())
139 ossia::remove_erase(it->second, handler);
140 if(it->second.empty())
144 static GlobalSocketHandlers& instance()
146 static GlobalSocketHandlers handlers;
151 :
virtual public Steinberg::IPlugFrame
152 ,
virtual public Steinberg::Linux::IRunLoop
155 using TUID = Steinberg::TUID;
156 using FUID = Steinberg::FUID;
157 using uint32 = Steinberg::uint32;
158 using tresult = Steinberg::tresult;
160 QObject internalContextObject;
162 std::vector<std::pair<Linux::ITimerHandler*, QTimer*>> timers;
164 std::vector<SocketHandler> handlers;
166 tresult queryInterface(
const TUID _iid,
void** obj)
override
168 using namespace Steinberg;
169 if(FUnknownPrivate::iidEqual(_iid, FUnknown::iid))
174 if(FUnknownPrivate::iidEqual(_iid, IPlugFrame::iid))
179 if(FUnknownPrivate::iidEqual(_iid, Linux::IRunLoop::iid))
181 *obj =
static_cast<Linux::IRunLoop*
>(
this);
188 uint32 addRef()
override {
return 1; }
189 uint32 release()
override {
return 1; }
191 tresult registerEventHandler(Linux::IEventHandler* handler, Linux::FileDescriptor fd)
194 using namespace Steinberg;
196 return kInvalidArgument;
197 auto existing = ossia::find_if(handlers, [=](
auto p) {
return p.fd == fd; });
199 if(existing != handlers.end())
200 return kInvalidArgument;
202 auto readnotifier =
new QSocketNotifier{fd, QSocketNotifier::Read};
203 readnotifier->setEnabled(
true);
204 readnotifier->setParent(&internalContextObject);
205 auto on_fd = [=] { handler->onFDIsSet(fd); };
207 readnotifier, &QSocketNotifier::activated, &internalContextObject, on_fd);
209 auto writenotifier =
new QSocketNotifier{fd, QSocketNotifier::Write};
210 writenotifier->setEnabled(
true);
211 writenotifier->setParent(&internalContextObject);
213 writenotifier, &QSocketNotifier::activated, &internalContextObject, on_fd);
215 auto errnotifier =
new QSocketNotifier{fd, QSocketNotifier::Exception};
216 errnotifier->setEnabled(
true);
217 errnotifier->setParent(&internalContextObject);
219 errnotifier, &QSocketNotifier::activated, &internalContextObject, on_fd);
222 SocketHandler{handler, readnotifier, writenotifier, errnotifier, fd});
223 GlobalSocketHandlers::instance().registerHandler(handler, fd);
228 tresult unregisterEventHandler(Linux::IEventHandler* handler) SMTG_OVERRIDE
230 using namespace Steinberg;
233 return kInvalidArgument;
236 tresult res{kResultFalse};
237 for(
auto it = handlers.begin(); it != handlers.end();)
239 if(it->handler == handler)
241 GlobalSocketHandlers::instance().unregisterHandler(handler, it->fd);
242 delete it->r_notifier;
243 delete it->w_notifier;
244 delete it->e_notifier;
245 it = handlers.erase(it);
257 tresult registerTimer(
258 Linux::ITimerHandler* handler, Linux::TimerInterval milliseconds)
override
261 t->setParent(&internalContextObject);
262 QObject::connect(t, &QTimer::timeout, [=] { handler->onTimer(); });
263 t->start(milliseconds);
264 timers.push_back({handler, t});
265 return Steinberg::kResultOk;
268 tresult unregisterTimer(Linux::ITimerHandler* handler)
override
270 auto t = ossia::find_if(timers, [=](
auto& p1) {
return p1.first == handler; });
271 if(t != timers.end())
276 return Steinberg::kResultOk;
281 explicit PlugFrame(QDialog& w, WindowContainer wc)
289 for(
auto timer : timers)
292 for(
auto handler : handlers)
294 GlobalSocketHandlers::instance().unregisterHandler(handler.handler, handler.fd);
295 delete handler.r_notifier;
296 delete handler.w_notifier;
297 delete handler.e_notifier;
302 ~PlugFrame() { cleanup(); }
304 tresult resizeView(Steinberg::IPlugView* view, Steinberg::ViewRect* newSize)
override
306 wc.setSizeFromVst(*view, *newSize, *w);
307 return Steinberg::kResultOk;