diff --git a/CMakeLists.txt b/CMakeLists.txt index eed02b88..de9c5429 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -19,9 +19,9 @@ endif() set(DAWN_ROOT_DIR "${CMAKE_SOURCE_DIR}") set(DAWN_BUILD_DIR "${CMAKE_BINARY_DIR}") set(DAWN_SOURCES_DIR "${DAWN_ROOT_DIR}/src") -# set(DAWN_TOOLS_DIR "${DAWN_ROOT_DIR}/tools") -# set(DAWN_ASSETS_SOURCE_DIR "${DAWN_ROOT_DIR}/assets") -# set(DAWN_ASSETS_BUILD_DIR "${DAWN_BUILD_DIR}/assets") +set(DAWN_TOOLS_DIR "${DAWN_ROOT_DIR}/tools") +set(DAWN_ASSETS_SOURCE_DIR "${DAWN_ROOT_DIR}/assets") +set(DAWN_ASSETS_BUILD_DIR "${DAWN_BUILD_DIR}/assets") set(DAWN_GENERATED_DIR "${DAWN_BUILD_DIR}/generated") set(DAWN_TEMP_DIR "${DAWN_BUILD_DIR}/temp") @@ -32,7 +32,10 @@ project(Dawn ) # Add tools -# add_subdirectory(tools) +add_subdirectory(tools) + +# Add assets +add_subdirectory(assets) # Add Libraries add_subdirectory(lib) diff --git a/assets/CMakeLists.txt b/assets/CMakeLists.txt new file mode 100644 index 00000000..8c867df2 --- /dev/null +++ b/assets/CMakeLists.txt @@ -0,0 +1,10 @@ +# Copyright (c) 2024 Dominic Masters +# +# This software is released under the MIT License. +# https://opensource.org/licenses/MIT + +tool_copy( + testmap + ${CMAKE_CURRENT_SOURCE_DIR}/testmap.json + testmap.json +) \ No newline at end of file diff --git a/lib/CMakeLists.txt b/lib/CMakeLists.txt index dc0051b9..b201a6d8 100644 --- a/lib/CMakeLists.txt +++ b/lib/CMakeLists.txt @@ -21,3 +21,11 @@ FetchContent_Declare( GIT_TAG 1796cc5ce298235b615dc7a4750b8c3ba56a05dd ) FetchContent_MakeAvailable(cglm) + +#LibArchive +FetchContent_Declare( + libarchive + GIT_REPOSITORY https://github.com/libarchive/libarchive + GIT_TAG v3.7.6 +) +FetchContent_MakeAvailable(libarchive) \ No newline at end of file diff --git a/src/dawn/CMakeLists.txt b/src/dawn/CMakeLists.txt index e991ac36..ca6cf341 100644 --- a/src/dawn/CMakeLists.txt +++ b/src/dawn/CMakeLists.txt @@ -6,6 +6,7 @@ # Libraries target_link_libraries(${DAWN_TARGET_NAME} PUBLIC + archive_static ) # Includes @@ -16,6 +17,7 @@ target_include_directories(${DAWN_TARGET_NAME} # Subdirs add_subdirectory(assert) +add_subdirectory(asset) add_subdirectory(display) add_subdirectory(game) add_subdirectory(rpg) @@ -25,4 +27,7 @@ add_subdirectory(ui) target_sources(${DAWN_TARGET_NAME} PRIVATE input.c -) \ No newline at end of file +) + +# Assets +add_dependencies(${DAWN_TARGET_NAME} dawnassets) \ No newline at end of file diff --git a/src/dawn/asset/CMakeLists.txt b/src/dawn/asset/CMakeLists.txt new file mode 100644 index 00000000..5e35bfe9 --- /dev/null +++ b/src/dawn/asset/CMakeLists.txt @@ -0,0 +1,11 @@ +# Copyright (c) 2024 Dominic Masters +# +# This software is released under the MIT License. +# https:#opensource.org/licenses/MIT + +# Sources +target_sources(${DAWN_TARGET_NAME} + PRIVATE + asset.c + assetarchive.c +) \ No newline at end of file diff --git a/src/dawn/asset/asset.c b/src/dawn/asset/asset.c new file mode 100644 index 00000000..af9360a7 --- /dev/null +++ b/src/dawn/asset/asset.c @@ -0,0 +1,158 @@ +/** + * Copyright (c) 2024 Dominic Masters + * + * This software is released under the MIT License. + * https://opensource.org/licenses/MIT + */ + +#include "asset.h" +#include "assetarchive.h" +#include "assert/assert.h" +#include "util/math.h" + +void assetInit() { + // TODO: Works on Windows? path sep probs wrong. + // const char_t *assetFilename = "dawn.tar"; + // char_t *assetPath = malloc(sizeof(char_t) * ( + // strlen(SYSTEM.executableDirectory) + strlen(assetFilename) + 1 + // )); + // sprintf(assetPath, "%s/%s", SYSTEM.executableDirectory, assetFilename); + char_t *assetPath = "/home/yourwishes/htdocs/Dawn/build/dawn.tar"; + ASSET_FILE = fopen(assetPath, "rb"); + // free(assetPath); + assertNotNull(ASSET_FILE, "assetInit: Failed to open asset file!"); + + ASSET_ARCHIVE = NULL; + ASSET_ENTRY = NULL; +} + +size_t assetReadUntil(uint8_t *buffer, const char_t c, const size_t maxLength) { + if(buffer == NULL) { + assertTrue( + maxLength == -1, "If no buffer is provided, maxLength must be -1." + ); + uint8_t tBuffer[1]; + size_t read = 0; + while(assetRead(tBuffer, 1) == 1 && (char_t)tBuffer[0] != c) read++; + return read; + } else { + size_t read = 0; + while(read < maxLength) { + // TODO: Read more than 1 char at a time. + read += assetRead(buffer + read, 1); + if((char_t)buffer[read-1] == c) return read - 1; + } + return -1; + } +} + +void assetOpen(const char_t *path) { + assertNull(ASSET_ARCHIVE, "assetOpenFile: Archive is not NULL!"); + assertNull(ASSET_ENTRY, "assetOpenFile: Entry is not NULL!"); + assertStringValid(path, 1024, "assetOpenFile: Path is not valid!"); + + // Store path + strcpy(ASSET_PATH_CURRENT, path); + + // Prepare data + ASSET_ARCHIVE = archive_read_new(); + assertNotNull(ASSET_ARCHIVE, "assetOpenFile: Failed to create archive!"); + + // Set up the reader + // archive_read_support_filter_bzip2(ASSET_ARCHIVE); + archive_read_support_format_tar(ASSET_ARCHIVE); + + // Open reader + archive_read_set_open_callback(ASSET_ARCHIVE, &assetArchiveOpen); + archive_read_set_read_callback(ASSET_ARCHIVE, &assetArchiveRead); + archive_read_set_seek_callback(ASSET_ARCHIVE, &assetArchiveSeek); + archive_read_set_close_callback(ASSET_ARCHIVE, &assetArchiveOpen); + archive_read_set_callback_data(ASSET_ARCHIVE, ASSET_ARCHIVE_BUFFER); + + int32_t ret = archive_read_open1(ASSET_ARCHIVE); + assertTrue(ret == ARCHIVE_OK, "assetOpenFile: Failed to open archive!"); + + // Iterate over each file. + while(archive_read_next_header(ASSET_ARCHIVE, &ASSET_ENTRY) == ARCHIVE_OK) { + const char_t *headerFile = (char_t*)archive_entry_pathname(ASSET_ENTRY); + if(strcmp(headerFile, ASSET_PATH_CURRENT) == 0) return; + int32_t ret = archive_read_data_skip(ASSET_ARCHIVE); + assertTrue(ret == ARCHIVE_OK, "assetOpenFile: Failed to skip data!"); + } + + assertUnreachable("assetOpenFile: Failed to find file!"); +} + +size_t assetGetSize() { + assertNotNull(ASSET_ARCHIVE, "assetGetSize: Archive is NULL!"); + assertNotNull(ASSET_ENTRY, "assetGetSize: Entry is NULL!"); + assertTrue( + archive_entry_size_is_set(ASSET_ENTRY), + "assetGetSize: Entry size is not set!" + ); + + size_t n = archive_entry_size(ASSET_ENTRY); + + char_t path[2048]; + sprintf( + path, "/home/yourwishes/htdocs/dusk/build/assets/%s", ASSET_PATH_CURRENT + ); + FILE *temp = fopen(path, "rb"); + assertNotNull(temp, "assetGetSize: Failed to open temp file!"); + fseek(temp, 0, SEEK_END); + size_t size = ftell(temp); + assertTrue(size == n, "assetGetSize: Size is not equal!"); + fclose(temp); + + return n; +} + +size_t assetRead(uint8_t *buffer, size_t bufferSize) { + assertNotNull(ASSET_ARCHIVE, "assetRead: Archive is NULL!"); + assertNotNull(ASSET_ENTRY, "assetRead: Entry is NULL!"); + assertNotNull(buffer, "assetRead: Buffer is NULL!"); + assertTrue(bufferSize > 0, "assetRead: Buffer size must be greater than 0!"); + ssize_t read = archive_read_data(ASSET_ARCHIVE, buffer, bufferSize); + + if(read == ARCHIVE_FATAL) { + assertUnreachable(archive_error_string(ASSET_ARCHIVE)); + } + + assertTrue(read != ARCHIVE_RETRY, "assetRead: Failed to read data (RETRY)!"); + assertTrue(read != ARCHIVE_WARN, "assetRead: Failed to read data (WARN)!"); + + return read; +} + +void assetSkip(const size_t length) { + assertNotNull(ASSET_ARCHIVE, "assetSkip: Archive is NULL!"); + assertNotNull(ASSET_ENTRY, "assetSkip: Entry is NULL!"); + assertTrue(length > 0, "assetSkip: Length must be greater than 0!"); + + // Asset archive does not support skipping, so we have to read and discard. + uint8_t buffer[1024]; + size_t remaining = length; + do { + size_t toRead = mathMin(remaining, 1024); + size_t read = assetRead(buffer, toRead); + assertTrue(read == toRead, "assetSkip: Failed to skip data! (overskip?)"); + remaining -= read; + } while(remaining > 0); +} + +void assetClose() { + assertNotNull(ASSET_ARCHIVE, "assetClose: Archive is NULL!"); + assertNotNull(ASSET_ENTRY, "assetClose: Entry is NULL!"); + int32_t ret = archive_read_free(ASSET_ARCHIVE); + assertTrue(ret == ARCHIVE_OK, "assetClose: Failed to close archive!"); + ASSET_ARCHIVE = NULL; + ASSET_ENTRY = NULL; +} + +void assetDispose() { + assertNull(ASSET_ARCHIVE, "assetDestroy: Archive is not NULL!"); + assertNull(ASSET_ENTRY, "assetDestroy: Entry is not NULL!"); + + int32_t result = fclose(ASSET_FILE); + assertTrue(result == 0, "assetDestroy: Failed to close asset file!"); +} \ No newline at end of file diff --git a/src/dawn/asset/asset.h b/src/dawn/asset/asset.h new file mode 100644 index 00000000..2059f67d --- /dev/null +++ b/src/dawn/asset/asset.h @@ -0,0 +1,73 @@ +/** + * Copyright (c) 2023 Dominic Masters + * + * This software is released under the MIT License. + * https://opensource.org/licenses/MIT + */ + +#pragma once +#include "dawn.h" + +/** + * Initializes the asset manager. + */ +void assetInit(); + +/** + * Opens an asset by its filename (within the asset archive). Asset paths should + * always use the unix forward slash '/' as a path separator. + * + * @param path The path to the asset within the archive. + */ +void assetOpen(const char_t *path); + +/** + * Returns the size of the asset. + * + * @return The size of the asset. + */ +size_t assetGetSize(); + +/** + * Reads the asset into the buffer. + * + * @param buffer The buffer to read the asset into. + * @param bufferSize The size of the buffer. + * @return The amount of data read. + */ +size_t assetRead(uint8_t *buffer, size_t bufferSize); + +/** + * Reads ahead in the buffer until either the end of the buffer, or the + * specified character is found. Return value will be -1 if the character was + * not found. + * + * Buffer can be NULL if you just want to skip ahead. + * + * Returned value will be either the amount of data read into the buffer, which + * excludes the extra 1 character that was read from the asset. If the character + * was not found, -1 will be returned. + * + * @param buffer Buffer to read into. + * @param c Character to read until. + * @param maxLength Maximum length to read. + * @return -1 if the character was not found, otherwise the amount of data read. + */ +size_t assetReadUntil(uint8_t *buffer, const char_t c, const size_t maxLength); + +/** + * Skips ahead in the buffer by the specified length. + * + * @param length The length to skip ahead by. + */ +void assetSkip(const size_t length); + +/** + * Closes the asset. + */ +void assetClose(); + +/** + * Destroys and cleans up the asset manager. + */ +void assetDispose(); \ No newline at end of file diff --git a/src/dawn/asset/assetarchive.c b/src/dawn/asset/assetarchive.c new file mode 100644 index 00000000..db135b5a --- /dev/null +++ b/src/dawn/asset/assetarchive.c @@ -0,0 +1,55 @@ +/** + * Copyright (c) 2023 Dominic Masters + * + * This software is released under the MIT License. + * https://opensource.org/licenses/MIT + */ + +#include "asset/assetarchive.h" +#include "assert/assert.h" +#include "util/math.h" + +FILE *ASSET_FILE; +struct archive *ASSET_ARCHIVE; +struct archive_entry *ASSET_ENTRY; +uint8_t ASSET_ARCHIVE_BUFFER[ASSET_BUFFER_SIZE]; +char_t ASSET_PATH_CURRENT[ASSET_PATH_MAX]; + +ssize_t assetArchiveRead( + struct archive *archive, + void *data, + const void **buffer +) { + assertNotNull(archive, "assetArchiveRead: Archive is NULL!"); + assertNotNull(data, "assetArchiveRead: Data is NULL!"); + assertNotNull(buffer, "assetArchiveRead: Buffer is NULL!"); + + *buffer = data; + size_t read = fread(data, 1, ASSET_BUFFER_SIZE, ASSET_FILE); + if(ferror(ASSET_FILE)) return ARCHIVE_FATAL; + return read; +} + +int64_t assetArchiveSeek( + struct archive *archive, + void *data, + int64_t offset, + int32_t whence +) { + assertNotNull(archive, "assetArchiveSeek: Archive is NULL!"); + assertNotNull(data, "assetArchiveSeek: Data is NULL!"); + assertTrue(offset > 0, "assetArchiveSeek: Offset must be greater than 0!"); + int32_t ret = fseek(ASSET_FILE, offset, whence); + assertTrue(ret == 0, "assetArchiveSeek: Failed to seek!"); + return ftell(ASSET_FILE); +} + +int32_t assetArchiveOpen(struct archive *a, void *data) { + int32_t ret = fseek(ASSET_FILE, 0, SEEK_SET); + assertTrue(ret == 0, "assetArchiveOpen: Failed to seek to start of file!"); + return ARCHIVE_OK; +} + +int32_t assetArchiveClose(struct archive *a, void *data) { + return assetArchiveOpen(a, data); +} \ No newline at end of file diff --git a/src/dawn/asset/assetarchive.h b/src/dawn/asset/assetarchive.h new file mode 100644 index 00000000..57e08b53 --- /dev/null +++ b/src/dawn/asset/assetarchive.h @@ -0,0 +1,68 @@ +/** + * Copyright (c) 2023 Dominic Masters + * + * This software is released under the MIT License. + * https://opensource.org/licenses/MIT + */ + +#pragma once +#include "asset/asset.h" +#include +#include + +#define ASSET_BUFFER_SIZE 32768 +#define ASSET_PATH_MAX 1024 + +extern FILE *ASSET_FILE; +extern struct archive *ASSET_ARCHIVE; +extern struct archive_entry *ASSET_ENTRY; +extern uint8_t ASSET_ARCHIVE_BUFFER[ASSET_BUFFER_SIZE]; +extern char_t ASSET_PATH_CURRENT[ASSET_PATH_MAX]; + +/** + * Internal read method provided to libarchive api. + * + * @param archive The archive to read from. + * @param data The data to read into. + * @param buffer The buffer to read from. + * @return The amount of data read. + */ +ssize_t assetArchiveRead( + struct archive *archive, + void *data, + const void **buffer +); + +/** + * Internal seek method provided to libarchive api. + * + * @param archive The archive to seek in. + * @param data The data to seek in. + * @param offset Offset bytes to seek. + * @param whence Relative to whence to seek. + * @return The new position. + */ +int64_t assetArchiveSeek( + struct archive *archive, + void *data, + int64_t offset, + int32_t whence +); + +/** + * Internal open method provided to libarchive api. + * + * @param archive The archive to open. + * @param data The data to open. + * @return The result of the open. + */ +int32_t assetArchiveOpen(struct archive *a, void *data); + +/** + * Internal close method provided to libarchive api. + * + * @param archive The archive to close. + * @param data The data to close. + * @return The result of the close. + */ +int32_t assetArchiveClose(struct archive *a, void *data); \ No newline at end of file diff --git a/src/dawn/game/game.c b/src/dawn/game/game.c index 05c1d441..b71833b5 100644 --- a/src/dawn/game/game.c +++ b/src/dawn/game/game.c @@ -11,6 +11,7 @@ #include "display/display.h" #include "rpg/world/maps/testmap.h" #include "ui/textbox.h" +#include "asset/asset.h" map_t MAP; game_t GAME; @@ -21,6 +22,7 @@ void gameInit() { timeInit(); inputInit(); displayInit(); + assetInit(); textboxInit(); testMapInit(&MAP); @@ -62,5 +64,6 @@ void gameSetMap(map_t *map) { } void gameDispose() { + assetDispose(); displayDispose(); } \ No newline at end of file diff --git a/tools/CMakeLists.txt b/tools/CMakeLists.txt new file mode 100644 index 00000000..710b24cc --- /dev/null +++ b/tools/CMakeLists.txt @@ -0,0 +1,14 @@ +# Copyright (c) 2023 Dominic Msters +# +# This software is released under the MIT License. +# https://opensource.org/licenses/MIT + +# Tool Level Values +set( + DAWN_TOOL_GENERATED_DEPENDENCIES + CACHE INTERNAL ${DAWN_CACHE_TARGET} +) + +# Tools +add_subdirectory(assetstool) +add_subdirectory(copytool) \ No newline at end of file diff --git a/tools/assetstool/CMakeLists.txt b/tools/assetstool/CMakeLists.txt new file mode 100644 index 00000000..f26a801c --- /dev/null +++ b/tools/assetstool/CMakeLists.txt @@ -0,0 +1,17 @@ +# Copyright (c) 2023 Dominic Masters +# +# This software is released under the MIT License. +# https://opensource.org/licenses/MIT + +find_package(Python3 REQUIRED COMPONENTS Interpreter) + +add_custom_target(dawnassets + COMMAND + ${Python3_EXECUTABLE} + ${DAWN_TOOLS_DIR}/assetstool/assetstool.py + --input=${DAWN_ASSETS_BUILD_DIR} + --output=${DAWN_BUILD_DIR}/dawn.tar + COMMENT "Bundling assets..." + USES_TERMINAL + DEPENDS ${DAWN_ASSETS} +) \ No newline at end of file diff --git a/tools/assetstool/assetstool.py b/tools/assetstool/assetstool.py new file mode 100755 index 00000000..6519617c --- /dev/null +++ b/tools/assetstool/assetstool.py @@ -0,0 +1,62 @@ +#!/usr/bin/env python +# Copyright (c) 2023 Dominic Masters +# +# This software is released under the MIT License. +# https://opensource.org/licenses/MIT + +import os +import tarfile +import argparse + +# Args +parser = argparse.ArgumentParser( + description='Bundles all assets into the internal archive format.' +) +parser.add_argument('-i', '--input'); +parser.add_argument('-o', '--output'); +args = parser.parse_args() + +# Ensure the directory for the output path exists +if not os.path.exists(os.path.dirname(args.output)): + os.makedirs(os.path.dirname(args.output)) + +# Create a ZIP archive and add the specified directory +# archive = tarfile.open(args.output, 'w:bz2') # BZ2 Compression + +# Does the archive already exist? +filesInArchive = [] + +if os.path.exists(args.output): + # Yes, open it + archive = tarfile.open(args.output, 'r:') + + # Get all the files in the archive + for member in archive.getmembers(): + filesInArchive.append(member.name) + + archive.close() + + # Open archive for appending. + archive = tarfile.open(args.output, 'a:') +else: + # No, create it + archive = tarfile.open(args.output, 'w:') + +# Add all files in the input directory +for foldername, subfolders, filenames in os.walk(args.input): + for filename in filenames: + + # Is the file already in the archive? + absolute_path = os.path.join(foldername, filename) + relative_path = os.path.relpath(absolute_path, args.input) + + if relative_path in filesInArchive: + # Yes, skip it + continue + + # No, add it + print(f"Archiving asset {filename}...") + archive.add(absolute_path, arcname=relative_path) + +# Close the archive +archive.close() diff --git a/tools/copytool/CMakeLists.txt b/tools/copytool/CMakeLists.txt new file mode 100644 index 00000000..82900dde --- /dev/null +++ b/tools/copytool/CMakeLists.txt @@ -0,0 +1,11 @@ +# Copyright (c) 2023 Dominic Msters +# +# This software is released under the MIT License. +# https://opensource.org/licenses/MIT + +function(tool_copy target input output) + add_custom_target(${target} + COMMAND ${CMAKE_COMMAND} -E copy ${input} ${output} + ) + add_dependencies(dawnassets ${target}) +endfunction()