2009-07-28 23:32:10 +02:00
|
|
|
// Copyright (C) 2003 Dolphin Project.
|
2009-07-13 09:31:43 +02:00
|
|
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, version 2.0.
|
|
|
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License 2.0 for more details.
|
|
|
|
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
|
|
|
|
// Official SVN repository and contact information can be found at
|
|
|
|
// http://code.google.com/p/dolphin-emu/
|
|
|
|
|
2008-09-07 22:26:38 +02:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2008-12-07 23:42:49 +01:00
|
|
|
#include <stdarg.h>
|
|
|
|
|
2008-09-07 22:26:38 +02:00
|
|
|
#ifndef _WIN32
|
2009-07-13 08:34:12 +02:00
|
|
|
#include <sys/param.h>
|
2008-09-07 22:26:38 +02:00
|
|
|
#endif
|
|
|
|
|
2009-01-13 00:59:27 +01:00
|
|
|
#include "Common.h"
|
2009-05-15 10:55:46 +02:00
|
|
|
#include "FileUtil.h"
|
|
|
|
|
2010-03-16 04:34:27 +01:00
|
|
|
#if defined HAVE_X11 && HAVE_X11
|
2010-04-12 03:33:10 +02:00
|
|
|
#include <X11/keysym.h>
|
|
|
|
#include "State.h"
|
2010-04-22 06:28:34 +02:00
|
|
|
#include "X11Utils.h"
|
2010-03-16 04:34:27 +01:00
|
|
|
#endif
|
|
|
|
|
2010-06-04 06:59:07 +02:00
|
|
|
#ifdef __APPLE__
|
|
|
|
#import "cocoaApp.h"
|
|
|
|
#endif
|
|
|
|
|
2010-02-05 11:52:27 +01:00
|
|
|
#include "Core.h"
|
2008-09-10 03:01:28 +02:00
|
|
|
#include "Host.h"
|
2008-09-07 22:26:38 +02:00
|
|
|
#include "CPUDetect.h"
|
2008-09-07 23:06:55 +02:00
|
|
|
#include "cmdline.h"
|
|
|
|
#include "Thread.h"
|
|
|
|
#include "PowerPC/PowerPC.h"
|
2009-01-12 21:52:45 +01:00
|
|
|
|
2010-02-19 18:05:26 +01:00
|
|
|
#include "PluginManager.h"
|
|
|
|
#include "ConfigManager.h"
|
|
|
|
#include "LogManager.h"
|
2008-09-07 22:26:38 +02:00
|
|
|
#include "BootManager.h"
|
|
|
|
|
2010-04-12 03:33:10 +02:00
|
|
|
#if defined HAVE_X11 && HAVE_X11
|
|
|
|
bool running = true;
|
|
|
|
#endif
|
|
|
|
|
2010-05-26 23:23:44 +02:00
|
|
|
bool rendererHasFocus = true;
|
|
|
|
|
2008-09-07 22:26:38 +02:00
|
|
|
void Host_NotifyMapLoaded(){}
|
|
|
|
|
2009-08-01 21:16:15 +02:00
|
|
|
void Host_ShowJitResults(unsigned int address){}
|
|
|
|
|
2010-03-16 14:18:52 +01:00
|
|
|
Common::Event updateMainFrameEvent;
|
|
|
|
void Host_Message(int Id)
|
|
|
|
{
|
|
|
|
#if defined(HAVE_X11) && HAVE_X11
|
2010-03-16 14:30:52 +01:00
|
|
|
switch (Id)
|
|
|
|
{
|
|
|
|
case WM_USER_STOP:
|
2010-04-12 03:33:10 +02:00
|
|
|
running = false;
|
2010-03-16 14:30:52 +01:00
|
|
|
break;
|
|
|
|
}
|
2010-03-16 14:18:52 +01:00
|
|
|
#endif
|
|
|
|
}
|
2008-09-07 22:26:38 +02:00
|
|
|
|
2010-04-19 05:06:18 +02:00
|
|
|
void Host_UpdateTitle(const char* title){};
|
|
|
|
|
2008-09-07 22:26:38 +02:00
|
|
|
void Host_UpdateLogDisplay(){}
|
|
|
|
|
|
|
|
|
|
|
|
void Host_UpdateDisasmDialog(){}
|
|
|
|
|
|
|
|
|
2008-09-07 23:06:55 +02:00
|
|
|
void Host_UpdateMainFrame()
|
|
|
|
{
|
|
|
|
updateMainFrameEvent.Set();
|
|
|
|
}
|
2008-09-07 22:26:38 +02:00
|
|
|
|
|
|
|
void Host_UpdateBreakPointView(){}
|
|
|
|
|
|
|
|
|
|
|
|
void Host_UpdateMemoryView(){}
|
|
|
|
|
|
|
|
|
|
|
|
void Host_SetDebugMode(bool){}
|
|
|
|
|
2010-04-12 03:33:10 +02:00
|
|
|
void Host_RequestWindowSize(int& x, int& y, int& width, int& height)
|
|
|
|
{
|
|
|
|
x = SConfig::GetInstance().m_LocalCoreStartupParameter.iRenderWindowXPos;
|
|
|
|
y = SConfig::GetInstance().m_LocalCoreStartupParameter.iRenderWindowYPos;
|
|
|
|
width = SConfig::GetInstance().m_LocalCoreStartupParameter.iRenderWindowWidth;
|
|
|
|
height = SConfig::GetInstance().m_LocalCoreStartupParameter.iRenderWindowHeight;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Host_RendererHasFocus()
|
|
|
|
{
|
|
|
|
return rendererHasFocus;
|
|
|
|
}
|
2008-09-07 22:26:38 +02:00
|
|
|
|
|
|
|
void Host_SetWaitCursor(bool enable){}
|
|
|
|
|
|
|
|
|
2009-02-16 10:26:47 +01:00
|
|
|
void Host_UpdateStatusBar(const char* _pText, int Filed){}
|
2008-09-07 22:26:38 +02:00
|
|
|
|
2008-09-10 03:01:28 +02:00
|
|
|
void Host_SysMessage(const char *fmt, ...)
|
|
|
|
{
|
|
|
|
va_list list;
|
|
|
|
char msg[512];
|
|
|
|
|
|
|
|
va_start(list, fmt);
|
|
|
|
vsprintf(msg, fmt, list);
|
|
|
|
va_end(list);
|
|
|
|
|
|
|
|
size_t len = strlen(msg);
|
|
|
|
if (msg[len - 1] != '\n') {
|
|
|
|
msg[len - 1] = '\n';
|
|
|
|
msg[len] = '\0';
|
|
|
|
}
|
2010-02-19 18:05:26 +01:00
|
|
|
fprintf(stderr, "%s", msg);
|
2008-09-10 03:01:28 +02:00
|
|
|
}
|
|
|
|
|
2008-10-30 17:36:45 +01:00
|
|
|
void Host_SetWiiMoteConnectionState(int _State) {}
|
|
|
|
|
2010-04-22 06:28:34 +02:00
|
|
|
#if defined(HAVE_X11) && HAVE_X11
|
|
|
|
void X11_MainLoop()
|
|
|
|
{
|
|
|
|
bool fullscreen = SConfig::GetInstance().m_LocalCoreStartupParameter.bFullscreen;
|
|
|
|
while (Core::GetState() == Core::CORE_UNINITIALIZED)
|
|
|
|
updateMainFrameEvent.Wait();
|
|
|
|
|
|
|
|
Display *dpy = XOpenDisplay(0);
|
2010-07-16 16:14:57 +02:00
|
|
|
Window win = (Window)Core::GetWindowHandle();
|
2010-04-22 06:28:34 +02:00
|
|
|
XSelectInput(dpy, win, KeyPressMask | KeyReleaseMask | FocusChangeMask);
|
|
|
|
|
|
|
|
#if defined(HAVE_XRANDR) && HAVE_XRANDR
|
|
|
|
X11Utils::XRRConfiguration *XRRConfig = new X11Utils::XRRConfiguration(dpy, win);
|
|
|
|
#endif
|
|
|
|
|
2010-07-24 01:51:34 +02:00
|
|
|
Cursor blankCursor = None;
|
2010-04-22 06:28:34 +02:00
|
|
|
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bHideCursor)
|
|
|
|
{
|
|
|
|
// make a blank cursor
|
|
|
|
Pixmap Blank;
|
|
|
|
XColor DummyColor;
|
|
|
|
char ZeroData[1] = {0};
|
|
|
|
Blank = XCreateBitmapFromData (dpy, win, ZeroData, 1, 1);
|
|
|
|
blankCursor = XCreatePixmapCursor(dpy, Blank, Blank, &DummyColor, &DummyColor, 0, 0);
|
|
|
|
XFreePixmap (dpy, Blank);
|
|
|
|
XDefineCursor(dpy, win, blankCursor);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fullscreen)
|
|
|
|
{
|
2010-07-16 16:14:57 +02:00
|
|
|
X11Utils::EWMH_Fullscreen(dpy, _NET_WM_STATE_TOGGLE);
|
2010-04-22 06:28:34 +02:00
|
|
|
#if defined(HAVE_XRANDR) && HAVE_XRANDR
|
|
|
|
XRRConfig->ToggleDisplayMode(True);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
// The actual loop
|
|
|
|
while (running)
|
|
|
|
{
|
|
|
|
XEvent event;
|
|
|
|
KeySym key;
|
|
|
|
for (int num_events = XPending(dpy); num_events > 0; num_events--)
|
|
|
|
{
|
|
|
|
XNextEvent(dpy, &event);
|
|
|
|
switch(event.type)
|
|
|
|
{
|
|
|
|
case KeyPress:
|
|
|
|
key = XLookupKeysym((XKeyEvent*)&event, 0);
|
|
|
|
if (key == XK_Escape)
|
|
|
|
{
|
|
|
|
if (Core::GetState() == Core::CORE_RUN)
|
|
|
|
{
|
|
|
|
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bHideCursor)
|
|
|
|
XUndefineCursor(dpy, win);
|
|
|
|
Core::SetState(Core::CORE_PAUSE);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bHideCursor)
|
|
|
|
XDefineCursor(dpy, win, blankCursor);
|
|
|
|
Core::SetState(Core::CORE_RUN);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if ((key == XK_Return) && (event.xkey.state & Mod1Mask))
|
|
|
|
{
|
|
|
|
fullscreen = !fullscreen;
|
2010-07-16 16:14:57 +02:00
|
|
|
X11Utils::EWMH_Fullscreen(dpy, _NET_WM_STATE_TOGGLE);
|
2010-04-22 06:28:34 +02:00
|
|
|
#if defined(HAVE_XRANDR) && HAVE_XRANDR
|
|
|
|
XRRConfig->ToggleDisplayMode(fullscreen);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else if (key >= XK_F1 && key <= XK_F8)
|
|
|
|
{
|
|
|
|
int slot_number = key - XK_F1 + 1;
|
|
|
|
if (event.xkey.state & ShiftMask)
|
|
|
|
State_Save(slot_number);
|
|
|
|
else
|
|
|
|
State_Load(slot_number);
|
|
|
|
}
|
|
|
|
else if (key == XK_F9)
|
|
|
|
Core::ScreenShot();
|
|
|
|
else if (key == XK_F11)
|
|
|
|
State_LoadLastSaved();
|
|
|
|
else if (key == XK_F12)
|
|
|
|
{
|
|
|
|
if (event.xkey.state & ShiftMask)
|
|
|
|
State_UndoLoadState();
|
|
|
|
else
|
|
|
|
State_UndoSaveState();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case FocusIn:
|
|
|
|
rendererHasFocus = true;
|
|
|
|
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bHideCursor &&
|
|
|
|
Core::GetState() != Core::CORE_PAUSE)
|
|
|
|
XDefineCursor(dpy, win, blankCursor);
|
|
|
|
break;
|
|
|
|
case FocusOut:
|
|
|
|
rendererHasFocus = false;
|
|
|
|
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bHideCursor)
|
|
|
|
XUndefineCursor(dpy, win);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!fullscreen)
|
|
|
|
{
|
|
|
|
Window winDummy;
|
|
|
|
unsigned int borderDummy, depthDummy;
|
|
|
|
XGetGeometry(dpy, win, &winDummy,
|
|
|
|
&SConfig::GetInstance().m_LocalCoreStartupParameter.iRenderWindowXPos,
|
|
|
|
&SConfig::GetInstance().m_LocalCoreStartupParameter.iRenderWindowYPos,
|
|
|
|
(unsigned int *)&SConfig::GetInstance().m_LocalCoreStartupParameter.iRenderWindowWidth,
|
|
|
|
(unsigned int *)&SConfig::GetInstance().m_LocalCoreStartupParameter.iRenderWindowHeight,
|
|
|
|
&borderDummy, &depthDummy);
|
|
|
|
}
|
|
|
|
usleep(100000);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(HAVE_XRANDR) && HAVE_XRANDR
|
|
|
|
delete XRRConfig;
|
|
|
|
#endif
|
|
|
|
if (SConfig::GetInstance().m_LocalCoreStartupParameter.bHideCursor)
|
|
|
|
XFreeCursor(dpy, blankCursor);
|
|
|
|
XCloseDisplay(dpy);
|
|
|
|
Core::Stop();
|
|
|
|
}
|
|
|
|
#endif
|
2009-01-12 21:52:45 +01:00
|
|
|
|
2010-06-04 06:59:07 +02:00
|
|
|
//for cocoa we need to hijack the main to get event
|
|
|
|
#ifdef __APPLE__
|
|
|
|
|
|
|
|
@interface CocoaThread : NSObject
|
|
|
|
{
|
|
|
|
NSThread *Thread;
|
|
|
|
}
|
|
|
|
- (void)cocoaThreadStart;
|
|
|
|
- (void)cocoaThreadRun:(id)sender;
|
|
|
|
- (void)cocoaThreadQuit:(NSNotification*)note;
|
|
|
|
- (bool)cocoaThreadRunning;
|
|
|
|
@end
|
|
|
|
|
|
|
|
static NSString *CocoaThreadHaveFinish = @"CocoaThreadHaveFinish";
|
|
|
|
|
|
|
|
int cocoaArgc;
|
|
|
|
char **cocoaArgv;
|
|
|
|
int appleMain(int argc, char *argv[]);
|
|
|
|
|
|
|
|
@implementation CocoaThread
|
|
|
|
|
|
|
|
- (void)cocoaThreadStart
|
|
|
|
{
|
|
|
|
|
|
|
|
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(cocoaThreadQuit:) name:CocoaThreadHaveFinish object:nil];
|
|
|
|
[NSThread detachNewThreadSelector:@selector(cocoaThreadRun:) toTarget:self withObject:nil];
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)cocoaThreadRun:(id)sender
|
|
|
|
{
|
|
|
|
|
|
|
|
NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
|
|
|
|
Thread = [NSThread currentThread];
|
|
|
|
//launch main
|
|
|
|
appleMain(cocoaArgc,cocoaArgv);
|
|
|
|
|
|
|
|
[[NSNotificationCenter defaultCenter] postNotificationName:CocoaThreadHaveFinish object:nil];
|
|
|
|
|
|
|
|
[pool release];
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)cocoaThreadQuit:(NSNotification*)note
|
|
|
|
{
|
|
|
|
|
|
|
|
[[NSNotificationCenter defaultCenter] removeObserver:self];
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
- (bool)cocoaThreadRunning
|
|
|
|
{
|
|
|
|
if([Thread isFinished])
|
|
|
|
return false;
|
|
|
|
else
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
|
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
2010-07-10 00:31:04 +02:00
|
|
|
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
|
2010-06-04 06:59:07 +02:00
|
|
|
|
|
|
|
cocoaArgc = argc;
|
|
|
|
cocoaArgv = argv;
|
|
|
|
|
|
|
|
cocoaCreateApp();
|
|
|
|
|
|
|
|
CocoaThread *thread = [[CocoaThread alloc] init];
|
|
|
|
NSEvent *event = [[NSEvent alloc] init];
|
|
|
|
|
|
|
|
[thread cocoaThreadStart];
|
|
|
|
|
|
|
|
//cocoa event loop
|
|
|
|
while(1)
|
|
|
|
{
|
|
|
|
event = [NSApp nextEventMatchingMask:NSAnyEventMask untilDate:[NSDate distantPast] inMode:NSDefaultRunLoopMode dequeue:YES ];
|
|
|
|
if(cocoaSendEvent(event))
|
|
|
|
{
|
|
|
|
Core::Stop();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if(![thread cocoaThreadRunning])
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
[event release];
|
|
|
|
[thread release];
|
|
|
|
[pool release];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int appleMain(int argc, char *argv[])
|
|
|
|
#else
|
2008-09-07 22:26:38 +02:00
|
|
|
int main(int argc, char* argv[])
|
2010-06-04 06:59:07 +02:00
|
|
|
#endif
|
2008-09-07 22:26:38 +02:00
|
|
|
{
|
2008-09-07 23:02:57 +02:00
|
|
|
gengetopt_args_info args_info;
|
|
|
|
|
2010-02-19 18:05:26 +01:00
|
|
|
if (cmdline_parser(argc, argv, &args_info) != 0)
|
2008-09-07 23:02:57 +02:00
|
|
|
return(1);
|
|
|
|
|
|
|
|
if (args_info.inputs_num < 1)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Please supply at least one argument - the ISO to boot.\n");
|
|
|
|
return(1);
|
|
|
|
}
|
|
|
|
std::string bootFile(args_info.inputs[0]);
|
|
|
|
|
2008-09-07 23:06:55 +02:00
|
|
|
updateMainFrameEvent.Init();
|
2010-02-19 18:05:26 +01:00
|
|
|
|
|
|
|
LogManager::Init();
|
|
|
|
SConfig::Init();
|
|
|
|
CPluginManager::Init();
|
|
|
|
|
2009-02-17 21:28:49 +01:00
|
|
|
CPluginManager::GetInstance().ScanForPlugins();
|
2009-05-15 10:55:46 +02:00
|
|
|
|
2010-03-16 04:34:27 +01:00
|
|
|
#if defined HAVE_X11 && HAVE_X11
|
|
|
|
XInitThreads();
|
|
|
|
#endif
|
|
|
|
|
2010-02-19 18:05:26 +01:00
|
|
|
if (BootManager::BootCore(bootFile)) //no use running the loop when booting fails
|
2008-09-07 23:06:55 +02:00
|
|
|
{
|
2010-03-16 14:18:52 +01:00
|
|
|
#if defined(HAVE_X11) && HAVE_X11
|
2010-04-22 06:28:34 +02:00
|
|
|
X11_MainLoop();
|
2010-03-16 14:18:52 +01:00
|
|
|
#else
|
2010-02-19 18:05:26 +01:00
|
|
|
while (PowerPC::GetState() != PowerPC::CPU_POWERDOWN)
|
|
|
|
updateMainFrameEvent.Wait();
|
2010-03-16 14:18:52 +01:00
|
|
|
#endif
|
2008-09-07 23:06:55 +02:00
|
|
|
}
|
2010-03-16 14:18:52 +01:00
|
|
|
updateMainFrameEvent.Shutdown();
|
2010-02-19 18:05:26 +01:00
|
|
|
|
|
|
|
CPluginManager::Shutdown();
|
|
|
|
SConfig::Shutdown();
|
|
|
|
LogManager::Shutdown();
|
2008-09-07 22:26:38 +02:00
|
|
|
|
2008-09-07 23:02:57 +02:00
|
|
|
cmdline_parser_free (&args_info);
|
|
|
|
return(0);
|
2008-09-07 22:26:38 +02:00
|
|
|
}
|