⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 hog.c

📁 嵌入式环境下的GUI
💻 C
📖 第 1 页 / 共 2 页
字号:
/* * HOG support routines for PhysicsFS. * * This driver handles Descent I/II HOG archives. * * The format is very simple: * *   The file always starts with the 3-byte signature "DHF" (Descent *   HOG file). After that the files of a HOG are just attached after *   another, divided by a 17 bytes header, which specifies the name *   and length (in bytes) of the forthcoming file! So you just read *   the header with its information of how big the following file is, *   and then skip exact that number of bytes to get to the next file *   in that HOG. * *    char sig[3] = {'D', 'H', 'F'}; // "DHF"=Descent HOG File * *    struct { *     char file_name[13]; // Filename, padded to 13 bytes with 0s *     int file_size; // filesize in bytes *     char data[file_size]; // The file data *    } FILE_STRUCT; // Repeated until the end of the file. * * (That info is from http://www.descent2.com/ddn/specs/hog/) * * Please see the file LICENSE in the source's root directory. * *  This file written by Bradley Bell. *  Based on grp.c by Ryan C. Gordon. */#if HAVE_CONFIG_H#  include <config.h>#endif#if (defined PHYSFS_SUPPORTS_HOG)#include <stdio.h>#include <stdlib.h>#include <string.h>#include "physfs.h"#define __PHYSICSFS_INTERNAL__#include "physfs_internal.h"/* * One HOGentry is kept for each file in an open HOG archive. */typedef struct{    char name[13];    PHYSFS_uint32 startPos;    PHYSFS_uint32 size;} HOGentry;/* * One HOGinfo is kept for each open HOG archive. */typedef struct{    char *filename;    PHYSFS_sint64 last_mod_time;    PHYSFS_uint32 entryCount;    HOGentry *entries;} HOGinfo;/* * One HOGfileinfo is kept for each open file in a HOG archive. */typedef struct{    void *handle;    HOGentry *entry;    PHYSFS_uint32 curPos;} HOGfileinfo;static void HOG_dirClose(DirHandle *h);static PHYSFS_sint64 HOG_read(FileHandle *handle, void *buffer,                              PHYSFS_uint32 objSize, PHYSFS_uint32 objCount);static PHYSFS_sint64 HOG_write(FileHandle *handle, const void *buffer,                               PHYSFS_uint32 objSize, PHYSFS_uint32 objCount);static int HOG_eof(FileHandle *handle);static PHYSFS_sint64 HOG_tell(FileHandle *handle);static int HOG_seek(FileHandle *handle, PHYSFS_uint64 offset);static PHYSFS_sint64 HOG_fileLength(FileHandle *handle);static int HOG_fileClose(FileHandle *handle);static int HOG_isArchive(const char *filename, int forWriting);static DirHandle *HOG_openArchive(const char *name, int forWriting);static LinkedStringList *HOG_enumerateFiles(DirHandle *h,                                            const char *dirname,                                            int omitSymLinks);static int HOG_exists(DirHandle *h, const char *name);static int HOG_isDirectory(DirHandle *h, const char *name, int *fileExists);static int HOG_isSymLink(DirHandle *h, const char *name, int *fileExists);static PHYSFS_sint64 HOG_getLastModTime(DirHandle *h, const char *n, int *e);static FileHandle *HOG_openRead(DirHandle *h, const char *name, int *exist);static FileHandle *HOG_openWrite(DirHandle *h, const char *name);static FileHandle *HOG_openAppend(DirHandle *h, const char *name);static int HOG_remove(DirHandle *h, const char *name);static int HOG_mkdir(DirHandle *h, const char *name);const PHYSFS_ArchiveInfo __PHYSFS_ArchiveInfo_HOG ={    "HOG",    HOG_ARCHIVE_DESCRIPTION,    "Bradley Bell <btb@icculus.org>",    "http://icculus.org/physfs/",};static const FileFunctions __PHYSFS_FileFunctions_HOG ={    HOG_read,       /* read() method       */    HOG_write,      /* write() method      */    HOG_eof,        /* eof() method        */    HOG_tell,       /* tell() method       */    HOG_seek,       /* seek() method       */    HOG_fileLength, /* fileLength() method */    HOG_fileClose   /* fileClose() method  */};const DirFunctions __PHYSFS_DirFunctions_HOG ={    &__PHYSFS_ArchiveInfo_HOG,    HOG_isArchive,          /* isArchive() method      */    HOG_openArchive,        /* openArchive() method    */    HOG_enumerateFiles,     /* enumerateFiles() method */    HOG_exists,             /* exists() method         */    HOG_isDirectory,        /* isDirectory() method    */    HOG_isSymLink,          /* isSymLink() method      */    HOG_getLastModTime,     /* getLastModTime() method */    HOG_openRead,           /* openRead() method       */    HOG_openWrite,          /* openWrite() method      */    HOG_openAppend,         /* openAppend() method     */    HOG_remove,             /* remove() method         */    HOG_mkdir,              /* mkdir() method          */    HOG_dirClose            /* dirClose() method       */};static void HOG_dirClose(DirHandle *h){    HOGinfo *info = ((HOGinfo *) h->opaque);    free(info->filename);    free(info->entries);    free(info);    free(h);} /* HOG_dirClose */static PHYSFS_sint64 HOG_read(FileHandle *handle, void *buffer,                              PHYSFS_uint32 objSize, PHYSFS_uint32 objCount){    HOGfileinfo *finfo = (HOGfileinfo *) (handle->opaque);    HOGentry *entry = finfo->entry;    PHYSFS_uint32 bytesLeft = entry->size - finfo->curPos;    PHYSFS_uint32 objsLeft = (bytesLeft / objSize);    PHYSFS_sint64 rc;    if (objsLeft < objCount)        objCount = objsLeft;    rc = __PHYSFS_platformRead(finfo->handle, buffer, objSize, objCount);    if (rc > 0)        finfo->curPos += (PHYSFS_uint32) (rc * objSize);    return(rc);} /* HOG_read */static PHYSFS_sint64 HOG_write(FileHandle *handle, const void *buffer,                               PHYSFS_uint32 objSize, PHYSFS_uint32 objCount){    BAIL_MACRO(ERR_NOT_SUPPORTED, -1);} /* HOG_write */static int HOG_eof(FileHandle *handle){    HOGfileinfo *finfo = (HOGfileinfo *) (handle->opaque);    HOGentry *entry = finfo->entry;    return(finfo->curPos >= entry->size);} /* HOG_eof */static PHYSFS_sint64 HOG_tell(FileHandle *handle){    return(((HOGfileinfo *) (handle->opaque))->curPos);} /* HOG_tell */static int HOG_seek(FileHandle *handle, PHYSFS_uint64 offset){    HOGfileinfo *finfo = (HOGfileinfo *) (handle->opaque);    HOGentry *entry = finfo->entry;    int rc;    BAIL_IF_MACRO(offset < 0, ERR_INVALID_ARGUMENT, 0);    BAIL_IF_MACRO(offset >= entry->size, ERR_PAST_EOF, 0);    rc = __PHYSFS_platformSeek(finfo->handle, entry->startPos + offset);    if (rc)        finfo->curPos = (PHYSFS_uint32) offset;    return(rc);} /* HOG_seek */static PHYSFS_sint64 HOG_fileLength(FileHandle *handle){    HOGfileinfo *finfo = ((HOGfileinfo *) handle->opaque);    return((PHYSFS_sint64) finfo->entry->size);} /* HOG_fileLength */static int HOG_fileClose(FileHandle *handle){    HOGfileinfo *finfo = ((HOGfileinfo *) handle->opaque);    BAIL_IF_MACRO(!__PHYSFS_platformClose(finfo->handle), NULL, 0);    free(finfo);    free(handle);    return(1);} /* HOG_fileClose */static int hog_open(const char *filename, int forWriting,                    void **fh, PHYSFS_uint32 *count){    PHYSFS_uint8 buf[13];    PHYSFS_uint32 size;    PHYSFS_sint64 pos;    *count = 0;    *fh = NULL;    BAIL_IF_MACRO(forWriting, ERR_ARC_IS_READ_ONLY, 0);    *fh = __PHYSFS_platformOpenRead(filename);    BAIL_IF_MACRO(*fh == NULL, NULL, 0);    if (__PHYSFS_platformRead(*fh, buf, 3, 1) != 1)        goto openHog_failed;    if (memcmp(buf, "DHF", 3) != 0)    {        __PHYSFS_setError(ERR_UNSUPPORTED_ARCHIVE);        goto openHog_failed;    } /* if */    while( 1 )    {        if (__PHYSFS_platformRead(*fh, buf, 13, 1) != 1)            break;             //eof here is ok        if (__PHYSFS_platformRead(*fh, &size, 4, 1) != 1)            goto openHog_failed;        size = PHYSFS_swapULE32(size);        (*count)++;        // Skip over entry        pos = __PHYSFS_platformTell(*fh);        if (pos == -1)            goto openHog_failed;        if (!__PHYSFS_platformSeek(*fh, pos + size))            goto openHog_failed;    }    // Rewind to start of entries    if (!__PHYSFS_platformSeek(*fh, 3))        goto openHog_failed;    return(1);openHog_failed:    if (*fh != NULL)        __PHYSFS_platformClose(*fh);    *count = -1;    *fh = NULL;    return(0);} /* hog_open */

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -