mirror of
https://github.com/PCSX2/pcsx2-sourceforge.git
synced 2026-02-04 03:11:18 +01:00
620 lines
15 KiB
C
620 lines
15 KiB
C
/* Pcsx2 - Pc Ps2 Emulator
|
|
* Copyright (C) 2002-2003 Pcsx2 Team
|
|
*
|
|
* 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; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* 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 for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdarg.h>
|
|
#include <stdlib.h>
|
|
#include <dirent.h>
|
|
#include <dlfcn.h>
|
|
#include <fcntl.h>
|
|
#include <errno.h>
|
|
//#include <error.h>
|
|
#include <signal.h>
|
|
#include <string.h>
|
|
#include <sys/mman.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/types.h>
|
|
#include <time.h>
|
|
#include <X11/keysym.h>
|
|
#include <gtk/gtk.h>
|
|
#include <glib/gthread.h>
|
|
#include <unistd.h>
|
|
|
|
|
|
#include "Common.h"
|
|
#include "PsxCommon.h"
|
|
#include "Linux.h"
|
|
|
|
int UseGui = 1;
|
|
int needReset = 1;
|
|
int RunExe = 0;
|
|
extern int g_SaveGSStream;
|
|
extern int g_ZeroGSOptions;
|
|
|
|
#ifdef PCSX2_DEVBUILD
|
|
|
|
TESTRUNARGS g_TestRun;
|
|
char* g_pRunGSState = NULL;
|
|
|
|
#endif
|
|
|
|
int main(int argc, char *argv[]) {
|
|
char *file = NULL;
|
|
char elfname[256];
|
|
char *lang;
|
|
int runcd=0;
|
|
int efile = 0;
|
|
char* g_pRunGSState = NULL;
|
|
int i;
|
|
struct stat buf;
|
|
|
|
#ifdef PCSX2_VIRTUAL_MEM
|
|
void* pmem = mmap(PS2MEM_BASE, 0x40000000, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, 0, 0);
|
|
|
|
if( pmem != PS2MEM_BASE ) {
|
|
SysMessage("Failed to allocate virtual memory %x-%x. Use TLB build",
|
|
PS2MEM_BASE, PS2MEM_BASE+0x40000000);
|
|
return -1;
|
|
}
|
|
#endif
|
|
|
|
|
|
#ifdef ENABLE_NLS
|
|
setlocale(LC_ALL, "");
|
|
bindtextdomain(PACKAGE, "Langs");
|
|
textdomain(PACKAGE);
|
|
#endif
|
|
|
|
printf("\n");
|
|
mkdir("inis", 0755);
|
|
|
|
strcpy(cfgfile, "inis/pcsx2.cfg");
|
|
|
|
#ifdef PCSX2_DEVBUILD
|
|
memset(&g_TestRun, 0, sizeof(g_TestRun));
|
|
#endif
|
|
|
|
i = 1;
|
|
while(i < argc) {
|
|
char* token = argv[i++];
|
|
|
|
if( stricmp(token, "-help") == 0 || stricmp(token, "--help") == 0 || stricmp(token, "-h") == 0 ) {
|
|
char* phelpmsg =
|
|
"\tpcsx2 [options] [file]\n\n"
|
|
"-cfg [file] {configuration file}\n"
|
|
"-efile [efile] {0 - reset, 1 - runcd (default), 2 - loadelf}\n"
|
|
"-help {display this help file}\n"
|
|
"-nogui {Don't use gui when launching}\n"
|
|
"-loadgs [file} {Loads a gsstate}\n"
|
|
"\n"
|
|
#ifdef PCSX2_DEVBUILD
|
|
"Testing Options: \n"
|
|
"\t-frame [frame] {game will run up to this frame before exiting}\n"
|
|
"\t-image [name] {path and base name of image (do not include the .ext)}\n"
|
|
"\t-jpg {save images to jpg format}\n"
|
|
"\t-log [name] {log path to save log file in}\n"
|
|
"\t-logopt [hex] {log options in hex (see debug.h) }\n"
|
|
"\t-numimages [num] {after hitting frame, this many images will be captures every 20 frames}\n"
|
|
"\t-test {Triggers testing mode (only for dev builds)}\n"
|
|
"\n"
|
|
#endif
|
|
|
|
"Load Plugins:\n"
|
|
"\t-cdvd [dllpath] {specify the dll load path of the CDVD plugin}\n"
|
|
"\t-gs [dllpath] {specify the dll load path of the GS plugin}\n"
|
|
"-pad [tsxcal] {specify to hold down on the triangle, square, circle, x, start, select buttons}\n"
|
|
"\t-spu [dllpath] {specify the dll load path of the SPU2 plugin}\n"
|
|
"\n";
|
|
|
|
//SysMessage(phelpmsg);
|
|
return 0;
|
|
}
|
|
else if( stricmp(token, "-efile") == 0 ) {
|
|
token = argv[i++];
|
|
if( token != NULL ) {
|
|
efile = atoi(token);
|
|
}
|
|
}
|
|
else if( stricmp(token, "-nogui") == 0 ) {
|
|
UseGui = 0;
|
|
}
|
|
else if( stricmp(token, "-loadgs") == 0 ) {
|
|
g_pRunGSState = argv[i++];
|
|
}
|
|
#ifdef PCSX2_DEVBUILD
|
|
else if( stricmp(token, "-image") == 0 ) {
|
|
g_TestRun.pimagename = argv[i++];
|
|
}
|
|
else if( stricmp(token, "-log") == 0 ) {
|
|
g_TestRun.plogname = argv[i++];
|
|
}
|
|
else if( stricmp(token, "-logopt") == 0 ) {
|
|
token = argv[i++];
|
|
if( token != NULL ) {
|
|
if( token[0] == '0' && token[1] == 'x' ) token += 2;
|
|
sscanf(token, "%x", &varLog);
|
|
}
|
|
}
|
|
else if( stricmp(token, "-frame") == 0 ) {
|
|
token = argv[i++];
|
|
if( token != NULL ) {
|
|
g_TestRun.frame = atoi(token);
|
|
}
|
|
}
|
|
else if( stricmp(token, "-numimages") == 0 ) {
|
|
token = argv[i++];
|
|
if( token != NULL ) {
|
|
g_TestRun.numimages = atoi(token);
|
|
}
|
|
}
|
|
else if( stricmp(token, "-jpg") == 0 ) {
|
|
g_TestRun.jpgcapture = 1;
|
|
}
|
|
else if( stricmp(token, "-gs") == 0 ) {
|
|
token = argv[i++];
|
|
g_TestRun.pgsdll = token;
|
|
}
|
|
else if( stricmp(token, "-cdvd") == 0 ) {
|
|
token = argv[i++];
|
|
g_TestRun.pcdvddll = token;
|
|
}
|
|
else if( stricmp(token, "-spu") == 0 ) {
|
|
token = argv[i++];
|
|
g_TestRun.pspudll = token;
|
|
}
|
|
else if( stricmp(token, "-test") == 0 ) {
|
|
g_TestRun.enabled = 1;
|
|
}
|
|
#endif
|
|
else if( stricmp(token, "-pad") == 0 ) {
|
|
token = argv[i++];
|
|
printf("-pad ignored\n");
|
|
}
|
|
else if( stricmp(token, "-loadgs") == 0 ) {
|
|
token = argv[i++];
|
|
g_pRunGSState = token;
|
|
}
|
|
else {
|
|
file = token;
|
|
printf("opening file %s\n", file);
|
|
}
|
|
}
|
|
|
|
#ifdef PCSX2_DEVBUILD
|
|
g_TestRun.efile = efile;
|
|
g_TestRun.ptitle = file;
|
|
#endif
|
|
|
|
if (LoadConfig() == -1) {
|
|
|
|
if (UseGui) gtk_init(NULL, NULL);
|
|
|
|
memset(&Config, 0, sizeof(Config));
|
|
strcpy(Config.BiosDir, "bios/");
|
|
strcpy(Config.PluginsDir, "plugins/");
|
|
Config.Patch = 1;
|
|
Config.Options = PCSX2_EEREC|PCSX2_VU0REC|PCSX2_VU1REC|PCSX2_COP2REC;
|
|
|
|
SysMessage(_("Pcsx2 needs to be configured"));
|
|
Pcsx2Configure();
|
|
|
|
return 0;
|
|
}
|
|
|
|
// make gtk thread safe if using MTGS
|
|
if( CHECK_MULTIGS ) {
|
|
g_thread_init(NULL);
|
|
gdk_threads_init();
|
|
}
|
|
|
|
if (UseGui) gtk_init(NULL, NULL);
|
|
|
|
if (Config.Lang[0] == 0) {
|
|
strcpy(Config.Lang, "en");
|
|
}
|
|
|
|
langs = (_langs*)malloc(sizeof(_langs));
|
|
strcpy(langs[0].lang, "en");
|
|
InitLanguages(); i=1;
|
|
while ((lang = GetLanguageNext()) != NULL) {
|
|
langs = (_langs*)realloc(langs, sizeof(_langs)*(i+1));
|
|
strcpy(langs[i].lang, lang);
|
|
i++;
|
|
}
|
|
CloseLanguages();
|
|
langsMax = i;
|
|
|
|
if( Config.PsxOut ) {
|
|
// output the help commands
|
|
SysPrintf("\tF1 - save state\n");
|
|
SysPrintf("\t(Shift +) F2 - cycle states\n");
|
|
SysPrintf("\tF3 - load state\n");
|
|
|
|
#ifdef PCSX2_DEVBUILD
|
|
SysPrintf("\tF10 - dump performance counters\n");
|
|
SysPrintf("\tF11 - save GS state\n");
|
|
SysPrintf("\tF12 - dump hardware registers\n");
|
|
#endif
|
|
}
|
|
|
|
if (SysInit() == -1) return 1;
|
|
|
|
#ifdef PCSX2_DEVBUILD
|
|
if( g_pRunGSState ) {
|
|
LoadGSState(g_pRunGSState);
|
|
SysClose();
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
if (UseGui&&file==NULL) {
|
|
StartGui();
|
|
return 0;
|
|
}
|
|
|
|
if (OpenPlugins(file) == -1) {
|
|
return -1;
|
|
}
|
|
SysReset();
|
|
|
|
cpuExecuteBios();
|
|
if (file)
|
|
strcpy(elfname, file);
|
|
if (!efile)
|
|
efile=GetPS2ElfName(elfname);
|
|
loadElfFile(elfname);
|
|
|
|
if( GSsetGameCRC != NULL )
|
|
GSsetGameCRC(ElfCRC, g_ZeroGSOptions);
|
|
|
|
Cpu->Execute();
|
|
|
|
return 0;
|
|
}
|
|
|
|
DIR *dir;
|
|
|
|
void InitLanguages() {
|
|
dir = opendir("Langs");
|
|
}
|
|
|
|
char *GetLanguageNext() {
|
|
struct dirent *ent;
|
|
|
|
if (dir == NULL) return NULL;
|
|
for (;;) {
|
|
ent = readdir(dir);
|
|
if (ent == NULL) return NULL;
|
|
|
|
if (!strcmp(ent->d_name, ".")) continue;
|
|
if (!strcmp(ent->d_name, "..")) continue;
|
|
break;
|
|
}
|
|
|
|
return ent->d_name;
|
|
}
|
|
|
|
void CloseLanguages() {
|
|
if (dir) closedir(dir);
|
|
}
|
|
|
|
void ChangeLanguage(char *lang) {
|
|
strcpy(Config.Lang, lang);
|
|
SaveConfig();
|
|
LoadConfig();
|
|
}
|
|
|
|
void KeyEvent(keyEvent* ev) {
|
|
char Text[256];
|
|
int ret;
|
|
static int shift = 0;
|
|
|
|
if (ev == NULL)
|
|
return;
|
|
|
|
if( GSkeyEvent != NULL )
|
|
GSkeyEvent(ev);
|
|
|
|
if( ev->event == KEYPRESS ) {
|
|
if( ev->key == XK_Shift_L || ev->key == XK_Shift_R )
|
|
shift = 1;
|
|
}
|
|
else {
|
|
if (ev->event == KEYRELEASE) {
|
|
if( ev->key == XK_Shift_L || ev->key == XK_Shift_R )
|
|
shift = 0;
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
switch (ev->key) {
|
|
case XK_F1: ProcessFKeys(1, shift); break;
|
|
case XK_F2: ProcessFKeys(2, shift); break;
|
|
case XK_F3: ProcessFKeys(3, shift); break;
|
|
case XK_F4: ProcessFKeys(4, shift); break;
|
|
case XK_F5: ProcessFKeys(5, shift); break;
|
|
case XK_F6: ProcessFKeys(6, shift); break;
|
|
case XK_F7: ProcessFKeys(7, shift); break;
|
|
case XK_F8: ProcessFKeys(8, shift); break;
|
|
case XK_F9: ProcessFKeys(9, shift); break;
|
|
case XK_F10: ProcessFKeys(10, shift); break;
|
|
case XK_F11: ProcessFKeys(11, shift); break;
|
|
case XK_F12: ProcessFKeys(12, shift); break;
|
|
|
|
case XK_Escape:
|
|
signal(SIGINT, SIG_DFL);
|
|
signal(SIGPIPE, SIG_DFL);
|
|
|
|
#ifdef PCSX2_DEVBUILD
|
|
if( g_SaveGSStream >= 3 ) {
|
|
// gs state
|
|
g_SaveGSStream = 4;
|
|
break;
|
|
}
|
|
#endif
|
|
|
|
ClosePlugins();
|
|
if (!UseGui)
|
|
exit(0);
|
|
RunGui();
|
|
break;
|
|
default:
|
|
GSkeyEvent(ev);
|
|
break;
|
|
}
|
|
}
|
|
|
|
int SysInit() {
|
|
mkdir("sstates", 0755);
|
|
mkdir("memcards", 0755);
|
|
|
|
#ifdef EMU_LOG
|
|
mkdir("logs", 0755);
|
|
|
|
#ifdef PCSX2_DEVBUILD
|
|
if( g_TestRun.plogname != NULL )
|
|
emuLog = fopen(g_TestRun.plogname, "w");
|
|
if( emuLog == NULL )
|
|
emuLog = fopen("logs/emuLog.txt","wb");
|
|
#endif
|
|
|
|
if( emuLog != NULL )
|
|
setvbuf(emuLog, NULL, _IONBF, 0);
|
|
#endif
|
|
|
|
if(cpuInit() == -1 )
|
|
return -1;
|
|
|
|
while (LoadPlugins() == -1) {
|
|
if (Pcsx2Configure() == 0)
|
|
exit(1);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void SysReset() {
|
|
cpuReset();
|
|
}
|
|
|
|
void SysClose() {
|
|
cpuShutdown();
|
|
ReleasePlugins();
|
|
|
|
if (emuLog != NULL) {
|
|
fclose(emuLog);
|
|
emuLog = NULL;
|
|
}
|
|
}
|
|
|
|
void SysPrintf(char *fmt, ...) {
|
|
va_list list;
|
|
char msg[512];
|
|
char* ptr, *src;
|
|
int len, i, j, s;
|
|
|
|
va_start(list, fmt);
|
|
vsprintf(msg, fmt, list);
|
|
va_end(list);
|
|
|
|
if (Config.PsxOut == 0) {
|
|
#ifdef EMU_LOG
|
|
#ifndef LOG_STDOUT
|
|
if (emuLog != NULL && !(varLog & 0x80000000)) {
|
|
fprintf(emuLog, "%s", msg);
|
|
}
|
|
#endif
|
|
#endif
|
|
return;
|
|
}
|
|
|
|
// due to various linux ways of writing lines, convert all \r\n to \n
|
|
// src = msg;
|
|
// while( (ptr = strstr(src, "\r\n")) != NULL ) {
|
|
// ptr[0] = 0;
|
|
// printf("%s\n", src);
|
|
// src = ptr+2;
|
|
// }
|
|
//
|
|
// if( src[0] != 0 )
|
|
// printf("%s", src);
|
|
printf("%s", msg);
|
|
|
|
#ifdef EMU_LOG
|
|
#ifndef LOG_STDOUT
|
|
if (emuLog != NULL && !(varLog & 0x80000000)) {
|
|
ptr = msg; len = strlen(msg);
|
|
for (i=0, j=0; i<len; i++, j++) {
|
|
if (ptr[j] == '\033') {
|
|
ptr[j] = 0;
|
|
fprintf(emuLog, "%s", ptr);
|
|
if (ptr[j+2] == '0') {
|
|
s = 4;
|
|
} else {
|
|
s = 5;
|
|
}
|
|
ptr+= j+s;
|
|
j = 0;
|
|
}
|
|
}
|
|
|
|
fprintf(emuLog, "%s", ptr);
|
|
}
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
void *SysLoadLibrary(char *lib) {
|
|
return dlopen(lib, RTLD_NOW);
|
|
}
|
|
|
|
void *SysLoadSym(void *lib, char *sym) {
|
|
return dlsym(lib, sym);
|
|
}
|
|
|
|
char *SysLibError() {
|
|
return dlerror();
|
|
}
|
|
|
|
void SysCloseLibrary(void *lib) {
|
|
dlclose(lib);
|
|
}
|
|
|
|
void SysUpdate() {
|
|
KeyEvent(PAD1keyEvent());
|
|
KeyEvent(PAD2keyEvent());
|
|
}
|
|
|
|
void SysRunGui() {
|
|
RunGui();
|
|
}
|
|
|
|
void *SysMmap(uptr base, u32 size) {
|
|
return mmap((uptr*)base, size, PROT_EXEC | PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANONYMOUS, 0, 0);
|
|
}
|
|
|
|
void SysMunmap(uptr base, u32 size) {
|
|
munmap((uptr*)base, size);
|
|
}
|
|
|
|
#ifdef PCSX2_VIRTUAL_MEM
|
|
|
|
static int s_nPageSize = 0;
|
|
static int s_nShmCounter = 0;
|
|
|
|
int SysPhysicalAlloc(u32 size, PSMEMORYBLOCK* pblock)
|
|
{
|
|
assert( pblock != NULL );
|
|
|
|
if( s_nPageSize == 0 ) {
|
|
s_nPageSize = getpagesize();
|
|
|
|
// has to be 4k
|
|
if( s_nPageSize != 0x1000 ) {
|
|
perror("page size failed");
|
|
SysMessage("Error! OS page size must be 4Kb!\n"
|
|
"If for some reason the OS cannot have 4Kb pages, then run the TLB build.");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
pblock->pname = (char*)malloc(20);
|
|
sprintf(pblock->pname, "/pcsx2_shm%d", s_nShmCounter++);
|
|
shm_unlink(pblock->pname); // unlink to make sure one can be created
|
|
pblock->fd = shm_open(pblock->pname, O_RDWR|O_CREAT, 0);
|
|
|
|
if( pblock->fd < 0 ) {
|
|
perror("shm_open failed");
|
|
SysMessage("Failed to create shared memory object %s\n", pblock->pname);
|
|
return -1;
|
|
}
|
|
|
|
if( ftruncate(fd, size) < 0 ) {
|
|
perror("ftruncate failed");
|
|
SysMessage("Failed to allocate 0x%x bytes to shm\n", size);
|
|
}
|
|
|
|
pblock->size = size;
|
|
|
|
return 0;
|
|
}
|
|
|
|
void SysPhysicalFree(PSMEMORYBLOCK* pblock)
|
|
{
|
|
assert( pblock != NULL );
|
|
if( pblock->fd )
|
|
close(pblock->fd);
|
|
if( pblock->pname ) {
|
|
shm_unlink(pblock->pname);
|
|
free(pblock->pname);
|
|
}
|
|
memset(pblock, 0, sizeof(PSMEMORYBLOCK));
|
|
}
|
|
|
|
int SysVirtualPhyAlloc(void* base, u32 size, PSMEMORYBLOCK* pblock)
|
|
{
|
|
void* pmem;
|
|
assert(pblock != NULL && pblock->fd >= 0 && pblock->size >= size );
|
|
pmem = mmap(base, size, PROT_READ|PROT_WRITE, MAP_SHARED, pblock->fd, 0);
|
|
|
|
if( pmem != base ) {
|
|
|
|
if( pmem == (void*)-1 ) {
|
|
munmap(base, size);
|
|
}
|
|
|
|
SysPrintf("Failed to map memory at 0x%x of size 0x%x with shm %s\n", (uptr)base, (uptr)size, pblock->pname);
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void SysVirtualFree(void* lpMemReserved, u32 size)
|
|
{
|
|
if( munmap(lpMemReserved, size) < 0 )
|
|
SysPrintf("Failed to unmap %x\n", lpMemReserved);
|
|
}
|
|
|
|
|
|
int SysMapUserPhysicalPages(void* Addr, uptr NumPages, PSMEMORYBLOCK* pblock, int pageoffset)
|
|
{
|
|
void* pmem;
|
|
if( pblock == NULL ) {
|
|
// unmap
|
|
if( munmap(base, size) < 0 ) {
|
|
SysPrintf("SysMapUserPhysicalPages: Failed to unmap %x\n", Addr);
|
|
return 0;
|
|
}
|
|
}
|
|
else {
|
|
// map
|
|
pmem = mmap(Addr, NumPages*s_nPageSize, PROT_READ|PROT_WRITE, MAP_SHARED, pblock->fd, pageoffset*s_nPageSize);
|
|
if( pmem != base ) {
|
|
SysPrintf("SysMapUserPhysicalPages: Failed to map 0x%x, size=0x%x, offset=0x%x\n",
|
|
Addr, NumPages*s_nPageSize, pageoffset*s_nPageSize);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
#endif
|