Compare commits
23 Commits
7fd7bc4dfa
...
main
Author | SHA1 | Date | |
---|---|---|---|
90b5acd002 | |||
e8fbe275f7 | |||
ae61eef10a | |||
fb1d1697f1 | |||
2f3d8106ea | |||
1f423b1d9b | |||
07a223f250 | |||
b2dc73148e | |||
1be318c239 | |||
7b472cac4b | |||
db4df86756 | |||
f8f08c7473 | |||
ca55dc2549 | |||
ce4be85d14 | |||
3ec25758ef | |||
f3ae1aef64 | |||
c04e8bd4a3 | |||
c5c447d5c5 | |||
490ccfc3c6 | |||
041167d26f | |||
7beac7e475 | |||
31d78bfcd3 | |||
07daa2ca19 |
2
.gitignore
vendored
2
.gitignore
vendored
@@ -1,3 +1,5 @@
|
|||||||
.cache
|
.cache
|
||||||
compile_commands.json
|
compile_commands.json
|
||||||
pckr
|
pckr
|
||||||
|
libpakrd.so
|
||||||
|
dist
|
||||||
|
8
.gitmodules
vendored
Normal file
8
.gitmodules
vendored
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
[submodule "intern/c-cpp-aliases"]
|
||||||
|
path = intern/c-cpp-aliases
|
||||||
|
url = https://git.thewizardapprentice.com/abdelrahman/c-cpp-aliases.git
|
||||||
|
branch = main
|
||||||
|
[submodule "intern/cpath"]
|
||||||
|
path = intern/cpath
|
||||||
|
url = https://git.thewizardapprentice.com/abdelrahman/cpath.git
|
||||||
|
branch = main
|
24
.vscode/launch.json
vendored
24
.vscode/launch.json
vendored
@@ -54,6 +54,30 @@
|
|||||||
"ignoreFailures": true
|
"ignoreFailures": true
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "(gdb) Debug pakrd_test",
|
||||||
|
"type": "cppdbg",
|
||||||
|
"request": "launch",
|
||||||
|
"program": "${workspaceFolder}/pakrd_test",
|
||||||
|
"args": [],
|
||||||
|
"stopAtEntry": false,
|
||||||
|
"cwd": "${workspaceFolder}",
|
||||||
|
"environment": [],
|
||||||
|
"externalConsole": false,
|
||||||
|
"MIMode": "gdb",
|
||||||
|
"setupCommands": [
|
||||||
|
{
|
||||||
|
"description": "Enable pretty-printing for gdb",
|
||||||
|
"text": "-enable-pretty-printing",
|
||||||
|
"ignoreFailures": true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"description": "Set Disassembly Flavor to Intel",
|
||||||
|
"text": "-gdb-set disassembly-flavor intel",
|
||||||
|
"ignoreFailures": true
|
||||||
|
}
|
||||||
|
]
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
68
compile
68
compile
@@ -1,30 +1,10 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
#ESCAPE SEQUENCES
|
CC=gcc
|
||||||
CLEAR="\033[0m"
|
CFLAGS="-Wall -Werror -pedantic -Iinclude -Iintern/c-cpp-aliases -Iintern/cpath/include "
|
||||||
BOLD="\033[1"
|
|
||||||
FG_BLACK="\033[30"
|
|
||||||
FG_RED="\033[31"
|
|
||||||
FG_GREEN="\033[32"
|
|
||||||
FG_YELLOW="\033[33"
|
|
||||||
FG_BLUE="\033[34"
|
|
||||||
FG_MAGENTA="\033[35"
|
|
||||||
FG_CYAN="\033[36"
|
|
||||||
FG_WHITE="\033[37"
|
|
||||||
FG_BR_BLACK="\033[90"
|
|
||||||
FG_BR_RED="\033[91"
|
|
||||||
FG_BR_GREEN="\033[92"
|
|
||||||
FG_BR_YELLOW="\033[93"
|
|
||||||
FG_BR_BLUE="\033[94"
|
|
||||||
FG_BR_MAGENTA="\033[95"
|
|
||||||
FG_BR_CYAN="\033[96"
|
|
||||||
FG_BR_WHITE="\033[97"
|
|
||||||
|
|
||||||
CC=clang
|
|
||||||
CFLAGS="-Wall -Werror -pedantic -Iinclude "
|
|
||||||
|
|
||||||
PCKR_SRC="\
|
PCKR_SRC="\
|
||||||
src/path_utils.c \
|
intern/cpath/src/cpath.c \
|
||||||
src/argparse.c \
|
src/argparse.c \
|
||||||
src/io.c \
|
src/io.c \
|
||||||
src/darr.c \
|
src/darr.c \
|
||||||
@@ -32,14 +12,21 @@ PCKR_SRC="\
|
|||||||
src/pckr.c \
|
src/pckr.c \
|
||||||
src/main.c"
|
src/main.c"
|
||||||
|
|
||||||
|
PAKRD_SRC="\
|
||||||
|
intern/cpath/src/cpath.c \
|
||||||
|
src/io.c \
|
||||||
|
src/darr.c \
|
||||||
|
src/pak.c"
|
||||||
|
PAKRD_FLAGS="-fPIC -rdynamic -shared "
|
||||||
|
|
||||||
PCKR_TEST_SRC="\
|
PCKR_TEST_SRC="\
|
||||||
src/path_utils.c \
|
intern/cpath/src/cpath.c \
|
||||||
src/io.c \
|
src/io.c \
|
||||||
src/pckr_test.c"
|
src/pckr_test.c"
|
||||||
PCKR_TEST_OUT="pckr_test"
|
PCKR_TEST_OUT="pckr_test"
|
||||||
|
|
||||||
PAKRD_TEST_SRC="\
|
PAKRD_TEST_SRC="\
|
||||||
src/path_utils.c \
|
intern/cpath/src/cpath.c \
|
||||||
src/io.c \
|
src/io.c \
|
||||||
src/darr.c \
|
src/darr.c \
|
||||||
src/pak.c \
|
src/pak.c \
|
||||||
@@ -66,9 +53,13 @@ done
|
|||||||
if [[ $BUILD_TYPE == "debug" ]]; then
|
if [[ $BUILD_TYPE == "debug" ]]; then
|
||||||
CFLAGS+="-g "
|
CFLAGS+="-g "
|
||||||
PCKR_OUT="pckr"
|
PCKR_OUT="pckr"
|
||||||
|
PAKRD_OUT="libpakrd.so"
|
||||||
else
|
else
|
||||||
|
mkdir -p dist
|
||||||
|
cp ./include/pak.h dist
|
||||||
CFLAGS+="-O3 "
|
CFLAGS+="-O3 "
|
||||||
PCKR_OUT="../proj/pckr"
|
PCKR_OUT="dist/pckr"
|
||||||
|
PAKRD_OUT="dist/libpakrd.so"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
|
||||||
@@ -76,6 +67,10 @@ fi
|
|||||||
(set -x ; $CC $CFLAGS $PCKR_SRC -o $PCKR_OUT)
|
(set -x ; $CC $CFLAGS $PCKR_SRC -o $PCKR_OUT)
|
||||||
|
|
||||||
|
|
||||||
|
# Build pakrd shared library
|
||||||
|
(set -x ; $CC $CFLAGS $PAKRD_FLAGS $PAKRD_SRC -o $PAKRD_OUT)
|
||||||
|
|
||||||
|
|
||||||
# Build pckr_test executable
|
# Build pckr_test executable
|
||||||
(set -x ; $CC $CFLAGS $PCKR_TEST_SRC -o $PCKR_TEST_OUT)
|
(set -x ; $CC $CFLAGS $PCKR_TEST_SRC -o $PCKR_TEST_OUT)
|
||||||
|
|
||||||
@@ -88,6 +83,27 @@ fi
|
|||||||
./pckr ./test_assets ./assets
|
./pckr ./test_assets ./assets
|
||||||
|
|
||||||
|
|
||||||
|
#ESCAPE SEQUENCES
|
||||||
|
CLEAR="\033[0m"
|
||||||
|
BOLD="\033[1"
|
||||||
|
FG_BLACK="\033[30"
|
||||||
|
FG_RED="\033[31"
|
||||||
|
FG_GREEN="\033[32"
|
||||||
|
FG_YELLOW="\033[33"
|
||||||
|
FG_BLUE="\033[34"
|
||||||
|
FG_MAGENTA="\033[35"
|
||||||
|
FG_CYAN="\033[36"
|
||||||
|
FG_WHITE="\033[37"
|
||||||
|
FG_BR_BLACK="\033[90"
|
||||||
|
FG_BR_RED="\033[91"
|
||||||
|
FG_BR_GREEN="\033[92"
|
||||||
|
FG_BR_YELLOW="\033[93"
|
||||||
|
FG_BR_BLUE="\033[94"
|
||||||
|
FG_BR_MAGENTA="\033[95"
|
||||||
|
FG_BR_CYAN="\033[96"
|
||||||
|
FG_BR_WHITE="\033[97"
|
||||||
|
|
||||||
|
|
||||||
# Run pckr_test
|
# Run pckr_test
|
||||||
(./$PCKR_TEST_OUT && echo -e "\npckr_test [${FG_BR_GREEN}m${BOLD}m SUCCESS $CLEAR]") || echo -e "\npckr_test [${FG_BR_RED}m${BOLD}m FAILURE $CLEAR]"
|
(./$PCKR_TEST_OUT && echo -e "\npckr_test [${FG_BR_GREEN}m${BOLD}m SUCCESS $CLEAR]") || echo -e "\npckr_test [${FG_BR_RED}m${BOLD}m FAILURE $CLEAR]"
|
||||||
|
|
||||||
|
@@ -1,27 +0,0 @@
|
|||||||
#ifndef ALIASES_H
|
|
||||||
#define ALIASES_H
|
|
||||||
|
|
||||||
#include <stdint.h>
|
|
||||||
|
|
||||||
#define u8 uint8_t
|
|
||||||
#define u16 uint16_t
|
|
||||||
#define u32 uint32_t
|
|
||||||
#define u64 uint64_t
|
|
||||||
|
|
||||||
#define i8 int8_t
|
|
||||||
#define i16 int16_t
|
|
||||||
#define i32 int32_t
|
|
||||||
#define i64 int64_t
|
|
||||||
|
|
||||||
#define f32 float
|
|
||||||
#define f64 double
|
|
||||||
#define f128 long double
|
|
||||||
|
|
||||||
#define INTERNAL static
|
|
||||||
#define PERSISTENT static
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
#define CLASS_MEMBER static
|
|
||||||
#endif // __cplusplus
|
|
||||||
|
|
||||||
#endif // !ALIASES_H
|
|
@@ -1,6 +1,10 @@
|
|||||||
#ifndef ARGPARSE_H
|
#ifndef ARGPARSE_H
|
||||||
#define ARGPARSE_H
|
#define ARGPARSE_H
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif // __cplusplus
|
||||||
|
|
||||||
#include "aliases.h"
|
#include "aliases.h"
|
||||||
#include <linux/limits.h>
|
#include <linux/limits.h>
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
@@ -17,4 +21,8 @@ typedef struct {
|
|||||||
|
|
||||||
argparser_t parse_args(i32 argc, char *argv[]);
|
argparser_t parse_args(i32 argc, char *argv[]);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif // __cplusplus
|
||||||
|
|
||||||
#endif // !ARGPARSE_H
|
#endif // !ARGPARSE_H
|
||||||
|
@@ -1,6 +1,10 @@
|
|||||||
#ifndef D_ARR_H
|
#ifndef D_ARR_H
|
||||||
#define D_ARR_H
|
#define D_ARR_H
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif // __cplusplus
|
||||||
|
|
||||||
#include "aliases.h"
|
#include "aliases.h"
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
|
|
||||||
@@ -13,4 +17,8 @@ i64 darr_add(darr_t **darr, void *item);
|
|||||||
void **darr_get_items(darr_t *darr);
|
void **darr_get_items(darr_t *darr);
|
||||||
void darr_free(darr_t **darr);
|
void darr_free(darr_t **darr);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif // __cplusplus
|
||||||
|
|
||||||
#endif // !D_ARR_H
|
#endif // !D_ARR_H
|
||||||
|
10
include/io.h
10
include/io.h
@@ -1,12 +1,18 @@
|
|||||||
#ifndef IO_H
|
#ifndef IO_H
|
||||||
#define IO_H
|
#define IO_H
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif // __cplusplus
|
||||||
|
|
||||||
#include "aliases.h"
|
#include "aliases.h"
|
||||||
#include <linux/limits.h>
|
#include <linux/limits.h>
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
#define STREQ(S1, S2) (strcmp(S1, S2) == 0)
|
#define STREQ(S1, S2) (strcmp(S1, S2) == 0)
|
||||||
|
#define STRNEQ(S1, S2, COUNT) (strncmp(S1, S2, COUNT) == 0)
|
||||||
|
|
||||||
typedef struct dirwalk dirwalk_t;
|
typedef struct dirwalk dirwalk_t;
|
||||||
typedef struct {
|
typedef struct {
|
||||||
@@ -19,4 +25,8 @@ u64 get_file_length(FILE *fp);
|
|||||||
void read_entire_file(void *dst, FILE *fp);
|
void read_entire_file(void *dst, FILE *fp);
|
||||||
dirwalk_result_t walk_dir(dirwalk_t **dirwalk, const char *filepath);
|
dirwalk_result_t walk_dir(dirwalk_t **dirwalk, const char *filepath);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif // __cplusplus
|
||||||
|
|
||||||
#endif // !IO_H
|
#endif // !IO_H
|
||||||
|
@@ -1,12 +1,16 @@
|
|||||||
#ifndef PAK_H
|
#ifndef PAK_H
|
||||||
#define PAK_H
|
#define PAK_H
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif // __cplusplus
|
||||||
|
|
||||||
#include "aliases.h"
|
#include "aliases.h"
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
u64 size;
|
u64 size;
|
||||||
u8 *data;
|
u8 data[];
|
||||||
} buf_t;
|
} buf_t;
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
@@ -41,7 +45,12 @@ typedef struct asset_pack asset_pack_t;
|
|||||||
|
|
||||||
bool create_asset_pack(const char *dirpath, const char *output);
|
bool create_asset_pack(const char *dirpath, const char *output);
|
||||||
asset_pack_t *load_asset_pack(const char *filepath);
|
asset_pack_t *load_asset_pack(const char *filepath);
|
||||||
// buf_t read_file_from_pack(pak_read_t *pack, const char *filename);
|
buf_t *read_file_from_pack(asset_pack_t *pack, const char *filename);
|
||||||
|
void close_pack_file(buf_t **buf);
|
||||||
void close_asset_pack(asset_pack_t **pack);
|
void close_asset_pack(asset_pack_t **pack);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif // __cplusplus
|
||||||
|
|
||||||
#endif // !PAK_H
|
#endif // !PAK_H
|
||||||
|
@@ -1,18 +0,0 @@
|
|||||||
#ifndef PATH_UTILS_H
|
|
||||||
#define PATH_UTILS_H
|
|
||||||
|
|
||||||
#include "aliases.h"
|
|
||||||
|
|
||||||
#define NUMPARTS(...) \
|
|
||||||
(sizeof((const char *[]){"", __VA_ARGS__}) / sizeof(const char *) - 1)
|
|
||||||
|
|
||||||
#define JOIN_PATH(DST, ...) join_path(DST, NUMPARTS(__VA_ARGS__), __VA_ARGS__)
|
|
||||||
|
|
||||||
void join_path(char *dst, u64 count, ...);
|
|
||||||
|
|
||||||
#define DIRNAME(DST, PATH) dirup(DST, 1, PATH)
|
|
||||||
#define DIRUP(DST, COUNT, PATH) dirup(DST, COUNT, PATH)
|
|
||||||
|
|
||||||
void dirup(char *dst, u64 levels, const char *path);
|
|
||||||
|
|
||||||
#endif // !PATH_UTILS_H
|
|
@@ -1,8 +1,16 @@
|
|||||||
#ifndef PCKR_H
|
#ifndef PCKR_H
|
||||||
#define PCKR_H
|
#define PCKR_H
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif // __cplusplus
|
||||||
|
|
||||||
#include "aliases.h"
|
#include "aliases.h"
|
||||||
|
|
||||||
i32 run_pckr(i32 argc, char *argv[]);
|
i32 run_pckr(i32 argc, char *argv[]);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif // __cplusplus
|
||||||
|
|
||||||
#endif // !PCKR_H
|
#endif // !PCKR_H
|
||||||
|
19
include/test_utils.h
Normal file
19
include/test_utils.h
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
#ifndef TEST_UTILS_H
|
||||||
|
#define TEST_UTILS_H
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif // __cplusplus
|
||||||
|
|
||||||
|
#define ARRLEN(ARR) (sizeof ARR / sizeof ARR[0])
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
const char *name;
|
||||||
|
const char *contents;
|
||||||
|
} asset_t;
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif // __cplusplus
|
||||||
|
|
||||||
|
#endif // !TEST_UTILS_H
|
1
intern/c-cpp-aliases
Submodule
1
intern/c-cpp-aliases
Submodule
Submodule intern/c-cpp-aliases added at f95f3aa499
1
intern/cpath
Submodule
1
intern/cpath
Submodule
Submodule intern/cpath added at 076b1310e3
@@ -7,10 +7,9 @@ argparser_t parse_args(i32 argc, char *argv[]) {
|
|||||||
return (argparser_t){0};
|
return (argparser_t){0};
|
||||||
}
|
}
|
||||||
|
|
||||||
argparser_t output = {
|
argparser_t output = {0};
|
||||||
.succeeded = true,
|
output.succeeded = true;
|
||||||
.args = (pckr_args_t){0},
|
output.args = (pckr_args_t){0};
|
||||||
};
|
|
||||||
|
|
||||||
realpath(argv[1], output.args.dir);
|
realpath(argv[1], output.args.dir);
|
||||||
realpath(argv[2], output.args.output);
|
realpath(argv[2], output.args.output);
|
||||||
|
@@ -40,7 +40,7 @@ i64 darr_add(darr_t **darr, void *item) {
|
|||||||
return INVALID_IDX;
|
return INVALID_IDX;
|
||||||
}
|
}
|
||||||
|
|
||||||
(*darr)->capacity = orig->capacity + BASE_CAPACITY;
|
(*darr)->capacity += BASE_CAPACITY;
|
||||||
}
|
}
|
||||||
|
|
||||||
u64 index = (*darr)->count;
|
u64 index = (*darr)->count;
|
||||||
|
16
src/io.c
16
src/io.c
@@ -1,6 +1,6 @@
|
|||||||
#include "io.h"
|
#include "io.h"
|
||||||
#include "aliases.h"
|
#include "aliases.h"
|
||||||
#include "path_utils.h"
|
#include "cpath.h"
|
||||||
#include <dirent.h>
|
#include <dirent.h>
|
||||||
#include <linux/limits.h>
|
#include <linux/limits.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
@@ -16,8 +16,8 @@ struct dirwalk {
|
|||||||
u64 count;
|
u64 count;
|
||||||
FILE *current_file;
|
FILE *current_file;
|
||||||
DIR *current_dir;
|
DIR *current_dir;
|
||||||
char dst[PATH_MAX];
|
char dst[PATH_MAX + 1];
|
||||||
DIR *directories[MAX_DIR_DEPTH];
|
DIR *directories[MAX_DIR_DEPTH + 1];
|
||||||
};
|
};
|
||||||
|
|
||||||
u64 get_file_length(FILE *fp) {
|
u64 get_file_length(FILE *fp) {
|
||||||
@@ -73,7 +73,7 @@ dirwalk_result_t walk_dir(dirwalk_t **dirwalk, const char *filepath) {
|
|||||||
|
|
||||||
dirwalk_t *dw = *dirwalk;
|
dirwalk_t *dw = *dirwalk;
|
||||||
|
|
||||||
char tmp[PATH_MAX] = {0};
|
char tmp[PATH_MAX + 1] = {0};
|
||||||
|
|
||||||
dirent_t *ep = NULL;
|
dirent_t *ep = NULL;
|
||||||
stat_t dirstat = {0};
|
stat_t dirstat = {0};
|
||||||
@@ -102,7 +102,7 @@ dirwalk_result_t walk_dir(dirwalk_t **dirwalk, const char *filepath) {
|
|||||||
|
|
||||||
result.reading = true;
|
result.reading = true;
|
||||||
result.fp = dw->current_file;
|
result.fp = dw->current_file;
|
||||||
strncpy(result.name, ep->d_name, NAME_MAX);
|
strncpy(result.name, ep->d_name, NAME_MAX + 1);
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@@ -116,7 +116,7 @@ dirwalk_result_t walk_dir(dirwalk_t **dirwalk, const char *filepath) {
|
|||||||
if ((dirstat.st_mode & S_IFDIR) == S_IFDIR) {
|
if ((dirstat.st_mode & S_IFDIR) == S_IFDIR) {
|
||||||
memset(dw->dst, 0, length);
|
memset(dw->dst, 0, length);
|
||||||
|
|
||||||
strncpy(dw->dst, tmp, length);
|
strncpy(dw->dst, tmp, length + 1);
|
||||||
|
|
||||||
dw->directories[(dw->count)++] = opendir(dw->dst);
|
dw->directories[(dw->count)++] = opendir(dw->dst);
|
||||||
|
|
||||||
@@ -128,7 +128,7 @@ dirwalk_result_t walk_dir(dirwalk_t **dirwalk, const char *filepath) {
|
|||||||
|
|
||||||
result.reading = true;
|
result.reading = true;
|
||||||
result.fp = dw->current_file;
|
result.fp = dw->current_file;
|
||||||
strncpy(result.name, ep->d_name, NAME_MAX);
|
strncpy(result.name, ep->d_name, NAME_MAX + 1);
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@@ -150,7 +150,7 @@ dirwalk_result_t walk_dir(dirwalk_t **dirwalk, const char *filepath) {
|
|||||||
|
|
||||||
u64 length = strlen(dw->dst);
|
u64 length = strlen(dw->dst);
|
||||||
|
|
||||||
strncpy(tmp, dw->dst, length);
|
strcpy(tmp, dw->dst);
|
||||||
|
|
||||||
memset(dw->dst, 0, length);
|
memset(dw->dst, 0, length);
|
||||||
|
|
||||||
|
81
src/pak.c
81
src/pak.c
@@ -34,13 +34,11 @@ void write_pak(const pak_t *pak, toc_entry_t **toc_entries,
|
|||||||
bool create_asset_pack(const char *dirpath, const char *output) {
|
bool create_asset_pack(const char *dirpath, const char *output) {
|
||||||
bool created = false;
|
bool created = false;
|
||||||
|
|
||||||
char dst[PATH_MAX] = {0};
|
char dst[PATH_MAX + 1] = {0};
|
||||||
|
|
||||||
u64 output_length = strlen(output);
|
strcpy(dst, output);
|
||||||
strncpy(dst, output, output_length);
|
|
||||||
|
|
||||||
u64 ext_length = strlen(PAK_EXT);
|
strcat(dst, PAK_EXT);
|
||||||
strncat(dst, PAK_EXT, ext_length);
|
|
||||||
|
|
||||||
FILE *fp = fopen(dst, "wb");
|
FILE *fp = fopen(dst, "wb");
|
||||||
if (!fp) {
|
if (!fp) {
|
||||||
@@ -130,9 +128,11 @@ asset_pack_t *load_asset_pack(const char *filepath) {
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
memset((void *)pack, 0, sizeof(asset_pack_t));
|
||||||
|
|
||||||
pack->fp = fopen(full_path, "rb");
|
pack->fp = fopen(full_path, "rb");
|
||||||
if (!(pack->fp)) {
|
if (!(pack->fp)) {
|
||||||
free(pack);
|
close_asset_pack(&pack);
|
||||||
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
@@ -145,16 +145,29 @@ asset_pack_t *load_asset_pack(const char *filepath) {
|
|||||||
|
|
||||||
pack->toc = (u8 *)malloc(toc_size);
|
pack->toc = (u8 *)malloc(toc_size);
|
||||||
if (!(pack->toc)) {
|
if (!(pack->toc)) {
|
||||||
fclose(pack->fp);
|
close_asset_pack(&pack);
|
||||||
|
|
||||||
free(pack);
|
|
||||||
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
fread((void *)(pack->toc), toc_size, 1, pack->fp);
|
fread((void *)(pack->toc), toc_size, 1, pack->fp);
|
||||||
|
|
||||||
// TODO (Abdelrahman): Populate the entries array
|
u64 entries_size = sizeof(toc_entry_t *) * pack->header.toc.count;
|
||||||
|
|
||||||
|
pack->toc_entries = (toc_entry_t **)malloc(entries_size);
|
||||||
|
if (!(pack->toc_entries)) {
|
||||||
|
close_asset_pack(&pack);
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
toc_entry_t *next = (toc_entry_t *)pack->toc;
|
||||||
|
|
||||||
|
for (u64 i = 0; i < pack->header.toc.count; ++i) {
|
||||||
|
pack->toc_entries[i] = next;
|
||||||
|
|
||||||
|
next = (toc_entry_t *)((u8 *)next + sizeof(toc_entry_t) + next->length);
|
||||||
|
}
|
||||||
|
|
||||||
pack->assets_offset = ftell(pack->fp);
|
pack->assets_offset = ftell(pack->fp);
|
||||||
|
|
||||||
@@ -163,18 +176,62 @@ asset_pack_t *load_asset_pack(const char *filepath) {
|
|||||||
return pack;
|
return pack;
|
||||||
}
|
}
|
||||||
|
|
||||||
// buf_t read_file_from_pack(pak_read_t *pack, const char *filename) {}
|
buf_t *read_file_from_pack(asset_pack_t *pack, const char *filename) {
|
||||||
|
toc_entry_t *entry = NULL;
|
||||||
|
buf_t *output = NULL;
|
||||||
|
|
||||||
|
for (u64 i = 0; i < pack->header.toc.count; ++i) {
|
||||||
|
entry = pack->toc_entries[i];
|
||||||
|
|
||||||
|
if (STRNEQ(filename, entry->name, entry->length)) {
|
||||||
|
fseek(pack->fp, pack->assets_offset + entry->offset, SEEK_SET);
|
||||||
|
|
||||||
|
u64 size = 0;
|
||||||
|
fread(&size, sizeof(u64), 1, pack->fp);
|
||||||
|
|
||||||
|
output = (buf_t *)malloc(sizeof(buf_t) + size);
|
||||||
|
if (!output) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
output->size = size;
|
||||||
|
|
||||||
|
fread(&(output->data), size, 1, pack->fp);
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
void close_pack_file(buf_t **buf) {
|
||||||
|
buf_t *b = *buf;
|
||||||
|
|
||||||
|
if (!b || b->size == 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
free(*buf);
|
||||||
|
*buf = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
void close_asset_pack(asset_pack_t **pack) {
|
void close_asset_pack(asset_pack_t **pack) {
|
||||||
if (!(*pack)) {
|
if (!(*pack)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ((*pack)->toc_entries) {
|
||||||
|
free((*pack)->toc_entries);
|
||||||
|
}
|
||||||
|
|
||||||
if ((*pack)->toc) {
|
if ((*pack)->toc) {
|
||||||
free((*pack)->toc);
|
free((*pack)->toc);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ((*pack)->fp) {
|
||||||
fclose((*pack)->fp);
|
fclose((*pack)->fp);
|
||||||
|
}
|
||||||
|
|
||||||
free(*pack);
|
free(*pack);
|
||||||
|
|
||||||
@@ -195,7 +252,7 @@ toc_entry_t *create_toc_entry(const char *name, u64 offset) {
|
|||||||
entry->offset = offset;
|
entry->offset = offset;
|
||||||
entry->length = length + 1;
|
entry->length = length + 1;
|
||||||
|
|
||||||
strncpy(entry->name, name, length);
|
strcpy(entry->name, name);
|
||||||
|
|
||||||
return entry;
|
return entry;
|
||||||
}
|
}
|
||||||
|
@@ -1,4 +1,6 @@
|
|||||||
|
#include "io.h"
|
||||||
#include "pak.h"
|
#include "pak.h"
|
||||||
|
#include "test_utils.h"
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|
||||||
@@ -10,5 +12,40 @@ int main(int argc, char *argv[]) {
|
|||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
asset_t assets[] = {
|
||||||
|
(asset_t){.name = "file01", .contents = "Hello\n"},
|
||||||
|
(asset_t){.name = "name.txt", .contents = "Abdelrahman\n"},
|
||||||
|
(asset_t){.name = "file02", .contents = "This is the second file\n"},
|
||||||
|
};
|
||||||
|
|
||||||
|
for (u64 i = 0; i < ARRLEN(assets); ++i) {
|
||||||
|
buf_t *buf = read_file_from_pack(pack, assets[i].name);
|
||||||
|
if (!buf || buf->size == 0) {
|
||||||
|
printf("Failed to read file %s\n", assets[i].name);
|
||||||
|
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!(STREQ((char *)(buf->data), assets[i].contents))) {
|
||||||
|
printf("File contents mismatch for file %s\n", assets[i].name);
|
||||||
|
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
close_pack_file(&buf);
|
||||||
|
if (buf) {
|
||||||
|
printf("Failed to close file %s\n", assets[i].name);
|
||||||
|
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
close_asset_pack(&pack);
|
||||||
|
if (pack) {
|
||||||
|
printf("Failed to close asset pack");
|
||||||
|
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
return EXIT_SUCCESS;
|
return EXIT_SUCCESS;
|
||||||
}
|
}
|
||||||
|
@@ -1,76 +0,0 @@
|
|||||||
#include "path_utils.h"
|
|
||||||
#include "aliases.h"
|
|
||||||
#include <stdarg.h>
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <string.h>
|
|
||||||
|
|
||||||
#if defined(__unix__) || defined(__APPLE__) || defined(__ANDROID__)
|
|
||||||
INTERNAL char path_sep = '/';
|
|
||||||
#elif defined(_WIN32) || defined(_WIN64)
|
|
||||||
INTERNAL char path_sep = '\';
|
|
||||||
#endif
|
|
||||||
|
|
||||||
void join_root_and_leaf(const char *root, const char *leaf, char *dst);
|
|
||||||
|
|
||||||
void join_path(char *dst, u64 count, ...) {
|
|
||||||
va_list args;
|
|
||||||
|
|
||||||
va_start(args, count);
|
|
||||||
|
|
||||||
for (u64 i = 0; i < count; ++i) {
|
|
||||||
join_root_and_leaf(dst, va_arg(args, const char *), dst);
|
|
||||||
}
|
|
||||||
|
|
||||||
va_end(args);
|
|
||||||
}
|
|
||||||
|
|
||||||
void dirup(char *dst, u64 levels, const char *path) {
|
|
||||||
if (levels < 1) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
u64 end_index = 0;
|
|
||||||
u64 sep_count = 0;
|
|
||||||
|
|
||||||
u64 length = strlen(path);
|
|
||||||
if (path[length - 1] == path_sep) {
|
|
||||||
--length;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (u64 i = length - 1; i >= 0; --i) {
|
|
||||||
if (path[i] == path_sep) {
|
|
||||||
++sep_count;
|
|
||||||
end_index = i;
|
|
||||||
|
|
||||||
if (sep_count == levels) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (sep_count < levels) {
|
|
||||||
end_index = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
strncpy(dst, path, end_index);
|
|
||||||
}
|
|
||||||
|
|
||||||
void join_root_and_leaf(const char *root, const char *leaf, char *dst) {
|
|
||||||
u64 root_length = strlen(root);
|
|
||||||
u64 root_end = root_length - 1;
|
|
||||||
|
|
||||||
u64 leaf_length = strlen(leaf);
|
|
||||||
u64 leaf_start = 0;
|
|
||||||
|
|
||||||
if (root[root_end] == path_sep) {
|
|
||||||
--root_end;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (leaf[leaf_start] == path_sep) {
|
|
||||||
++leaf_start;
|
|
||||||
}
|
|
||||||
|
|
||||||
strncpy(dst, root, ++root_end);
|
|
||||||
dst[root_end] = path_sep;
|
|
||||||
strncpy(&(dst[++root_end]), &(leaf[leaf_start]), leaf_length - leaf_start);
|
|
||||||
}
|
|
@@ -1,19 +1,13 @@
|
|||||||
#include "aliases.h"
|
#include "aliases.h"
|
||||||
#include "io.h"
|
#include "io.h"
|
||||||
#include "pak.h"
|
#include "pak.h"
|
||||||
|
#include "test_utils.h"
|
||||||
#include <linux/limits.h>
|
#include <linux/limits.h>
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
#define ARRLEN(ARR) (sizeof ARR / sizeof ARR[0])
|
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
const char *name;
|
|
||||||
const char *contents;
|
|
||||||
} asset_t;
|
|
||||||
|
|
||||||
i32 main(i32 argc, char *argv[]) {
|
i32 main(i32 argc, char *argv[]) {
|
||||||
char filepath[PATH_MAX] = {0};
|
char filepath[PATH_MAX] = {0};
|
||||||
realpath("assets.pak", filepath);
|
realpath("assets.pak", filepath);
|
||||||
|
Reference in New Issue
Block a user