CringeDB/System Abstraction/linux/cdb_file.c

105 lines
2.1 KiB
C
Raw Normal View History

2024-02-18 17:42:05 +01:00
#include "../cdb_system.h"
2022-08-28 15:50:14 +02:00
#define _GNU_SOURCE
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <sys/errno.h>
2022-08-28 20:31:53 +02:00
#include <sys/file.h>
2022-08-28 15:50:14 +02:00
#include <fcntl.h>
2024-02-18 17:42:05 +01:00
struct sys_File_impl{
void * memory;
2022-08-28 15:50:14 +02:00
size_t length;
int fd;
};
2024-02-18 17:42:05 +01:00
sys_File sys_fileOpenInMem(char * fileName, unsigned long parameter) {
2022-08-28 15:50:14 +02:00
int flags = O_RDWR;
2024-02-18 17:42:05 +01:00
if(parameter & SYS_FILE_NEW) flags |= O_CREAT | O_EXCL;
2022-08-28 15:50:14 +02:00
int fd = open(fileName, flags, S_IRWXU);
if(fd == -1) return NULL;
2022-08-29 17:27:24 +02:00
if(flock(fd, LOCK_EX | LOCK_NB)) {
close(fd);
return NULL;
}
2022-08-28 20:31:53 +02:00
2022-08-28 15:50:14 +02:00
struct stat info;
if(stat(fileName, &info)) {
close(fd);
return NULL;
}
2024-02-18 17:42:05 +01:00
if((parameter & SYS_FILE_TEMP) == SYS_FILE_TEMP) unlink(fileName);
2022-08-28 15:50:14 +02:00
if(info.st_size == 0) {
write(fd, "", 1);
info.st_size++;
}
void *mem = mmap(NULL, info.st_size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
if(mem == MAP_FAILED) {
2022-08-29 15:24:28 +02:00
close(fd);
2022-08-28 15:50:14 +02:00
return NULL;
}
2024-02-18 17:42:05 +01:00
sys_File file = (sys_File) sys_heapAlloc(sizeof(struct sys_File_impl));
file->memory = mem;
2022-08-28 15:50:14 +02:00
file->length = info.st_size;
file->fd = fd;
return file;
}
2024-02-18 17:42:05 +01:00
void * sys_fileFileToMemory(sys_File file) {
return file->memory;
2022-08-28 15:50:14 +02:00
}
2024-02-18 17:42:05 +01:00
sys_Bool sys_fileExists(char * fileName) {
2022-08-28 15:50:14 +02:00
struct stat bro;
return !stat(fileName, &bro);
}
2024-02-18 17:42:05 +01:00
void sys_fileClose(sys_File file) {
if(file->memory != NULL) {
msync(file->memory, file->length, MS_SYNC);
munmap(file->memory, file->length);
2022-08-29 17:27:24 +02:00
close(file->fd);
}
2024-02-18 17:42:05 +01:00
sys_heapFree(file);
2022-08-28 15:50:14 +02:00
}
2024-02-18 17:42:05 +01:00
void sys_fileFlush(sys_File file) {
msync(file->memory, file->length, MS_SYNC);
2022-08-28 15:50:14 +02:00
}
2024-02-18 17:42:05 +01:00
void * sys_fileResize(sys_File file, signed long appendBytes) {
2022-08-28 15:50:14 +02:00
size_t newSize = file->length + appendBytes;
lseek(file->fd, newSize - 1, SEEK_SET);
write(file->fd, "", 1);
2024-02-18 17:42:05 +01:00
munmap(file->memory, file->length);
2022-08-28 15:50:14 +02:00
void *newMem = mmap(NULL, newSize, PROT_READ | PROT_WRITE, MAP_SHARED, file->fd, 0);
if(newMem == MAP_FAILED) {
2024-02-18 17:42:05 +01:00
sys_fileClose(file);
2022-08-28 15:50:14 +02:00
file->fd = 0;
file->length = 0;
2024-02-18 17:42:05 +01:00
file->memory = NULL;
2022-08-28 15:50:14 +02:00
return NULL;
}
file->length = newSize;
2024-02-18 17:42:05 +01:00
file->memory = newMem;
2022-08-28 15:50:14 +02:00
return newMem;
}
2024-02-18 17:42:05 +01:00
void sys_fileDelete(char * fileName) {
2022-08-28 15:50:14 +02:00
unlink(fileName);
}
2024-02-18 17:42:05 +01:00
void sys_fileRename(char * fileName, char * newFileName) {
2022-08-28 15:50:14 +02:00
link(fileName, newFileName);
unlink(fileName);
}