flycast/core/linux-dist/main.cpp

441 lines
9.5 KiB
C++
Raw Normal View History

2013-12-19 17:10:14 +00:00
#include "types.h"
#include "cfg/cfg.h"
#if HOST_OS==OS_LINUX
#include <poll.h>
#include <termios.h>
//#include <curses.h>
#include <fcntl.h>
#include <semaphore.h>
#include <stdarg.h>
#include <signal.h>
#include <sys/param.h>
#include <sys/mman.h>
#include <sys/time.h>
#include "hw/sh4/dyna/blockmanager.h"
#include <unistd.h>
2015-07-29 04:22:59 +00:00
#if defined(TARGET_EMSCRIPTEN)
2015-08-17 21:46:28 +00:00
#include <emscripten.h>
2015-07-29 04:22:59 +00:00
#endif
2013-12-19 17:10:14 +00:00
#if defined(SUPPORT_X11)
2015-08-17 21:46:28 +00:00
#include "linux-dist/x11.h"
2013-12-19 17:10:14 +00:00
#endif
#if defined(USES_HOMEDIR)
2015-08-17 21:46:28 +00:00
#include <sys/stat.h>
2013-12-19 17:10:14 +00:00
#endif
#if defined(USE_EVDEV)
2015-08-17 21:46:28 +00:00
#include "linux-dist/evdev.h"
#endif
#if defined(USE_JOYSTICK)
2015-08-17 21:46:28 +00:00
#include "linux-dist/joystick.h"
#endif
#ifdef TARGET_PANDORA
2015-08-17 21:46:28 +00:00
#include <signal.h>
#include <execinfo.h>
2015-08-21 00:52:49 +00:00
#include <sys/soundcard.h>
#endif
2013-12-19 17:10:14 +00:00
int msgboxf(const wchar* text, unsigned int type, ...)
2013-12-19 17:10:14 +00:00
{
2015-08-17 21:46:28 +00:00
va_list args;
2013-12-19 17:10:14 +00:00
2015-08-17 21:46:28 +00:00
wchar temp[2048];
va_start(args, type);
vsprintf(temp, text, args);
va_end(args);
2013-12-19 17:10:14 +00:00
2015-08-17 21:46:28 +00:00
//printf(NULL,temp,VER_SHORTNAME,type | MB_TASKMODAL);
puts(temp);
return MBX_OK;
2013-12-19 17:10:14 +00:00
}
void* x11_win = 0;
void* x11_disp = 0;
void* libPvr_GetRenderTarget()
{
2015-08-17 21:46:28 +00:00
return x11_win;
}
void* libPvr_GetRenderSurface()
{
2015-08-17 21:46:28 +00:00
return x11_disp;
}
u16 kcode[4] = {0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF};
u8 rt[4] = {0, 0, 0, 0};
u8 lt[4] = {0, 0, 0, 0};
2013-12-19 17:10:14 +00:00
u32 vks[4];
s8 joyx[4], joyy[4];
2013-12-19 17:10:14 +00:00
void emit_WriteCodeCache();
#if defined(USE_EVDEV)
2015-08-17 21:46:28 +00:00
/* evdev input */
static EvdevController evdev_controllers[4] = {
2015-08-17 21:46:28 +00:00
{ -1, NULL },
{ -1, NULL },
{ -1, NULL },
{ -1, NULL }
};
#endif
2013-12-19 17:10:14 +00:00
#if defined(USE_JOYSTICK)
2015-08-17 21:46:28 +00:00
/* legacy joystick input */
static int joystick_fd = -1; // Joystick file descriptor
#endif
void SetupInput()
{
2015-08-17 21:46:28 +00:00
#if defined(USE_EVDEV)
int evdev_device_id[4] = { -1, -1, -1, -1 };
size_t size_needed;
int port, i;
2015-08-17 21:46:28 +00:00
char* evdev_device;
2015-08-21 00:52:49 +00:00
2015-08-17 21:46:28 +00:00
for (port = 0; port < 4; port++)
{
size_needed = snprintf(NULL, 0, EVDEV_DEVICE_CONFIG_KEY, port+1) + 1;
char* evdev_config_key = (char*)malloc(size_needed);
sprintf(evdev_config_key, EVDEV_DEVICE_CONFIG_KEY, port+1);
2015-08-17 21:46:28 +00:00
evdev_device_id[port] = cfgLoadInt("input", evdev_config_key, EVDEV_DEFAULT_DEVICE_ID(port+1));
free(evdev_config_key);
2015-08-21 00:52:49 +00:00
2015-08-17 21:46:28 +00:00
// Check if the same device is already in use on another port
if (evdev_device_id[port] < 0)
{
printf("evdev: Controller %d disabled by config.\n", port + 1);
}
else
{
for (i = 0; i < port; i++)
{
if (evdev_device_id[port] == evdev_device_id[i])
{
die("You can't assign the same device to multiple ports!\n");
}
}
size_needed = snprintf(NULL, 0, EVDEV_DEVICE_STRING, evdev_device_id[port]) + 1;
evdev_device = (char*)malloc(size_needed);
2015-08-17 21:46:28 +00:00
sprintf(evdev_device, EVDEV_DEVICE_STRING, evdev_device_id[port]);
size_needed = snprintf(NULL, 0, EVDEV_MAPPING_CONFIG_KEY, port+1) + 1;
evdev_config_key = (char*)malloc(size_needed);
sprintf(evdev_config_key, EVDEV_MAPPING_CONFIG_KEY, port+1);
const char* mapping = (cfgExists("input", evdev_config_key) == 2 ? cfgLoadStr("input", evdev_config_key, "").c_str() : NULL);
free(evdev_config_key);
input_evdev_init(&evdev_controllers[port], evdev_device, mapping);
2015-08-17 21:46:28 +00:00
free(evdev_device);
}
}
#endif
#if defined(USE_JOYSTICK)
int joystick_device_id = cfgLoadInt("input", "joystick_device_id", JOYSTICK_DEFAULT_DEVICE_ID);
if (joystick_device_id < 0) {
puts("Legacy Joystick input disabled by config.\n");
2015-08-17 21:46:28 +00:00
}
else
{
int joystick_device_length = snprintf(NULL, 0, JOYSTICK_DEVICE_STRING, joystick_device_id);
char* joystick_device = (char*)malloc(joystick_device_length + 1);
sprintf(joystick_device, JOYSTICK_DEVICE_STRING, joystick_device_id);
joystick_fd = input_joystick_init(joystick_device);
free(joystick_device);
}
#endif
#if defined(SUPPORT_X11)
input_x11_init();
#endif
2013-12-19 17:10:14 +00:00
}
void UpdateInputState(u32 port)
{
2015-08-17 21:46:28 +00:00
#if defined(TARGET_EMSCRIPTEN)
return;
#endif
2015-08-11 16:08:06 +00:00
2015-08-17 21:46:28 +00:00
#if defined(USE_JOYSTICK)
input_joystick_handle(joystick_fd, port);
#endif
2015-08-17 21:46:28 +00:00
#if defined(USE_EVDEV)
input_evdev_handle(&evdev_controllers[port], port);
2015-08-17 21:46:28 +00:00
#endif
2013-12-19 17:10:14 +00:00
}
void os_DoEvents()
{
2015-08-17 21:46:28 +00:00
#if defined(SUPPORT_X11)
input_x11_handle();
#endif
2013-12-19 17:10:14 +00:00
}
void os_SetWindowText(const char * text)
{
2015-08-17 21:46:28 +00:00
printf("%s\n",text);
#if defined(SUPPORT_X11)
x11_window_set_text(text);
#endif
2013-12-19 17:10:14 +00:00
}
void os_CreateWindow()
{
2015-08-17 21:46:28 +00:00
#if defined(SUPPORT_X11)
x11_window_create();
#endif
2013-12-19 17:10:14 +00:00
}
void common_linux_setup();
int dc_init(int argc,wchar* argv[]);
void dc_run();
2013-12-19 17:10:14 +00:00
#ifdef TARGET_PANDORA
2015-08-17 21:46:28 +00:00
void gl_term();
void clean_exit(int sig_num)
{
void* array[10];
size_t size;
if (joystick_fd >= 0) { close(joystick_fd); }
for (int port = 0; port < 4 ; port++)
{
if (evdev_controllers[port]->fd >= 0)
2015-08-17 21:46:28 +00:00
{
close(evdev_controllers[port]->fd);
2015-08-17 21:46:28 +00:00
}
}
// Close EGL context ???
if (sig_num!=0)
{
gl_term();
}
x11_window_destroy():
// finish cleaning
if (sig_num!=0)
{
write(2, "\nSignal received\n", sizeof("\nSignal received\n"));
size = backtrace(array, 10);
backtrace_symbols_fd(array, size, STDERR_FILENO);
exit(1);
}
}
#endif
string find_user_config_dir()
{
#ifdef USES_HOMEDIR
struct stat info;
string home = "";
if(getenv("HOME") != NULL)
{
// Support for the legacy config dir at "$HOME/.reicast"
string legacy_home = (string)getenv("HOME") + "/.reicast";
if((stat(legacy_home.c_str(), &info) == 0) && (info.st_mode & S_IFDIR))
{
// "$HOME/.reicast" already exists, let's use it!
return legacy_home;
}
/* If $XDG_CONFIG_HOME is not set, we're supposed to use "$HOME/.config" instead.
* Consult the XDG Base Directory Specification for details:
* http://standards.freedesktop.org/basedir-spec/basedir-spec-latest.html#variables
*/
home = (string)getenv("HOME") + "/.config/reicast";
}
if(getenv("XDG_CONFIG_HOME") != NULL)
{
// If XDG_CONFIG_HOME is set explicitly, we'll use that instead of $HOME/.config
home = (string)getenv("XDG_CONFIG_HOME") + "/reicast";
}
if(!home.empty())
{
if((stat(home.c_str(), &info) != 0) || !(info.st_mode & S_IFDIR))
{
// If the directory doesn't exist yet, create it!
mkdir(home.c_str(), 0755);
}
return home;
}
#endif
// Unable to detect config dir, use the current folder
return ".";
}
string find_user_data_dir()
{
#ifdef USES_HOMEDIR
struct stat info;
string data = "";
if(getenv("HOME") != NULL)
{
// Support for the legacy config dir at "$HOME/.reicast"
string legacy_data = (string)getenv("HOME") + "/.reicast";
if((stat(legacy_data.c_str(), &info) == 0) && (info.st_mode & S_IFDIR))
{
// "$HOME/.reicast" already exists, let's use it!
return legacy_data;
}
/* If $XDG_DATA_HOME is not set, we're supposed to use "$HOME/.local/share" instead.
* Consult the XDG Base Directory Specification for details:
* http://standards.freedesktop.org/basedir-spec/basedir-spec-latest.html#variables
*/
data = (string)getenv("HOME") + "/.local/share/reicast";
}
if(getenv("XDG_DATA_HOME") != NULL)
{
// If XDG_DATA_HOME is set explicitly, we'll use that instead of $HOME/.config
data = (string)getenv("XDG_DATA_HOME") + "/reicast";
}
if(!data.empty())
{
if((stat(data.c_str(), &info) != 0) || !(info.st_mode & S_IFDIR))
{
// If the directory doesn't exist yet, create it!
mkdir(data.c_str(), 0755);
}
return data;
}
#endif
// Unable to detect config dir, use the current folder
return ".";
}
std::vector<string> find_system_config_dirs()
{
std::vector<string> dirs;
if (getenv("XDG_DATA_DIRS") != NULL)
{
string s = (string)getenv("XDG_CONFIG_DIRS");
string::size_type pos = 0;
string::size_type n = s.find(":", pos);
while(n != std::string::npos)
{
dirs.push_back(s.substr(pos, n-pos) + "/reicast");
pos = n + 1;
n = s.find(":", pos);
}
// Separator not found
dirs.push_back(s.substr(pos) + "/reicast");
}
else
{
dirs.push_back("/etc/reicast"); // This isn't part of the XDG spec, but much more common than /etc/xdg/
dirs.push_back("/etc/xdg/reicast");
}
return dirs;
}
std::vector<string> find_system_data_dirs()
{
std::vector<string> dirs;
if (getenv("XDG_DATA_DIRS") != NULL)
{
string s = (string)getenv("XDG_DATA_DIRS");
string::size_type pos = 0;
string::size_type n = s.find(":", pos);
while(n != std::string::npos)
{
dirs.push_back(s.substr(pos, n-pos) + "/reicast");
pos = n + 1;
n = s.find(":", pos);
}
// Separator not found
dirs.push_back(s.substr(pos) + "/reicast");
}
else
{
dirs.push_back("/usr/local/share/reicast");
dirs.push_back("/usr/share/reicast");
}
return dirs;
}
2013-12-19 17:10:14 +00:00
int main(int argc, wchar* argv[])
{
2015-08-17 21:46:28 +00:00
#ifdef TARGET_PANDORA
signal(SIGSEGV, clean_exit);
signal(SIGKILL, clean_exit);
#endif
/* Set directories */
set_user_config_dir(find_user_config_dir());
set_user_data_dir(find_user_data_dir());
std::vector<string> dirs;
dirs = find_system_config_dirs();
for(unsigned int i = 0; i < dirs.size(); i++)
{
add_system_data_dir(dirs[i]);
}
dirs = find_system_data_dirs();
for(unsigned int i = 0; i < dirs.size(); i++)
{
add_system_data_dir(dirs[i]);
}
printf("Config dir is: %s\n", get_writable_config_path("/").c_str());
printf("Data dir is: %s\n", get_writable_data_path("/").c_str());
2015-08-17 21:46:28 +00:00
common_linux_setup();
settings.profile.run_counts=0;
dc_init(argc,argv);
SetupInput();
#if !defined(TARGET_EMSCRIPTEN)
dc_run();
#else
emscripten_set_main_loop(&dc_run, 100, false);
#endif
#ifdef TARGET_PANDORA
clean_exit(0);
#endif
return 0;
}
#endif
2014-03-20 15:13:53 +00:00
int get_mic_data(u8* buffer) { return 0; }
int push_vmu_screen(u8* buffer) { return 0; }
void os_DebugBreak()
{
2015-08-17 21:46:28 +00:00
#if !defined(TARGET_EMSCRIPTEN)
raise(SIGTRAP);
#else
printf("DEBUGBREAK!\n");
exit(-1);
#endif
}