/* * Mr. 4th Dimention - Allen Webster * * 30.06.2017 * * General unix functions * */ // TOP struct Unix_Vars{ b32 did_first_log; }; global Unix_Vars unixvars; // // 4ed Path // internal Sys_Get_4ed_Path_Sig(system_get_4ed_path){ ssize_t size = readlink("/proc/self/exe", out, capacity - 1); if (size != -1 && size < capacity - 1){ String str = make_string(out, size); remove_last_folder(&str); terminate_with_null(&str); size = str.size; } else{ size = 0; } return(size); } // // Logging // internal Sys_Log_Sig(system_log){ if (plat_settings.use_log == LogTo_LogFile){ char file_path_space[1024]; String file_path = make_fixed_width_string(file_path_space); file_path.size = system_get_4ed_path(file_path.str, file_path.memory_size); append_sc(&file_path, "4coder_log.txt"); terminate_with_null(&file_path); i32 fd = -1; if (unixvars.did_first_log){ fd = open(file_path.str, O_WRONLY | O_CREAT | O_APPEND, 00640); } else{ fd = open(file_path.str, O_WRONLY | O_CREAT | O_TRUNC, 00640); unixvars.did_first_log = true; } if (fd >= 0){ do{ ssize_t written = write(fd, message, length); if (written != -1){ length -= written; message += written; } } while(length > 0); close(fd); } } else if (plat_settings.use_log == LogTo_Stdout){ fwrite(message, 1, length, stdout); } } // // Shared system functions (system_shared.h) // internal Sys_File_Can_Be_Made_Sig(system_file_can_be_made){ b32 result = access((char*)filename, W_OK) == 0; LOGF("%s = %d\n", filename, result); return(result); } // // Memory // internal void* system_memory_allocate_extended(void *base, umem size){ // NOTE(allen): This must return the exact base of the vpage. // We will count on the user to keep track of size themselves. void *result = mmap(base, size, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANONYMOUS, -1, 0); if (result == MAP_FAILED){ LOG("error: mmap failed\n"); result = 0; } return(result); } internal Sys_Memory_Allocate_Sig(system_memory_allocate){ void *result = system_memory_allocate_extended(0, size); return(result); } internal Sys_Memory_Set_Protection_Sig(system_memory_set_protection){ bool32 result = true; int protect = 0; switch (flags & 0x7){ case 0: protect = PROT_NONE; break; case MemProtect_Read: protect = PROT_READ; break; case MemProtect_Write: case MemProtect_Read|MemProtect_Write: protect = PROT_READ | PROT_WRITE; break; case MemProtect_Execute: protect = PROT_EXEC; break; case MemProtect_Execute|MemProtect_Read: protect = PROT_READ | PROT_EXEC; break; // NOTE(inso): some W^X protection things might be unhappy about this one case MemProtect_Execute|MemProtect_Write: case MemProtect_Execute|MemProtect_Write|MemProtect_Read: protect = PROT_READ | PROT_WRITE | PROT_EXEC; break; } if(mprotect(ptr, size, protect) == -1){ result = 0; LOG("error: mprotect\n"); } return(result); } internal Sys_Memory_Free_Sig(system_memory_free){ // NOTE(allen): This must take the exact base of the vpage. munmap(ptr, size); } // // Files // internal Sys_Set_File_List_Sig(system_set_file_list){ if (directory == 0){ system_memory_free(file_list->block, file_list->block_size); file_list->block = 0; file_list->block_size = 0; file_list->infos = 0; file_list->count = 0; return; } LOGF("set_file_list: %s\n", directory); DIR *d = opendir(directory); if (d != 0){ if (canon_directory_out != 0){ u32 length = copy_fast_unsafe_cc(canon_directory_out, directory); if (canon_directory_out[length-1] != '/'){ canon_directory_out[length++] = '/'; } canon_directory_out[length] = 0; *canon_directory_size_out = length; } i32 character_count = 0; i32 file_count = 0; for (struct dirent *entry = readdir(d); entry != 0; entry = readdir(d)){ char *fname = entry->d_name; if (match_cc(fname, ".") || match_cc(fname, "..")){ continue; } ++file_count; i32 size = 0; for (; fname[size]; ++size); character_count += size + 1; } i32 required_size = character_count + file_count * sizeof(File_Info); if (file_list->block_size < required_size){ system_memory_free(file_list->block, file_list->block_size); file_list->block = system_memory_allocate(required_size); file_list->block_size = required_size; } file_list->infos = (File_Info*)file_list->block; char *cursor = (char*)(file_list->infos + file_count); if (file_list->block != 0){ rewinddir(d); File_Info *info_ptr = file_list->infos; for (struct dirent *entry = readdir(d); entry != 0; entry = readdir(d)){ char *fname = entry->d_name; if (match_cc(fname, ".") || match_cc(fname, "..")){ continue; } char *cursor_start = cursor; i32 length = copy_fast_unsafe_cc(cursor_start, fname); cursor += length; if(entry->d_type == DT_LNK){ struct stat st; if(stat(entry->d_name, &st) != -1){ info_ptr->folder = S_ISDIR(st.st_mode); } else{ info_ptr->folder = 0; } } else{ info_ptr->folder = (entry->d_type == DT_DIR); } info_ptr->filename = cursor_start; info_ptr->filename_len = length; *cursor++ = 0; ++info_ptr; } } file_list->count = file_count; closedir(d); } else{ system_memory_free(file_list->block, file_list->block_size); file_list->block = 0; file_list->block_size = 0; file_list->infos = 0; file_list->count = 0; } } internal Sys_Get_Canonical_Sig(system_get_canonical){ char* path = (char*) alloca(len + 1); char* write_p = path; const char* read_p = filename; // return 0 for relative paths (e.g. cmdline args) if(len > 0 && filename[0] != '/'){ return 0; } if (max == 0){ return 0; } max -= 1; while(read_p < filename + len){ if(read_p == filename || read_p[0] == '/'){ if(read_p[1] == '/'){ ++read_p; } else if(read_p[1] == '.'){ if(read_p[2] == '/' || !read_p[2]){ read_p += 2; } else if(read_p[2] == '.' && (read_p[3] == '/' || !read_p[3])){ while(write_p > path && *--write_p != '/'); read_p += 3; } else { *write_p++ = *read_p++; } } else { *write_p++ = *read_p++; } } else { *write_p++ = *read_p++; } } if(write_p == path) *write_p++ = '/'; if(max >= (write_p - path)){ memcpy(buffer, path, write_p - path); } else { write_p = path; } #if defined(FRED_INTERNAL) if(len != (write_p - path) || memcmp(filename, path, len) != 0){ LOGF("[%.*s] -> [%.*s]\n", len, filename, (int)(write_p - path), path); } #endif u32 length = (i32)(write_p - path); buffer[length] = 0; return(length); } internal Sys_Load_Handle_Sig(system_load_handle){ b32 result = false; i32 fd = open(filename, O_RDONLY); if (fd == -1){ LOG("error: open\n"); } else{ *(int*)handle_out = fd; result = true; } return(result); } internal Sys_Load_Size_Sig(system_load_size){ u32 result = 0; int fd = *(int*)&handle; struct stat st; if(fstat(fd, &st) == -1){ LOG("error: fstat\n"); } else{ result = st.st_size; } return result; } internal Sys_Load_File_Sig(system_load_file){ int fd = *(int*)&handle; do{ ssize_t n = read(fd, buffer, size); if(n == -1){ if(errno != EINTR){ LOG("error: read\n"); break; } } else{ size -= n; buffer += n; } } while(size); return size == 0; } internal Sys_Load_Close_Sig(system_load_close){ b32 result = 1; int fd = *(int*)&handle; if(close(fd) == -1){ LOG("error: close\n"); result = 0; } return result; } internal Sys_Save_File_Sig(system_save_file){ int fd = open(filename, O_WRONLY | O_TRUNC | O_CREAT, 00640); LOGF("%s %d\n", filename, size); if(fd < 0){ LOGF("error: open '%s': %s\n", filename, strerror(errno)); } else{ do { ssize_t written = write(fd, buffer, size); if(written == -1){ if(errno != EINTR){ LOG("error: write\n"); break; } } else { size -= written; buffer += written; } } while(size); close(fd); } return (size == 0); } // // File System // internal Sys_File_Exists_Sig(system_file_exists){ int result = 0; char buff[PATH_MAX] = {}; if (len + 1 > PATH_MAX){ LOG("system_directory_has_file: path too long\n"); } else{ memcpy(buff, filename, len); buff[len] = 0; struct stat st; result = stat(buff, &st) == 0 && S_ISREG(st.st_mode); } LOGF("%s: %d\n", buff, result); return(result); } internal b32 system_directory_exists(char *path){ struct stat st; b32 result = (stat(path, &st) == 0 && S_ISDIR(st.st_mode)); return(result); } // // Time // internal Sys_Now_Time_Sig(system_now_time){ struct timespec spec; clock_gettime(CLOCK_REALTIME, &spec); u64 result = (spec.tv_sec * UINT64_C(1000000)) + (spec.tv_nsec / UINT64_C(1000)); return(result); } // BOTTOM