linux file loading / saving working (with questionable threading)
parent
654e2ab8e1
commit
c1b0521e81
336
linux_4ed.cpp
336
linux_4ed.cpp
|
@ -57,6 +57,32 @@
|
||||||
#include <sys/stat.h>
|
#include <sys/stat.h>
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
|
#include <pthread.h>
|
||||||
|
#include <signal.h>
|
||||||
|
|
||||||
|
struct Linux_Semaphore {
|
||||||
|
pthread_mutex_t mutex;
|
||||||
|
pthread_cond_t cond;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct Linux_Semaphore_Handle {
|
||||||
|
pthread_mutex_t *mutex_p;
|
||||||
|
pthread_cond_t *cond_p;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct Thread_Context{
|
||||||
|
u32 job_id;
|
||||||
|
b32 running;
|
||||||
|
|
||||||
|
Work_Queue *queue;
|
||||||
|
u32 id;
|
||||||
|
pthread_t handle;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct Thread_Group{
|
||||||
|
Thread_Context *threads;
|
||||||
|
i32 count;
|
||||||
|
};
|
||||||
|
|
||||||
struct Linux_Vars{
|
struct Linux_Vars{
|
||||||
Display *XDisplay;
|
Display *XDisplay;
|
||||||
|
@ -79,6 +105,11 @@ struct Linux_Vars{
|
||||||
|
|
||||||
void *app_code;
|
void *app_code;
|
||||||
void *custom;
|
void *custom;
|
||||||
|
|
||||||
|
Thread_Memory *thread_memory;
|
||||||
|
Thread_Group groups[THREAD_GROUP_COUNT];
|
||||||
|
Linux_Semaphore thread_locks[THREAD_GROUP_COUNT];
|
||||||
|
Linux_Semaphore locks[LOCK_COUNT];
|
||||||
|
|
||||||
Plat_Settings settings;
|
Plat_Settings settings;
|
||||||
System_Functions *system;
|
System_Functions *system;
|
||||||
|
@ -99,6 +130,8 @@ struct Linux_Vars{
|
||||||
#define FPS 60
|
#define FPS 60
|
||||||
#define frame_useconds (1000000 / FPS)
|
#define frame_useconds (1000000 / FPS)
|
||||||
|
|
||||||
|
#define DBG_FN do { fprintf(stderr, "Fn called: %s\n", __PRETTY_FUNCTION__); } while(0)
|
||||||
|
|
||||||
globalvar Linux_Vars linuxvars;
|
globalvar Linux_Vars linuxvars;
|
||||||
globalvar Application_Memory memory_vars;
|
globalvar Application_Memory memory_vars;
|
||||||
globalvar Exchange exchange_vars;
|
globalvar Exchange exchange_vars;
|
||||||
|
@ -328,6 +361,7 @@ Sys_Post_Clipboard_Sig(system_post_clipboard){
|
||||||
|
|
||||||
Sys_CLI_Call_Sig(system_cli_call){
|
Sys_CLI_Call_Sig(system_cli_call){
|
||||||
// TODO(allen): Implement
|
// TODO(allen): Implement
|
||||||
|
DBG_FN;
|
||||||
AllowLocal(path);
|
AllowLocal(path);
|
||||||
AllowLocal(script_name);
|
AllowLocal(script_name);
|
||||||
AllowLocal(cli_out);
|
AllowLocal(cli_out);
|
||||||
|
@ -335,11 +369,13 @@ Sys_CLI_Call_Sig(system_cli_call){
|
||||||
|
|
||||||
Sys_CLI_Begin_Update_Sig(system_cli_begin_update){
|
Sys_CLI_Begin_Update_Sig(system_cli_begin_update){
|
||||||
// TODO(allen): Implement
|
// TODO(allen): Implement
|
||||||
|
DBG_FN;
|
||||||
AllowLocal(cli);
|
AllowLocal(cli);
|
||||||
}
|
}
|
||||||
|
|
||||||
Sys_CLI_Update_Step_Sig(system_cli_update_step){
|
Sys_CLI_Update_Step_Sig(system_cli_update_step){
|
||||||
// TODO(allen): Implement
|
// TODO(allen): Implement
|
||||||
|
DBG_FN;
|
||||||
AllowLocal(cli);
|
AllowLocal(cli);
|
||||||
AllowLocal(dest);
|
AllowLocal(dest);
|
||||||
AllowLocal(max);
|
AllowLocal(max);
|
||||||
|
@ -348,31 +384,156 @@ Sys_CLI_Update_Step_Sig(system_cli_update_step){
|
||||||
|
|
||||||
Sys_CLI_End_Update_Sig(system_cli_end_update){
|
Sys_CLI_End_Update_Sig(system_cli_end_update){
|
||||||
// TODO(allen): Implement
|
// TODO(allen): Implement
|
||||||
|
DBG_FN;
|
||||||
AllowLocal(cli);
|
AllowLocal(cli);
|
||||||
}
|
}
|
||||||
|
|
||||||
Sys_Post_Job_Sig(system_post_job){
|
static_assert(sizeof(Plat_Handle) >= sizeof(Linux_Semaphore_Handle), "Plat_Handle not big enough");
|
||||||
// TODO(allen): Implement
|
|
||||||
AllowLocal(group_id);
|
internal Plat_Handle
|
||||||
AllowLocal(job);
|
LinuxSemToHandle(Linux_Semaphore* sem){
|
||||||
|
Linux_Semaphore_Handle h = { &sem->mutex, &sem->cond };
|
||||||
|
return *(Plat_Handle*)&h;
|
||||||
}
|
}
|
||||||
|
|
||||||
Sys_Cancel_Job_Sig(system_cancel_job){
|
internal void*
|
||||||
|
ThreadProc(void* arg){
|
||||||
|
Thread_Context *thread = (Thread_Context*)arg;
|
||||||
|
Work_Queue *queue = thread->queue;
|
||||||
|
|
||||||
|
for (;;){
|
||||||
|
u32 read_index = queue->read_position;
|
||||||
|
u32 write_index = queue->write_position;
|
||||||
|
|
||||||
|
if (read_index != write_index){
|
||||||
|
u32 next_read_index = (read_index + 1) % JOB_ID_WRAP;
|
||||||
|
u32 safe_read_index =
|
||||||
|
__sync_val_compare_and_swap(&queue->read_position,
|
||||||
|
read_index, next_read_index);
|
||||||
|
|
||||||
|
if (safe_read_index == read_index){
|
||||||
|
Full_Job_Data *full_job = queue->jobs + (safe_read_index % QUEUE_WRAP);
|
||||||
|
// NOTE(allen): This is interlocked so that it plays nice
|
||||||
|
// with the cancel job routine, which may try to cancel this job
|
||||||
|
// at the same time that we try to run it
|
||||||
|
|
||||||
|
i32 safe_running_thread =
|
||||||
|
__sync_val_compare_and_swap(&full_job->running_thread,
|
||||||
|
THREAD_NOT_ASSIGNED, thread->id);
|
||||||
|
|
||||||
|
if (safe_running_thread == THREAD_NOT_ASSIGNED){
|
||||||
|
thread->job_id = full_job->id;
|
||||||
|
thread->running = 1;
|
||||||
|
Thread_Memory *thread_memory = 0;
|
||||||
|
|
||||||
|
// TODO(allen): remove memory_request
|
||||||
|
if (full_job->job.memory_request != 0){
|
||||||
|
thread_memory = linuxvars.thread_memory + thread->id - 1;
|
||||||
|
if (thread_memory->size < full_job->job.memory_request){
|
||||||
|
if (thread_memory->data){
|
||||||
|
LinuxFreeMemory(thread_memory->data);
|
||||||
|
}
|
||||||
|
i32 new_size = LargeRoundUp(full_job->job.memory_request, Kbytes(4));
|
||||||
|
thread_memory->data = LinuxGetMemory(new_size);
|
||||||
|
thread_memory->size = new_size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
full_job->job.callback(linuxvars.system, thread, thread_memory,
|
||||||
|
&exchange_vars.thread, full_job->job.data);
|
||||||
|
full_job->running_thread = 0;
|
||||||
|
thread->running = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else{
|
||||||
|
Linux_Semaphore_Handle* h = (Linux_Semaphore_Handle*)&(queue->semaphore);
|
||||||
|
pthread_cond_wait(h->cond_p, h->mutex_p);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Sys_Post_Job_Sig(system_post_job){
|
||||||
// TODO(allen): Implement
|
// TODO(allen): Implement
|
||||||
|
DBG_FN;
|
||||||
AllowLocal(group_id);
|
AllowLocal(group_id);
|
||||||
AllowLocal(job_id);
|
AllowLocal(job);
|
||||||
|
|
||||||
|
Work_Queue *queue = exchange_vars.thread.queues + group_id;
|
||||||
|
|
||||||
|
Assert((queue->write_position + 1) % QUEUE_WRAP != queue->read_position % QUEUE_WRAP);
|
||||||
|
|
||||||
|
b32 success = 0;
|
||||||
|
u32 result = 0;
|
||||||
|
while (!success){
|
||||||
|
u32 write_index = queue->write_position;
|
||||||
|
u32 next_write_index = (write_index + 1) % JOB_ID_WRAP;
|
||||||
|
u32 safe_write_index =
|
||||||
|
__sync_val_compare_and_swap(&queue->write_position,
|
||||||
|
write_index, next_write_index);
|
||||||
|
if (safe_write_index == write_index){
|
||||||
|
result = write_index;
|
||||||
|
write_index = write_index % QUEUE_WRAP;
|
||||||
|
queue->jobs[write_index].job = job;
|
||||||
|
queue->jobs[write_index].running_thread = THREAD_NOT_ASSIGNED;
|
||||||
|
queue->jobs[write_index].id = result;
|
||||||
|
success = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Linux_Semaphore_Handle* h = (Linux_Semaphore_Handle*)&(queue->semaphore);
|
||||||
|
pthread_cond_broadcast(h->cond_p);
|
||||||
|
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
Sys_Acquire_Lock_Sig(system_acquire_lock){
|
Sys_Acquire_Lock_Sig(system_acquire_lock){
|
||||||
// TODO(allen): Implement
|
// printf("%s: id: %d\n", __PRETTY_FUNCTION__, id);
|
||||||
AllowLocal(id);
|
pthread_mutex_lock(&linuxvars.locks[id].mutex);
|
||||||
|
//pthread_cond_wait(&linuxvars.locks[id].cond, &linuxvars.locks[id].mutex);
|
||||||
|
pthread_mutex_unlock(&linuxvars.locks[id].mutex);
|
||||||
}
|
}
|
||||||
|
|
||||||
Sys_Release_Lock_Sig(system_release_lock){
|
Sys_Release_Lock_Sig(system_release_lock){
|
||||||
// TODO(allen): Implement
|
//printf("%s: id: %d\n", __PRETTY_FUNCTION__, id);
|
||||||
AllowLocal(id);
|
pthread_mutex_lock(&linuxvars.locks[id].mutex);
|
||||||
|
pthread_cond_broadcast(&linuxvars.locks[id].cond);
|
||||||
|
pthread_mutex_unlock(&linuxvars.locks[id].mutex);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Sys_Cancel_Job_Sig(system_cancel_job){
|
||||||
|
DBG_FN;
|
||||||
|
AllowLocal(group_id);
|
||||||
|
AllowLocal(job_id);
|
||||||
|
|
||||||
|
Work_Queue *queue = exchange_vars.thread.queues + group_id;
|
||||||
|
Thread_Group *group = linuxvars.groups + group_id;
|
||||||
|
|
||||||
|
u32 job_index;
|
||||||
|
u32 thread_id;
|
||||||
|
Full_Job_Data *full_job;
|
||||||
|
Thread_Context *thread;
|
||||||
|
|
||||||
|
job_index = job_id % QUEUE_WRAP;
|
||||||
|
full_job = queue->jobs + job_index;
|
||||||
|
|
||||||
|
Assert(full_job->id == job_id);
|
||||||
|
thread_id =
|
||||||
|
__sync_val_compare_and_swap(&full_job->running_thread,
|
||||||
|
THREAD_NOT_ASSIGNED, 0);
|
||||||
|
|
||||||
|
if (thread_id != THREAD_NOT_ASSIGNED){
|
||||||
|
system_acquire_lock(CANCEL_LOCK0 + thread_id - 1);
|
||||||
|
thread = group->threads + thread_id - 1;
|
||||||
|
pthread_kill(thread->handle, SIGINT); //NOTE(inso) SIGKILL if you really want it to die.
|
||||||
|
pthread_create(&thread->handle, NULL, &ThreadProc, thread);
|
||||||
|
system_release_lock(CANCEL_LOCK0 + thread_id - 1);
|
||||||
|
thread->running = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
Sys_Grow_Thread_Memory_Sig(system_grow_thread_memory){
|
Sys_Grow_Thread_Memory_Sig(system_grow_thread_memory){
|
||||||
void *old_data;
|
void *old_data;
|
||||||
i32 old_size, new_size;
|
i32 old_size, new_size;
|
||||||
|
@ -402,6 +563,7 @@ INTERNAL_Sys_Sentinel_Sig(internal_sentinel){
|
||||||
|
|
||||||
INTERNAL_Sys_Get_Thread_States_Sig(internal_get_thread_states){
|
INTERNAL_Sys_Get_Thread_States_Sig(internal_get_thread_states){
|
||||||
// TODO(allen): Implement
|
// TODO(allen): Implement
|
||||||
|
DBG_FN;
|
||||||
AllowLocal(id);
|
AllowLocal(id);
|
||||||
AllowLocal(running);
|
AllowLocal(running);
|
||||||
AllowLocal(pending);
|
AllowLocal(pending);
|
||||||
|
@ -502,10 +664,44 @@ out:
|
||||||
internal
|
internal
|
||||||
Sys_Save_File_Sig(system_save_file){
|
Sys_Save_File_Sig(system_save_file){
|
||||||
b32 result = 0;
|
b32 result = 0;
|
||||||
// TODO(allen): Implement
|
DBG_FN;
|
||||||
AllowLocal(filename);
|
|
||||||
AllowLocal(data);
|
const size_t save_fsz = strlen(filename);
|
||||||
AllowLocal(size);
|
const char tmp_end[] = ".4ed.XXXXXX";
|
||||||
|
char* tmp_fname = (char*) alloca(save_fsz + sizeof(tmp_end));
|
||||||
|
|
||||||
|
memcpy(tmp_fname, filename, save_fsz);
|
||||||
|
memcpy(tmp_fname + save_fsz, tmp_end, sizeof(tmp_end));
|
||||||
|
|
||||||
|
int tmp_fd = mkstemp(tmp_fname);
|
||||||
|
if(tmp_fd == -1){
|
||||||
|
perror("system_save_file: mkstemp");
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t remaining = size;
|
||||||
|
do {
|
||||||
|
ssize_t written = write(tmp_fd, data, size);
|
||||||
|
if(written == -1){
|
||||||
|
if(errno == EINTR){
|
||||||
|
continue;
|
||||||
|
} else {
|
||||||
|
perror("system_save_file: write");
|
||||||
|
unlink(tmp_fname);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
remaining -= written;
|
||||||
|
}
|
||||||
|
} while(remaining);
|
||||||
|
|
||||||
|
if(rename(tmp_fname, filename) == -1){
|
||||||
|
perror("system_save_file: rename");
|
||||||
|
unlink(tmp_fname);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
result = 1;
|
||||||
return(result);
|
return(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1060,7 +1256,8 @@ InitializeXInput(Display *dpy, Window XWindow)
|
||||||
PointerMotionMask |
|
PointerMotionMask |
|
||||||
FocusChangeMask |
|
FocusChangeMask |
|
||||||
StructureNotifyMask |
|
StructureNotifyMask |
|
||||||
MappingNotify
|
MappingNotify |
|
||||||
|
ExposureMask
|
||||||
);
|
);
|
||||||
|
|
||||||
result.input_method = XOpenIM(dpy, 0, 0, 0);
|
result.input_method = XOpenIM(dpy, 0, 0, 0);
|
||||||
|
@ -1230,7 +1427,38 @@ main(int argc, char **argv)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// TODO(allen): Setup background threads and locks
|
// TODO(allen): Setup background threads and locks
|
||||||
|
|
||||||
|
Thread_Context background[4] = {};
|
||||||
|
linuxvars.groups[BACKGROUND_THREADS].threads = background;
|
||||||
|
linuxvars.groups[BACKGROUND_THREADS].count = ArrayCount(background);
|
||||||
|
|
||||||
|
Thread_Memory thread_memory[ArrayCount(background)];
|
||||||
|
linuxvars.thread_memory = thread_memory;
|
||||||
|
|
||||||
|
pthread_mutex_init(&linuxvars.thread_locks[BACKGROUND_THREADS].mutex, NULL);
|
||||||
|
pthread_cond_init(&linuxvars.thread_locks[BACKGROUND_THREADS].cond, NULL);
|
||||||
|
|
||||||
|
exchange_vars.thread.queues[BACKGROUND_THREADS].semaphore =
|
||||||
|
LinuxSemToHandle(&linuxvars.thread_locks[BACKGROUND_THREADS]);
|
||||||
|
|
||||||
|
for(i32 i = 0; i < linuxvars.groups[BACKGROUND_THREADS].count; ++i){
|
||||||
|
Thread_Context *thread = linuxvars.groups[BACKGROUND_THREADS].threads + i;
|
||||||
|
thread->id = i + 1;
|
||||||
|
|
||||||
|
Thread_Memory *memory = linuxvars.thread_memory + i;
|
||||||
|
*memory = {};
|
||||||
|
memory->id = thread->id;
|
||||||
|
|
||||||
|
thread->queue = &exchange_vars.thread.queues[BACKGROUND_THREADS];
|
||||||
|
pthread_create(&thread->handle, NULL, &ThreadProc, thread);
|
||||||
|
}
|
||||||
|
|
||||||
|
Assert(linuxvars.locks);
|
||||||
|
for(i32 i = 0; i < LOCK_COUNT; ++i){
|
||||||
|
pthread_mutex_init(&linuxvars.locks[i].mutex, NULL);
|
||||||
|
pthread_cond_init(&linuxvars.locks[i].cond, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
LinuxLoadRenderCode();
|
LinuxLoadRenderCode();
|
||||||
linuxvars.target.max = Mbytes(1);
|
linuxvars.target.max = Mbytes(1);
|
||||||
linuxvars.target.push_buffer = (byte*)system_get_memory(linuxvars.target.max);
|
linuxvars.target.push_buffer = (byte*)system_get_memory(linuxvars.target.max);
|
||||||
|
@ -1400,17 +1628,11 @@ main(int argc, char **argv)
|
||||||
push_key(0, 0, 0, &mods, is_hold);
|
push_key(0, 0, 0, &mods, is_hold);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
linuxvars.redraw = 1;
|
|
||||||
}break;
|
|
||||||
|
|
||||||
case KeyRelease: {
|
|
||||||
linuxvars.redraw = 1;
|
|
||||||
}break;
|
}break;
|
||||||
|
|
||||||
case MotionNotify: {
|
case MotionNotify: {
|
||||||
linuxvars.mouse_data.x = Event.xmotion.x;
|
linuxvars.mouse_data.x = Event.xmotion.x;
|
||||||
linuxvars.mouse_data.y = Event.xmotion.y;
|
linuxvars.mouse_data.y = Event.xmotion.y;
|
||||||
linuxvars.redraw = 1;
|
|
||||||
}break;
|
}break;
|
||||||
|
|
||||||
case ButtonPress: {
|
case ButtonPress: {
|
||||||
|
@ -1424,7 +1646,6 @@ main(int argc, char **argv)
|
||||||
linuxvars.mouse_data.right_button = 1;
|
linuxvars.mouse_data.right_button = 1;
|
||||||
} break;
|
} break;
|
||||||
}
|
}
|
||||||
linuxvars.redraw = 1;
|
|
||||||
}break;
|
}break;
|
||||||
|
|
||||||
case ButtonRelease: {
|
case ButtonRelease: {
|
||||||
|
@ -1438,24 +1659,20 @@ main(int argc, char **argv)
|
||||||
linuxvars.mouse_data.right_button = 0;
|
linuxvars.mouse_data.right_button = 0;
|
||||||
} break;
|
} break;
|
||||||
}
|
}
|
||||||
linuxvars.redraw = 1;
|
|
||||||
}break;
|
}break;
|
||||||
|
|
||||||
case EnterNotify: {
|
case EnterNotify: {
|
||||||
linuxvars.mouse_data.out_of_window = 0;
|
linuxvars.mouse_data.out_of_window = 0;
|
||||||
linuxvars.redraw = 1;
|
|
||||||
}break;
|
}break;
|
||||||
|
|
||||||
case LeaveNotify: {
|
case LeaveNotify: {
|
||||||
linuxvars.mouse_data.out_of_window = 1;
|
linuxvars.mouse_data.out_of_window = 1;
|
||||||
linuxvars.redraw = 1;
|
|
||||||
}break;
|
}break;
|
||||||
|
|
||||||
case FocusIn:
|
case FocusIn:
|
||||||
case FocusOut: {
|
case FocusOut: {
|
||||||
linuxvars.mouse_data.left_button = 0;
|
linuxvars.mouse_data.left_button = 0;
|
||||||
linuxvars.mouse_data.right_button = 0;
|
linuxvars.mouse_data.right_button = 0;
|
||||||
linuxvars.redraw = 1;
|
|
||||||
}break;
|
}break;
|
||||||
|
|
||||||
case ConfigureNotify: {
|
case ConfigureNotify: {
|
||||||
|
@ -1559,6 +1776,10 @@ main(int argc, char **argv)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}break;
|
}break;
|
||||||
|
|
||||||
|
case Expose: {
|
||||||
|
linuxvars.redraw = 1;
|
||||||
|
}break;
|
||||||
}
|
}
|
||||||
|
|
||||||
PrevEvent = Event;
|
PrevEvent = Event;
|
||||||
|
@ -1582,6 +1803,11 @@ main(int argc, char **argv)
|
||||||
mouse = linuxvars.mouse_data;
|
mouse = linuxvars.mouse_data;
|
||||||
|
|
||||||
result.mouse_cursor_type = APP_MOUSE_CURSOR_DEFAULT;
|
result.mouse_cursor_type = APP_MOUSE_CURSOR_DEFAULT;
|
||||||
|
|
||||||
|
if(__sync_bool_compare_and_swap(&exchange_vars.thread.force_redraw, 1, 0)){
|
||||||
|
linuxvars.redraw = 1;
|
||||||
|
}
|
||||||
|
|
||||||
result.redraw = linuxvars.redraw;
|
result.redraw = linuxvars.redraw;
|
||||||
result.lctrl_lalt_is_altgr = 0;
|
result.lctrl_lalt_is_altgr = 0;
|
||||||
|
|
||||||
|
@ -1613,6 +1839,62 @@ main(int argc, char **argv)
|
||||||
linuxvars.mouse_data.left_button_released = 0;
|
linuxvars.mouse_data.left_button_released = 0;
|
||||||
linuxvars.mouse_data.right_button_pressed = 0;
|
linuxvars.mouse_data.right_button_pressed = 0;
|
||||||
linuxvars.mouse_data.right_button_released = 0;
|
linuxvars.mouse_data.right_button_released = 0;
|
||||||
|
|
||||||
|
ProfileStart(OS_file_process);
|
||||||
|
{
|
||||||
|
File_Slot *file;
|
||||||
|
int d = 0;
|
||||||
|
|
||||||
|
for (file = exchange_vars.file.active.next;
|
||||||
|
file != &exchange_vars.file.active;
|
||||||
|
file = file->next){
|
||||||
|
++d;
|
||||||
|
|
||||||
|
if (file->flags & FEx_Save){
|
||||||
|
Assert((file->flags & FEx_Request) == 0);
|
||||||
|
file->flags &= (~FEx_Save);
|
||||||
|
if (system_save_file(file->filename, (char*)file->data, file->size)){
|
||||||
|
file->flags |= FEx_Save_Complete;
|
||||||
|
}
|
||||||
|
else{
|
||||||
|
file->flags |= FEx_Save_Failed;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (file->flags & FEx_Request){
|
||||||
|
Assert((file->flags & FEx_Save) == 0);
|
||||||
|
file->flags &= (~FEx_Request);
|
||||||
|
Data sysfile =
|
||||||
|
system_load_file(file->filename);
|
||||||
|
if (sysfile.data == 0){
|
||||||
|
file->flags |= FEx_Not_Exist;
|
||||||
|
}
|
||||||
|
else{
|
||||||
|
file->flags |= FEx_Ready;
|
||||||
|
file->data = sysfile.data;
|
||||||
|
file->size = sysfile.size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Assert(d == exchange_vars.file.num_active);
|
||||||
|
|
||||||
|
for (file = exchange_vars.file.free_list.next;
|
||||||
|
file != &exchange_vars.file.free_list;
|
||||||
|
file = file->next){
|
||||||
|
if (file->data){
|
||||||
|
system_free_memory(file->data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (exchange_vars.file.free_list.next != &exchange_vars.file.free_list){
|
||||||
|
ex__insert_range(exchange_vars.file.free_list.next, exchange_vars.file.free_list.prev,
|
||||||
|
&exchange_vars.file.available);
|
||||||
|
}
|
||||||
|
|
||||||
|
ex__check(&exchange_vars.file);
|
||||||
|
}
|
||||||
|
ProfileEnd(OS_file_process);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue