2023-12-27 17:35:59 +03:00
|
|
|
/*
|
2024-01-03 22:02:31 +03:00
|
|
|
* C programming idea: Handling temporary files like memory allocations
|
|
|
|
* (allocating -> creating empty file, using -> locking for R/W,
|
|
|
|
* freeing -> deleting).
|
|
|
|
*
|
|
|
|
* 20% AI, 80% human (the code is tested and reviewed)
|
|
|
|
*
|
|
|
|
* Warning: The current result is quick and dirty. Not for educational or
|
|
|
|
* production purposes.
|
|
|
|
*
|
|
|
|
* GCC/Clang/TCC: Compile with -DTEST to set macro TEST as defined, with
|
|
|
|
* -DDEBUG to enable debug mode
|
|
|
|
*
|
|
|
|
* To-Do: Add thread-safe versions of functions (use postfix `_r`)
|
|
|
|
*/
|
2023-12-27 17:35:59 +03:00
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
#include <inttypes.h>
|
2023-12-27 21:25:39 +03:00
|
|
|
#include <math.h>
|
2024-01-03 20:05:14 +03:00
|
|
|
#include <stdbool.h>
|
2023-12-27 17:35:59 +03:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2023-12-28 00:02:52 +03:00
|
|
|
|
2023-12-28 00:39:59 +03:00
|
|
|
#if _POSIX_VERSION >= 200112L
|
2023-12-28 00:02:52 +03:00
|
|
|
# include <fcntl.h>
|
|
|
|
# include <unistd.h>
|
|
|
|
# define IS_POSIX 1
|
|
|
|
#else
|
|
|
|
# define IS_POSIX 0
|
|
|
|
#endif
|
2023-12-27 17:35:59 +03:00
|
|
|
|
2023-12-27 19:11:51 +03:00
|
|
|
#ifdef DEBUG
|
2024-01-03 21:21:29 +03:00
|
|
|
# define LINE_FAIL(x) printf("failed on line %d\n", __LINE__ + x)
|
2024-01-03 21:57:49 +03:00
|
|
|
# define DBG_PRINT(...) do { \
|
|
|
|
printf(__VA_ARGS__); \
|
|
|
|
fflush(stdout); \
|
|
|
|
} while (0)
|
2023-12-27 19:11:51 +03:00
|
|
|
#else
|
2024-01-03 21:21:29 +03:00
|
|
|
# define LINE_FAIL(x)
|
2024-01-03 21:57:49 +03:00
|
|
|
# define DBG_PRINT(...)
|
2023-12-27 19:11:51 +03:00
|
|
|
#endif
|
|
|
|
|
2024-01-03 21:21:29 +03:00
|
|
|
#define RETREAT(s) do { \
|
2023-12-27 19:11:51 +03:00
|
|
|
perror(s); \
|
|
|
|
exit(EXIT_FAILURE); \
|
|
|
|
} while (0)
|
|
|
|
|
2023-12-28 00:24:29 +03:00
|
|
|
typedef struct {
|
2024-01-03 20:05:14 +03:00
|
|
|
bool locked;
|
2023-12-28 00:24:29 +03:00
|
|
|
int ID;
|
|
|
|
char* file_path;
|
2024-01-03 19:51:43 +03:00
|
|
|
#if IS_POSIX
|
2023-12-28 00:24:29 +03:00
|
|
|
int file;
|
|
|
|
#else
|
2023-12-27 17:35:59 +03:00
|
|
|
FILE* file;
|
2023-12-28 00:24:29 +03:00
|
|
|
#endif
|
2024-01-03 19:51:43 +03:00
|
|
|
} TempFile;
|
2023-12-27 17:35:59 +03:00
|
|
|
|
|
|
|
TempFile* temp_files = NULL;
|
|
|
|
size_t num_temp_files = 0;
|
|
|
|
|
2023-12-27 19:11:00 +03:00
|
|
|
int tf_alloc(size_t n, size_t type_size);
|
|
|
|
int tf_free(int ID);
|
2023-12-30 16:04:25 +03:00
|
|
|
int tf_write(int ID, size_t offset, void* src, size_t data_size);
|
2023-12-27 19:11:00 +03:00
|
|
|
int tf_read(int ID, size_t offset, void* dest, size_t data_size);
|
|
|
|
|
2023-12-27 17:35:59 +03:00
|
|
|
int tf_alloc(size_t n, size_t type_size) {
|
2024-01-03 21:57:49 +03:00
|
|
|
DBG_PRINT("tf_alloc(%zu, %zu)\n", n, type_size);
|
|
|
|
|
2023-12-27 17:35:59 +03:00
|
|
|
// Create an empty file
|
2023-12-27 17:55:20 +03:00
|
|
|
size_t len_digit;
|
|
|
|
if (num_temp_files == 0)
|
|
|
|
len_digit = 1;
|
|
|
|
else
|
2024-01-03 21:07:45 +03:00
|
|
|
len_digit = (size_t) floor(log10((double) num_temp_files)) + 1;
|
2023-12-27 17:55:20 +03:00
|
|
|
size_t file_path_len = len_digit + strlen("tf_.tmp");
|
|
|
|
char* file_path = malloc((file_path_len + 1) * sizeof(char));
|
|
|
|
if (file_path == NULL) {
|
2024-01-03 21:21:29 +03:00
|
|
|
LINE_FAIL(-2);
|
2023-12-27 17:55:20 +03:00
|
|
|
return -1;
|
|
|
|
}
|
2023-12-28 00:04:22 +03:00
|
|
|
int res = snprintf(file_path, file_path_len + 1, "tf_%" PRIuMAX ".tmp",
|
2023-12-27 21:29:16 +03:00
|
|
|
(uintmax_t) num_temp_files);
|
2023-12-27 17:55:20 +03:00
|
|
|
if ((size_t) res != file_path_len) {
|
2024-01-03 21:21:29 +03:00
|
|
|
LINE_FAIL(-2);
|
2023-12-27 17:55:20 +03:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2023-12-28 00:24:29 +03:00
|
|
|
#if IS_POSIX
|
|
|
|
int file = open(file_path, O_RDWR | O_CREAT);
|
|
|
|
if (file == -1) {
|
|
|
|
#else
|
2023-12-27 17:35:59 +03:00
|
|
|
FILE* file = fopen(file_path, "w+b");
|
|
|
|
if (file == NULL) {
|
2023-12-28 00:24:29 +03:00
|
|
|
#endif
|
2024-01-03 21:21:29 +03:00
|
|
|
LINE_FAIL(-2);
|
2023-12-27 17:35:59 +03:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Allocate memory for the TempFile struct
|
2023-12-27 17:55:20 +03:00
|
|
|
TempFile* temp_file = malloc(sizeof(TempFile));
|
2023-12-27 17:35:59 +03:00
|
|
|
if (temp_file == NULL) {
|
2024-01-03 21:21:29 +03:00
|
|
|
LINE_FAIL(-2);
|
2023-12-27 17:35:59 +03:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2023-12-27 21:58:50 +03:00
|
|
|
// Assign the ID, file path, file handler
|
2024-01-03 20:05:14 +03:00
|
|
|
temp_file->locked = false;
|
2023-12-27 17:35:59 +03:00
|
|
|
temp_file->ID = num_temp_files;
|
|
|
|
temp_file->file_path = strdup(file_path);
|
|
|
|
temp_file->file = file;
|
|
|
|
|
2023-12-27 21:45:53 +03:00
|
|
|
// Allocate/reallocate memory for the temp_files structure
|
|
|
|
if (temp_files == NULL)
|
|
|
|
temp_files = malloc(sizeof(TempFile));
|
|
|
|
else
|
2024-01-03 21:43:22 +03:00
|
|
|
temp_files = realloc(temp_files, num_temp_files * sizeof(TempFile));
|
2023-12-27 17:35:59 +03:00
|
|
|
if (temp_files == NULL) {
|
2024-01-03 21:21:29 +03:00
|
|
|
LINE_FAIL(-2);
|
2023-12-27 17:35:59 +03:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2023-12-28 00:07:29 +03:00
|
|
|
// Add the temp file to the array
|
|
|
|
temp_files[num_temp_files++] = *temp_file;
|
|
|
|
|
2023-12-27 17:35:59 +03:00
|
|
|
return temp_file->ID;
|
|
|
|
}
|
|
|
|
|
|
|
|
int tf_free(int ID) {
|
2024-01-03 21:57:49 +03:00
|
|
|
DBG_PRINT("tf_free(%d)\n", ID);
|
|
|
|
|
2023-12-27 17:55:20 +03:00
|
|
|
size_t index = (size_t) ID;
|
2023-12-27 17:35:59 +03:00
|
|
|
|
2024-01-03 20:05:14 +03:00
|
|
|
if (temp_files[index].locked) {
|
|
|
|
errno = EBUSY;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
temp_files[index].locked = true;
|
|
|
|
|
2023-12-28 00:24:29 +03:00
|
|
|
#if IS_POSIX
|
|
|
|
close(temp_files[index].file);
|
|
|
|
#else
|
2023-12-27 17:35:59 +03:00
|
|
|
fclose(temp_files[index].file);
|
2023-12-28 00:24:29 +03:00
|
|
|
#endif
|
2023-12-27 17:35:59 +03:00
|
|
|
|
|
|
|
// Delete the file
|
|
|
|
if (remove(temp_files[index].file_path) != 0) {
|
2024-01-03 21:21:29 +03:00
|
|
|
LINE_FAIL(-1);
|
2023-12-27 17:35:59 +03:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
free(temp_files[index].file_path);
|
|
|
|
|
|
|
|
// Shift the remaining temp files in the array
|
2024-01-03 21:10:22 +03:00
|
|
|
for (size_t i = index; i < num_temp_files - 1; i++)
|
|
|
|
temp_files[i] = temp_files[i + 1];
|
2023-12-27 17:35:59 +03:00
|
|
|
|
|
|
|
// Reallocate memory for the temp_files array
|
2024-01-03 21:03:43 +03:00
|
|
|
if (--num_temp_files > 0) {
|
2024-01-03 21:57:49 +03:00
|
|
|
DBG_PRINT("num_temp_files = %zu\n", num_temp_files);
|
2024-01-03 21:03:43 +03:00
|
|
|
if ((temp_files = realloc(temp_files, num_temp_files
|
|
|
|
* sizeof(TempFile))) == NULL) {
|
2024-01-03 21:21:29 +03:00
|
|
|
LINE_FAIL(-2);
|
2024-01-03 21:03:43 +03:00
|
|
|
return -1;
|
|
|
|
}
|
2023-12-27 17:35:59 +03:00
|
|
|
}
|
|
|
|
|
2024-01-03 20:05:14 +03:00
|
|
|
temp_files[index].locked = false;
|
|
|
|
|
2023-12-27 17:35:59 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2023-12-30 16:04:25 +03:00
|
|
|
int tf_write(int ID, size_t offset, void* src, size_t data_size) {
|
2024-01-03 21:57:49 +03:00
|
|
|
DBG_PRINT("tf_write(%d, %zu, %p, %zu)\n", ID, offset, src, data_size);
|
|
|
|
|
2023-12-27 17:55:20 +03:00
|
|
|
size_t index = (size_t) ID;
|
2023-12-27 17:35:59 +03:00
|
|
|
|
2024-01-03 20:05:14 +03:00
|
|
|
if (temp_files[index].locked) {
|
|
|
|
errno = EBUSY;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
temp_files[index].locked = true;
|
|
|
|
|
2023-12-28 00:24:29 +03:00
|
|
|
#if IS_POSIX
|
|
|
|
// Check file handler for -1
|
|
|
|
int file = temp_files[index].file;
|
|
|
|
if (file == -1)
|
|
|
|
#else
|
2023-12-27 17:55:20 +03:00
|
|
|
// Check file handler for NULL
|
2023-12-27 17:35:59 +03:00
|
|
|
FILE* file = temp_files[index].file;
|
|
|
|
if (file == NULL)
|
2023-12-28 00:24:29 +03:00
|
|
|
#endif
|
2023-12-27 17:35:59 +03:00
|
|
|
return -1;
|
|
|
|
|
2023-12-27 17:55:20 +03:00
|
|
|
// Set the position
|
2023-12-28 00:24:29 +03:00
|
|
|
#if IS_POSIX
|
|
|
|
if (lseek(file, offset, SEEK_SET) == -1) {
|
|
|
|
#else
|
|
|
|
if (fseek(file, offset, SEEK_SET) == -1) {
|
|
|
|
#endif
|
2024-01-03 21:21:29 +03:00
|
|
|
LINE_FAIL(-1);
|
2023-12-28 00:24:29 +03:00
|
|
|
return -1;
|
|
|
|
}
|
2023-12-27 17:35:59 +03:00
|
|
|
|
|
|
|
// Write the data to the file
|
2023-12-28 00:24:29 +03:00
|
|
|
#if IS_POSIX
|
2024-01-03 21:49:23 +03:00
|
|
|
ssize_t
|
2023-12-28 00:24:29 +03:00
|
|
|
#else
|
2024-01-03 21:49:23 +03:00
|
|
|
size_t
|
|
|
|
#endif
|
|
|
|
bytes_written =
|
|
|
|
#if IS_POSIX
|
|
|
|
write(file, src, data_size);
|
|
|
|
#else
|
|
|
|
fwrite(src, 1, data_size, file);
|
2023-12-28 00:24:29 +03:00
|
|
|
#endif
|
2023-12-27 17:35:59 +03:00
|
|
|
|
2023-12-28 00:24:29 +03:00
|
|
|
if (
|
|
|
|
#if IS_POSIX
|
|
|
|
(size_t)
|
|
|
|
#endif
|
|
|
|
bytes_written != data_size) {
|
|
|
|
errno = EIO;
|
2023-12-27 17:35:59 +03:00
|
|
|
return -1;
|
2023-12-28 00:24:29 +03:00
|
|
|
}
|
2023-12-27 17:35:59 +03:00
|
|
|
|
2023-12-28 00:24:29 +03:00
|
|
|
#if IS_POSIX
|
|
|
|
if (fsync(file) == -1) {
|
2024-01-03 21:21:29 +03:00
|
|
|
LINE_FAIL(-1);
|
2023-12-27 21:30:42 +03:00
|
|
|
return -1;
|
|
|
|
}
|
2023-12-28 00:24:29 +03:00
|
|
|
#else
|
|
|
|
fflush(file);
|
|
|
|
#endif
|
2023-12-27 21:30:42 +03:00
|
|
|
|
2024-01-03 20:05:14 +03:00
|
|
|
temp_files[index].locked = false;
|
|
|
|
|
2023-12-27 17:35:59 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int tf_read(int ID, size_t offset, void* dest, size_t data_size) {
|
2024-01-03 21:57:49 +03:00
|
|
|
DBG_PRINT("tf_read(%d, %zu, %p, %zu)\n", ID, offset, dest, data_size);
|
|
|
|
|
2023-12-27 17:55:20 +03:00
|
|
|
size_t index = (size_t) ID;
|
2023-12-27 17:35:59 +03:00
|
|
|
|
2024-01-03 20:05:14 +03:00
|
|
|
if (temp_files[index].locked) {
|
|
|
|
errno = EBUSY;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
temp_files[index].locked = true;
|
|
|
|
|
2023-12-28 00:24:29 +03:00
|
|
|
#if IS_POSIX
|
|
|
|
int file = temp_files[index].file;
|
|
|
|
if (file == -1)
|
|
|
|
#else
|
2023-12-27 17:35:59 +03:00
|
|
|
FILE* file = temp_files[index].file;
|
|
|
|
if (file == NULL)
|
2023-12-28 00:24:29 +03:00
|
|
|
#endif
|
2023-12-27 17:35:59 +03:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
// Read the data from the file
|
2023-12-30 16:04:25 +03:00
|
|
|
void* src = malloc(data_size);
|
|
|
|
if (src == NULL) {
|
2023-12-28 00:24:29 +03:00
|
|
|
#if IS_POSIX
|
|
|
|
close(file);
|
|
|
|
#else
|
2023-12-27 17:35:59 +03:00
|
|
|
fclose(file);
|
2023-12-28 00:24:29 +03:00
|
|
|
#endif
|
2024-01-03 21:21:29 +03:00
|
|
|
LINE_FAIL(-7);
|
2023-12-27 17:35:59 +03:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2023-12-30 16:04:25 +03:00
|
|
|
memset(src, 0, data_size); // clear destination
|
2023-12-28 00:24:29 +03:00
|
|
|
// Set the position
|
|
|
|
#if IS_POSIX
|
|
|
|
if (lseek(file, offset, SEEK_SET) == -1) {
|
|
|
|
#else
|
|
|
|
if (fseek(file, offset, SEEK_SET) == -1) {
|
|
|
|
#endif
|
2024-01-03 21:21:29 +03:00
|
|
|
LINE_FAIL(-1);
|
2023-12-28 00:24:29 +03:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// read bytes
|
|
|
|
#if IS_POSIX
|
2023-12-30 16:04:25 +03:00
|
|
|
ssize_t bytes_read = read(file, src, data_size);
|
2023-12-28 00:24:29 +03:00
|
|
|
#else
|
2023-12-30 16:04:25 +03:00
|
|
|
size_t bytes_read = fread(src, 1, data_size, file);
|
2023-12-28 00:24:29 +03:00
|
|
|
#endif
|
2023-12-30 16:04:25 +03:00
|
|
|
memcpy(dest, src, data_size);
|
2023-12-27 17:35:59 +03:00
|
|
|
|
2023-12-30 16:04:25 +03:00
|
|
|
free(src); // Free the allocated memory
|
2023-12-27 17:35:59 +03:00
|
|
|
|
2023-12-28 00:24:29 +03:00
|
|
|
if (
|
|
|
|
#if IS_POSIX
|
|
|
|
(size_t)
|
|
|
|
#endif
|
|
|
|
bytes_read != data_size) {
|
2023-12-27 17:35:59 +03:00
|
|
|
errno = EIO;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2023-12-27 17:55:20 +03:00
|
|
|
#ifdef DEBUG
|
2024-01-03 21:26:13 +03:00
|
|
|
printf("Read: ID = %d, src = %p, size = %zu -> '",
|
|
|
|
ID, dest, data_size);
|
2023-12-27 17:35:59 +03:00
|
|
|
for (size_t i = 0; i < data_size; i++)
|
2024-01-03 21:30:07 +03:00
|
|
|
printf("0x%02" PRIX8 "%c", *((uint8_t*)((uint8_t*)dest + i)),
|
|
|
|
i == (data_size - 1) ? '\'' : ' ');
|
|
|
|
printf("\n");
|
2023-12-27 21:29:16 +03:00
|
|
|
fflush(stdout);
|
2023-12-27 17:55:20 +03:00
|
|
|
#endif
|
2024-01-03 20:05:14 +03:00
|
|
|
|
|
|
|
temp_files[index].locked = false;
|
|
|
|
|
2023-12-27 17:35:59 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef TEST
|
2023-12-27 18:38:05 +03:00
|
|
|
int main(void) {
|
2024-01-03 21:57:49 +03:00
|
|
|
DBG_PRINT("started\n");
|
|
|
|
|
|
|
|
int ID_1 = tf_alloc(4, sizeof(int));
|
|
|
|
int ID_2 = tf_alloc(16, sizeof(uint16_t));
|
|
|
|
if (ID_1 == -1 || ID_2 == -1)
|
2024-01-03 21:21:29 +03:00
|
|
|
RETREAT("tf_alloc");
|
2024-01-03 21:57:49 +03:00
|
|
|
DBG_PRINT("allocated memory\n");
|
2023-12-27 17:35:59 +03:00
|
|
|
|
2024-01-03 21:57:49 +03:00
|
|
|
int test_data_1[4] = {123, 456, 789, -123};
|
|
|
|
DBG_PRINT("initialized array 1\n");
|
|
|
|
|
|
|
|
uint16_t test_data_2[16];
|
|
|
|
for (size_t i = 0; i < 16; i++)
|
|
|
|
test_data_2[i] = 1 << i;
|
|
|
|
DBG_PRINT("initialized array 2\n");
|
2023-12-27 17:35:59 +03:00
|
|
|
|
|
|
|
for (size_t i = 0; i < 4; i++)
|
2024-01-03 21:57:49 +03:00
|
|
|
if (tf_write(ID_1, i * sizeof(int), &test_data_1[i], sizeof(int)) == -1)
|
|
|
|
RETREAT("tf_write");
|
|
|
|
DBG_PRINT("wrote array 1\n");
|
|
|
|
|
|
|
|
for (size_t i = 0; i < 16; i++)
|
|
|
|
if (tf_write(ID_2, i * sizeof(uint16_t), &test_data_2[i],
|
|
|
|
sizeof(uint16_t)) == -1)
|
2024-01-03 21:21:29 +03:00
|
|
|
RETREAT("tf_write");
|
2024-01-03 21:57:49 +03:00
|
|
|
DBG_PRINT("wrote array 2\n");
|
2023-12-27 17:35:59 +03:00
|
|
|
|
|
|
|
// round-trip
|
2024-01-03 21:57:49 +03:00
|
|
|
test_data_1[0] = 111;
|
|
|
|
test_data_1[1] = 222;
|
|
|
|
test_data_1[2] = 333;
|
|
|
|
test_data_1[3] = 444;
|
|
|
|
DBG_PRINT("filled array 1 with garbage\n");
|
|
|
|
|
|
|
|
for (size_t i = 0; i < 16; i++)
|
|
|
|
test_data_2[i] ^= 1;
|
|
|
|
DBG_PRINT("filled array 2 with garbage\n");
|
2023-12-27 17:35:59 +03:00
|
|
|
|
|
|
|
for (size_t i = 0; i < 4; i++)
|
2024-01-03 21:57:49 +03:00
|
|
|
if (tf_read(ID_1, i * sizeof(int), &test_data_1[i], sizeof(int)) == -1)
|
2024-01-03 21:21:29 +03:00
|
|
|
RETREAT("tf_read");
|
2024-01-03 21:57:49 +03:00
|
|
|
DBG_PRINT("restored array 1\n");
|
2023-12-27 17:35:59 +03:00
|
|
|
|
2024-01-03 21:57:49 +03:00
|
|
|
for (size_t i = 0; i < 16; i++)
|
|
|
|
if (tf_read(ID_2, i * sizeof(uint16_t), &test_data_2[i],
|
|
|
|
sizeof(uint16_t)) == -1)
|
|
|
|
RETREAT("tf_read");
|
|
|
|
DBG_PRINT("restored array 2\n");
|
2023-12-27 17:35:59 +03:00
|
|
|
|
2024-01-03 21:57:49 +03:00
|
|
|
DBG_PRINT("Values (1): ");
|
|
|
|
for (size_t i = 0; i < 4; i++)
|
|
|
|
DBG_PRINT("%d%c", test_data_1[i], i == 3 ? '\n' : ' ');
|
|
|
|
DBG_PRINT("Values (2): ");
|
|
|
|
for (size_t i = 0; i < 16; i++)
|
|
|
|
DBG_PRINT("%d%c", test_data_2[i], i == 15 ? '\n' : ' ');
|
|
|
|
|
|
|
|
tf_free(ID_1);
|
|
|
|
tf_free(ID_2);
|
|
|
|
DBG_PRINT("freed both files\n");
|
2023-12-27 17:35:59 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|