381 lines
10 KiB
C++
381 lines
10 KiB
C++
#include "stdafx.h"
|
|
#include <Common/path.h>
|
|
#include <Project64-core/Plugins/PluginBase.h>
|
|
|
|
CPlugin::CPlugin() :
|
|
DllAbout(nullptr),
|
|
DllConfig(nullptr),
|
|
CloseDLL(nullptr),
|
|
RomOpen(nullptr),
|
|
RomClosed(nullptr),
|
|
PluginOpened(nullptr),
|
|
SetSettingInfo(nullptr),
|
|
SetSettingInfo2(nullptr),
|
|
SetSettingInfo3(nullptr),
|
|
m_LibHandle(nullptr),
|
|
m_Initialized(false),
|
|
m_RomOpen(false)
|
|
{
|
|
memset(&m_PluginInfo, 0, sizeof(m_PluginInfo));
|
|
}
|
|
|
|
CPlugin::~CPlugin()
|
|
{
|
|
WriteTrace(PluginTraceType(), TraceDebug, "Start");
|
|
UnloadPlugin();
|
|
WriteTrace(PluginTraceType(), TraceDebug, "Done");
|
|
}
|
|
|
|
bool CPlugin::Load(const char * FileName)
|
|
{
|
|
WriteTrace(PluginTraceType(), TraceDebug, "Loading: %s", FileName);
|
|
|
|
// Already loaded, so unload first
|
|
if (m_LibHandle != nullptr)
|
|
{
|
|
UnloadPlugin();
|
|
}
|
|
|
|
// Try to load the DLL library
|
|
m_LibHandle = DynamicLibraryOpen(FileName, HaveDebugger());
|
|
WriteTrace(PluginTraceType(), TraceDebug, "Loaded: %s LibHandle: %X", FileName, m_LibHandle);
|
|
|
|
if (m_LibHandle == nullptr)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
// Get DLL information
|
|
void(CALL * GetDllInfo)(PLUGIN_INFO * PluginInfo);
|
|
LoadFunction(GetDllInfo);
|
|
if (GetDllInfo == nullptr)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
GetDllInfo(&m_PluginInfo);
|
|
if (!ValidPluginVersion(m_PluginInfo))
|
|
{
|
|
return false;
|
|
}
|
|
if (m_PluginInfo.Type != type())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
LoadFunction(CloseDLL);
|
|
LoadFunction(RomOpen);
|
|
LoadFunction(RomClosed);
|
|
_LoadFunction("PluginLoaded", PluginOpened);
|
|
LoadFunction(DllConfig);
|
|
LoadFunction(DllAbout);
|
|
|
|
LoadFunction(SetPluginNotification);
|
|
if (SetPluginNotification)
|
|
{
|
|
WriteTrace(PluginTraceType(), TraceDebug, "Found SetPluginNotification");
|
|
PLUGIN_NOTIFICATION info;
|
|
info.DisplayError = DisplayError;
|
|
info.FatalError = FatalError;
|
|
info.DisplayMessage = DisplayMessage;
|
|
info.DisplayMessage2 = DisplayMessage2;
|
|
info.BreakPoint = BreakPoint;
|
|
SetPluginNotification(&info);
|
|
}
|
|
|
|
LoadFunction(SetSettingNotificationInfo);
|
|
if (SetSettingNotificationInfo)
|
|
{
|
|
WriteTrace(PluginTraceType(), TraceDebug, "Found SetSettingNotificationInfo");
|
|
PLUGIN_SETTINGS_NOTIFICATION info;
|
|
info.RegisterChangeCB = (void (*)(void *, int ID, void * Data, PLUGIN_SETTINGS_NOTIFICATION::SettingChangedFunc Func))CSettings::sRegisterChangeCB;
|
|
info.UnregisterChangeCB = (void (*)(void *, int ID, void * Data, PLUGIN_SETTINGS_NOTIFICATION::SettingChangedFunc Func))CSettings::sUnregisterChangeCB;
|
|
SetSettingNotificationInfo(&info);
|
|
}
|
|
|
|
LoadFunction(SetSettingInfo3);
|
|
if (SetSettingInfo3)
|
|
{
|
|
WriteTrace(PluginTraceType(), TraceDebug, "Found SetSettingInfo3");
|
|
PLUGIN_SETTINGS3 info;
|
|
info.FlushSettings = (void (*)(void * handle))CSettings::FlushSettings;
|
|
SetSettingInfo3(&info);
|
|
}
|
|
|
|
LoadFunction(SetSettingInfo2);
|
|
if (SetSettingInfo2)
|
|
{
|
|
WriteTrace(PluginTraceType(), TraceDebug, "Found SetSettingInfo2");
|
|
PLUGIN_SETTINGS2 info;
|
|
info.FindSystemSettingId = (uint32_t(*)(void * handle, const char *))CSettings::FindSetting;
|
|
SetSettingInfo2(&info);
|
|
}
|
|
|
|
LoadFunction(SetSettingInfo);
|
|
if (SetSettingInfo)
|
|
{
|
|
WriteTrace(PluginTraceType(), TraceDebug, "Found SetSettingInfo");
|
|
PLUGIN_SETTINGS info;
|
|
info.dwSize = sizeof(PLUGIN_SETTINGS);
|
|
info.DefaultStartRange = GetDefaultSettingStartRange();
|
|
info.SettingStartRange = GetSettingStartRange();
|
|
info.MaximumSettings = MaxPluginSetting;
|
|
info.NoDefault = Default_None;
|
|
info.DefaultLocation = g_Settings->LoadDword(Setting_UseFromRegistry) ? SettingType_Registry : SettingType_CfgFile;
|
|
info.handle = g_Settings;
|
|
info.RegisterSetting = (void (*)(void *, int, int, SettingDataType, SettingType, const char *, const char *, uint32_t)) & CSettings::RegisterSetting;
|
|
info.GetSetting = (uint32_t(*)(void *, int)) & CSettings::GetSetting;
|
|
info.GetSettingSz = (const char * (*)(void *, int, char *, int)) & CSettings::GetSettingSz;
|
|
info.SetSetting = (void (*)(void *, int, uint32_t)) & CSettings::SetSetting;
|
|
info.SetSettingSz = (void (*)(void *, int, const char *)) & CSettings::SetSettingSz;
|
|
info.UseUnregisteredSetting = nullptr;
|
|
|
|
SetSettingInfo(&info);
|
|
}
|
|
|
|
if (RomClosed == nullptr)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (!LoadFunctions())
|
|
{
|
|
WriteTrace(PluginTraceType(), TraceWarning, "Failed to load functions");
|
|
return false;
|
|
}
|
|
WriteTrace(PluginTraceType(), TraceDebug, "Functions loaded");
|
|
|
|
if (PluginOpened)
|
|
{
|
|
WriteTrace(PluginTraceType(), TraceDebug, "Before plugin opened");
|
|
PluginOpened();
|
|
WriteTrace(PluginTraceType(), TraceDebug, "After plugin opened");
|
|
}
|
|
WriteTrace(PluginTraceType(), TraceDebug, "Loaded");
|
|
return true;
|
|
}
|
|
|
|
void CPlugin::RomOpened(RenderWindow * Render)
|
|
{
|
|
if (m_RomOpen)
|
|
{
|
|
return;
|
|
}
|
|
|
|
#ifdef ANDROID
|
|
if (m_PluginInfo.Type == PLUGIN_TYPE_VIDEO)
|
|
{
|
|
WriteTrace(PluginTraceType(), TraceDebug, "Render = %p", Render);
|
|
if (Render != nullptr)
|
|
{
|
|
WriteTrace(PluginTraceType(), TraceDebug, "Calling GfxThreadInit");
|
|
Render->GfxThreadInit();
|
|
WriteTrace(PluginTraceType(), TraceDebug, "GfxThreadInit Done");
|
|
}
|
|
}
|
|
#else
|
|
Render = Render; // Used just for the Android port
|
|
#endif
|
|
|
|
if (RomOpen != nullptr)
|
|
{
|
|
WriteTrace(PluginTraceType(), TraceDebug, "Before ROM open");
|
|
RomOpen();
|
|
WriteTrace(PluginTraceType(), TraceDebug, "After ROM open");
|
|
}
|
|
|
|
m_RomOpen = true;
|
|
}
|
|
|
|
void CPlugin::RomClose(RenderWindow * Render)
|
|
{
|
|
if (!m_RomOpen)
|
|
{
|
|
return;
|
|
}
|
|
|
|
#ifdef ANDROID
|
|
if (m_PluginInfo.Type == PLUGIN_TYPE_VIDEO)
|
|
{
|
|
WriteTrace(PluginTraceType(), TraceDebug, "Render = %p", Render);
|
|
if (Render != NULL)
|
|
{
|
|
WriteTrace(PluginTraceType(), TraceDebug, "Calling GfxThreadDone");
|
|
Render->GfxThreadDone();
|
|
WriteTrace(PluginTraceType(), TraceDebug, "GfxThreadDone Done");
|
|
}
|
|
}
|
|
#else
|
|
Render = Render; // Used just for the Android port
|
|
#endif
|
|
|
|
WriteTrace(PluginTraceType(), TraceDebug, "Before ROM close");
|
|
RomClosed();
|
|
m_RomOpen = false;
|
|
WriteTrace(PluginTraceType(), TraceDebug, "After ROM close");
|
|
}
|
|
|
|
void CPlugin::GameReset(RenderWindow * Render)
|
|
{
|
|
if (m_RomOpen)
|
|
{
|
|
RomClose(Render);
|
|
RomOpened(Render);
|
|
}
|
|
}
|
|
|
|
void CPlugin::Close(RenderWindow * Render)
|
|
{
|
|
WriteTrace(PluginTraceType(), TraceDebug, "(%s): Start", PluginType());
|
|
RomClose(Render);
|
|
m_Initialized = false;
|
|
if (CloseDLL != nullptr)
|
|
{
|
|
CloseDLL();
|
|
}
|
|
WriteTrace(PluginTraceType(), TraceDebug, "(%s): Done", PluginType());
|
|
}
|
|
|
|
void CPlugin::UnloadPlugin()
|
|
{
|
|
WriteTrace(PluginTraceType(), TraceDebug, "(%s): Start", PluginType());
|
|
memset(&m_PluginInfo, 0, sizeof(m_PluginInfo));
|
|
if (m_LibHandle != nullptr)
|
|
{
|
|
UnloadPluginDetails();
|
|
}
|
|
if (m_LibHandle != nullptr)
|
|
{
|
|
DynamicLibraryClose(m_LibHandle);
|
|
m_LibHandle = nullptr;
|
|
}
|
|
|
|
DllAbout = nullptr;
|
|
CloseDLL = nullptr;
|
|
RomOpen = nullptr;
|
|
RomClosed = nullptr;
|
|
PluginOpened = nullptr;
|
|
DllConfig = nullptr;
|
|
SetSettingInfo = nullptr;
|
|
SetSettingInfo2 = nullptr;
|
|
SetSettingInfo3 = nullptr;
|
|
WriteTrace(PluginTraceType(), TraceDebug, "(%s): Done", PluginType());
|
|
}
|
|
|
|
const char * CPlugin::PluginType() const
|
|
{
|
|
switch (m_PluginInfo.Type)
|
|
{
|
|
case PLUGIN_TYPE_RSP: return "RSP";
|
|
case PLUGIN_TYPE_VIDEO: return "Video";
|
|
case PLUGIN_TYPE_AUDIO: return "Audio";
|
|
case PLUGIN_TYPE_CONTROLLER: return "Control";
|
|
}
|
|
return "Unknown";
|
|
}
|
|
|
|
TraceModuleProject64 CPlugin::PluginTraceType() const
|
|
{
|
|
switch (m_PluginInfo.Type)
|
|
{
|
|
case PLUGIN_TYPE_RSP: return TraceRSPPlugin;
|
|
case PLUGIN_TYPE_VIDEO: return TraceVideoPlugin;
|
|
case PLUGIN_TYPE_AUDIO: return TraceAudioPlugin;
|
|
case PLUGIN_TYPE_CONTROLLER: return TraceControllerPlugin;
|
|
}
|
|
return TracePlugins;
|
|
}
|
|
|
|
bool CPlugin::ValidPluginVersion(PLUGIN_INFO & PluginInfo)
|
|
{
|
|
switch (PluginInfo.Type)
|
|
{
|
|
case PLUGIN_TYPE_RSP:
|
|
if (PluginInfo.Version == 0x0001)
|
|
{
|
|
return true;
|
|
}
|
|
if (PluginInfo.Version == 0x0100)
|
|
{
|
|
return true;
|
|
}
|
|
if (PluginInfo.Version == 0x0101)
|
|
{
|
|
return true;
|
|
}
|
|
if (PluginInfo.Version == 0x0102)
|
|
{
|
|
return true;
|
|
}
|
|
if (PluginInfo.Version == 0x0103)
|
|
{
|
|
return true;
|
|
}
|
|
break;
|
|
case PLUGIN_TYPE_VIDEO:
|
|
if (PluginInfo.Version == 0x0102)
|
|
{
|
|
return true;
|
|
}
|
|
if (PluginInfo.Version == 0x0103)
|
|
{
|
|
return true;
|
|
}
|
|
if (PluginInfo.Version == 0x0104)
|
|
{
|
|
return true;
|
|
}
|
|
break;
|
|
case PLUGIN_TYPE_AUDIO:
|
|
if (PluginInfo.Version == 0x0101)
|
|
{
|
|
return true;
|
|
}
|
|
if (PluginInfo.Version == 0x0102)
|
|
{
|
|
return true;
|
|
}
|
|
break;
|
|
case PLUGIN_TYPE_CONTROLLER:
|
|
if (PluginInfo.Version == 0x0100)
|
|
{
|
|
return true;
|
|
}
|
|
if (PluginInfo.Version == 0x0101)
|
|
{
|
|
return true;
|
|
}
|
|
if (PluginInfo.Version == 0x0102)
|
|
{
|
|
return true;
|
|
}
|
|
break;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void CPlugin::DisplayError(const char * Message)
|
|
{
|
|
g_Notify->DisplayError(Message);
|
|
}
|
|
|
|
void CPlugin::FatalError(const char * Message)
|
|
{
|
|
g_Notify->FatalError(Message);
|
|
}
|
|
|
|
void CPlugin::DisplayMessage(int DisplayTime, const char * Message)
|
|
{
|
|
g_Notify->DisplayMessage(DisplayTime, Message);
|
|
}
|
|
|
|
void CPlugin::DisplayMessage2(const char * Message)
|
|
{
|
|
g_Notify->DisplayMessage2(Message);
|
|
}
|
|
|
|
void CPlugin::BreakPoint(const char * FileName, int32_t LineNumber)
|
|
{
|
|
g_Notify->BreakPoint(FileName, LineNumber);
|
|
}
|