[286] | 1 | // This file is a part of Framsticks SDK. http://www.framsticks.com/ |
---|
[1328] | 2 | // Copyright (C) 1999-2024 Maciej Komosinski and Szymon Ulatowski. |
---|
[286] | 3 | // See LICENSE.txt for details. |
---|
[109] | 4 | |
---|
| 5 | #ifndef _VIRTFILE_H_ |
---|
| 6 | #define _VIRTFILE_H_ |
---|
| 7 | |
---|
| 8 | #include <stdio.h> |
---|
| 9 | #include <stdarg.h> |
---|
| 10 | #include <common/nonstd_dir.h> |
---|
[295] | 11 | #include <string> |
---|
[410] | 12 | #include <set> |
---|
[295] | 13 | using std::string; |
---|
[109] | 14 | |
---|
[301] | 15 | #ifdef DLLEXPORTACTIVE //defined in the project that makes the DLL |
---|
[109] | 16 | #define DLLEXP __declspec(dllexport) |
---|
| 17 | #else |
---|
[301] | 18 | #ifdef __BORLANDC__ //assuming that all executables produced by borland use the DLL |
---|
| 19 | #define DLLEXP __declspec(dllimport) //without dllimport, fields in objects would have separate instances in DLL and in EXE |
---|
| 20 | #else |
---|
[109] | 21 | #define DLLEXP |
---|
| 22 | #endif |
---|
[301] | 23 | #endif |
---|
[109] | 24 | |
---|
| 25 | class DLLEXP VirtFileSystem; |
---|
| 26 | |
---|
| 27 | class DLLEXP VirtFILE |
---|
| 28 | { |
---|
[302] | 29 | protected: |
---|
| 30 | string path; |
---|
| 31 | public: |
---|
| 32 | virtual size_t Vread(void *ptr, size_t size, size_t nmemb) = 0; |
---|
| 33 | virtual size_t Vwrite(const void *ptr, size_t size, size_t nmemb) = 0; |
---|
| 34 | virtual int Veof() = 0; |
---|
| 35 | virtual int Vputc(int c) { unsigned char data = (unsigned char)c; return (Vwrite(&data, 1, 1) == 1) ? data : EOF; } |
---|
| 36 | virtual int Vputs(const char *s) = 0; |
---|
| 37 | virtual int Vgetc() { unsigned char data; if (Vread(&data, 1, 1) == 1) return data; else return EOF; } |
---|
| 38 | virtual int Vseek(long offset, int whence) = 0; |
---|
| 39 | virtual long Vtell() = 0; |
---|
| 40 | virtual void Vrewind() { Vseek(0, SEEK_SET); } |
---|
| 41 | virtual int Vflush() = 0; |
---|
| 42 | virtual char *Vgets(char *s, int size) = 0; |
---|
| 43 | virtual int Vprintf(const char *format, va_list args); |
---|
| 44 | int printf(const char *format, ...); |
---|
| 45 | virtual const char *VgetPath() { return path.c_str(); } |
---|
| 46 | virtual int getSize(); |
---|
| 47 | VirtFILE(const char* _path) :path(_path) {} |
---|
| 48 | virtual ~VirtFILE(); |
---|
| 49 | static VirtFILE *Vstdin, *Vstdout, *Vstderr; |
---|
| 50 | static void setVstdin(VirtFILE *); |
---|
| 51 | static void setVstdout(VirtFILE *); |
---|
| 52 | static void setVstderr(VirtFILE *); |
---|
| 53 | static VirtFILE* getVstdin(); |
---|
| 54 | static VirtFILE* getVstdout(); |
---|
| 55 | static VirtFILE* getVstderr(); |
---|
| 56 | static VirtFileSystem *vfs; |
---|
| 57 | static void selectFileSystem(VirtFileSystem *s); |
---|
[109] | 58 | }; |
---|
| 59 | |
---|
[295] | 60 | /** can be used directly or as a base class for implementations delegating VirtFILE calls to another VirtFILE object */ |
---|
[302] | 61 | class DLLEXP DelegatedFILE : public VirtFILE |
---|
[295] | 62 | { |
---|
[302] | 63 | VirtFILE *delegate; |
---|
| 64 | public: |
---|
| 65 | size_t Vread(void *ptr, size_t size, size_t nmemb) { return delegate->Vread(ptr, size, nmemb); } |
---|
| 66 | size_t Vwrite(const void *ptr, size_t size, size_t nmemb) { return delegate->Vwrite(ptr, size, nmemb); } |
---|
| 67 | int Veof() { return delegate->Veof(); } |
---|
| 68 | int Vputc(int c) { return delegate->Vputc(c); } |
---|
| 69 | int Vputs(const char *s) { return delegate->Vputs(s); } |
---|
| 70 | int Vgetc() { return delegate->Vgetc(); } |
---|
| 71 | int Vseek(long offset, int whence) { return delegate->Vseek(offset, whence); } |
---|
| 72 | long Vtell() { return delegate->Vtell(); } |
---|
| 73 | void Vrewind() { delegate->Vrewind(); } |
---|
| 74 | int Vflush() { return delegate->Vflush(); } |
---|
| 75 | char *Vgets(char *s, int size) { return delegate->Vgets(s, size); } |
---|
| 76 | int Vprintf(const char *format, va_list args) { return delegate->Vprintf(format, args); } |
---|
| 77 | int getSize() { return delegate->getSize(); } |
---|
| 78 | // not overriden: VgetPath() |
---|
[295] | 79 | |
---|
[302] | 80 | DelegatedFILE(const char* _path, VirtFILE *_delegate) :VirtFILE(_path), delegate(_delegate) {} |
---|
| 81 | virtual ~DelegatedFILE() { if (delegate) delete delegate; delegate = NULL; } |
---|
[295] | 82 | }; |
---|
| 83 | |
---|
[109] | 84 | class DLLEXP VirtDIR |
---|
| 85 | { |
---|
[302] | 86 | public: |
---|
| 87 | virtual ~VirtDIR() {} |
---|
[427] | 88 | virtual dirent* Vreaddir() { return NULL; } |
---|
[109] | 89 | }; |
---|
| 90 | |
---|
| 91 | class DLLEXP VirtFileSystem |
---|
| 92 | { |
---|
| 93 | public: |
---|
[302] | 94 | virtual VirtFILE *Vfopen(const char* path, const char*mode); |
---|
| 95 | virtual bool Vfexists(const char* path); |
---|
| 96 | virtual VirtDIR *Vopendir(const char* path); |
---|
| 97 | virtual bool Vmkdir(const char* path); |
---|
| 98 | virtual bool Vmkdirs(const char* path); |
---|
| 99 | virtual bool Vdirexists(const char* path, bool is_writable); |
---|
[1328] | 100 | virtual bool Vdelete(const char* path); |
---|
| 101 | struct Stat { bool is_file; time_t modification_time; }; |
---|
| 102 | virtual bool Vstat(const char* path, Stat* stat); |
---|
| 103 | virtual bool Vsettime(const char* path, double timestamp); |
---|
[109] | 104 | }; |
---|
| 105 | |
---|
[295] | 106 | /// base class for chained filesystems - redirect unimplemented calls -> chain |
---|
| 107 | class DLLEXP ChainFileSystem : public VirtFileSystem |
---|
| 108 | { |
---|
| 109 | public: |
---|
| 110 | VirtFileSystem *chain; |
---|
[301] | 111 | ChainFileSystem(VirtFileSystem *_chain = NULL); |
---|
[295] | 112 | VirtFILE *Vfopen(const char* path, const char*mode); |
---|
| 113 | bool Vfexists(const char* path); |
---|
| 114 | VirtDIR *Vopendir(const char* path); |
---|
[891] | 115 | virtual VirtDIR *internalopendir(const char* path) { return NULL; } |
---|
[295] | 116 | bool Vmkdir(const char* path); |
---|
| 117 | bool Vmkdirs(const char* path); |
---|
[302] | 118 | bool Vdirexists(const char* path, bool is_writable); |
---|
[1328] | 119 | bool Vdelete(const char* path); |
---|
| 120 | bool Vstat(const char* path, Stat* stat); |
---|
| 121 | bool Vsettime(const char* path, double timestamp); |
---|
[410] | 122 | |
---|
[891] | 123 | class Dir : public VirtDIR |
---|
[410] | 124 | { |
---|
[891] | 125 | ChainFileSystem *first; |
---|
| 126 | VirtFileSystem *second; |
---|
| 127 | string path; |
---|
| 128 | std::set<string> duplicates; |
---|
| 129 | VirtDIR *dir; |
---|
| 130 | public: |
---|
| 131 | Dir(string _path, ChainFileSystem *_first, VirtFileSystem *_second) :first(_first), second(_second), path(_path), dir(NULL) {} |
---|
| 132 | ~Dir(); |
---|
| 133 | dirent* Vreaddir(); |
---|
[410] | 134 | }; |
---|
[295] | 135 | }; |
---|
| 136 | |
---|
| 137 | |
---|
[302] | 138 | DLLEXP VirtFILE *Vfopen(const char* path, const char*mode); |
---|
[109] | 139 | DLLEXP VirtDIR *Vopendir(const char* path); |
---|
[295] | 140 | DLLEXP bool Vfexists(const char* path); |
---|
| 141 | DLLEXP bool Vmkdir(const char* path); |
---|
| 142 | DLLEXP bool Vmkdirs(const char* path); |
---|
[302] | 143 | DLLEXP bool Vdirexists(const char* path, bool is_writable); |
---|
[1328] | 144 | DLLEXP bool Vdelete(const char* path); |
---|
| 145 | DLLEXP bool Vstat(const char* path, struct VirtFileSystem::Stat *stat); |
---|
| 146 | DLLEXP bool Vsettime(const char* path, double timestamp); |
---|
[109] | 147 | |
---|
[523] | 148 | #ifdef VIRTFILE_OVERLOADING |
---|
[655] | 149 | //required for redefine_stdio.h and for building virtfile.dll, can also be used if someone prefers the standard f-functions instead of explicitly using virtfile objects |
---|
[523] | 150 | |
---|
[891] | 151 | DLLEXP size_t fread(void *ptr, size_t size, size_t nmemb, VirtFILE* f); |
---|
| 152 | DLLEXP size_t fwrite(const void *ptr, size_t size, size_t nmemb, VirtFILE* f); |
---|
[123] | 153 | |
---|
[131] | 154 | //since we want our own feof(VirtFILE*) function and some systems unfortunately define feof as a macro, we need to #undef it. Same as in virtfile.cpp |
---|
[225] | 155 | #if defined _MSC_VER || defined __CYGWIN__ || defined SHP || defined __ANDROID__ |
---|
[302] | 156 | #pragma push_macro("feof") |
---|
| 157 | #undef feof |
---|
[123] | 158 | #endif |
---|
[131] | 159 | #if defined __BORLANDC__ //does not support #pragma push_macro/pop_macro |
---|
[302] | 160 | #undef feof |
---|
[131] | 161 | #endif |
---|
[302] | 162 | |
---|
[123] | 163 | DLLEXP int feof(VirtFILE* f);// {return f->Veof();} |
---|
| 164 | |
---|
| 165 | //...and then restore the original macro: |
---|
[225] | 166 | #if defined _MSC_VER || defined __CYGWIN__ || defined SHP || defined __ANDROID__ |
---|
[302] | 167 | #pragma pop_macro("feof") |
---|
[131] | 168 | #endif |
---|
[225] | 169 | #if defined __BORLANDC__ |
---|
[302] | 170 | #define feof(__f) ((__f)->flags & _F_EOF) |
---|
[109] | 171 | #endif |
---|
| 172 | |
---|
[123] | 173 | |
---|
[302] | 174 | DLLEXP int fputc(int c, VirtFILE* f); |
---|
| 175 | DLLEXP int fputs(const char *s, VirtFILE* f); |
---|
[109] | 176 | DLLEXP int fgetc(VirtFILE* f); |
---|
[302] | 177 | DLLEXP int fseek(VirtFILE* f, long offset, int whence); |
---|
[109] | 178 | DLLEXP int ftell(VirtFILE* f); |
---|
| 179 | DLLEXP void rewind(VirtFILE* f); |
---|
| 180 | DLLEXP int fflush(VirtFILE* f); |
---|
| 181 | DLLEXP char *fgets(char *s, int size, VirtFILE* f); |
---|
[302] | 182 | DLLEXP int fprintf(VirtFILE* f, const char *format, ...); |
---|
[109] | 183 | DLLEXP int fclose(VirtFILE* f); |
---|
| 184 | |
---|
| 185 | DLLEXP dirent* readdir(VirtDIR* d); |
---|
| 186 | DLLEXP int closedir(VirtDIR* d); |
---|
| 187 | |
---|
[523] | 188 | #endif //VIRTFILE_OVERLOADING |
---|
| 189 | |
---|
[109] | 190 | #endif |
---|