#ifdef _WIN32
#include <windows.h>
#else
#include <dlfcn.h>
#endif
#include <cstdio>
#include <cstring>
#include <QDirIterator>
#include "qmpplugin.hpp"
#include "qmpmainwindow.hpp"
#include "qmpmidioutfluid.hpp"
#include "qmpsettingswindow.hpp"
qmpPluginAPIImpl *qmpPluginManager::pluginAPI = nullptr;
qmpMainWindow *qmpPluginManager::mainwindow = nullptr;
#ifdef _WIN32
#include <codecvt>
#include <locale>
std::string wstr2str(std::wstring s)
{
std::wstring_convert<std::codecvt_utf8<wchar_t>, wchar_t> wsc;
return wsc.to_bytes(s);
}
void qmpPluginManager::scanPlugins(const std::vector<std::string> &pp)
{
QDirIterator *dir;
std::vector<std::wstring> cpluginpaths;
std::wstring_convert<std::codecvt_utf8<wchar_t>, wchar_t> wsc;
for (auto p : pp)
cpluginpaths.push_back(wsc.from_bytes(p));
dir = new QDirIterator(QCoreApplication::applicationDirPath() + "/plugins/");
while (dir->hasNext())
{
dir->next();
if (dir->fileInfo().suffix() == "dll")
cpluginpaths.push_back(QCoreApplication::applicationDirPath().toStdWString() + std::wstring(L"/plugins/") + dir->fileName().toStdWString());
}
delete dir;
for (unsigned i = 0; i < cpluginpaths.size(); ++i)
{
HMODULE hso = LoadLibraryW(cpluginpaths[i].c_str());
if (!hso)
{
fprintf(stderr, "Error while loading library: %d\n", GetLastError());
continue;
}
FARPROC hndi = GetProcAddress(hso, "qmpPluginGetInterface");
if (!hndi)
{
fprintf(stderr, "plugin %s doesn't seem to be a qmidiplayer plugin.\n", wstr2str(cpluginpaths[i]).c_str());
continue;
}
FARPROC hndiv = GetProcAddress(hso, "qmpPluginGetAPIRev");
if (!hndiv)
{
fprintf(stderr, "plugin %s is incompatible with this version of qmidiplayer.\n", wstr2str(cpluginpaths[i]).c_str());
continue;
}
qmpPluginAPIRevEntry getv = (qmpPluginAPIRevEntry)hndiv;
if (strcmp(getv(), QMP_PLUGIN_API_REV))
{
fprintf(stderr, "plugin %s is incompatible with this version of qmidiplayer.\n", wstr2str(cpluginpaths[i]).c_str());
continue;
}
qmpPluginEntry e = (qmpPluginEntry)hndi;
qmpPluginIntf *intf = e(pluginAPI);
plugins.push_back(qmpPlugin(std::string(intf->pluginGetName()), std::string(intf->pluginGetVersion()), wstr2str(cpluginpaths[i]), intf));
}
}
#else
void qmpPluginManager::scanPlugins(const std::vector<std::string> &pp)
{
QDirIterator *dir;
std::vector<std::string> cpluginpaths(pp);
#ifdef NON_PORTABLE
QString pdir = QString(QT_STRINGIFY(INSTALL_PREFIX)) + "/lib/qmidiplayer/";
dir = new QDirIterator(pdir);
while (dir->hasNext())
{
dir->next();
if (dir->fileInfo().suffix() == "so")
cpluginpaths.push_back((pdir + dir->fileName()).toStdString());
}
delete dir;
#endif
dir = new QDirIterator(QCoreApplication::applicationDirPath() + "/plugins/");
while (dir->hasNext())
{
dir->next();
if (dir->fileInfo().suffix() == "so")
cpluginpaths.push_back(QCoreApplication::applicationDirPath().toStdString() + std::string("/plugins/") + dir->fileName().toStdString());
}
delete dir;
for (unsigned i = 0; i < cpluginpaths.size(); ++i)
{
void *hso = dlopen(cpluginpaths[i].c_str(), RTLD_LAZY);
if (!hso)
{
fprintf(stderr, "%s\n", dlerror());
continue;
}
void *hndi = dlsym(hso, "qmpPluginGetInterface");
if (!hndi)
{
fprintf(stderr, "file %s doesn't seem to be a qmidiplayer plugin.\n", cpluginpaths[i].c_str());
continue;
}
void *hndiv = dlsym(hso, "qmpPluginGetAPIRev");
if (!hndiv)
{
fprintf(stderr, "file %s is incompatible with this version of qmidiplayer.\n", cpluginpaths[i].c_str());
continue;
}
qmpPluginAPIRevEntry getv = (qmpPluginAPIRevEntry)hndiv;
if (strcmp(getv(), QMP_PLUGIN_API_REV))
{
fprintf(stderr, "file %s is incompatible with this version of qmidiplayer.\n", cpluginpaths[i].c_str());
continue;
}
qmpPluginEntry e = (qmpPluginEntry)hndi;
qmpPluginIntf *intf = e(pluginAPI);
plugins.push_back(qmpPlugin(std::string(intf->pluginGetName()), std::string(intf->pluginGetVersion()), std::string(cpluginpaths[i]), intf));
}
}
#endif
qmpPluginManager::qmpPluginManager()
{
mainwindow = qmpMainWindow::getInstance();
pluginAPI = new qmpPluginAPIImpl();
}
qmpPluginManager::~qmpPluginManager()
{
for (unsigned i = 0; i < plugins.size(); ++i)
{
if (plugins[i].initialized)
plugins[i].pinterface->deinit();
delete plugins[i].pinterface;
}
mainwindow = nullptr;
delete pluginAPI;
}
std::vector<qmpPlugin> *qmpPluginManager::getPlugins()
{
return &plugins;
}
void qmpPluginManager::initPlugins()
{
for (unsigned i = 0; i < plugins.size(); ++i)
{
if (!plugins[i].enabled)
continue;
fprintf(stderr, "Loaded plugin: %s\n", plugins[i].path.c_str());
plugins[i].pinterface->init();
plugins[i].initialized = true;
}
}
void qmpPluginManager::deinitPlugins()
{
for (unsigned i = 0; i < plugins.size(); ++i)
{
if (plugins[i].initialized)
plugins[i].pinterface->deinit();
plugins[i].enabled = plugins[i].initialized = false;
}
}
qmpPluginAPIImpl::qmpPluginAPIImpl() {}
qmpPluginAPIImpl::~qmpPluginAPIImpl() {}
#define qmw qmpPluginManager::mainwindow
uint32_t qmpPluginAPIImpl::getDivision()
{
return qmw && qmw->getPlayer() ? qmw->getPlayer()->getDivision() : 0;
}
uint32_t qmpPluginAPIImpl::getRawTempo()
{
return qmw && qmw->getPlayer() ? qmw->getPlayer()->getRawTempo() : 0;
}
double qmpPluginAPIImpl::getRealTempo()
{
return qmw && qmw->getPlayer() ? qmw->getPlayer()->getTempo() : 0;
}
uint32_t qmpPluginAPIImpl::getTimeSig()
{
int n, d = 0, t = 1;
qmw &&qmw->getPlayer() ? qmw->getPlayer()->getCurrentTimeSignature(&n, &t) : void(0);
for (; t >>= 1; ++d);
return n << 8 | d;
}
int qmpPluginAPIImpl::getKeySig()
{
return qmw && qmw->getPlayer() ? qmw->getPlayer()->getCurrentKeySignature() : 0;
}
uint32_t qmpPluginAPIImpl::getNoteCount()
{
return qmw && qmw->getPlayer() ? qmw->getPlayer()->getFileNoteCount() : 0;
}
uint32_t qmpPluginAPIImpl::getMaxTick()
{
return qmw && qmw->getPlayer() ? qmw->getPlayer()->getMaxTick() : 0;
}
uint32_t qmpPluginAPIImpl::getCurrentPolyphone()
{
return qmw && qmw->getPlayer() ? qmw->getFluid()->getPolyphone() : 0;
}
uint32_t qmpPluginAPIImpl::getMaxPolyphone()
{
return qmw && qmw->getPlayer() ? qmw->getFluid()->getMaxPolyphone() : 0;
}
uint32_t qmpPluginAPIImpl::getCurrentTimeStamp()
{
return qmw && qmw->getPlayer() ? qmw->getPlayer()->getTick() : 0;
}
uint32_t qmpPluginAPIImpl::getCurrentPlaybackPercentage()
{
return qmw ? qmw->getPlaybackPercentage() : 0;
}
PlaybackStatus qmpPluginAPIImpl::getPlaybackStatus()
{
return qmw ? qmw->getPlaybackStatus() : PlaybackStatus{false, 0, 0, 0, 0};
}
int qmpPluginAPIImpl::getChannelCC(int ch, int cc)
{
return qmw && qmw->getPlayer() ? qmw->getPlayer()->getCC(ch, cc) : 0;
}
int qmpPluginAPIImpl::getChannelPreset(int ch)
{
uint16_t b;
uint8_t p;
std::string nm;
if (qmw && qmw->getPlayer())
{
if (qmw->getPlayer()->getChannelOutputDevice(ch)->getChannelPreset(ch, &b, &p, nm))
return p;
return qmw->getPlayer()->getCC(ch, 128);
}
return 0;
}
void qmpPluginAPIImpl::playerSeek(uint32_t percentage)
{
if (qmw)
qmw->playerSeek(percentage);
}
double qmpPluginAPIImpl::getPitchBend(int ch)
{
return qmw && qmw->getPlayer() ? qmw->getPlayer()->getPitchBend(ch) : 0;
}
void qmpPluginAPIImpl::getPitchBendRaw(int ch, uint32_t *pb, uint32_t *pbr)
{
if (qmw && qmw->getPlayer())
qmw->getPlayer()->getPitchBendRaw(ch, pb, pbr);
}
bool qmpPluginAPIImpl::getChannelMask(int ch)
{
return qmw && qmw->getPlayer() ? qmw->getPlayer()->getChannelMask(ch) : false;
}
std::string qmpPluginAPIImpl::getTitle()
{
return qmw ? qmw->getTitle() : "";
}
std::wstring qmpPluginAPIImpl::getWTitle()
{
return qmw ? qmw->getWTitle() : L"";
}
std::string qmpPluginAPIImpl::getFilePath()
{
return qmw ? qmw->getFilePath().toString().toStdString() : "";
}
std::wstring qmpPluginAPIImpl::getWFilePath()
{
return qmw ? qmw->getFilePath().toString().toStdWString() : L"";
}
std::string qmpPluginAPIImpl::getChannelPresetString(int ch)
{
uint16_t b;
uint8_t p;
char ret[320];
ret[0] = 0;
std::string nm;
if (qmw && qmw->getPlayer())
{
int r = qmw->getPlayer()->getChannelOutputDevice(ch)->getChannelPreset(ch, &b, &p, nm);
if (!r)
{
b = qmw->getPlayer()->getCC(ch, 0) << 7 | qmw->getPlayer()->getCC(ch, 32);
p = qmw->getPlayer()->getCC(ch, 128);
nm = qmw->getPlayer()->getChannelOutputDevice(ch)->getPresetName(b, p);
}
snprintf(ret, 320, "%03d:%03d %s", b, p, nm.c_str());
}
return std::string(ret);
}
bool qmpPluginAPIImpl::isDarkTheme()
{
return qmw ? qmw->isDarkTheme() : false;
}
void *qmpPluginAPIImpl::getMainWindow()
{
return (void *)qmw;
}
void qmpPluginAPIImpl::playbackControl(PlaybackControlCommand cmd, void *data)
{
if (!qmw) return;
switch (cmd)
{
case PlaybackControlCommand::Pause:
qmw->setPaused(true);
break;
case PlaybackControlCommand::Play:
qmw->setPaused(false);
break;
case PlaybackControlCommand::TogglePause:
qmw->setPaused(!qmw->getPlaybackStatus().paused);
break;
case PlaybackControlCommand::Stop:
qmw->stop();
break;
case PlaybackControlCommand::Seek:
qmw->playerSeek(*static_cast<uint32_t*>(data));
break;
case PlaybackControlCommand::SeekAbs:
{
double t = *static_cast<double*>(data);
uint32_t p = 100. * t / (qmw->getPlaybackStatus().maxtime_ms / 1000.);
qmw->playerSeek(p);
}
break;
case PlaybackControlCommand::NextTrack:
qmw->nextTrack();
break;
case PlaybackControlCommand::PrevTrack:
qmw->prevTrack();
break;
}
}
void qmpPluginAPIImpl::discardCurrentEvent()
{
if (qmw && qmw->getPlayer())
qmw->getPlayer()->discardCurrentEvent();
}
void qmpPluginAPIImpl::commitEventChange(SEvent d)
{
if (qmw && qmw->getPlayer())
qmw->getPlayer()->commitEventChange(d);
}
void qmpPluginAPIImpl::callEventReaderCB(SEvent d)
{
if (qmw && qmw->getPlayer())
qmw->getPlayer()->callEventReaderCB(d);
}
void qmpPluginAPIImpl::setFuncState(std::string name, bool state)
{
if (qmw)
qmw->setFuncState(name, state);
}
void qmpPluginAPIImpl::setFuncEnabled(std::string name, bool enable)
{
if (qmw)
qmw->setFuncEnabled(name, enable);
}
void qmpPluginAPIImpl::registerMidiOutDevice(qmpMidiOutDevice *dev, std::string name)
{
qmw->getPlayer()->registerMidiOutDevice(dev, name);
}
void qmpPluginAPIImpl::unregisterMidiOutDevice(std::string name)
{
qmw->getPlayer()->unregisterMidiOutDevice(name);
}
int qmpPluginAPIImpl::registerEventHandlerIntf(ICallBack *cb, void *userdata)
{
return qmw->getPlayer()->setEventHandlerCB(cb, userdata);
}
void qmpPluginAPIImpl::unregisterEventHandlerIntf(int intfhandle)
{
qmw->getPlayer()->unsetEventHandlerCB(intfhandle);
}
int qmpPluginAPIImpl::registerEventReaderIntf(ICallBack *cb, void *userdata)
{
return qmw->getPlayer()->setEventReaderCB(cb, userdata);
}
void qmpPluginAPIImpl::unregisterEventReaderIntf(int intfhandle)
{
qmw->getPlayer()->unsetEventReaderCB(intfhandle);
}
int qmpPluginAPIImpl::registerUIHook(std::string e, ICallBack *cb, void *userdat)
{
return qmw->registerUIHook(e, cb, userdat);
}
int qmpPluginAPIImpl::registerUIHook(std::string e, callback_t cb, void *userdat)
{
return qmw->registerUIHook(e, cb, userdat);
}
void qmpPluginAPIImpl::unregisterUIHook(std::string e, int hook)
{
qmw->unregisterUIHook(e, hook);
}
void qmpPluginAPIImpl::registerFunctionality(qmpFuncBaseIntf *i, std::string name, std::string desc, const char *icon, int iconlen, bool checkable)
{
qmw->registerFunctionality(i, name, desc, icon, iconlen, checkable);
}
void qmpPluginAPIImpl::unregisterFunctionality(std::string name)
{
qmw->unregisterFunctionality(name);
}
int qmpPluginAPIImpl::registerFileReadFinishedHandlerIntf(ICallBack *cb, void *userdata)
{
return qmw->getPlayer()->setFileReadFinishedCB(cb, userdata);
}
void qmpPluginAPIImpl::unregisterFileReadFinishedHandlerIntf(int intfhandle)
{
qmw->getPlayer()->unsetFileReadFinishedCB(intfhandle);
}
void qmpPluginAPIImpl::registerFileReader(qmpFileReader *reader, std::string name)
{
qmw->getPlayer()->registerReader(reader, name);
}
void qmpPluginAPIImpl::unregisterFileReader(std::string name)
{
qmw->getPlayer()->unregisterReader(name);
}
int qmpPluginAPIImpl::registerEventHandler(callback_t cb, void *userdata, bool post)
{
return qmw->getPlayer()->registerEventHandler(cb, userdata, post);
}
void qmpPluginAPIImpl::unregisterEventHandler(int id)
{
qmw->getPlayer()->unregisterEventHandler(id);
}
int qmpPluginAPIImpl::registerEventReadHandler(callback_t cb, void *userdata)
{
return qmw->getPlayer()->registerEventReadHandler(cb, userdata);
}
void qmpPluginAPIImpl::unregisterEventReadHandler(int id)
{
qmw->getPlayer()->unregisterEventReadHandler(id);
}
int qmpPluginAPIImpl::registerFileReadFinishHook(callback_t cb, void *userdata)
{
return qmw->getPlayer()->registerFileReadFinishHook(cb, userdata);
}
void qmpPluginAPIImpl::unregisterFileReadFinishHook(int id)
{
qmw->getPlayer()->unregisterFileReadFinishHook(id);
}
void qmpPluginAPIImpl::registerOptionInt(std::string tab, std::string desc, std::string key, int min, int max, int defaultval)
{
qmw->getSettings()->registerOptionInt(tab, desc, key, min, max, defaultval);
}
int qmpPluginAPIImpl::getOptionInt(std::string key)
{
return qmw->getSettings()->getOptionInt(key);
}
void qmpPluginAPIImpl::setOptionInt(std::string key, int val)
{
qmw->getSettings()->setOptionInt(key, val);
}
void qmpPluginAPIImpl::registerOptionUint(std::string tab, std::string desc, std::string key, unsigned min, unsigned max, unsigned defaultval)
{
qmw->getSettings()->registerOptionUint(tab, desc, key, min, max, defaultval);
}
unsigned qmpPluginAPIImpl::getOptionUint(std::string key)
{
return qmw->getSettings()->getOptionUint(key);
}
void qmpPluginAPIImpl::setOptionUint(std::string key, unsigned val)
{
qmw->getSettings()->setOptionUint(key, val);
}
void qmpPluginAPIImpl::registerOptionBool(std::string tab, std::string desc, std::string key, bool defaultval)
{
qmw->getSettings()->registerOptionBool(tab, desc, key, defaultval);
}
bool qmpPluginAPIImpl::getOptionBool(std::string key)
{
return qmw->getSettings()->getOptionBool(key);
}
void qmpPluginAPIImpl::setOptionBool(std::string key, bool val)
{
qmw->getSettings()->setOptionBool(key, val);
}
void qmpPluginAPIImpl::registerOptionDouble(std::string tab, std::string desc, std::string key, double min, double max, double defaultval)
{
qmw->getSettings()->registerOptionDouble(tab, desc, key, min, max, defaultval);
}
double qmpPluginAPIImpl::getOptionDouble(std::string key)
{
return qmw->getSettings()->getOptionDouble(key);
}
void qmpPluginAPIImpl::setOptionDouble(std::string key, double val)
{
qmw->getSettings()->setOptionDouble(key, val);
}
void qmpPluginAPIImpl::registerOptionString(std::string tab, std::string desc, std::string key, std::string defaultval, bool ispath)
{
qmw->getSettings()->registerOptionString(tab, desc, key, defaultval, ispath);
}
std::string qmpPluginAPIImpl::getOptionString(std::string key)
{
return qmw->getSettings()->getOptionString(key);
}
void qmpPluginAPIImpl::setOptionString(std::string key, std::string val)
{
return qmw->getSettings()->setOptionString(key, val);
}
void qmpPluginAPIImpl::registerOptionEnumInt(std::string tab, std::string desc, std::string key, std::vector<std::string> options, int defaultval)
{
qmw->getSettings()->registerOptionEnumInt(tab, desc, key, options, defaultval);
}
int qmpPluginAPIImpl::getOptionEnumInt(std::string key)
{
return qmw->getSettings()->getOptionEnumInt(key);
}
void qmpPluginAPIImpl::setOptionEnumInt(std::string key, int val)
{
return qmw->getSettings()->setOptionEnumInt(key, val);
}