/* 4tech_file_moving.h - Code for moving files around on the file system. By Allen Webster 21.01.2017 (dd.mm.yyyy) */ // TOP #if !defined(FTECH_FILE_MOVING_H) #define FTECH_FILE_MOVING_H #include "../4ed_os_comp_cracking.h" #include // include system for windows #include // include system for linux (YAY!) #include #include // // API // // System commands static char SF_CMD[4096]; static i32 error_state = 0; static i32 prev_error = 0; #if defined(FM_PRINT_COMMANDS) #define SYSTEMF_PRINTF(...) printf(__VA_ARGS__); #else #define SYSTEMF_PRINTF(...) #endif #define systemf(...) do{ \ int32_t n = snprintf(SF_CMD, sizeof(SF_CMD), __VA_ARGS__); \ AllowLocal(n); \ Assert(n < sizeof(SF_CMD)); \ SYSTEMF_PRINTF("%s\n", SF_CMD); \ prev_error = system(SF_CMD); \ if (prev_error != 0) error_state = 1; \ }while(0) internal void fm_execute_in_dir(char *dir, char *str, char *args); // Init internal void fm_init_system(); // Timing internal u64 fm_get_time(); #define LLU_CAST(n) (long long unsigned int)(n) #define BEGIN_TIME_SECTION() uint64_t start = fm_get_time() #define END_TIME_SECTION(n) uint64_t total = fm_get_time() - start; printf("%-20s: %.2llu.%.6llu\n", (n), LLU_CAST(total/1000000), LLU_CAST(total%1000000)); // Files and Folders Manipulation internal void fm_make_folder_if_missing(char *dir); internal void fm_clear_folder(char *folder); internal void fm_delete_file(char *file); internal void fm_copy_file(char *file, char *newname); internal void fm_copy_all(char *source, char *tag, char *folder); internal void fm_copy_folder(char *src_dir, char *dst_dir, char *src_folder); // Zip internal void fm_zip(char *parent, char *folder, char *dest); // Slash Correction internal void fm_slash_fix(char *path); // Memory concat helpers internal char *fm_prepare_string_internal(char *s1, ...); #define fm_str(...) fm_prepare_string_internal(__VA_ARGS__, 0) internal char **fm_prepare_list_internal(char **l1, ...); #define fm_list(...) fm_prepare_list_internal(__VA_ARGS__, 0) internal char **fm_list_one_item(char *item); internal void *fm__push(umem size); internal void fm_align(); #define fm_push_array(T,c) (T*)fm__push(sizeof(T)*c) // File System Navigation typedef umem Temp; internal Temp fm_begin_temp(); internal void fm_end_temp(Temp temp); internal i32 fm_get_current_directory(char *buffer, i32 max); typedef struct Temp_Dir{ char dir[512]; } Temp_Dir; internal Temp_Dir fm_pushdir(char *dir); internal void fm_popdir(Temp_Dir temp); // Build Line #define BUILD_LINE_MAX 4096 typedef struct Build_Line{ char build_optionsA[BUILD_LINE_MAX]; char build_optionsB[BUILD_LINE_MAX]; char *build_options; char *build_options_prev; i32 build_max; } Build_Line; internal void fm_init_build_line(Build_Line *line); internal void fm_finish_build_line(Build_Line *line); internal void fm__swap_ptr(char **A, char **B); #if defined(IS_CL) #define fm_add_to_line(line, str, ...) do{ \ snprintf(line.build_options, \ line.build_max, "%s "str, \ line.build_options_prev, __VA_ARGS__); \ fm__swap_ptr(&line.build_options, &line.build_options_prev); \ }while(0) #elif defined(IS_GCC) #define fm_add_to_line(line, str, ...) do{ \ snprintf(line.build_options, line.build_max, "%s "str, \ line.build_options_prev, ##__VA_ARGS__); \ fm__swap_ptr(&line.build_options, &line.build_options_prev); \ }while(0) #endif // Slashes #if defined(IS_WINDOWS) #define SLASH "\\" static char platform_correct_slash = '\\'; #elif defined(IS_LINUX) || defined(IS_MAC) #define SLASH "/" static char platform_correct_slash = '/'; #else #error Slash not set for this platform. #endif // File Extensions #if defined(IS_WINDOWS) # define EXE ".exe" #elif defined(IS_LINUX) || defined(IS_MAC) # define EXE "" #else # error No EXE format specified for this OS #endif #if defined(IS_WINDOWS) # define PDB ".pdb" #elif defined(IS_LINUX) || defined(IS_MAC) # define PDB "" #else # error No PDB format specified for this OS #endif #if defined(IS_WINDOWS) # define DLL ".dll" #elif defined(IS_LINUX) || defined(IS_MAC) # define DLL ".so" #else # error No DLL format specified for this OS #endif #if defined(IS_WINDOWS) # define BAT ".bat" #elif defined(IS_LINUX) || defined(IS_MAC) # define BAT ".sh" #else # error No BAT format specified for this OS #endif #endif // // Implementation // #if defined(FTECH_FILE_MOVING_IMPLEMENTATION) && !defined(FTECH_FILE_MOVING_IMPL_GUARD) #define FTECH_FILE_MOVING_IMPL_GUARD char *fm_arena_memory = 0; umem fm_arena_pos = 0; umem fm_arena_max = 0; internal void fm__init_memory(){ Assert(fm_arena_memory == 0); fm_arena_max = MB(16); fm_arena_memory = (char*)malloc(fm_arena_max); } internal Temp fm_begin_temp(){ return(fm_arena_pos); } internal void fm_end_temp(Temp temp){ fm_arena_pos = temp; } internal void* fm__push(umem size){ void *result = fm_arena_memory + fm_arena_pos; if (size + fm_arena_pos > fm_arena_max){ result = 0; } else{ fm_arena_pos += size; } return(result); } internal void fm_align(){ fm_arena_pos = (fm_arena_pos+7)&(~7); } #if defined(IS_WINDOWS) typedef uint32_t DWORD; typedef int32_t LONG; typedef int64_t LONGLONG; typedef char* LPTSTR; typedef char* LPCTSTR; typedef int32_t BOOL; typedef void* LPSECURITY_ATTRIBUTES; typedef union _LARGE_INTEGER { struct { DWORD LowPart; LONG HighPart; }; struct { DWORD LowPart; LONG HighPart; } u; LONGLONG QuadPart; } LARGE_INTEGER, *PLARGE_INTEGER; #define WINAPI extern "C"{ DWORD WINAPI GetCurrentDirectoryA(_In_ DWORD nBufferLength, _Out_ LPTSTR lpBuffer); BOOL WINAPI SetCurrentDirectoryA(_In_ LPCTSTR lpPathName); BOOL WINAPI QueryPerformanceCounter(_Out_ LARGE_INTEGER *lpPerformanceCount); BOOL WINAPI QueryPerformanceFrequency(_Out_ LARGE_INTEGER *lpFrequency); BOOL WINAPI CreateDirectoryA(_In_ LPCTSTR lpPathName, _In_opt_ LPSECURITY_ATTRIBUTES lpSecurityAttributes); BOOL WINAPI CopyFileA(_In_ LPCTSTR lpExistingFileName, _In_ LPCTSTR lpNewFileName, _In_ BOOL bFailIfExists); } static uint64_t perf_frequency; static void fm_init_system(){ LARGE_INTEGER lint; if (QueryPerformanceFrequency(&lint)){ perf_frequency = lint.QuadPart; } fm__init_memory(); } static Temp_Dir fm_pushdir(char *dir){ Temp_Dir temp = {0}; GetCurrentDirectoryA(sizeof(temp.dir), temp.dir); SetCurrentDirectoryA(dir); return(temp); } static void fm_popdir(Temp_Dir temp){ SetCurrentDirectoryA(temp.dir); } static uint64_t fm_get_time(){ uint64_t time = 0; LARGE_INTEGER lint; if (QueryPerformanceCounter(&lint)){ time = lint.QuadPart; time = (time * 1000000) / perf_frequency; } return(time); } static int32_t fm_get_current_directory(char *buffer, int32_t max){ int32_t result = GetCurrentDirectoryA(max, buffer); return(result); } static void fm_execute_in_dir(char *dir, char *str, char *args){ if (dir){ Temp_Dir temp = fm_pushdir(dir); if (args){ systemf("call \"%s\" %s", str, args); } else{ systemf("call \"%s\"", str); } fm_popdir(temp); } else{ if (args){ systemf("call \"%s\" %s", str, args); } else{ systemf("call \"%s\"", str); } } } static void fm_slash_fix(char *path){ if (path != 0){ for (int32_t i = 0; path[i]; ++i){ if (path[i] == '/') path[i] = '\\'; } } } static void fm_make_folder_if_missing(char *dir){ char *path = fm_str(dir); char *p = path; for (; *p; ++p){ if (*p == '\\'){ *p = 0; CreateDirectoryA(path, 0); *p = '\\'; } } CreateDirectoryA(path, 0); } static void fm_clear_folder(char *folder){ systemf("del /S /Q /F %s\\* & rmdir /S /Q %s & mkdir %s", folder, folder, folder); } static void fm_delete_file(char *file){ systemf("del %s", file); } static void fm_copy_file(char *file, char *newname){ CopyFileA(file, newname, 0); } static void fm_copy_all(char *source, char *tag, char *folder){ if (source){ systemf("copy %s\\%s %s\\*", source, tag, folder); } else{ systemf("copy %s %s\\*", tag, folder); } } static void fm_zip(char *parent, char *folder, char *dest){ char cdir[512]; fm_get_current_directory(cdir, sizeof(cdir)); Temp_Dir temp = fm_pushdir(parent); systemf("%s\\zip %s\\4ed_gobble.zip", cdir, cdir); fm_popdir(temp); systemf("copy %s\\4ed_gobble.zip %s & del %s\\4ed_gobble.zip", cdir, dest, cdir); } #elif defined(IS_LINUX) || defined(IS_MAC) #include #include static Temp_Dir fm_pushdir(char *dir){ Temp_Dir temp; char *result = getcwd(temp.dir, sizeof(temp.dir)); int32_t chresult = chdir(dir); if (result == 0 || chresult != 0){ printf("trying pushdir %s\n", dir); Assert(result != 0); Assert(chresult == 0); } return(temp); } static void fm_popdir(Temp_Dir temp){ chdir(temp.dir); } static void fm_init_system(){ fm__init_memory(); } static uint64_t fm_get_time(){ struct timespec spec; uint64_t result; clock_gettime(CLOCK_MONOTONIC, &spec); result = (spec.tv_sec * (uint64_t)(1000000)) + (spec.tv_nsec / (uint64_t)(1000)); return(result); } static int32_t fm_get_current_directory(char *buffer, int32_t max){ int32_t result = 0; char *d = getcwd(buffer, max); if (d == buffer){ result = strlen(buffer); } return(result); } static void fm_execute_in_dir(char *dir, char *str, char *args){ if (dir){ if (args){ Temp_Dir temp = fm_pushdir(dir); systemf("%s %s", str, args); fm_popdir(temp); } else{ Temp_Dir temp = fm_pushdir(dir); systemf("%s", str); fm_popdir(temp); } } else{ if (args){ systemf("%s %s", str, args); } else{ systemf("%s", str); } } } static void fm_slash_fix(char *path){} static void fm_make_folder_if_missing(char *dir){ systemf("mkdir -p %s", dir); } static void fm_clear_folder(char *folder){ systemf("rm -rf %s*", folder); } static void fm_delete_file(char *file){ systemf("rm %s", file); } static void fm_copy_file(char *file, char *newname){ systemf("cp %s %s", file, newname); } static void fm_copy_all(char *source, char *tag, char *folder){ if (source){ systemf("cp -f %s/%s %s", source, tag, folder); } else{ systemf("cp -f %s %s", tag, folder); } } static void fm_zip(char *parent, char *folder, char *file){ Temp_Dir temp = fm_pushdir(parent); printf("PARENT DIR: %s\n", parent); systemf("zip -r %s %s", file, folder); fm_popdir(temp); } #else #error This OS is not supported yet #endif internal void fm_copy_folder(char *src_dir, char *dst_dir, char *src_folder){ Temp_Dir temp = fm_pushdir(src_dir); fm_make_folder_if_missing(fm_str(dst_dir, "/", src_folder)); char *copy_name = fm_str(dst_dir, "/", src_folder); fm_copy_all(src_folder, "*", copy_name); fm_popdir(temp); } // List Helpers internal umem listsize(void *p, umem item_size){ u64 zero = 0; u8 *ptr = (u8*)p; for (;memcmp(ptr, &zero, item_size) != 0; ptr += item_size); umem size = (ptr - (u8*)p); return(size); } internal void* fm__prepare(umem item_size, void *i1, va_list list){ umem size = listsize(i1, item_size); void *result = (void*)fm__push(size); memcpy(result, i1, size); void *ln = va_arg(list, void*); for (;ln != 0;){ size = listsize(ln, item_size); void *new_str = (void*)fm__push(size); memcpy(new_str, ln, size); ln = va_arg(list, void*); } void *terminator = (void*)fm__push(item_size); memset(terminator, 0, item_size); return(result); } internal char* fm_prepare_string_internal(char *s1, ...){ umem item_size = sizeof(*s1); va_list list; va_start(list, s1); char *result = (char*)fm__prepare(item_size, s1, list); va_end(list); fm_slash_fix(result); return(result); } internal char** fm_prepare_list_internal(char **p1, ...){ umem item_size = sizeof(*p1); va_list list; va_start(list, p1); char **result = (char**)fm__prepare(item_size, p1, list); va_end(list); return(result); } internal char** fm_list_one_item(char *item){ char **result = (char**)fm__push(sizeof(char*)*2); result[0] = item; result[1] = 0; return(result); } // Build Line internal void fm_init_build_line(Build_Line *line){ line->build_options = line->build_optionsA; line->build_options_prev = line->build_optionsB; line->build_optionsA[0] = 0; line->build_optionsB[0] = 0; line->build_max = BUILD_LINE_MAX; } internal void fm_finish_build_line(Build_Line *line){ fm__swap_ptr(&line->build_options, &line->build_options_prev); } internal void fm__swap_ptr(char **A, char **B){ char *a = *A; char *b = *B; *A = b; *B = a; } #endif // BOTTOM