|
|
@@ -28,15 +28,17 @@ OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
|
|
#if CROWN_PLATFORM_LINUX
|
|
|
|
|
|
-#include "args.h"
|
|
|
-#include "bundle_compiler.h"
|
|
|
#include "device.h"
|
|
|
-#include "json_parser.h"
|
|
|
-#include "log.h"
|
|
|
#include "memory.h"
|
|
|
#include "os_event_queue.h"
|
|
|
#include "os_window_linux.h"
|
|
|
#include "thread.h"
|
|
|
+#include "main.h"
|
|
|
+#include "command_line.h"
|
|
|
+#include "disk_filesystem.h"
|
|
|
+#include "crown.h"
|
|
|
+#include "bundle_compiler.h"
|
|
|
+#include "console_server.h"
|
|
|
#include <X11/Xutil.h>
|
|
|
#include <X11/Xatom.h>
|
|
|
#include <X11/Xlib.h>
|
|
|
@@ -48,17 +50,53 @@ OTHER DEALINGS IN THE SOFTWARE.
|
|
|
namespace crown
|
|
|
{
|
|
|
|
|
|
-//-----------------------------------------------------------------------------
|
|
|
-void init()
|
|
|
-{
|
|
|
- crown::memory::init();
|
|
|
-}
|
|
|
-
|
|
|
-//-----------------------------------------------------------------------------
|
|
|
-void shutdown()
|
|
|
-{
|
|
|
- crown::memory::shutdown();
|
|
|
-}
|
|
|
+// //-----------------------------------------------------------------------------
|
|
|
+// void display_modes(Array<DisplayMode>& modes)
|
|
|
+// {
|
|
|
+// int num_rrsizes = 0;
|
|
|
+// XRRScreenSize* rrsizes = XRRConfigSizes(m_screen_config, &num_rrsizes);
|
|
|
+
|
|
|
+// for (int i = 0; i < num_rrsizes; i++)
|
|
|
+// {
|
|
|
+// DisplayMode dm;
|
|
|
+// dm.id = (uint32_t) i;
|
|
|
+// dm.width = rrsizes[i].width;
|
|
|
+// dm.height = rrsizes[i].height;
|
|
|
+// array::push_back(modes, dm);
|
|
|
+// }
|
|
|
+// }
|
|
|
+
|
|
|
+// //-----------------------------------------------------------------------------
|
|
|
+// void set_display_mode(uint32_t id)
|
|
|
+// {
|
|
|
+// // Check if id is valid
|
|
|
+// int num_rrsizes = 0;
|
|
|
+// XRRScreenSize* rrsizes = XRRConfigSizes(m_screen_config, &num_rrsizes);
|
|
|
+// (void) rrsizes;
|
|
|
+
|
|
|
+// if ((int) id >= num_rrsizes)
|
|
|
+// return;
|
|
|
+
|
|
|
+// XRRSetScreenConfig(m_x11_display,
|
|
|
+// m_screen_config,
|
|
|
+// RootWindow(m_x11_display, DefaultScreen(m_x11_display)),
|
|
|
+// (int) id,
|
|
|
+// RR_Rotate_0,
|
|
|
+// CurrentTime);
|
|
|
+// }
|
|
|
+
|
|
|
+// //-----------------------------------------------------------------------------
|
|
|
+// void set_fullscreen(bool full)
|
|
|
+// {
|
|
|
+// XEvent e;
|
|
|
+// e.xclient.type = ClientMessage;
|
|
|
+// e.xclient.window = m_x11_window;
|
|
|
+// e.xclient.message_type = XInternAtom(m_x11_display, "_NET_WM_STATE", False );
|
|
|
+// e.xclient.format = 32;
|
|
|
+// e.xclient.data.l[0] = full ? 1 : 0;
|
|
|
+// e.xclient.data.l[1] = XInternAtom(m_x11_display, "_NET_WM_STATE_FULLSCREEN", False);
|
|
|
+// XSendEvent(m_x11_display, DefaultRootWindow(m_x11_display), False, SubstructureNotifyMask, &e);
|
|
|
+// }
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
static KeyboardButton::Enum x11_translate_key(KeySym x11_key)
|
|
|
@@ -117,140 +155,46 @@ static KeyboardButton::Enum x11_translate_key(KeySym x11_key)
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-struct MainArgs
|
|
|
+static int x11_error_handler(Display* dpy, XErrorEvent* ev)
|
|
|
+{
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+static bool s_exit = false;
|
|
|
+
|
|
|
+struct MainThreadArgs
|
|
|
{
|
|
|
- int argc;
|
|
|
- char** argv;
|
|
|
- class LinuxDevice* device;
|
|
|
+ Filesystem* fs;
|
|
|
+ ConfigSettings* cs;
|
|
|
};
|
|
|
|
|
|
-class LinuxDevice : public Device
|
|
|
+int32_t func(void* data)
|
|
|
{
|
|
|
-public:
|
|
|
+ MainThreadArgs* args = (MainThreadArgs*) data;
|
|
|
+ crown::init(*args->fs, *args->cs);
|
|
|
+ crown::update();
|
|
|
+ crown::shutdown();
|
|
|
+ s_exit = true;
|
|
|
+ return EXIT_SUCCESS;
|
|
|
+}
|
|
|
|
|
|
- //-----------------------------------------------------------------------------
|
|
|
+struct LinuxDevice
|
|
|
+{
|
|
|
LinuxDevice()
|
|
|
: m_x11_display(NULL)
|
|
|
, m_x11_window(None)
|
|
|
, m_x11_parent_window(None)
|
|
|
, m_x11_hidden_cursor(None)
|
|
|
, m_screen_config(NULL)
|
|
|
- , m_exit(false)
|
|
|
- , m_x(0)
|
|
|
- , m_y(0)
|
|
|
- , m_width(1000)
|
|
|
- , m_height(625)
|
|
|
, m_parent_window_handle(0)
|
|
|
- , m_fullscreen(0)
|
|
|
- , m_compile(0)
|
|
|
- , m_continue(0)
|
|
|
- , m_wait_console(0)
|
|
|
- , m_platform("linux")
|
|
|
- {
|
|
|
- }
|
|
|
-
|
|
|
- //-----------------------------------------------------------------------------
|
|
|
- void init(int argc, char** argv)
|
|
|
- {
|
|
|
- parse_command_line(argc, argv);
|
|
|
- check_preferred_settings();
|
|
|
-
|
|
|
- #if defined(CROWN_DEBUG)
|
|
|
- m_console = CE_NEW(default_allocator(), ConsoleServer)();
|
|
|
- m_console->init(m_console_port, (bool) m_wait_console);
|
|
|
-
|
|
|
- if (m_compile == 1)
|
|
|
- {
|
|
|
- m_bundle_compiler = CE_NEW(default_allocator(), BundleCompiler);
|
|
|
- if (!m_bundle_compiler->compile(m_bundle_dir, m_source_dir, m_platform))
|
|
|
- {
|
|
|
- CE_DELETE(default_allocator(), m_bundle_compiler);
|
|
|
- CE_LOGE("Exiting.");
|
|
|
- exit(EXIT_FAILURE);
|
|
|
- }
|
|
|
-
|
|
|
- if (!m_continue)
|
|
|
- {
|
|
|
- CE_DELETE(default_allocator(), m_bundle_compiler);
|
|
|
- exit(EXIT_SUCCESS);
|
|
|
- }
|
|
|
- }
|
|
|
- #endif
|
|
|
-
|
|
|
- read_configuration();
|
|
|
-
|
|
|
- m_argc = argc;
|
|
|
- m_argv = argv;
|
|
|
- }
|
|
|
-
|
|
|
- //-----------------------------------------------------------------------------
|
|
|
- void shutdown()
|
|
|
- {
|
|
|
- #if defined(CROWN_DEBUG)
|
|
|
- CE_DELETE(default_allocator(), m_bundle_compiler);
|
|
|
-
|
|
|
- m_console->shutdown();
|
|
|
- CE_DELETE(default_allocator(), m_console);
|
|
|
- #endif
|
|
|
- }
|
|
|
-
|
|
|
- //-----------------------------------------------------------------------------
|
|
|
- void display_modes(Array<DisplayMode>& modes)
|
|
|
- {
|
|
|
- int num_rrsizes = 0;
|
|
|
- XRRScreenSize* rrsizes = XRRConfigSizes(m_screen_config, &num_rrsizes);
|
|
|
-
|
|
|
- for (int i = 0; i < num_rrsizes; i++)
|
|
|
- {
|
|
|
- DisplayMode dm;
|
|
|
- dm.id = (uint32_t) i;
|
|
|
- dm.width = rrsizes[i].width;
|
|
|
- dm.height = rrsizes[i].height;
|
|
|
- array::push_back(modes, dm);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- //-----------------------------------------------------------------------------
|
|
|
- void set_display_mode(uint32_t id)
|
|
|
- {
|
|
|
- // Check if id is valid
|
|
|
- int num_rrsizes = 0;
|
|
|
- XRRScreenSize* rrsizes = XRRConfigSizes(m_screen_config, &num_rrsizes);
|
|
|
- (void) rrsizes;
|
|
|
-
|
|
|
- if ((int) id >= num_rrsizes)
|
|
|
- return;
|
|
|
-
|
|
|
- XRRSetScreenConfig(m_x11_display,
|
|
|
- m_screen_config,
|
|
|
- RootWindow(m_x11_display, DefaultScreen(m_x11_display)),
|
|
|
- (int) id,
|
|
|
- RR_Rotate_0,
|
|
|
- CurrentTime);
|
|
|
- }
|
|
|
-
|
|
|
- //-----------------------------------------------------------------------------
|
|
|
- void set_fullscreen(bool full)
|
|
|
{
|
|
|
- XEvent e;
|
|
|
- e.xclient.type = ClientMessage;
|
|
|
- e.xclient.window = m_x11_window;
|
|
|
- e.xclient.message_type = XInternAtom(m_x11_display, "_NET_WM_STATE", False );
|
|
|
- e.xclient.format = 32;
|
|
|
- e.xclient.data.l[0] = full ? 1 : 0;
|
|
|
- e.xclient.data.l[1] = XInternAtom(m_x11_display, "_NET_WM_STATE_FULLSCREEN", False);
|
|
|
-
|
|
|
- XSendEvent(m_x11_display, DefaultRootWindow(m_x11_display), False, SubstructureNotifyMask, &e);
|
|
|
}
|
|
|
|
|
|
- //-----------------------------------------------------------------------------
|
|
|
- int32_t run(int argc, char** argv)
|
|
|
+ int32_t run(Filesystem* fs, ConfigSettings* cs)
|
|
|
{
|
|
|
- init(argc, argv);
|
|
|
-
|
|
|
+ // Create main window
|
|
|
XInitThreads();
|
|
|
- CE_ASSERT(m_width != 0 || m_height != 0, "Width and height must differ from zero");
|
|
|
-
|
|
|
+ XSetErrorHandler(x11_error_handler);
|
|
|
m_x11_display = XOpenDisplay(NULL);
|
|
|
|
|
|
CE_ASSERT(m_x11_display != NULL, "Unable to open X11 display");
|
|
|
@@ -262,29 +206,33 @@ public:
|
|
|
m_x11_parent_window = (m_parent_window_handle == 0) ? RootWindow(m_x11_display, screen) :
|
|
|
(Window) m_parent_window_handle;
|
|
|
|
|
|
- // We want to track keyboard and mouse events
|
|
|
+ // Create main window
|
|
|
XSetWindowAttributes win_attribs;
|
|
|
win_attribs.background_pixmap = 0;
|
|
|
win_attribs.border_pixel = 0;
|
|
|
- win_attribs.event_mask = FocusChangeMask | StructureNotifyMask | KeyPressMask |
|
|
|
- KeyReleaseMask | ButtonPressMask | ButtonReleaseMask | PointerMotionMask;
|
|
|
-
|
|
|
- m_x11_window = XCreateWindow(
|
|
|
- m_x11_display,
|
|
|
- m_x11_parent_window,
|
|
|
- 0, 0,
|
|
|
- m_width, m_height,
|
|
|
- 0,
|
|
|
- depth,
|
|
|
- InputOutput,
|
|
|
- visual,
|
|
|
- CWBorderPixel | CWEventMask,
|
|
|
- &win_attribs
|
|
|
- );
|
|
|
-
|
|
|
+ win_attribs.event_mask = FocusChangeMask
|
|
|
+ | StructureNotifyMask
|
|
|
+ | KeyPressMask
|
|
|
+ | KeyReleaseMask
|
|
|
+ | ButtonPressMask
|
|
|
+ | ButtonReleaseMask
|
|
|
+ | PointerMotionMask;
|
|
|
+
|
|
|
+ m_x11_window = XCreateWindow(m_x11_display,
|
|
|
+ m_x11_parent_window,
|
|
|
+ 0, 0,
|
|
|
+ math::max((uint16_t) 1, cs->window_width),
|
|
|
+ math::max((uint16_t) 1, cs->window_height),
|
|
|
+ 0,
|
|
|
+ depth,
|
|
|
+ InputOutput,
|
|
|
+ visual,
|
|
|
+ CWBorderPixel | CWEventMask,
|
|
|
+ &win_attribs
|
|
|
+ );
|
|
|
CE_ASSERT(m_x11_window != None, "Unable to create X window");
|
|
|
|
|
|
- // Check presence of detectable autorepeat
|
|
|
+ // Do we have detectable autorepeat?
|
|
|
Bool detectable;
|
|
|
m_x11_detectable_autorepeat = (bool) XkbSetDetectableAutoRepeat(m_x11_display, true, &detectable);
|
|
|
|
|
|
@@ -302,10 +250,9 @@ public:
|
|
|
m_wm_delete_message = XInternAtom(m_x11_display, "WM_DELETE_WINDOW", False);
|
|
|
XSetWMProtocols(m_x11_display, m_x11_window, &m_wm_delete_message, 1);
|
|
|
|
|
|
- XMapRaised(m_x11_display, m_x11_window);
|
|
|
-
|
|
|
oswindow_set_window(m_x11_display, m_x11_window);
|
|
|
bgfx::x11SetDisplayWindow(m_x11_display, m_x11_window);
|
|
|
+ XMapRaised(m_x11_display, m_x11_window);
|
|
|
|
|
|
// Get screen configuration
|
|
|
m_screen_config = XRRGetScreenInfo(m_x11_display, RootWindow(m_x11_display, screen));
|
|
|
@@ -313,19 +260,20 @@ public:
|
|
|
Rotation rr_old_rot;
|
|
|
const SizeID rr_old_sizeid = XRRConfigCurrentConfiguration(m_screen_config, &rr_old_rot);
|
|
|
|
|
|
- Thread game_thread;
|
|
|
- game_thread.start(main_loop, (void*)this);
|
|
|
+ // Start main thread
|
|
|
+ MainThreadArgs mta;
|
|
|
+ mta.fs = fs;
|
|
|
+ mta.cs = cs;
|
|
|
|
|
|
- // Create window
|
|
|
- CE_LOGD("Creating main window...");
|
|
|
- m_window = CE_NEW(m_allocator, OsWindow);
|
|
|
+ Thread main_thread;
|
|
|
+ main_thread.start(func, &mta);
|
|
|
|
|
|
- while (!m_exit)
|
|
|
+ while (!s_exit)
|
|
|
{
|
|
|
- LinuxDevice::pump_events();
|
|
|
+ pump_events();
|
|
|
}
|
|
|
|
|
|
- game_thread.stop();
|
|
|
+ main_thread.stop();
|
|
|
|
|
|
// Restore previous screen configuration if changed
|
|
|
Rotation rr_cur_rot;
|
|
|
@@ -334,109 +282,19 @@ public:
|
|
|
if (rr_cur_rot != rr_old_rot || rr_cur_sizeid != rr_old_sizeid)
|
|
|
{
|
|
|
XRRSetScreenConfig(m_x11_display,
|
|
|
- m_screen_config,
|
|
|
- RootWindow(m_x11_display, screen),
|
|
|
- rr_old_sizeid,
|
|
|
- rr_old_rot,
|
|
|
- CurrentTime);
|
|
|
+ m_screen_config,
|
|
|
+ RootWindow(m_x11_display, screen),
|
|
|
+ rr_old_sizeid,
|
|
|
+ rr_old_rot,
|
|
|
+ CurrentTime);
|
|
|
}
|
|
|
XRRFreeScreenConfigInfo(m_screen_config);
|
|
|
|
|
|
- LinuxDevice::shutdown();
|
|
|
XDestroyWindow(m_x11_display, m_x11_window);
|
|
|
XCloseDisplay(m_x11_display);
|
|
|
-
|
|
|
return EXIT_SUCCESS;
|
|
|
}
|
|
|
|
|
|
- //-----------------------------------------------------------------------------
|
|
|
- int32_t loop()
|
|
|
- {
|
|
|
- Device::init();
|
|
|
-
|
|
|
- while(!process_events() && is_running())
|
|
|
- {
|
|
|
- #if defined(CROWN_DEBUG)
|
|
|
- m_console->update();
|
|
|
- #endif
|
|
|
-
|
|
|
- Device::frame();
|
|
|
-
|
|
|
- m_keyboard->update();
|
|
|
- m_mouse->update();
|
|
|
- }
|
|
|
-
|
|
|
- Device::shutdown();
|
|
|
-
|
|
|
- m_exit = true;
|
|
|
-
|
|
|
- return 0;
|
|
|
- }
|
|
|
-
|
|
|
- //-----------------------------------------------------------------------------
|
|
|
- static int32_t main_loop(void* thiz)
|
|
|
- {
|
|
|
- return ((LinuxDevice*)thiz)->loop();
|
|
|
- }
|
|
|
-
|
|
|
- //-----------------------------------------------------------------------------
|
|
|
- bool process_events()
|
|
|
- {
|
|
|
- OsEvent event;
|
|
|
- bool exit = false;
|
|
|
-
|
|
|
- while(m_queue.pop_event(event))
|
|
|
- {
|
|
|
- if (event.type == OsEvent::NONE) continue;
|
|
|
-
|
|
|
- switch (event.type)
|
|
|
- {
|
|
|
- case OsEvent::MOUSE:
|
|
|
- {
|
|
|
- const OsMouseEvent& ev = event.mouse;
|
|
|
- switch (ev.type)
|
|
|
- {
|
|
|
- case OsMouseEvent::BUTTON: m_mouse->set_button_state(ev.x, ev.y, ev.button, ev.pressed); break;
|
|
|
- case OsMouseEvent::MOVE: m_mouse->set_position(ev.x, ev.y); break;
|
|
|
- default: CE_FATAL("Oops, unknown mouse event type"); break;
|
|
|
- }
|
|
|
-
|
|
|
- break;
|
|
|
- }
|
|
|
- case OsEvent::KEYBOARD:
|
|
|
- {
|
|
|
- const OsKeyboardEvent& ev = event.keyboard;
|
|
|
- m_keyboard->set_button_state(ev.button, ev.pressed);
|
|
|
- break;
|
|
|
- }
|
|
|
- case OsEvent::METRICS:
|
|
|
- {
|
|
|
- const OsMetricsEvent& ev = event.metrics;
|
|
|
- m_mouse->set_metrics(ev.width, ev.height);
|
|
|
- m_window->m_x = ev.x;
|
|
|
- m_window->m_y = ev.y;
|
|
|
- m_window->m_width = ev.width;
|
|
|
- m_window->m_height = ev.height;
|
|
|
-
|
|
|
- break;
|
|
|
- }
|
|
|
- case OsEvent::EXIT:
|
|
|
- {
|
|
|
- exit = true;
|
|
|
- break;
|
|
|
- }
|
|
|
- default:
|
|
|
- {
|
|
|
- CE_FATAL("Unknown Os Event");
|
|
|
- break;
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- return exit;
|
|
|
- }
|
|
|
-
|
|
|
- //-----------------------------------------------------------------------------
|
|
|
void pump_events()
|
|
|
{
|
|
|
while (XPending(m_x11_display))
|
|
|
@@ -456,14 +314,8 @@ public:
|
|
|
}
|
|
|
case ConfigureNotify:
|
|
|
{
|
|
|
- m_x = event.xconfigure.x;
|
|
|
- m_y = event.xconfigure.y;
|
|
|
- m_width = event.xconfigure.width;
|
|
|
- m_height = event.xconfigure.height;
|
|
|
-
|
|
|
m_queue.push_metrics_event(event.xconfigure.x, event.xconfigure.y,
|
|
|
- event.xconfigure.width, event.xconfigure.height);
|
|
|
-
|
|
|
+ event.xconfigure.width, event.xconfigure.height);
|
|
|
break;
|
|
|
}
|
|
|
case ButtonPress:
|
|
|
@@ -517,20 +369,6 @@ public:
|
|
|
}
|
|
|
|
|
|
m_queue.push_keyboard_event(modifier_mask, kb, event.type == KeyPress);
|
|
|
-
|
|
|
- // // Text input part
|
|
|
- // if (event.type == KeyPress && len > 0)
|
|
|
- // {
|
|
|
- // //crownEvent.event_type = ET_TEXT;m_queue
|
|
|
- // //crownEvent.text.type = TET_TEXT_INPUT;
|
|
|
- // strncpy(keyboardEvent.text, string, 4);
|
|
|
-
|
|
|
- // if (mListener)
|
|
|
- // {
|
|
|
- // mListener->TextInput(keyboardEvent);
|
|
|
- // }
|
|
|
- // }
|
|
|
-
|
|
|
break;
|
|
|
}
|
|
|
case KeymapNotify:
|
|
|
@@ -546,234 +384,58 @@ public:
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- //-----------------------------------------------------------------------------
|
|
|
- void parse_command_line(int argc, char** argv)
|
|
|
- {
|
|
|
- static const char* help_message =
|
|
|
- "Usage: crown [options]\n"
|
|
|
- "Options:\n\n"
|
|
|
-
|
|
|
- "All of the following options take precedence over\n"
|
|
|
- "environment variables and configuration files.\n\n"
|
|
|
-
|
|
|
- " --help Show this help.\n"
|
|
|
- " --bundle-dir <path> Use <path> as the source directory for compiled resources.\n"
|
|
|
- " --width <width> Set the <width> of the main window.\n"
|
|
|
- " --height <width> Set the <height> of the main window.\n"
|
|
|
- " --fullscreen Start in fullscreen.\n"
|
|
|
- " --parent-window <handle> Set the parent window <handle> of the main window.\n"
|
|
|
- " Used only by tools.\n"
|
|
|
-
|
|
|
- "\nAvailable only in debug and development builds:\n\n"
|
|
|
-
|
|
|
- " --source-dir <path> Use <path> as the source directory for resource compilation.\n"
|
|
|
- " --compile Do a full compile of the resources.\n"
|
|
|
- " --platform <platform> Compile resources for the given <platform>.\n"
|
|
|
- " Possible values for <platform> are:\n"
|
|
|
- " linux\n"
|
|
|
- " android\n"
|
|
|
- " windows\n"
|
|
|
- " --continue Continue the execution after the resource compilation step.\n"
|
|
|
- " --file-server Read resources from a remote engine instance.\n"
|
|
|
- " --console-port Set the network port of the console server.\n"
|
|
|
- " --wait-console Wait for a console connection before starting up.\n";
|
|
|
-
|
|
|
- static ArgsOption options[] =
|
|
|
- {
|
|
|
- { "help", AOA_NO_ARGUMENT, NULL, 'i' },
|
|
|
- { "source-dir", AOA_REQUIRED_ARGUMENT, NULL, 's' },
|
|
|
- { "bundle-dir", AOA_REQUIRED_ARGUMENT, NULL, 'b' },
|
|
|
- { "compile", AOA_NO_ARGUMENT, &m_compile, 1 },
|
|
|
- { "platform", AOA_REQUIRED_ARGUMENT, NULL, 'r' },
|
|
|
- { "continue", AOA_NO_ARGUMENT, &m_continue, 1 },
|
|
|
- { "width", AOA_REQUIRED_ARGUMENT, NULL, 'w' },
|
|
|
- { "height", AOA_REQUIRED_ARGUMENT, NULL, 'h' },
|
|
|
- { "fullscreen", AOA_NO_ARGUMENT, &m_fullscreen, 1 },
|
|
|
- { "parent-window", AOA_REQUIRED_ARGUMENT, NULL, 'p' },
|
|
|
- { "file-server", AOA_NO_ARGUMENT, &m_fileserver, 1 },
|
|
|
- { "console-port", AOA_REQUIRED_ARGUMENT, NULL, 'c' },
|
|
|
- { "wait-console", AOA_NO_ARGUMENT, &m_wait_console, 1 },
|
|
|
- { NULL, 0, NULL, 0 }
|
|
|
- };
|
|
|
-
|
|
|
- Args args(argc, argv, "", options);
|
|
|
-
|
|
|
- int32_t opt;
|
|
|
- while ((opt = args.getopt()) != -1)
|
|
|
- {
|
|
|
- switch (opt)
|
|
|
- {
|
|
|
- case 0:
|
|
|
- {
|
|
|
- break;
|
|
|
- }
|
|
|
- // Source directory
|
|
|
- case 's':
|
|
|
- {
|
|
|
- string::strncpy(m_source_dir, args.optarg(), MAX_PATH_LENGTH);
|
|
|
- break;
|
|
|
- }
|
|
|
- // Bundle directory
|
|
|
- case 'b':
|
|
|
- {
|
|
|
- string::strncpy(m_bundle_dir, args.optarg(), MAX_PATH_LENGTH);
|
|
|
- break;
|
|
|
- }
|
|
|
- // Window width
|
|
|
- case 'w':
|
|
|
- {
|
|
|
- m_width = atoi(args.optarg());
|
|
|
- break;
|
|
|
- }
|
|
|
- // Window height
|
|
|
- case 'h':
|
|
|
- {
|
|
|
- m_height = atoi(args.optarg());
|
|
|
- break;
|
|
|
- }
|
|
|
- // Parent window
|
|
|
- case 'p':
|
|
|
- {
|
|
|
- m_parent_window_handle = string::parse_uint(args.optarg());
|
|
|
- break;
|
|
|
- }
|
|
|
- // Console port
|
|
|
- case 'c':
|
|
|
- {
|
|
|
- m_console_port = string::parse_uint(args.optarg());
|
|
|
- break;
|
|
|
- }
|
|
|
- // Platform
|
|
|
- case 'r':
|
|
|
- {
|
|
|
- m_platform = args.optarg();
|
|
|
- break;
|
|
|
- }
|
|
|
- case 'i':
|
|
|
- case '?':
|
|
|
- default:
|
|
|
- {
|
|
|
- printf(help_message);
|
|
|
- exit(EXIT_FAILURE);
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- //-----------------------------------------------------------------------------
|
|
|
- void check_preferred_settings()
|
|
|
- {
|
|
|
- if (m_compile == 1)
|
|
|
- {
|
|
|
- if (string::strcmp(m_source_dir, "") == 0)
|
|
|
- {
|
|
|
- CE_LOGE("You have to specify the source directory when running in compile mode.");
|
|
|
- exit(EXIT_FAILURE);
|
|
|
- }
|
|
|
-
|
|
|
- if (!os::is_absolute_path(m_source_dir))
|
|
|
- {
|
|
|
- CE_LOGE("The source directory must be absolute.");
|
|
|
- exit(EXIT_FAILURE);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- if (!os::is_absolute_path(m_bundle_dir))
|
|
|
- {
|
|
|
- CE_LOGE("The bundle directory must be absolute.");
|
|
|
- exit(EXIT_FAILURE);
|
|
|
- }
|
|
|
-
|
|
|
- if (m_width == 0 || m_height == 0)
|
|
|
- {
|
|
|
- CE_LOGE("Window width and height must be greater than zero.");
|
|
|
- exit(EXIT_FAILURE);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- //-------------------------------------------------------------------------
|
|
|
- void read_configuration()
|
|
|
- {
|
|
|
- DiskFilesystem fs(m_bundle_dir);
|
|
|
-
|
|
|
- // crown.config is mandatory
|
|
|
- CE_ASSERT(fs.is_file("crown.config"), "Unable to open crown.config");
|
|
|
-
|
|
|
- File* config_file = fs.open("crown.config", FOM_READ);
|
|
|
-
|
|
|
- TempAllocator4096 alloc;
|
|
|
- char* json_string = (char*)alloc.allocate(config_file->size());
|
|
|
- config_file->read(json_string, config_file->size());
|
|
|
- fs.close(config_file);
|
|
|
-
|
|
|
- // Parse crown.config
|
|
|
- JSONParser parser(json_string);
|
|
|
- JSONElement root = parser.root();
|
|
|
-
|
|
|
- // Boot
|
|
|
- if (root.has_key("boot"))
|
|
|
- {
|
|
|
- DynamicString boot;
|
|
|
- root.key("boot").to_string(boot);
|
|
|
-
|
|
|
- string::strncpy(m_boot_file, boot.c_str(), (boot.length() > MAX_PATH_LENGTH) ? MAX_PATH_LENGTH : boot.length() + 1);
|
|
|
- }
|
|
|
-
|
|
|
- // Window width
|
|
|
- if (root.has_key("window_width"))
|
|
|
- {
|
|
|
- m_width = root.key("window_width").to_int();
|
|
|
- }
|
|
|
-
|
|
|
- // Window height
|
|
|
- if (root.has_key("window_height"))
|
|
|
- {
|
|
|
- m_height = root.key("window_height").to_int();
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
-private:
|
|
|
+public:
|
|
|
|
|
|
Display* m_x11_display;
|
|
|
Window m_x11_window;
|
|
|
Window m_x11_parent_window;
|
|
|
Cursor m_x11_hidden_cursor;
|
|
|
Atom m_wm_delete_message;
|
|
|
-
|
|
|
XRRScreenConfiguration* m_screen_config;
|
|
|
-
|
|
|
- bool m_exit;
|
|
|
- uint32_t m_x;
|
|
|
- uint32_t m_y;
|
|
|
- uint32_t m_width;
|
|
|
- uint32_t m_height;
|
|
|
bool m_x11_detectable_autorepeat;
|
|
|
-
|
|
|
uint32_t m_parent_window_handle;
|
|
|
- int32_t m_fullscreen;
|
|
|
- int32_t m_compile;
|
|
|
- int32_t m_continue;
|
|
|
- int32_t m_wait_console;
|
|
|
- const char* m_platform;
|
|
|
-
|
|
|
OsEventQueue m_queue;
|
|
|
};
|
|
|
|
|
|
+static LinuxDevice s_ldvc;
|
|
|
+
|
|
|
+bool next_event(OsEvent& ev)
|
|
|
+{
|
|
|
+ return s_ldvc.m_queue.pop_event(ev);
|
|
|
+}
|
|
|
+
|
|
|
} // namespace crown
|
|
|
|
|
|
int main(int argc, char** argv)
|
|
|
{
|
|
|
- crown::init();
|
|
|
+ using namespace crown;
|
|
|
|
|
|
- crown::LinuxDevice* engine = CE_NEW(crown::default_allocator(), crown::LinuxDevice)();
|
|
|
- crown::set_device(engine);
|
|
|
+ CommandLineSettings cls = parse_command_line(argc, argv);
|
|
|
|
|
|
- int32_t ret = engine->run(argc, argv);
|
|
|
+ memory_globals::init();
|
|
|
+ DiskFilesystem src_fs(cls.source_dir);
|
|
|
+ ConfigSettings cs = parse_config_file(src_fs);
|
|
|
|
|
|
- CE_DELETE(crown::default_allocator(), engine);
|
|
|
+ console_server_globals::init();
|
|
|
+ console_server_globals::console().init(cs.console_port, cls.wait_console);
|
|
|
|
|
|
- crown::shutdown();
|
|
|
- return ret;
|
|
|
+ bundle_compiler_globals::init();
|
|
|
+
|
|
|
+ bool do_continue = true;
|
|
|
+ int exitcode = EXIT_SUCCESS;
|
|
|
+
|
|
|
+ do_continue = bundle_compiler::main(cls);
|
|
|
+
|
|
|
+ if (do_continue)
|
|
|
+ {
|
|
|
+ DiskFilesystem dst_fs(cls.bundle_dir);
|
|
|
+ exitcode = crown::s_ldvc.run(&dst_fs, &cs);
|
|
|
+ }
|
|
|
+
|
|
|
+ bundle_compiler_globals::shutdown();
|
|
|
+ console_server_globals::shutdown();
|
|
|
+ memory_globals::shutdown();
|
|
|
+ return exitcode;
|
|
|
}
|
|
|
|
|
|
#endif // CROWN_PLATFORM_LINUX
|