2015-11-16 16:15:45 +00:00
|
|
|
/*
|
|
|
|
* Mr. 4th Dimention - Allen Webster
|
|
|
|
*
|
|
|
|
* 14.11.2015
|
|
|
|
*
|
|
|
|
* Linux layer for project codename "4ed"
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
// TOP
|
|
|
|
|
2016-02-01 05:03:42 +00:00
|
|
|
#include "4ed_config.h"
|
2015-11-16 16:15:45 +00:00
|
|
|
|
|
|
|
#include "4ed_meta.h"
|
|
|
|
|
|
|
|
#define FCPP_FORBID_MALLOC
|
|
|
|
|
|
|
|
#include "4cpp_types.h"
|
|
|
|
#define FCPP_STRING_IMPLEMENTATION
|
2016-02-01 05:03:42 +00:00
|
|
|
#include "4coder_string.h"
|
2015-11-16 16:15:45 +00:00
|
|
|
|
|
|
|
#include "4ed_mem.cpp"
|
|
|
|
#include "4ed_math.cpp"
|
2016-02-01 05:03:42 +00:00
|
|
|
|
2015-11-16 16:15:45 +00:00
|
|
|
#include "4coder_custom.h"
|
|
|
|
#include "4ed_system.h"
|
|
|
|
#include "4ed_rendering.h"
|
2016-02-01 05:03:42 +00:00
|
|
|
#include "4ed.h"
|
2015-11-16 16:15:45 +00:00
|
|
|
|
2015-12-01 02:51:53 +00:00
|
|
|
#include <stdio.h>
|
2016-02-01 05:03:42 +00:00
|
|
|
#include <memory.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <sys/time.h>
|
2016-02-07 15:19:18 +00:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <dirent.h>
|
2016-02-01 05:03:42 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <X11/Xlib.h>
|
|
|
|
#include <GL/glx.h>
|
|
|
|
#include <xmmintrin.h>
|
|
|
|
#include <linux/fs.h>
|
|
|
|
#include <X11/extensions/XInput2.h>
|
|
|
|
#include <linux/input.h>
|
2016-02-07 15:19:18 +00:00
|
|
|
#include <time.h>
|
2016-02-11 17:17:52 +00:00
|
|
|
#include <dlfcn.h>
|
|
|
|
#include <unistd.h>
|
2015-12-01 02:51:53 +00:00
|
|
|
|
2015-11-16 16:15:45 +00:00
|
|
|
#include "4ed_internal.h"
|
|
|
|
#include "4ed_linux_keyboard.cpp"
|
2016-02-11 17:17:52 +00:00
|
|
|
#include "system_shared.h"
|
2015-11-16 16:15:45 +00:00
|
|
|
|
2016-02-07 15:19:18 +00:00
|
|
|
#include <stdlib.h>
|
2015-11-16 16:15:45 +00:00
|
|
|
|
2016-02-11 17:17:52 +00:00
|
|
|
struct Linux_Vars{
|
|
|
|
void *app_code;
|
|
|
|
void *custom;
|
|
|
|
|
|
|
|
Plat_Settings settings;
|
|
|
|
System_Functions *system;
|
|
|
|
App_Functions app;
|
|
|
|
Custom_API custom_api;
|
|
|
|
b32 first;
|
|
|
|
|
|
|
|
#if FRED_INTERNAL
|
|
|
|
Sys_Bubble internal_bubble;
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
|
|
|
globalvar Linux_Vars linuxvars;
|
|
|
|
globalvar Application_Memory memory_vars;
|
|
|
|
globalvar Exchange exchange_vars;
|
|
|
|
|
|
|
|
internal
|
|
|
|
Sys_Get_Memory_Sig(system_get_memory_){
|
2016-02-07 15:19:18 +00:00
|
|
|
// TODO(allen): Implement without stdlib.h
|
2016-02-11 17:17:52 +00:00
|
|
|
void *result = 0;
|
|
|
|
|
|
|
|
if (size != 0){
|
|
|
|
#if FRED_INTERNAL
|
|
|
|
Sys_Bubble *bubble;
|
|
|
|
|
|
|
|
result = malloc(size + sizeof(Sys_Bubble));
|
|
|
|
bubble = (Sys_Bubble*)result;
|
|
|
|
bubble->flags = MEM_BUBBLE_SYS_DEBUG;
|
|
|
|
bubble->line_number = line_number;
|
|
|
|
bubble->file_name = file_name;
|
|
|
|
bubble->size = size;
|
|
|
|
|
|
|
|
// TODO(allen): make Sys_Bubble list thread safe
|
|
|
|
insert_bubble(&linuxvars.internal_bubble, bubble);
|
|
|
|
result = bubble + 1;
|
|
|
|
|
|
|
|
#else
|
|
|
|
result = malloc(size);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
return(result);
|
2016-02-07 15:19:18 +00:00
|
|
|
}
|
|
|
|
|
2016-02-11 17:17:52 +00:00
|
|
|
internal
|
|
|
|
Sys_Free_Memory_Sig(system_free_memory){
|
2016-02-07 15:19:18 +00:00
|
|
|
// TODO(allen): Implement without stdlib.h
|
2016-02-11 17:17:52 +00:00
|
|
|
|
|
|
|
if (block){
|
|
|
|
#if FRED_INTERNAL
|
|
|
|
Sys_Bubble *bubble;
|
|
|
|
|
|
|
|
bubble = (Sys_Bubble*)block;
|
|
|
|
--bubble;
|
|
|
|
Assert((bubble->flags & MEM_BUBBLE_DEBUG_MASK) == MEM_BUBBLE_SYS_DEBUG);
|
|
|
|
|
|
|
|
// TODO(allen): make Sys_Bubble list thread safe
|
|
|
|
remove_bubble(bubble);
|
|
|
|
|
|
|
|
free(bubble);
|
|
|
|
#else
|
|
|
|
free(block);
|
|
|
|
#endif
|
|
|
|
}
|
2016-02-07 15:19:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#if (defined(_BSD_SOURCE) || defined(_SVID_SOURCE))
|
|
|
|
#define TimeBySt
|
|
|
|
#endif
|
|
|
|
#if (_POSIX_C_SOURCE >= 200809L || _XOPEN_SOURCE >= 700)
|
|
|
|
#define TimeBySt
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef TimeBySt
|
|
|
|
#define nano_mtime_field st_mtim.tv_nsec
|
|
|
|
#undef TimeBySt
|
|
|
|
#else
|
|
|
|
#define nano_mtime_field st_mtimensec
|
|
|
|
#endif
|
|
|
|
|
|
|
|
Sys_File_Time_Stamp_Sig(system_file_time_stamp){
|
|
|
|
struct stat info;
|
|
|
|
i64 nanosecond_timestamp;
|
|
|
|
u64 result;
|
|
|
|
|
|
|
|
stat(filename, &info);
|
|
|
|
nanosecond_timestamp = info.nano_mtime_field;
|
|
|
|
if (nanosecond_timestamp != 0){
|
|
|
|
result = (u64)(nanosecond_timestamp / 1000);
|
|
|
|
}
|
|
|
|
else{
|
|
|
|
result = (u64)(info.st_mtime * 1000000);
|
|
|
|
}
|
|
|
|
|
|
|
|
return(result);
|
|
|
|
}
|
2015-11-16 16:15:45 +00:00
|
|
|
|
2016-02-11 17:17:52 +00:00
|
|
|
// TODO(allen): DOES THIS AGREE WITH THE FILESTAMP TIMES?
|
2016-02-07 15:19:18 +00:00
|
|
|
Sys_Time_Sig(system_time){
|
|
|
|
struct timespec spec;
|
|
|
|
u64 result;
|
|
|
|
|
|
|
|
clock_gettime(CLOCK_MONOTONIC, &spec);
|
|
|
|
result = (spec.tv_sec * 1000000) + (spec.tv_nsec / 1000);
|
|
|
|
|
|
|
|
return(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
Sys_Set_File_List_Sig(system_set_file_list){
|
|
|
|
DIR *d;
|
|
|
|
struct dirent *entry;
|
|
|
|
char *fname, *cursor, *cursor_start;
|
|
|
|
File_Info *info_ptr;
|
|
|
|
i32 count, file_count, size, required_size;
|
|
|
|
|
|
|
|
terminate_with_null(&directory);
|
|
|
|
d = opendir(directory.str);
|
|
|
|
if (d){
|
|
|
|
count = 0;
|
|
|
|
for (entry = readdir(d);
|
|
|
|
entry != 0;
|
|
|
|
entry = readdir(d)){
|
|
|
|
fname = entry->d_name;
|
|
|
|
++file_count;
|
|
|
|
for (size = 0; fname[size]; ++size);
|
|
|
|
count += size + 1;
|
|
|
|
}
|
|
|
|
closedir(d);
|
|
|
|
|
|
|
|
required_size = count + file_count * sizeof(File_Info);
|
|
|
|
if (file_list->block_size < required_size){
|
2016-02-11 17:17:52 +00:00
|
|
|
system_free_memory(file_list->block);
|
|
|
|
file_list->block = system_get_memory(required_size);
|
2016-02-07 15:19:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
file_list->infos = (File_Info*)file_list->block;
|
|
|
|
cursor = (char*)(file_list->infos + file_count);
|
|
|
|
|
|
|
|
d = opendir(directory.str);
|
|
|
|
if (d){
|
|
|
|
info_ptr = file_list->infos;
|
|
|
|
for (entry = readdir(d);
|
|
|
|
entry != 0;
|
|
|
|
entry = readdir(d), ++info_ptr){
|
|
|
|
fname = entry->d_name;
|
|
|
|
cursor_start = cursor;
|
|
|
|
for (; *fname; ) *cursor++ = *fname++;
|
2016-02-11 17:17:52 +00:00
|
|
|
|
|
|
|
// TODO(allen): detect file/folder status
|
|
|
|
// (also make sure this even GETS folders!!!)
|
2016-02-07 15:19:18 +00:00
|
|
|
info_ptr->folder = 0;
|
|
|
|
info_ptr->filename.str = cursor_start;
|
|
|
|
info_ptr->filename.size = (i32)(cursor - cursor_start);
|
2016-02-11 17:17:52 +00:00
|
|
|
*cursor++ = 0;
|
2016-02-07 15:19:18 +00:00
|
|
|
info_ptr->filename.memory_size = info_ptr->filename.size + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
closedir(d);
|
|
|
|
}
|
|
|
|
closedir(d);
|
|
|
|
}
|
|
|
|
|
|
|
|
Sys_Post_Clipboard_Sig(system_post_clipboard){
|
|
|
|
// TODO(allen): Implement
|
|
|
|
AllowLocal(str);
|
|
|
|
}
|
|
|
|
|
2016-02-11 17:17:52 +00:00
|
|
|
Sys_CLI_Call_Sig(system_cli_call){
|
|
|
|
// TODO(allen): Implement
|
|
|
|
AllowLocal(path);
|
|
|
|
AllowLocal(script_name);
|
|
|
|
AllowLocal(cli_out);
|
|
|
|
}
|
|
|
|
|
|
|
|
Sys_CLI_Begin_Update_Sig(system_cli_begin_update){
|
|
|
|
// TODO(allen): Implement
|
|
|
|
AllowLocal(cli);
|
|
|
|
}
|
|
|
|
|
|
|
|
Sys_CLI_Update_Step_Sig(system_cli_update_step){
|
|
|
|
// TODO(allen): Implement
|
|
|
|
AllowLocal(cli);
|
|
|
|
AllowLocal(dest);
|
|
|
|
AllowLocal(max);
|
|
|
|
AllowLocal(amount);
|
|
|
|
}
|
|
|
|
|
|
|
|
Sys_CLI_End_Update_Sig(system_cli_end_update){
|
|
|
|
// TODO(allen): Implement
|
|
|
|
AllowLocal(cli);
|
|
|
|
}
|
|
|
|
|
|
|
|
Sys_Post_Job_Sig(system_post_job){
|
|
|
|
// TODO(allen): Implement
|
|
|
|
AllowLocal(group_id);
|
|
|
|
AllowLocal(job);
|
|
|
|
}
|
|
|
|
|
|
|
|
Sys_Cancel_Job_Sig(system_cancel_job){
|
|
|
|
// TODO(allen): Implement
|
|
|
|
AllowLocal(group_id);
|
|
|
|
AllowLocal(job_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
Sys_Acquire_Lock_Sig(system_acquire_lock){
|
|
|
|
// TODO(allen): Implement
|
|
|
|
AllowLocal(id);
|
|
|
|
}
|
|
|
|
|
|
|
|
Sys_Release_Lock_Sig(system_release_lock){
|
|
|
|
// TODO(allen): Implement
|
|
|
|
AllowLocal(id);
|
|
|
|
}
|
|
|
|
|
|
|
|
Sys_Grow_Thread_Memory_Sig(system_grow_thread_memory){
|
|
|
|
void *old_data;
|
|
|
|
i32 old_size, new_size;
|
|
|
|
|
|
|
|
system_acquire_lock(CANCEL_LOCK0 + memory->id - 1);
|
|
|
|
old_data = memory->data;
|
|
|
|
old_size = memory->size;
|
|
|
|
new_size = LargeRoundUp(memory->size*2, Kbytes(4));
|
|
|
|
memory->data = system_get_memory(new_size);
|
|
|
|
memory->size = new_size;
|
|
|
|
if (old_data){
|
|
|
|
memcpy(memory->data, old_data, old_size);
|
|
|
|
system_free_memory(old_data);
|
|
|
|
}
|
|
|
|
system_release_lock(CANCEL_LOCK0 + memory->id - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
INTERNAL_Sys_Sentinel_Sig(internal_sentinel){
|
|
|
|
Bubble *result;
|
|
|
|
#if FRED_INTERNAL
|
|
|
|
result = &linuxvars.internal_bubble;
|
|
|
|
#else
|
|
|
|
result = 0;
|
|
|
|
#endif
|
|
|
|
return(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
INTERNAL_Sys_Get_Thread_States_Sig(internal_get_thread_states){
|
|
|
|
// TODO(allen): Implement
|
|
|
|
AllowLocal(id);
|
|
|
|
AllowLocal(running);
|
|
|
|
AllowLocal(pending);
|
|
|
|
}
|
|
|
|
|
|
|
|
INTERNAL_Sys_Debug_Message_Sig(internal_debug_message){
|
|
|
|
printf("%s", message);
|
|
|
|
}
|
|
|
|
|
|
|
|
DIRECTORY_HAS_FILE_SIG(system_directory_has_file){
|
|
|
|
int result = 0;
|
|
|
|
// TODO(allen): Implement
|
|
|
|
AllowLocal(dir);
|
|
|
|
AllowLocal(filename);
|
|
|
|
return(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
DIRECTORY_CD_SIG(system_directory_cd){
|
|
|
|
int result = 0;
|
|
|
|
// TODO(allen): Implement
|
|
|
|
AllowLocal(dir);
|
|
|
|
AllowLocal(rel_path);
|
|
|
|
return(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
internal
|
|
|
|
Sys_File_Can_Be_Made(system_file_can_be_made){
|
|
|
|
// TODO(allen): Implement
|
|
|
|
AllowLocal(filename);
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
internal
|
|
|
|
Sys_Load_File_Sig(system_load_file){
|
|
|
|
Data result = {};
|
|
|
|
// TODO(allen): Implement
|
|
|
|
AllowLocal(filename);
|
|
|
|
return(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
internal
|
|
|
|
Sys_Save_File_Sig(system_save_file){
|
|
|
|
b32 result = 0;
|
|
|
|
// TODO(allen): Implement
|
|
|
|
AllowLocal(filename);
|
|
|
|
AllowLocal(data);
|
|
|
|
AllowLocal(size);
|
|
|
|
return(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
internal b32
|
|
|
|
LinuxLoadAppCode(){
|
|
|
|
b32 result = 0;
|
|
|
|
App_Get_Functions *get_funcs = 0;
|
|
|
|
|
|
|
|
linuxvars.app_code = dlopen("./4ed_app.so", RTLD_LAZY);
|
|
|
|
if (linuxvars.app_code){
|
|
|
|
get_funcs = (App_Get_Functions*)
|
|
|
|
dlsym(linuxvars.app_code, "app_get_functions");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (get_funcs){
|
|
|
|
result = 1;
|
|
|
|
linuxvars.app = get_funcs();
|
|
|
|
}
|
|
|
|
|
|
|
|
return(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
internal void
|
|
|
|
LinuxLoadSystemCode(){
|
|
|
|
linuxvars.system->file_time_stamp = system_file_time_stamp;
|
|
|
|
linuxvars.system->set_file_list = system_set_file_list;
|
|
|
|
|
|
|
|
linuxvars.system->directory_has_file = system_directory_has_file;
|
|
|
|
linuxvars.system->directory_cd = system_directory_cd;
|
|
|
|
|
|
|
|
linuxvars.system->post_clipboard = system_post_clipboard;
|
|
|
|
linuxvars.system->time = system_time;
|
|
|
|
|
|
|
|
linuxvars.system->cli_call = system_cli_call;
|
|
|
|
linuxvars.system->cli_begin_update = system_cli_begin_update;
|
|
|
|
linuxvars.system->cli_update_step = system_cli_update_step;
|
|
|
|
linuxvars.system->cli_end_update = system_cli_end_update;
|
|
|
|
|
|
|
|
linuxvars.system->post_job = system_post_job;
|
|
|
|
linuxvars.system->cancel_job = system_cancel_job;
|
|
|
|
linuxvars.system->grow_thread_memory = system_grow_thread_memory;
|
|
|
|
linuxvars.system->acquire_lock = system_acquire_lock;
|
|
|
|
linuxvars.system->release_lock = system_release_lock;
|
|
|
|
|
|
|
|
linuxvars.system->internal_sentinel = internal_sentinel;
|
|
|
|
linuxvars.system->internal_get_thread_states = internal_get_thread_states;
|
|
|
|
linuxvars.system->internal_debug_message = internal_debug_message;
|
|
|
|
}
|
|
|
|
|
|
|
|
#include "system_shared.cpp"
|
|
|
|
#include "4ed_rendering.cpp"
|
|
|
|
|
2016-02-07 15:19:18 +00:00
|
|
|
// NOTE(allen): Thanks to Casey for providing the linux OpenGL launcher.
|
2016-02-01 05:03:42 +00:00
|
|
|
static bool ctxErrorOccurred = false;
|
|
|
|
static int XInput2OpCode = 0;
|
|
|
|
internal int
|
|
|
|
ctxErrorHandler( Display *dpy, XErrorEvent *ev )
|
|
|
|
{
|
|
|
|
ctxErrorOccurred = true;
|
|
|
|
return 0;
|
2015-11-16 16:15:45 +00:00
|
|
|
}
|
|
|
|
|
2016-02-01 05:03:42 +00:00
|
|
|
internal GLXContext
|
|
|
|
InitializeOpenGLContext(Display *XDisplay, Window XWindow, GLXFBConfig &bestFbc, b32 &IsLegacy)
|
|
|
|
{
|
|
|
|
IsLegacy = false;
|
|
|
|
|
|
|
|
typedef GLXContext (*glXCreateContextAttribsARBProc)(Display*, GLXFBConfig, GLXContext, Bool, const int*);
|
|
|
|
const char *glxExts = glXQueryExtensionsString(XDisplay, DefaultScreen(XDisplay));
|
|
|
|
|
|
|
|
glXCreateContextAttribsARBProc glXCreateContextAttribsARB = 0;
|
|
|
|
glXCreateContextAttribsARB = (glXCreateContextAttribsARBProc)
|
|
|
|
glXGetProcAddressARB( (const GLubyte *) "glXCreateContextAttribsARB" );
|
|
|
|
|
|
|
|
GLXContext ctx = 0;
|
|
|
|
ctxErrorOccurred = false;
|
|
|
|
int (*oldHandler)(Display*, XErrorEvent*) =
|
|
|
|
XSetErrorHandler(&ctxErrorHandler);
|
|
|
|
if (!glXCreateContextAttribsARB)
|
|
|
|
{
|
|
|
|
printf( "glXCreateContextAttribsARB() not found"
|
|
|
|
" ... using old-style GLX context\n" );
|
|
|
|
ctx = glXCreateNewContext( XDisplay, bestFbc, GLX_RGBA_TYPE, 0, True );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
int context_attribs[] =
|
|
|
|
{
|
|
|
|
GLX_CONTEXT_MAJOR_VERSION_ARB, 4,
|
|
|
|
GLX_CONTEXT_MINOR_VERSION_ARB, 3,
|
|
|
|
//GLX_CONTEXT_FLAGS_ARB , GLX_CONTEXT_FORWARD_COMPATIBLE_BIT_ARB,
|
|
|
|
None
|
|
|
|
};
|
|
|
|
|
|
|
|
printf("\nAttribs: %d %d %d %d %d\n",
|
|
|
|
context_attribs[0],
|
|
|
|
context_attribs[1],
|
|
|
|
context_attribs[2],
|
|
|
|
context_attribs[3],
|
|
|
|
context_attribs[4]);
|
|
|
|
|
|
|
|
printf( "Creating context\n" );
|
|
|
|
ctx = glXCreateContextAttribsARB( XDisplay, bestFbc, 0,
|
|
|
|
True, context_attribs );
|
|
|
|
|
|
|
|
XSync( XDisplay, False );
|
|
|
|
if ( !ctxErrorOccurred && ctx )
|
|
|
|
{
|
|
|
|
printf( "Created GL 4.3 context\n" );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ctxErrorOccurred = false;
|
|
|
|
|
|
|
|
context_attribs[1] = 4;
|
|
|
|
context_attribs[3] = 0;
|
|
|
|
|
|
|
|
printf( "Creating context\n" );
|
|
|
|
ctx = glXCreateContextAttribsARB( XDisplay, bestFbc, 0,
|
|
|
|
True, context_attribs );
|
|
|
|
|
|
|
|
XSync( XDisplay, False );
|
|
|
|
|
|
|
|
if ( !ctxErrorOccurred && ctx )
|
|
|
|
{
|
|
|
|
printf( "Created GL 4.0 context\n" );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
context_attribs[1] = 1;
|
|
|
|
context_attribs[3] = 0;
|
|
|
|
|
|
|
|
ctxErrorOccurred = false;
|
|
|
|
|
|
|
|
printf( "Failed to create GL 4.0 context"
|
|
|
|
" ... using old-style GLX context\n" );
|
|
|
|
ctx = glXCreateContextAttribsARB( XDisplay, bestFbc, 0,
|
|
|
|
True, context_attribs );
|
|
|
|
|
|
|
|
IsLegacy = true;
|
|
|
|
}
|
|
|
|
}
|
2015-11-16 16:15:45 +00:00
|
|
|
}
|
2016-02-01 05:03:42 +00:00
|
|
|
|
|
|
|
XSync( XDisplay, False );
|
|
|
|
XSetErrorHandler( oldHandler );
|
|
|
|
|
|
|
|
if ( ctxErrorOccurred || !ctx )
|
|
|
|
{
|
|
|
|
printf( "Failed to create an OpenGL context\n" );
|
|
|
|
exit(1);
|
2015-11-16 16:15:45 +00:00
|
|
|
}
|
2016-02-01 05:03:42 +00:00
|
|
|
|
|
|
|
if ( ! glXIsDirect ( XDisplay, ctx ) )
|
|
|
|
{
|
|
|
|
printf( "Indirect GLX rendering context obtained\n" );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf( "Direct GLX rendering context obtained\n" );
|
|
|
|
}
|
|
|
|
|
|
|
|
printf( "Making context current\n" );
|
|
|
|
glXMakeCurrent( XDisplay, XWindow, ctx );
|
|
|
|
|
|
|
|
GLint n;
|
|
|
|
char *Vendor = (char *)glGetString(GL_VENDOR);
|
|
|
|
char *Renderer = (char *)glGetString(GL_RENDERER);
|
|
|
|
char *Version = (char *)glGetString(GL_VERSION);
|
|
|
|
char *Extensions = (char *)glGetString(GL_EXTENSIONS);
|
|
|
|
|
|
|
|
printf("GL_VENDOR: %s\n", Vendor);
|
|
|
|
printf("GL_RENDERER: %s\n", Renderer);
|
|
|
|
printf("GL_VERSION: %s\n", Version);
|
|
|
|
printf("GL_EXTENSIONS: %s\n", Extensions);
|
|
|
|
|
|
|
|
return(ctx);
|
2015-11-16 16:15:45 +00:00
|
|
|
}
|
|
|
|
|
2016-02-01 05:03:42 +00:00
|
|
|
internal b32
|
|
|
|
GLXSupportsModernContexts(Display *XDisplay)
|
|
|
|
{
|
|
|
|
b32 Result = false;
|
2015-11-16 16:15:45 +00:00
|
|
|
|
2016-02-01 05:03:42 +00:00
|
|
|
int GLXMajor, GLXMinor;
|
|
|
|
|
|
|
|
char *XVendor = ServerVendor(XDisplay);
|
|
|
|
printf("XWindows vendor: %s\n", XVendor);
|
|
|
|
if(glXQueryVersion(XDisplay, &GLXMajor, &GLXMinor))
|
|
|
|
{
|
|
|
|
printf("GLX version %d.%d\n", GLXMajor, GLXMinor);
|
|
|
|
if(((GLXMajor == 1 ) && (GLXMinor >= 3)) ||
|
|
|
|
(GLXMajor > 1))
|
|
|
|
{
|
|
|
|
Result = true;
|
2015-11-16 16:15:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-01 05:03:42 +00:00
|
|
|
return(Result);
|
2015-11-16 16:15:45 +00:00
|
|
|
}
|
|
|
|
|
2016-02-01 05:03:42 +00:00
|
|
|
typedef struct glx_config_result{
|
|
|
|
b32 Found;
|
|
|
|
GLXFBConfig BestConfig;
|
|
|
|
XVisualInfo BestInfo;
|
|
|
|
} glx_config_result;
|
|
|
|
|
|
|
|
internal glx_config_result
|
|
|
|
ChooseGLXConfig(Display *XDisplay, int XScreenIndex)
|
|
|
|
{
|
|
|
|
glx_config_result Result = {0};
|
2015-11-16 16:15:45 +00:00
|
|
|
|
2016-02-01 05:03:42 +00:00
|
|
|
int DesiredAttributes[] =
|
|
|
|
{
|
|
|
|
GLX_X_RENDERABLE , True,
|
|
|
|
GLX_DRAWABLE_TYPE , GLX_WINDOW_BIT,
|
|
|
|
GLX_RENDER_TYPE , GLX_RGBA_BIT,
|
|
|
|
GLX_X_VISUAL_TYPE , GLX_TRUE_COLOR,
|
|
|
|
GLX_RED_SIZE , 8,
|
|
|
|
GLX_GREEN_SIZE , 8,
|
|
|
|
GLX_BLUE_SIZE , 8,
|
|
|
|
GLX_ALPHA_SIZE , 8,
|
|
|
|
GLX_DEPTH_SIZE , 24,
|
|
|
|
GLX_STENCIL_SIZE , 8,
|
|
|
|
GLX_DOUBLEBUFFER , True,
|
|
|
|
//GLX_SAMPLE_BUFFERS , 1,
|
|
|
|
//GLX_SAMPLES , 4,
|
|
|
|
None
|
|
|
|
};
|
2015-11-16 16:15:45 +00:00
|
|
|
|
2016-02-01 05:03:42 +00:00
|
|
|
{
|
|
|
|
int ConfigCount;
|
|
|
|
GLXFBConfig *Configs = glXChooseFBConfig(XDisplay,
|
|
|
|
XScreenIndex,
|
|
|
|
DesiredAttributes,
|
|
|
|
&ConfigCount);
|
2015-11-16 16:15:45 +00:00
|
|
|
|
2016-02-01 05:03:42 +00:00
|
|
|
#if 0
|
|
|
|
int DiffValues[GLXValueCount];
|
|
|
|
#endif
|
|
|
|
{for(int ConfigIndex = 0;
|
|
|
|
ConfigIndex < ConfigCount;
|
|
|
|
++ConfigIndex)
|
|
|
|
{
|
|
|
|
GLXFBConfig &Config = Configs[ConfigIndex];
|
|
|
|
XVisualInfo *VisualInfo = glXGetVisualFromFBConfig(XDisplay, Config);
|
2015-11-16 16:15:45 +00:00
|
|
|
|
2016-02-01 05:03:42 +00:00
|
|
|
#if 0
|
|
|
|
printf(" Option %d:\n", ConfigIndex);
|
|
|
|
printf(" Depth: %d\n", VisualInfo->depth);
|
|
|
|
printf(" Bits per channel: %d\n", VisualInfo->bits_per_rgb);
|
|
|
|
printf(" Mask: R%06x G%06x B%06x\n",
|
|
|
|
(uint32)VisualInfo->red_mask,
|
|
|
|
(uint32)VisualInfo->green_mask,
|
|
|
|
(uint32)VisualInfo->blue_mask);
|
|
|
|
printf(" Class: %d\n", VisualInfo->c_class);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
{for(int ValueIndex = 0;
|
|
|
|
ValueIndex < GLXValueCount;
|
|
|
|
++ValueIndex)
|
|
|
|
{
|
|
|
|
glx_value_info &ValueInfo = GLXValues[ValueIndex];
|
|
|
|
int Value;
|
|
|
|
glXGetFBConfigAttrib(XDisplay, Config, ValueInfo.ID, &Value);
|
|
|
|
if(DiffValues[ValueIndex] != Value)
|
|
|
|
{
|
|
|
|
printf(" %s: %d\n", ValueInfo.Name, Value);
|
|
|
|
DiffValues[ValueIndex] = Value;
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// TODO(casey): How the hell are you supposed to pick a config here??
|
|
|
|
if(ConfigIndex == 0)
|
|
|
|
{
|
|
|
|
Result.Found = true;
|
|
|
|
Result.BestConfig = Config;
|
|
|
|
Result.BestInfo = *VisualInfo;
|
|
|
|
}
|
|
|
|
|
|
|
|
XFree(VisualInfo);
|
|
|
|
}}
|
|
|
|
|
|
|
|
XFree(Configs);
|
|
|
|
}
|
|
|
|
|
|
|
|
return(Result);
|
2015-12-01 02:51:53 +00:00
|
|
|
}
|
|
|
|
|
2015-11-16 16:15:45 +00:00
|
|
|
internal void
|
2016-02-01 05:03:42 +00:00
|
|
|
InitializeXInput(Display *dpy, Window XWindow)
|
|
|
|
{
|
|
|
|
int event, error;
|
|
|
|
if(XQueryExtension(dpy, "XInputExtension", &XInput2OpCode, &event, &error))
|
|
|
|
{
|
|
|
|
int major = 2, minor = 0;
|
|
|
|
if(XIQueryVersion(dpy, &major, &minor) != BadRequest)
|
|
|
|
{
|
|
|
|
printf("XInput initialized");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf("XI2 not available. Server supports %d.%d\n", major, minor);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf("X Input extension not available.\n");
|
|
|
|
}
|
2015-11-16 16:15:45 +00:00
|
|
|
|
2016-02-01 05:03:42 +00:00
|
|
|
/*
|
|
|
|
TODO(casey): So, this is all one big clusterfuck I guess.
|
2015-11-16 16:15:45 +00:00
|
|
|
|
2016-02-01 05:03:42 +00:00
|
|
|
The problem here is that you want to be able to get input
|
|
|
|
from all possible devices that could be a mouse or keyboard
|
|
|
|
(or gamepad, or whatever). So you'd like to be able to
|
|
|
|
register events for XIAllDevices, so that when a new
|
|
|
|
input device is connected, you will start receiving
|
|
|
|
input from it automatically, without having to periodically
|
|
|
|
poll XIQueryDevice to see if a new device has appeared.
|
2015-11-16 16:15:45 +00:00
|
|
|
|
2016-02-01 05:03:42 +00:00
|
|
|
UNFORTUNATELY, this is not actually possible in Linux because
|
|
|
|
there was a bug in Xorg (as of early 2013, it is still not
|
|
|
|
fixed in most distributions people are using, AFAICT) which
|
|
|
|
makes the XServer return an error if you actually try to
|
|
|
|
do this :(
|
2015-11-16 16:15:45 +00:00
|
|
|
|
2016-02-01 05:03:42 +00:00
|
|
|
But EVENTUALLY, if that shit gets fixed, then that is
|
|
|
|
the way this should work.
|
|
|
|
*/
|
2015-12-01 02:51:53 +00:00
|
|
|
|
2016-02-01 05:03:42 +00:00
|
|
|
#if 0
|
|
|
|
int DeviceCount;
|
|
|
|
XIDeviceInfo *DeviceInfo = XIQueryDevice(dpy, XIAllDevices, &DeviceCount);
|
|
|
|
{for(int32x DeviceIndex = 0;
|
|
|
|
DeviceIndex < DeviceCount;
|
|
|
|
++DeviceIndex)
|
2015-11-16 16:15:45 +00:00
|
|
|
{
|
2016-02-01 05:03:42 +00:00
|
|
|
XIDeviceInfo *Device = DeviceInfo + DeviceIndex;
|
|
|
|
printf("Device %d: %s\n", Device->deviceid, Device->name);
|
|
|
|
}}
|
|
|
|
XIFreeDeviceInfo(DeviceInfo);
|
2015-11-16 16:15:45 +00:00
|
|
|
#endif
|
|
|
|
|
2016-02-01 05:03:42 +00:00
|
|
|
XIEventMask Mask = {0};
|
|
|
|
Mask.deviceid = XIAllDevices;
|
|
|
|
Mask.mask_len = XIMaskLen(XI_RawMotion);
|
|
|
|
size_t MaskSize = Mask.mask_len * sizeof(char unsigned);
|
|
|
|
Mask.mask = (char unsigned *)alloca(MaskSize);
|
|
|
|
memset(Mask.mask, 0, MaskSize);
|
|
|
|
if(Mask.mask)
|
|
|
|
{
|
|
|
|
XISetMask(Mask.mask, XI_ButtonPress);
|
|
|
|
XISetMask(Mask.mask, XI_ButtonRelease);
|
|
|
|
XISetMask(Mask.mask, XI_KeyPress);
|
|
|
|
XISetMask(Mask.mask, XI_KeyRelease);
|
|
|
|
XISetMask(Mask.mask, XI_Motion);
|
|
|
|
XISetMask(Mask.mask, XI_DeviceChanged);
|
|
|
|
XISetMask(Mask.mask, XI_Enter);
|
|
|
|
XISetMask(Mask.mask, XI_Leave);
|
|
|
|
XISetMask(Mask.mask, XI_FocusIn);
|
|
|
|
XISetMask(Mask.mask, XI_FocusOut);
|
|
|
|
XISetMask(Mask.mask, XI_HierarchyChanged);
|
|
|
|
XISetMask(Mask.mask, XI_PropertyEvent);
|
|
|
|
XISelectEvents(dpy, XWindow, &Mask, 1);
|
|
|
|
XSync(dpy, False);
|
|
|
|
|
|
|
|
Mask.deviceid = XIAllMasterDevices;
|
|
|
|
memset(Mask.mask, 0, MaskSize);
|
|
|
|
XISetMask(Mask.mask, XI_RawKeyPress);
|
|
|
|
XISetMask(Mask.mask, XI_RawKeyRelease);
|
|
|
|
XISetMask(Mask.mask, XI_RawButtonPress);
|
|
|
|
XISetMask(Mask.mask, XI_RawButtonRelease);
|
|
|
|
XISetMask(Mask.mask, XI_RawMotion);
|
|
|
|
XISelectEvents(dpy, DefaultRootWindow(dpy), &Mask, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2016-02-11 17:17:52 +00:00
|
|
|
main(int argc, char **argv)
|
2016-02-01 05:03:42 +00:00
|
|
|
{
|
2016-02-11 17:17:52 +00:00
|
|
|
linuxvars = {};
|
|
|
|
exchange_vars = {};
|
|
|
|
|
|
|
|
#if FRED_INTERNAL
|
|
|
|
linuxvars.internal_bubble.next = &linuxvars.internal_bubble;
|
|
|
|
linuxvars.internal_bubble.prev = &linuxvars.internal_bubble;
|
|
|
|
linuxvars.internal_bubble.flags = MEM_BUBBLE_SYS_DEBUG;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (!LinuxLoadAppCode()){
|
|
|
|
// TODO(allen): Failed to load app code, serious problem.
|
|
|
|
return 99;
|
|
|
|
}
|
|
|
|
|
|
|
|
System_Functions system_;
|
|
|
|
System_Functions *system = &system_;
|
|
|
|
linuxvars.system = system;
|
|
|
|
LinuxLoadSystemCode();
|
|
|
|
|
|
|
|
memory_vars.vars_memory_size = Mbytes(2);
|
|
|
|
memory_vars.vars_memory = system_get_memory(memory_vars.vars_memory_size);
|
|
|
|
memory_vars.target_memory_size = Mbytes(512);
|
|
|
|
memory_vars.target_memory = system_get_memory(memory_vars.target_memory_size);
|
|
|
|
|
|
|
|
String current_directory;
|
|
|
|
i32 curdir_req, curdir_size;
|
|
|
|
char *curdir_mem;
|
|
|
|
|
|
|
|
curdir_req = (1 << 9);
|
|
|
|
curdir_mem = (char*)system_get_memory(curdir_req);
|
|
|
|
for (; getcwd(curdir_mem, curdir_req) == 0 && curdir_req < (1 << 13);){
|
|
|
|
system_free_memory(curdir_mem);
|
|
|
|
curdir_req *= 4;
|
|
|
|
curdir_mem = (char*)system_get_memory(curdir_req);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (curdir_req >= (1 << 13)){
|
|
|
|
// TODO(allen): fuckin' bullshit string APIs makin' me pissed
|
|
|
|
return 57;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (curdir_size = 0; curdir_mem[curdir_size]; ++curdir_size);
|
|
|
|
|
|
|
|
current_directory = make_string(curdir_mem, curdir_size, curdir_req);
|
|
|
|
|
|
|
|
Command_Line_Parameters clparams;
|
|
|
|
clparams.argv = argv;
|
|
|
|
clparams.argc = argc;
|
|
|
|
|
|
|
|
char **files;
|
|
|
|
i32 *file_count;
|
|
|
|
i32 output_size;
|
|
|
|
|
|
|
|
output_size =
|
|
|
|
linuxvars.app.read_command_line(system,
|
|
|
|
&memory_vars,
|
|
|
|
current_directory,
|
|
|
|
&linuxvars.settings,
|
|
|
|
&files, &file_count,
|
|
|
|
clparams);
|
|
|
|
|
|
|
|
if (output_size > 0){
|
|
|
|
// TODO(allen): crt free version
|
|
|
|
printf("%.*s", output_size, (char*)memory_vars.target_memory);
|
|
|
|
}
|
|
|
|
if (output_size != 0) return 0;
|
|
|
|
|
|
|
|
system_filter_real_files(files, file_count);
|
|
|
|
|
2016-02-01 05:03:42 +00:00
|
|
|
Display *XDisplay = XOpenDisplay(0);
|
|
|
|
if(XDisplay && GLXSupportsModernContexts(XDisplay))
|
|
|
|
{
|
|
|
|
int XScreenCount = ScreenCount(XDisplay);
|
|
|
|
for(int XScreenIndex = 0;
|
|
|
|
XScreenIndex < XScreenCount;
|
|
|
|
++XScreenIndex)
|
|
|
|
{
|
|
|
|
Screen *XScreen = ScreenOfDisplay(XDisplay, XScreenIndex);
|
2015-12-01 02:51:53 +00:00
|
|
|
|
2016-02-01 05:03:42 +00:00
|
|
|
int WinWidth = WidthOfScreen(XScreen);
|
|
|
|
int WinHeight = HeightOfScreen(XScreen);
|
|
|
|
|
|
|
|
glx_config_result Config = ChooseGLXConfig(XDisplay, XScreenIndex);
|
|
|
|
if(Config.Found)
|
|
|
|
{
|
|
|
|
Colormap cmap;
|
|
|
|
XSetWindowAttributes swa;
|
|
|
|
swa.colormap = cmap = XCreateColormap(XDisplay,
|
|
|
|
RootWindow(XDisplay, Config.BestInfo.screen ),
|
|
|
|
Config.BestInfo.visual, AllocNone);
|
|
|
|
swa.background_pixmap = None;
|
|
|
|
swa.border_pixel = 0;
|
|
|
|
swa.event_mask = StructureNotifyMask;
|
|
|
|
|
|
|
|
Window XWindow = XCreateWindow(XDisplay,
|
|
|
|
RootWindow(XDisplay, Config.BestInfo.screen),
|
|
|
|
0, 0, WinWidth, WinHeight,
|
|
|
|
0, Config.BestInfo.depth, InputOutput,
|
|
|
|
Config.BestInfo.visual,
|
|
|
|
CWBorderPixel|CWColormap|CWEventMask, &swa );
|
|
|
|
if(XWindow)
|
|
|
|
{
|
|
|
|
XStoreName(XDisplay, XWindow, "4coder");
|
|
|
|
XMapWindow(XDisplay, XWindow);
|
|
|
|
|
|
|
|
InitializeXInput(XDisplay, XWindow);
|
|
|
|
|
|
|
|
b32 IsLegacy = false;
|
|
|
|
GLXContext GLContext =
|
|
|
|
InitializeOpenGLContext(XDisplay, XWindow, Config.BestConfig, IsLegacy);
|
|
|
|
|
|
|
|
XWindowAttributes WinAttribs;
|
|
|
|
if(XGetWindowAttributes(XDisplay, XWindow, &WinAttribs))
|
|
|
|
{
|
|
|
|
WinWidth = WinAttribs.width;
|
|
|
|
WinHeight = WinAttribs.height;
|
|
|
|
}
|
|
|
|
|
|
|
|
XRaiseWindow(XDisplay, XWindow);
|
|
|
|
XSync(XDisplay, False);
|
|
|
|
|
|
|
|
for(;;)
|
|
|
|
{
|
|
|
|
while(XPending(XDisplay))
|
|
|
|
{
|
|
|
|
XEvent Event;
|
|
|
|
XNextEvent(XDisplay, &Event);
|
|
|
|
|
|
|
|
if((Event.xcookie.type == GenericEvent) &&
|
|
|
|
(Event.xcookie.extension == XInput2OpCode) &&
|
|
|
|
XGetEventData(XDisplay, &Event.xcookie))
|
|
|
|
{
|
|
|
|
switch(Event.xcookie.evtype)
|
|
|
|
{
|
|
|
|
case XI_Motion:
|
|
|
|
{
|
|
|
|
Window root_return, child_return;
|
|
|
|
int root_x_return, root_y_return;
|
|
|
|
int MouseX, MouseY;
|
|
|
|
unsigned int mask_return;
|
|
|
|
XQueryPointer(XDisplay,
|
|
|
|
XWindow,
|
|
|
|
&root_return, &child_return,
|
|
|
|
&root_x_return, &root_y_return,
|
|
|
|
&MouseX, &MouseY,
|
|
|
|
&mask_return);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case XI_ButtonPress:
|
|
|
|
case XI_ButtonRelease:
|
|
|
|
{
|
|
|
|
b32 Down = (Event.xcookie.evtype == XI_ButtonPress);
|
|
|
|
XIDeviceEvent *DevEvent = (XIDeviceEvent *)Event.xcookie.data;
|
|
|
|
int Button = DevEvent->detail;
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case XI_KeyPress:
|
|
|
|
case XI_KeyRelease:
|
|
|
|
{
|
|
|
|
b32 Down = (Event.xcookie.evtype == XI_KeyPress);
|
|
|
|
XIDeviceEvent *DevEvent = (XIDeviceEvent *)Event.xcookie.data;
|
|
|
|
int VK = DevEvent->detail;
|
|
|
|
} break;
|
|
|
|
}
|
|
|
|
|
|
|
|
XFreeEventData(XDisplay, &Event.xcookie);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Draw some stuff here?
|
|
|
|
|
|
|
|
glXSwapBuffers(XDisplay, XWindow);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-11-16 16:15:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// BOTTOM
|
|
|
|
|