LCOV - code coverage report
Current view: top level - shared - stream.cpp (source / functions) Coverage Total Hit
Test: Libprimis Test Coverage Lines: 31.5 % 644 203
Test Date: 2025-04-04 04:51:56 Functions: 37.3 % 67 25

            Line data    Source code
       1              : /* stream.cpp: utilities for character streams
       2              :  *
       3              :  * stream.cpp defines character handling to enable character streams to be written
       4              :  * into and out of files
       5              :  * also included is utilities for gz archive support
       6              :  *
       7              :  */
       8              : #include <sstream>
       9              : 
      10              : #include "../libprimis-headers/cube.h"
      11              : #include "stream.h"
      12              : 
      13              : #include "../engine/interface/console.h"
      14              : 
      15              : ///////////////////////// character conversion /////////////////////////////////
      16              : 
      17              : #define CUBECTYPE(s, p, d, a, A, u, U) \
      18              :     0, U, U, U, U, U, U, U, U, s, s, s, s, s, U, U, \
      19              :     U, U, U, U, U, U, U, U, U, U, U, U, U, U, U, U, \
      20              :     s, p, p, p, p, p, p, p, p, p, p, p, p, p, p, p, \
      21              :     d, d, d, d, d, d, d, d, d, d, p, p, p, p, p, p, \
      22              :     p, A, A, A, A, A, A, A, A, A, A, A, A, A, A, A, \
      23              :     A, A, A, A, A, A, A, A, A, A, A, p, p, p, p, p, \
      24              :     p, a, a, a, a, a, a, a, a, a, a, a, a, a, a, a, \
      25              :     a, a, a, a, a, a, a, a, a, a, a, p, p, p, p, U, \
      26              :     U, u, u, u, u, u, u, u, u, u, u, u, u, u, u, u, \
      27              :     u, u, u, u, u, u, u, u, u, u, u, u, u, u, u, U, \
      28              :     u, U, u, U, u, U, u, U, u, U, u, U, u, U, u, U, \
      29              :     u, U, u, U, u, U, u, U, u, U, u, U, u, U, u, U, \
      30              :     u, U, u, U, u, U, u, U, U, u, U, u, U, u, U, U, \
      31              :     U, U, U, U, U, U, U, U, U, U, U, U, U, U, U, U, \
      32              :     U, U, U, U, u, u, u, u, u, u, u, u, u, u, u, u, \
      33              :     u, u, u, u, u, u, u, u, u, u, u, u, u, u, U, u
      34              : 
      35              : /* note here:
      36              :  * these vars are declared extern inline to allow a `const` (implicitly also
      37              :  * `static`) to be linked to other files as a `const`.
      38              :  *
      39              :  * these vars cannot be `constexpr` due to it not being legal to define constexpr
      40              :  * prototypes in headers
      41              :  */
      42              : 
      43              : extern const uchar cubectype[256] =
      44              : {
      45              :     CUBECTYPE(CubeType_Space,
      46              :               CubeType_Print,
      47              :               CubeType_Print | CubeType_Digit,
      48              :               CubeType_Print | CubeType_Alpha | CubeType_Lower,
      49              :               CubeType_Print | CubeType_Alpha | CubeType_Upper,
      50              :               CubeType_Print | CubeType_Unicode | CubeType_Alpha | CubeType_Lower,
      51              :               CubeType_Print | CubeType_Unicode | CubeType_Alpha | CubeType_Upper)
      52              : };
      53              : 
      54          697 : size_t encodeutf8(uchar *dstbuf, size_t dstlen, const uchar *srcbuf, size_t srclen, size_t *carry)
      55              : {
      56          697 :     uchar *dst = dstbuf,
      57          697 :           *dstend = &dstbuf[dstlen];
      58          697 :     const uchar *src = srcbuf,
      59          697 :                 *srcend = &srcbuf[srclen];
      60          697 :     if(src < srcend && dst < dstend)
      61              :     {
      62              :         do
      63              :         {
      64          697 :             int uni = cube2uni(*src);
      65          697 :             if(uni <= 0x7F)
      66              :             {
      67          697 :                 if(dst >= dstend)
      68              :                 {
      69            0 :                     goto done;
      70              :                 }
      71          697 :                 const uchar *end = std::min(srcend, &src[dstend-dst]);
      72              :                 do
      73              :                 {
      74        13139 :                     if(uni == '\f')
      75              :                     {
      76            0 :                         if(++src >= srcend)
      77              :                         {
      78            0 :                             goto done;
      79              :                         }
      80            0 :                         goto uni1;
      81              :                     }
      82        13139 :                     *dst++ = uni;
      83        13139 :                     if(++src >= end)
      84              :                     {
      85          697 :                         goto done;
      86              :                     }
      87        12442 :                     uni = cube2uni(*src);
      88        12442 :                 } while(uni <= 0x7F);
      89              :             }
      90            0 :             if(uni <= 0x7FF)
      91              :             {
      92            0 :                 if(dst + 2 > dstend)
      93              :                 {
      94            0 :                     goto done;
      95              :                 }
      96            0 :                 *dst++ = 0xC0 | (uni>>6);
      97            0 :                 goto uni2;
      98              :             }
      99            0 :             else if(uni <= 0xFFFF)
     100              :             {
     101            0 :                 if(dst + 3 > dstend)
     102              :                 {
     103            0 :                     goto done;
     104              :                 }
     105            0 :                 *dst++ = 0xE0 | (uni>>12);
     106            0 :                 goto uni3;
     107              :             }
     108            0 :             else if(uni <= 0x1FFFFF)
     109              :             {
     110            0 :                 if(dst + 4 > dstend)
     111              :                 {
     112            0 :                     goto done;
     113              :                 }
     114            0 :                 *dst++ = 0xF0 | (uni>>18);
     115            0 :                 goto uni4;
     116              :             }
     117            0 :             else if(uni <= 0x3FFFFFF)
     118              :             {
     119            0 :                 if(dst + 5 > dstend)
     120              :                 {
     121            0 :                     goto done;
     122              :                 }
     123            0 :                 *dst++ = 0xF8 | (uni>>24);
     124            0 :                 goto uni5;
     125              :             }
     126              :             else if(uni <= 0x7FFFFFFF)
     127              :             {
     128            0 :                 if(dst + 6 > dstend)
     129              :                 {
     130            0 :                     goto done;
     131              :                 }
     132            0 :                 *dst++ = 0xFC | (uni>>30);
     133            0 :                 goto uni6;
     134              :             }
     135              :             else
     136              :             {
     137              :                 goto uni1;
     138              :             }
     139            0 :         uni6: *dst++ = 0x80 | ((uni>>24)&0x3F);
     140            0 :         uni5: *dst++ = 0x80 | ((uni>>18)&0x3F);
     141            0 :         uni4: *dst++ = 0x80 | ((uni>>12)&0x3F);
     142            0 :         uni3: *dst++ = 0x80 | ((uni>>6)&0x3F);
     143            0 :         uni2: *dst++ = 0x80 | (uni&0x3F);
     144            0 :         uni1:;
     145            0 :         } while(++src < srcend);
     146              :     }
     147            0 : done:
     148          697 :     if(carry)
     149              :     {
     150          697 :         *carry += src - srcbuf;
     151              :     }
     152          697 :     return dst - dstbuf;
     153              : }
     154              : 
     155              : ///////////////////////// file system ///////////////////////
     156              : 
     157              : #ifdef WIN32
     158              : #include <shlobj.h>
     159              : #else
     160              : #include <unistd.h>
     161              : #include <sys/stat.h>
     162              : #include <sys/types.h>
     163              : #include <dirent.h>
     164              : #endif
     165              : 
     166              : std::string homedir = "";
     167              : struct packagedir
     168              : {
     169              :     std::string dir;
     170              :     std::string filter;
     171              : };
     172              : static std::vector<packagedir> packagedirs;
     173              : 
     174           12 : char *makerelpath(const char *dir, const char *file, const char *prefix, const char *cmd)
     175              : {
     176              :     static string tmp;
     177           12 :     if(prefix)
     178              :     {
     179            0 :         copystring(tmp, prefix);
     180              :     }
     181              :     else
     182              :     {
     183           12 :         tmp[0] = '\0';
     184              :     }
     185           12 :     if(file[0]=='<')
     186              :     {
     187            0 :         const char *end = std::strrchr(file, '>');
     188            0 :         if(end)
     189              :         {
     190            0 :             size_t len = std::strlen(tmp);
     191            0 :             copystring(&tmp[len], file, std::min(sizeof(tmp)-len, static_cast<size_t>(end+2-file)));
     192            0 :             file = end+1;
     193              :         }
     194              :     }
     195           12 :     if(cmd)
     196              :     {
     197            0 :         concatstring(tmp, cmd);
     198              :     }
     199           12 :     if(dir)
     200              :     {
     201           12 :         DEF_FORMAT_STRING(pname, "%s/%s", dir, file);
     202           12 :         concatstring(tmp, pname);
     203              :     }
     204              :     else
     205              :     {
     206            0 :         concatstring(tmp, file);
     207              :     }
     208           12 :     return tmp;
     209              : }
     210              : 
     211              : 
     212           30 : char *path(char *s)
     213              : {
     214           30 :     for(char *curpart = s;;)
     215              :     {
     216           34 :         char *endpart = std::strchr(curpart, '&');
     217           34 :         if(endpart)
     218              :         {
     219            4 :             *endpart = '\0';
     220              :         }
     221           34 :         if(curpart[0]=='<')
     222              :         {
     223            4 :             char *file = std::strrchr(curpart, '>');
     224            4 :             if(!file)
     225              :             {
     226            0 :                 return s;
     227              :             }
     228            4 :             curpart = file+1;
     229              :         }
     230          114 :         for(char *t = curpart; (t = std::strpbrk(t, "/\\")); *t++ = PATHDIV)
     231              :         {
     232              :             //(empty body)
     233              :         }
     234           34 :         for(char *prevdir = nullptr, *curdir = curpart;;)
     235              :         {
     236          107 :             prevdir = curdir[0]==PATHDIV ? curdir+1 : curdir;
     237          107 :             curdir = std::strchr(prevdir, PATHDIV);
     238          107 :             if(!curdir)
     239              :             {
     240           34 :                 break;
     241              :             }
     242           73 :             if(prevdir+1==curdir && prevdir[0]=='.')
     243              :             {
     244            4 :                 std::memmove(prevdir, curdir+1, std::strlen(curdir+1)+1);
     245            4 :                 curdir = prevdir;
     246              :             }
     247           69 :             else if(curdir[1]=='.' && curdir[2]=='.' && curdir[3]==PATHDIV)
     248              :             {
     249            8 :                 if(prevdir+2==curdir && prevdir[0]=='.' && prevdir[1]=='.')
     250              :                 {
     251            0 :                     continue;
     252              :                 }
     253            8 :                 std::memmove(prevdir, curdir+4, std::strlen(curdir+4)+1);
     254            8 :                 if(prevdir-2 >= curpart && prevdir[-1]==PATHDIV)
     255              :                 {
     256            1 :                     prevdir -= 2;
     257            2 :                     while(prevdir-1 >= curpart && prevdir[-1] != PATHDIV)
     258              :                     {
     259            1 :                         --prevdir;
     260              :                     }
     261              :                 }
     262            8 :                 curdir = prevdir;
     263              :             }
     264              :         }
     265           34 :         if(endpart)
     266              :         {
     267            4 :             *endpart = '&';
     268            4 :             curpart = endpart+1;
     269              :         }
     270              :         else
     271              :         {
     272           30 :             break;
     273              :         }
     274            4 :     }
     275           30 :     return s;
     276              : }
     277              : 
     278           58 : std::string path(std::string s)
     279              : {
     280           58 :     std::string truncated_path, processed_path;
     281              : 
     282           58 :     size_t path_begin = 0,
     283           58 :            path_end = s.length();
     284              : 
     285              :     // Find the in-line command segment and skip it
     286           58 :     if(s.find('<') != std::string::npos)
     287              :     {
     288            4 :         size_t command_end = s.rfind('>');
     289            4 :         if(command_end != std::string::npos)
     290              :         {
     291            4 :             path_begin = command_end + 1;
     292              :         }
     293              :     }
     294              : 
     295              :     // Find the conjugated path and cut it
     296           58 :     if(s.find('&') != std::string::npos)
     297              :     {
     298            4 :         path_end = s.find('&');
     299              :     }
     300              : 
     301           58 :     truncated_path = s.substr(path_begin, path_end - path_begin);
     302              : 
     303              :     // Handle "."" and ".."" in the path
     304           58 :     std::istringstream path_stream(truncated_path);
     305           58 :     std::stack<std::string> path_stack;
     306           58 :     std::string token;
     307              : 
     308              :     // Construct a stack of path tokens
     309          268 :     while(std::getline(path_stream, token, '/'))
     310              :     {
     311          210 :         if(token == "..")
     312              :         {
     313            8 :             if(!path_stack.empty())
     314              :             {
     315            6 :                 path_stack.pop();
     316              :             }
     317              :             else
     318              :             {
     319            2 :                 path_stack.push(token);
     320              :             }
     321              :         }
     322          202 :         else if(!token.empty() && token != ".")
     323              :         {
     324          195 :             path_stack.push(token);
     325              :         }
     326              :     }
     327              : 
     328              :     // Re-construct the processed path from the stack
     329          249 :     while(!path_stack.empty())
     330              :     {
     331          191 :         if(path_stack.size() > 1)
     332              :         {
     333          140 :             processed_path = "/" + path_stack.top() + processed_path;
     334              :         }
     335              :         else
     336              :         {
     337           51 :             processed_path = path_stack.top() + processed_path;
     338              :         }
     339          191 :         path_stack.pop();
     340              :     }
     341              : 
     342          116 :     return processed_path;
     343           58 : }
     344              : 
     345            2 : char *copypath(const char *s)
     346              : {
     347              :     static string tmp;
     348            2 :     copystring(tmp, s);
     349            2 :     path(tmp);
     350            2 :     return tmp;
     351              : }
     352              : 
     353            7 : const char *parentdir(const char *directory)
     354              : {
     355            7 :     const char *p = directory + std::strlen(directory);
     356           48 :     while(p > directory && *p != '/' && *p != '\\')
     357              :     {
     358           41 :         p--;
     359              :     }
     360              :     static string parent;
     361            7 :     size_t len = p-directory+1;
     362            7 :     copystring(parent, directory, len);
     363            7 :     return parent;
     364              : }
     365              : 
     366            2 : bool fileexists(const char *path, const char *mode)
     367              : {
     368            2 :     bool exists = true;
     369            2 :     if(mode[0]=='w' || mode[0]=='a')
     370              :     {
     371            1 :         path = parentdir(path);
     372              :     }
     373              : #ifdef WIN32
     374              :     if(GetFileAttributes(path[0] ? path : ".\\") == INVALID_FILE_ATTRIBUTES)
     375              :     {
     376              :         exists = false;
     377              :     }
     378              : #else
     379            2 :     if(access(path[0] ? path : ".", mode[0]=='w' || mode[0]=='a' ? W_OK : (mode[0]=='d' ? X_OK : R_OK)) == -1)
     380              :     {
     381            1 :         exists = false;
     382              :     }
     383              : #endif
     384            2 :     return exists;
     385              : }
     386              : 
     387            1 : bool createdir(const char *path)
     388              : {
     389            1 :     size_t len = std::strlen(path);
     390            1 :     if(path[len-1] == PATHDIV)
     391              :     {
     392              :         static string strip;
     393            1 :         path = copystring(strip, path, len);
     394              :     }
     395              : #ifdef WIN32
     396              :     return CreateDirectory(path, nullptr) != 0;
     397              : #else
     398            1 :     return mkdir(path, 0777) == 0;
     399              : #endif
     400              : }
     401              : 
     402            3 : size_t fixpackagedir(char *dir)
     403              : {
     404            3 :     path(dir);
     405            3 :     size_t len = std::strlen(dir);
     406            3 :     if(len > 0 && dir[len-1] != PATHDIV)
     407              :     {
     408            1 :         dir[len] = PATHDIV;
     409            1 :         dir[len+1] = '\0';
     410              :     }
     411            3 :     return len;
     412              : }
     413              : 
     414            0 : bool subhomedir(char *dst, int len, const char *src)
     415              : {
     416            0 :     const char *sub = std::strstr(src, "$HOME");
     417            0 :     if(!sub)
     418              :     {
     419            0 :         sub = std::strchr(src, '~');
     420              :     }
     421            0 :     if(sub && sub-src < len)
     422              :     {
     423              : #ifdef WIN32
     424              :         char home[MAX_PATH+1];
     425              :         home[0] = '\0';
     426              :         if(SHGetFolderPath(nullptr, CSIDL_PERSONAL, nullptr, 0, home) != S_OK || !home[0])
     427              :         {
     428              :             return false;
     429              :         }
     430              : #else
     431            0 :         const char *home = getenv("HOME");
     432            0 :         if(!home || !home[0])
     433              :         {
     434            0 :             return false;
     435              :         }
     436              : #endif
     437            0 :         dst[sub-src] = '\0';
     438            0 :         concatstring(dst, home, len);
     439            0 :         concatstring(dst, sub+(*sub == '~' ? 1 : std::strlen("$HOME")), len);
     440              :     }
     441            0 :     return true;
     442              : }
     443              : 
     444            0 : const char *sethomedir(const char *dir)
     445              : {
     446              :     string pdir;
     447            0 :     copystring(pdir, dir);
     448            0 :     if(!subhomedir(pdir, sizeof(pdir), dir) || !fixpackagedir(pdir))
     449              :     {
     450            0 :         return nullptr;
     451              :     }
     452            0 :     homedir = pdir;
     453            0 :     return homedir.c_str();
     454              : }
     455              : 
     456            0 : const char *addpackagedir(const char *dir)
     457              : {
     458              :     string pdir;
     459            0 :     copystring(pdir, dir);
     460            0 :     if(!subhomedir(pdir, sizeof(pdir), dir) || !fixpackagedir(pdir))
     461              :     {
     462            0 :         return nullptr;
     463              :     }
     464            0 :     char *filter = pdir;
     465              :     for(;;)
     466              :     {
     467              :         static int len = std::strlen("media");
     468            0 :         filter = std::strstr(filter, "media");
     469            0 :         if(!filter)
     470              :         {
     471            0 :             break;
     472              :         }
     473            0 :         if(filter > pdir && filter[-1] == PATHDIV && filter[len] == PATHDIV)
     474              :         {
     475            0 :             break;
     476              :         }
     477            0 :         filter += len;
     478            0 :     }
     479            0 :     packagedir pf;
     480            0 :     pf.dir = filter ? std::string(pdir, filter-pdir) : std::string(pdir);
     481            0 :     pf.filter = filter ? std::string(filter) : "";
     482            0 :     packagedirs.push_back(pf);
     483            0 :     return pf.dir.c_str();
     484            0 : }
     485              : 
     486           23 : const char *findfile(const char *filename, const char *mode)
     487              : {
     488              :     static string s;
     489           23 :     if(homedir[0])
     490              :     {
     491            0 :         formatstring(s, "%s%s", homedir.c_str(), filename);
     492            0 :         if(fileexists(s, mode))
     493              :         {
     494            0 :             return s;
     495              :         }
     496            0 :         if(mode[0] == 'w' || mode[0] == 'a')
     497              :         {
     498              :             string dirs;
     499            0 :             copystring(dirs, s);
     500            0 :             char *dir = std::strchr(dirs[0] == PATHDIV ? dirs+1 : dirs, PATHDIV);
     501            0 :             while(dir)
     502              :             {
     503            0 :                 *dir = '\0';
     504            0 :                 if(!fileexists(dirs, "d") && !createdir(dirs))
     505              :                 {
     506            0 :                     return s;
     507              :                 }
     508            0 :                 *dir = PATHDIV;
     509            0 :                 dir = std::strchr(dir+1, PATHDIV);
     510              :             }
     511            0 :             return s;
     512              :         }
     513              :     }
     514           23 :     if(mode[0] == 'w' || mode[0] == 'a')
     515              :     {
     516            1 :         return filename;
     517              :     }
     518           22 :     for(const packagedir &pf : packagedirs)
     519              :     {
     520            0 :         if(pf.filter.size() > 0 && std::strncmp(filename, pf.filter.c_str(), pf.filter.size()))
     521              :         {
     522            0 :             continue;
     523              :         }
     524            0 :         formatstring(s, "%s%s", pf.dir.c_str(), filename);
     525            0 :         if(fileexists(s, mode))
     526              :         {
     527            0 :             return s;
     528              :         }
     529              :     }
     530           22 :     if(mode[0]=='e')
     531              :     {
     532            0 :         return nullptr;
     533              :     }
     534           22 :     return filename;
     535              : }
     536              : 
     537            1 : bool listdir(const char *dirname, bool rel, const char *ext, std::vector<char *> &files)
     538              : {
     539            1 :     size_t extsize = ext ? std::strlen(ext)+1 : 0;
     540              : #ifdef WIN32
     541              :     DEF_FORMAT_STRING(pathname, rel ? ".\\%s\\*.%s" : "%s\\*.%s", dirname, ext ? ext : "*");
     542              :     WIN32_FIND_DATA FindFileData;
     543              :     HANDLE Find = FindFirstFile(pathname, &FindFileData);
     544              :     if(Find != INVALID_HANDLE_VALUE)
     545              :     {
     546              :         do {
     547              :             if(!ext)
     548              :             {
     549              :                 files.push_back(newstring(FindFileData.cFileName));
     550              :             }
     551              :             else
     552              :             {
     553              :                 size_t namelen = std::strlen(FindFileData.cFileName);
     554              :                 if(namelen > extsize)
     555              :                 {
     556              :                     namelen -= extsize;
     557              :                     if(FindFileData.cFileName[namelen] == '.' && std::strncmp(FindFileData.cFileName+namelen+1, ext, extsize-1)==0)
     558              :                     {
     559              :                         files.push_back(newstring(FindFileData.cFileName, namelen));
     560              :                     }
     561              :                 }
     562              :             }
     563              :         } while(FindNextFile(Find, &FindFileData));
     564              :         FindClose(Find);
     565              :         return true;
     566              :     }
     567              : #else
     568            1 :     DEF_FORMAT_STRING(pathname, rel ? "./%s" : "%s", dirname);
     569            1 :     DIR *d = opendir(pathname);
     570            1 :     if(d)
     571              :     {
     572              :         struct dirent *de;
     573           74 :         while((de = readdir(d)) != nullptr)
     574              :         {
     575           73 :             if(!ext)
     576              :             {
     577           73 :                 files.push_back(newstring(de->d_name));
     578              :             }
     579              :             else
     580              :             {
     581            0 :                 size_t namelen = std::strlen(de->d_name);
     582            0 :                 if(namelen > extsize)
     583              :                 {
     584            0 :                     namelen -= extsize;
     585            0 :                     if(de->d_name[namelen] == '.' && std::strncmp(de->d_name+namelen+1, ext, extsize-1)==0)
     586              :                     {
     587            0 :                         files.push_back(newstring(de->d_name, namelen));
     588              :                     }
     589              :                 }
     590              :             }
     591              :         }
     592            1 :         closedir(d);
     593            1 :         return true;
     594              :     }
     595              : #endif
     596              :     else
     597              :     {
     598            0 :         return false;
     599              :     }
     600              : }
     601              : 
     602            1 : int listfiles(const char *dir, const char *ext, std::vector<char *> &files)
     603              : {
     604              :     string dirname;
     605            1 :     copystring(dirname, dir);
     606            1 :     path(dirname);
     607            1 :     size_t dirlen = std::strlen(dirname);
     608            1 :     while(dirlen > 1 && dirname[dirlen-1] == PATHDIV)
     609              :     {
     610            0 :         dirname[--dirlen] = '\0';
     611              :     }
     612            1 :     int dirs = 0;
     613            1 :     if(listdir(dirname, true, ext, files))
     614              :     {
     615            1 :         dirs++;
     616              :     }
     617              :     string s;
     618            1 :     if(homedir[0])
     619              :     {
     620            0 :         formatstring(s, "%s%s", homedir.c_str(), dirname);
     621            0 :         if(listdir(s, false, ext, files))
     622              :         {
     623            0 :             dirs++;
     624              :         }
     625              :     }
     626            1 :     for(const packagedir &pf : packagedirs)
     627              :     {
     628            0 :         if(pf.filter.size() && std::strncmp(dirname, pf.filter.c_str(), dirlen == pf.filter.size()-1 ? dirlen : pf.filter.size()))
     629              :         {
     630            0 :             continue;
     631              :         }
     632            0 :         formatstring(s, "%s%s", pf.dir.c_str(), dirname);
     633            0 :         if(listdir(s, false, ext, files))
     634              :         {
     635            0 :             dirs++;
     636              :         }
     637              :     }
     638            1 :     dirs += listzipfiles(dirname, ext, files);
     639            1 :     return dirs;
     640              : }
     641              : 
     642            0 : static Sint64 rwopsseek(SDL_RWops *rw, Sint64 pos, int whence)
     643              : {
     644            0 :     stream *f = static_cast<stream *>(rw->hidden.unknown.data1);
     645            0 :     if((!pos && whence==SEEK_CUR) || f->seek(pos, whence))
     646              :     {
     647            0 :         return static_cast<int>(f->tell());
     648              :     }
     649            0 :     return -1;
     650              : }
     651              : 
     652            0 : static size_t rwopsread(SDL_RWops *rw, void *buf, size_t size, size_t nmemb)
     653              : {
     654            0 :     stream *f = static_cast<stream *>(rw->hidden.unknown.data1);
     655            0 :     return f->read(buf, size*nmemb)/size;
     656              : }
     657              : 
     658            0 : static size_t rwopswrite(SDL_RWops *rw, const void *buf, size_t size, size_t nmemb)
     659              : {
     660            0 :     stream *f = static_cast<stream *>(rw->hidden.unknown.data1);
     661            0 :     return f->write(buf, size*nmemb)/size;
     662              : }
     663              : 
     664            1 : SDL_RWops *stream::rwops()
     665              : {
     666            1 :     SDL_RWops *rw = SDL_AllocRW();
     667            1 :     if(!rw)
     668              :     {
     669            0 :         return nullptr;
     670              :     }
     671            1 :     rw->hidden.unknown.data1 = this;
     672            1 :     rw->seek = rwopsseek;
     673            1 :     rw->read = rwopsread;
     674            1 :     rw->write = rwopswrite;
     675            1 :     rw->close = 0;
     676            1 :     return rw;
     677              : }
     678              : 
     679            2 : stream::offset stream::size()
     680              : {
     681            2 :     offset pos = tell(),
     682              :                  endpos;
     683            2 :     if(pos < 0 || !seek(0, SEEK_END))
     684              :     {
     685            2 :         return -1;
     686              :     }
     687            0 :     endpos = tell();
     688            0 :     return pos == endpos || seek(pos, SEEK_SET) ? endpos : -1;
     689              : }
     690              : 
     691            1 : bool stream::getline(char *str, size_t len)
     692              : {
     693            1 :     for(int i = 0; i < static_cast<int>(len-1); ++i)
     694              :     {
     695            0 :         if(read(&str[i], 1) != 1)
     696              :         {
     697            0 :             str[i] = '\0';
     698            0 :             return i > 0;
     699              :         }
     700            0 :         else if(str[i] == '\n')
     701              :         {
     702            0 :             str[i+1] = '\0';
     703            0 :             return true;
     704              :         }
     705              :     }
     706            1 :     if(len > 0)
     707              :     {
     708            0 :         str[len-1] = '\0';
     709              :     }
     710            1 :     return true;
     711              : }
     712              : 
     713            1 : size_t stream::printf(const char *fmt, ...)
     714              : {
     715              :     char buf[512];
     716            1 :     char *str = buf;
     717              :     va_list args;
     718              : #if defined(WIN32) && !defined(__GNUC__)
     719              :     va_start(args, fmt);
     720              :     int len = _vscprintf(fmt, args);
     721              :     if(len <= 0)
     722              :     {
     723              :         va_end(args);
     724              :         return 0;
     725              :     }
     726              :     if(len >= static_cast<int>(sizeof(buf)))
     727              :     {
     728              :         str = new char[len+1];
     729              :     }
     730              :     _vsnprintf(str, len+1, fmt, args);
     731              :     va_end(args);
     732              : #else
     733            1 :     va_start(args, fmt);
     734            1 :     int len = vsnprintf(buf, sizeof(buf), fmt, args);
     735            1 :     va_end(args);
     736            1 :     if(len <= 0)
     737              :     {
     738            0 :         return 0;
     739              :     }
     740            1 :     if(len >= static_cast<int>(sizeof(buf)))
     741              :     {
     742            0 :         str = new char[len+1];
     743            0 :         va_start(args, fmt);
     744            0 :         vsnprintf(str, len+1, fmt, args);
     745            0 :         va_end(args);
     746              :     }
     747              : #endif
     748            1 :     size_t n = write(str, len);
     749            1 :     if(str != buf)
     750              :     {
     751            0 :         delete[] str;
     752              :     }
     753            1 :     return n;
     754              : }
     755              : 
     756              : struct filestream final : stream
     757              : {
     758              :     FILE *file;
     759              : 
     760           11 :     filestream() : file(nullptr) {}
     761           22 :     ~filestream() { close(); }
     762              : 
     763           11 :     bool open(const char *name, const char *mode)
     764              :     {
     765           11 :         if(file)
     766              :         {
     767            0 :             return false;
     768              :         }
     769           11 :         file = fopen(name, mode);
     770           11 :         return file!=nullptr;
     771              :     }
     772              :     #ifdef WIN32
     773              :         bool opentemp(const char *name, const char *mode)
     774              :         {
     775              :             if(file)
     776              :             {
     777              :                 return false;
     778              :             }
     779              :             file = fopen(name, mode);
     780              :             return file!=nullptr;
     781              :         }
     782              :     #else
     783              :         bool opentemp(const char *, const char *)
     784              :         {
     785              :             if(file)
     786              :             {
     787              :                 return false;
     788              :             }
     789              :             file = tmpfile();
     790              :             return file!=nullptr;
     791              :         }
     792              :     #endif
     793           11 :     void close() override final
     794              :     {
     795           11 :         if(file)
     796              :         {
     797            2 :             fclose(file);
     798            2 :             file = nullptr;
     799              :         }
     800           11 :     }
     801              : 
     802            0 :     bool end() override final
     803              :     {
     804            0 :         return feof(file)!=0;
     805              :     }
     806              : 
     807            0 :     offset tell() const override final
     808              :     {
     809              : #ifdef WIN32
     810              : #if defined(__GNUC__) && !defined(__MINGW32__)
     811              :         offset off = ftello64(file);
     812              : #else
     813              :         offset off = _ftelli64(file);
     814              : #endif
     815              : #else
     816            0 :         offset off = ftello(file);
     817              : #endif
     818              :         // ftello returns LONG_MAX for directories on some platforms
     819            0 :         return off + 1 >= 0 ? off : -1;
     820              :     }
     821              : 
     822            0 :     bool seek(offset pos, int whence) override final
     823              :     {
     824              : #ifdef WIN32
     825              : #if defined(__GNUC__) && !defined(__MINGW32__)
     826              :         return fseeko64(file, pos, whence) >= 0;
     827              : #else
     828              :         return _fseeki64(file, pos, whence) >= 0;
     829              : #endif
     830              : #else
     831            0 :         return fseeko(file, pos, whence) >= 0;
     832              : #endif
     833              :     }
     834              : 
     835            0 :     size_t read(void *buf, size_t len) override final
     836              :     {
     837            0 :         return fread(buf, 1, len, file);
     838              :     }
     839              : 
     840            0 :     size_t write(const void *buf, size_t len) override final
     841              :     {
     842            0 :         return fwrite(buf, 1, len, file);
     843              :     }
     844              : 
     845            0 :     bool flush() override final
     846              :     {
     847            0 :         return !fflush(file);
     848              :     }
     849              : 
     850            0 :     int getchar() override final
     851              :     {
     852            0 :         return fgetc(file);
     853              :     }
     854              : 
     855            0 :     bool putchar(int c) override final
     856              :     {
     857            0 :         return fputc(c, file)!=EOF;
     858              :     }
     859              : 
     860         1454 :     bool getline(char *str, size_t len) override final
     861              :     {
     862         1454 :         return fgets(str, len, file)!=nullptr;
     863              :     }
     864              : 
     865            0 :     bool putstring(const char *str) override final
     866              :     {
     867            0 :         return fputs(str, file)!=EOF;
     868              :     }
     869              : 
     870            0 :     size_t printf(const char *fmt, ...) override final
     871              :     {
     872              :         va_list v;
     873            0 :         va_start(v, fmt);
     874            0 :         int result = std::vfprintf(file, fmt, v);
     875            0 :         va_end(v);
     876            0 :         return std::max(result, 0);
     877              :     }
     878              : };
     879              : 
     880              : VAR(debuggz, 0, 0, 1); //toggles gz checking routines
     881              : 
     882              : class gzstream final : public stream
     883              : {
     884              :     public:
     885            0 :         gzstream() : file(nullptr), buf(nullptr), reading(false), writing(false), autoclose(false), crc(0), headersize(0)
     886              :         {
     887            0 :             zfile.zalloc = nullptr;
     888            0 :             zfile.zfree = nullptr;
     889            0 :             zfile.opaque = nullptr;
     890            0 :             zfile.next_in = zfile.next_out = nullptr;
     891            0 :             zfile.avail_in = zfile.avail_out = 0;
     892            0 :         }
     893              : 
     894            0 :         ~gzstream()
     895            0 :         {
     896            0 :             close();
     897            0 :         }
     898              : 
     899            0 :         void writeheader()
     900              :         {
     901            0 :             uchar header[] = { MAGIC1, MAGIC2, Z_DEFLATED, 0, 0, 0, 0, 0, 0, OS_UNIX };
     902            0 :             file->write(header, sizeof(header));
     903            0 :         }
     904              : 
     905            0 :         void readbuf(size_t size = BUFSIZE)
     906              :         {
     907            0 :             if(!zfile.avail_in)
     908              :             {
     909            0 :                 zfile.next_in = static_cast<Bytef *>(buf);
     910              :             }
     911            0 :             size = std::min(size, static_cast<size_t>(&buf[BUFSIZE] - &zfile.next_in[zfile.avail_in]));
     912            0 :             size_t n = file->read(zfile.next_in + zfile.avail_in, size);
     913            0 :             if(n > 0)
     914              :             {
     915            0 :                 zfile.avail_in += n;
     916              :             }
     917            0 :         }
     918              : 
     919            0 :         uchar readbyte(size_t size = BUFSIZE)
     920              :         {
     921            0 :             if(!zfile.avail_in)
     922              :             {
     923            0 :                 readbuf(size);
     924              :             }
     925            0 :             if(!zfile.avail_in)
     926              :             {
     927            0 :                 return 0;
     928              :             }
     929            0 :             zfile.avail_in--;
     930            0 :             return *static_cast<uchar *>(zfile.next_in++);
     931              :         }
     932              : 
     933            0 :         void skipbytes(size_t n)
     934              :         {
     935            0 :             while(n > 0 && zfile.avail_in > 0)
     936              :             {
     937            0 :                 size_t skipped = std::min(n, static_cast<size_t>(zfile.avail_in));
     938            0 :                 zfile.avail_in -= skipped;
     939            0 :                 zfile.next_in += skipped;
     940            0 :                 n -= skipped;
     941              :             }
     942            0 :             if(n <= 0)
     943              :             {
     944            0 :                 return;
     945              :             }
     946            0 :             file->seek(n, SEEK_CUR);
     947              :         }
     948              : 
     949            0 :         bool checkheader()
     950              :         {
     951            0 :             readbuf(10);
     952            0 :             if(readbyte() != MAGIC1 || readbyte() != MAGIC2 || readbyte() != Z_DEFLATED)
     953              :             {
     954            0 :                 return false;
     955              :             }
     956            0 :             uchar flags = readbyte();
     957            0 :             if(flags & F_RESERVED)
     958              :             {
     959            0 :                 return false;
     960              :             }
     961            0 :             skipbytes(6);
     962            0 :             if(flags & F_EXTRA)
     963              :             {
     964            0 :                 size_t len = readbyte(512);
     965            0 :                 len |= static_cast<size_t>(readbyte(512))<<8;
     966            0 :                 skipbytes(len);
     967              :             }
     968            0 :             if(flags & F_NAME)
     969              :             {
     970            0 :                 while(readbyte(512))
     971              :                 {
     972              :                     //(empty body)
     973              :                 }
     974              :             }
     975            0 :             if(flags & F_COMMENT)
     976              :             {
     977            0 :                 while(readbyte(512))
     978              :                 {
     979              :                     //(empty body)
     980              :                 }
     981              :             }
     982            0 :             if(flags & F_CRC)
     983              :             {
     984            0 :                 skipbytes(2);
     985              :             }
     986            0 :             headersize = static_cast<size_t>(file->tell() - zfile.avail_in);
     987            0 :             return zfile.avail_in > 0 || !file->end();
     988              :         }
     989              : 
     990            0 :         bool open(stream *f, const char *mode, bool needclose, int level)
     991              :         {
     992            0 :             if(file)
     993              :             {
     994            0 :                 return false;
     995              :             }
     996            0 :             for(; *mode; mode++)
     997              :             {
     998            0 :                 if(*mode=='r')
     999              :                 {
    1000            0 :                     reading = true;
    1001            0 :                     break;
    1002              :                 }
    1003            0 :                 else if(*mode=='w')
    1004              :                 {
    1005            0 :                     writing = true;
    1006            0 :                     break;
    1007              :                 }
    1008              :             }
    1009            0 :             if(reading)
    1010              :             {
    1011            0 :                 if(inflateInit2(&zfile, -MAX_WBITS) != Z_OK)
    1012              :                 {
    1013            0 :                     reading = false;
    1014              :                 }
    1015              :             }
    1016            0 :             else if(writing && deflateInit2(&zfile, level, Z_DEFLATED, -MAX_WBITS, std::min(MAX_MEM_LEVEL, 8), Z_DEFAULT_STRATEGY) != Z_OK)
    1017              :             {
    1018            0 :                 writing = false;
    1019              :             }
    1020            0 :             if(!reading && !writing)
    1021              :             {
    1022            0 :                 return false;
    1023              :             }
    1024            0 :             file = f;
    1025            0 :             crc = crc32(0, nullptr, 0);
    1026            0 :             buf = new uchar[BUFSIZE];
    1027            0 :             if(reading)
    1028              :             {
    1029            0 :                 if(!checkheader())
    1030              :                 {
    1031            0 :                     stopreading();
    1032            0 :                     return false;
    1033              :                 }
    1034              :             }
    1035            0 :             else if(writing)
    1036              :             {
    1037            0 :                 writeheader();
    1038              :             }
    1039            0 :             autoclose = needclose;
    1040            0 :             return true;
    1041              :         }
    1042              : 
    1043            0 :         uint getcrc() override final
    1044              :         {
    1045            0 :             return crc;
    1046              :         }
    1047              : 
    1048            0 :         void finishreading()
    1049              :         {
    1050            0 :             if(!reading)
    1051              :             {
    1052            0 :                 return;
    1053              :             }
    1054            0 :             if(debuggz)
    1055              :             {
    1056            0 :                 uint checkcrc = 0,
    1057            0 :                      checksize = 0;
    1058            0 :                 for(int i = 0; i < 4; ++i)
    1059              :                 {
    1060            0 :                     checkcrc |= static_cast<uint>(readbyte()) << (i*8);
    1061              :                 }
    1062            0 :                 for(int i = 0; i < 4; ++i)
    1063              :                 {
    1064            0 :                     checksize |= static_cast<uint>(readbyte()) << (i*8);
    1065              :                 }
    1066            0 :                 if(checkcrc != crc)
    1067              :                 {
    1068            0 :                     conoutf(Console_Debug, "gzip crc check failed: read %X, calculated %X", checkcrc, crc);
    1069              :                 }
    1070            0 :                 if(checksize != zfile.total_out)
    1071              :                 {
    1072            0 :                     conoutf(Console_Debug, "gzip size check failed: read %u, calculated %u", checksize, static_cast<uint>(zfile.total_out));
    1073              :                 }
    1074              :             }
    1075              :         }
    1076              : 
    1077            0 :         void stopreading()
    1078              :         {
    1079            0 :             if(!reading)
    1080              :             {
    1081            0 :                 return;
    1082              :             }
    1083            0 :             inflateEnd(&zfile);
    1084            0 :             reading = false;
    1085              :         }
    1086              : 
    1087            0 :         void finishwriting()
    1088              :         {
    1089            0 :             if(!writing)
    1090              :             {
    1091            0 :                 return;
    1092              :             }
    1093              :             for(;;)
    1094              :             {
    1095            0 :                 int err = zfile.avail_out > 0 ? deflate(&zfile, Z_FINISH) : Z_OK;
    1096            0 :                 if(err != Z_OK && err != Z_STREAM_END)
    1097              :                 {
    1098            0 :                     break;
    1099              :                 }
    1100            0 :                 flushbuf();
    1101            0 :                 if(err == Z_STREAM_END)
    1102              :                 {
    1103            0 :                     break;
    1104              :                 }
    1105            0 :             }
    1106              :             uchar trailer[8] =
    1107              :             {
    1108            0 :                 static_cast<uchar>(crc&0xFF), static_cast<uchar>((crc>>8)&0xFF), static_cast<uchar>((crc>>16)&0xFF), static_cast<uchar>((crc>>24)&0xFF),
    1109            0 :                 static_cast<uchar>(zfile.total_in&0xFF), static_cast<uchar>((zfile.total_in>>8)&0xFF), static_cast<uchar>((zfile.total_in>>16)&0xFF), static_cast<uchar>((zfile.total_in>>24)&0xFF)
    1110            0 :             };
    1111            0 :             file->write(trailer, sizeof(trailer));
    1112              :         }
    1113              : 
    1114            0 :         void stopwriting()
    1115              :         {
    1116            0 :             if(!writing)
    1117              :             {
    1118            0 :                 return;
    1119              :             }
    1120            0 :             deflateEnd(&zfile);
    1121            0 :             writing = false;
    1122              :         }
    1123              : 
    1124            0 :         void close() override final
    1125              :         {
    1126            0 :             if(reading)
    1127              :             {
    1128            0 :                 finishreading();
    1129              :             }
    1130            0 :             stopreading();
    1131            0 :             if(writing)
    1132              :             {
    1133            0 :                 finishwriting();
    1134              :             }
    1135            0 :             stopwriting();
    1136            0 :             delete[] buf;
    1137            0 :             buf = nullptr;
    1138            0 :             if(autoclose)
    1139              :             {
    1140            0 :                 if(file)
    1141              :                 {
    1142            0 :                     delete file;
    1143            0 :                     file = nullptr;
    1144              :                 }
    1145              :             }
    1146            0 :         }
    1147              : 
    1148            0 :         bool end() override final
    1149              :         {
    1150            0 :             return !reading && !writing;
    1151              :         }
    1152              : 
    1153            0 :         offset tell() const override final
    1154              :         {
    1155            0 :             return reading ? zfile.total_out : (writing ? zfile.total_in : offset(-1));
    1156              :         }
    1157              : 
    1158            0 :         offset rawtell() const override final
    1159              :         {
    1160            0 :             return file ? file->tell() : offset(-1);
    1161              :         }
    1162              : 
    1163            0 :         offset size() override final
    1164              :         {
    1165            0 :             if(!file)
    1166              :             {
    1167            0 :                 return -1;
    1168              :             }
    1169            0 :             offset pos = tell();
    1170            0 :             if(!file->seek(-4, SEEK_END))
    1171              :             {
    1172            0 :                 return -1;
    1173              :             }
    1174            0 :             uint isize = file->get<uint>();
    1175            0 :             return file->seek(pos, SEEK_SET) ? isize : offset(-1);
    1176              :         }
    1177              : 
    1178            0 :         offset rawsize() override final
    1179              :         {
    1180            0 :             return file ? file->size() : offset(-1);
    1181              :         }
    1182              : 
    1183            0 :         bool seek(offset pos, int whence) override final
    1184              :         {
    1185            0 :             if(writing || !reading)
    1186              :             {
    1187            0 :                 return false;
    1188              :             }
    1189              : 
    1190            0 :             if(whence == SEEK_END)
    1191              :             {
    1192              :                 uchar skip[512];
    1193            0 :                 while(read(skip, sizeof(skip)) == sizeof(skip))
    1194              :                 {
    1195              :                     //(empty body)
    1196              :                 }
    1197            0 :                 return !pos;
    1198              :             }
    1199            0 :             else if(whence == SEEK_CUR)
    1200              :             {
    1201            0 :                 pos += zfile.total_out;
    1202              :             }
    1203            0 :             if(pos >= static_cast<offset>(zfile.total_out))
    1204              :             {
    1205            0 :                 pos -= zfile.total_out;
    1206              :             }
    1207            0 :             else if(pos < 0 || !file->seek(headersize, SEEK_SET))
    1208              :             {
    1209            0 :                 return false;
    1210              :             }
    1211              :             else
    1212              :             {
    1213            0 :                 if(zfile.next_in && zfile.total_in <= static_cast<uint>(zfile.next_in - buf))
    1214              :                 {
    1215            0 :                     zfile.avail_in += zfile.total_in;
    1216            0 :                     zfile.next_in -= zfile.total_in;
    1217              :                 }
    1218              :                 else
    1219              :                 {
    1220            0 :                     zfile.avail_in = 0;
    1221            0 :                     zfile.next_in = nullptr;
    1222              :                 }
    1223            0 :                 inflateReset(&zfile);
    1224            0 :                 crc = crc32(0, nullptr, 0);
    1225              :             }
    1226              : 
    1227              :             uchar skip[512];
    1228            0 :             while(pos > 0)
    1229              :             {
    1230            0 :                 size_t skipped = static_cast<size_t>(std::min(pos, static_cast<offset>(sizeof(skip))));
    1231            0 :                 if(read(skip, skipped) != skipped)
    1232              :                 {
    1233            0 :                     stopreading();
    1234            0 :                     return false;
    1235              :                 }
    1236            0 :                 pos -= skipped;
    1237              :             }
    1238              : 
    1239            0 :             return true;
    1240              :         }
    1241              : 
    1242            0 :         size_t read(void *buf, size_t len) override final
    1243              :         {
    1244            0 :             if(!reading || !buf || !len)
    1245              :             {
    1246            0 :                 return 0;
    1247              :             }
    1248            0 :             zfile.next_out = static_cast<Bytef *>(buf);
    1249            0 :             zfile.avail_out = len;
    1250            0 :             while(zfile.avail_out > 0)
    1251              :             {
    1252            0 :                 if(!zfile.avail_in)
    1253              :                 {
    1254            0 :                     readbuf(BUFSIZE);
    1255            0 :                     if(!zfile.avail_in)
    1256              :                     {
    1257            0 :                         stopreading();
    1258            0 :                         break;
    1259              :                     }
    1260              :                 }
    1261            0 :                 int err = inflate(&zfile, Z_NO_FLUSH);
    1262            0 :                 if(err == Z_STREAM_END)
    1263              :                 {
    1264            0 :                     crc = crc32(crc, static_cast<Bytef *>(buf), len - zfile.avail_out);
    1265            0 :                     finishreading();
    1266            0 :                     stopreading();
    1267            0 :                     return len - zfile.avail_out;
    1268              :                 }
    1269            0 :                 else if(err != Z_OK)
    1270              :                 {
    1271            0 :                     stopreading();
    1272            0 :                     break;
    1273              :                 }
    1274              :             }
    1275            0 :             crc = crc32(crc, reinterpret_cast<Bytef *>(buf), len - zfile.avail_out);
    1276            0 :             return len - zfile.avail_out;
    1277              :         }
    1278              : 
    1279            0 :         bool flushbuf(bool full = false)
    1280              :         {
    1281            0 :             if(full)
    1282              :             {
    1283            0 :                 deflate(&zfile, Z_SYNC_FLUSH);
    1284              :             }
    1285            0 :             if(zfile.next_out && zfile.avail_out < BUFSIZE)
    1286              :             {
    1287            0 :                 if(file->write(buf, BUFSIZE - zfile.avail_out) != BUFSIZE - zfile.avail_out || (full && !file->flush()))
    1288              :                 {
    1289            0 :                     return false;
    1290              :                 }
    1291              :             }
    1292            0 :             zfile.next_out = buf;
    1293            0 :             zfile.avail_out = BUFSIZE;
    1294            0 :             return true;
    1295              :         }
    1296              : 
    1297            0 :         bool flush() override final
    1298              :         {
    1299            0 :             return flushbuf(true);
    1300              :         }
    1301              : 
    1302            0 :         size_t write(const void *buf, size_t len) override final
    1303              :         {
    1304            0 :             if(!writing || !buf || !len)
    1305              :             {
    1306            0 :                 return 0;
    1307              :             }
    1308            0 :             zfile.next_in = static_cast<Bytef *>(const_cast<void *>(buf)); //cast away constness, then to Bytef
    1309            0 :             zfile.avail_in = len;
    1310            0 :             while(zfile.avail_in > 0)
    1311              :             {
    1312            0 :                 if(!zfile.avail_out && !flushbuf())
    1313              :                 {
    1314            0 :                     stopwriting();
    1315            0 :                     break;
    1316              :                 }
    1317            0 :                 int err = deflate(&zfile, Z_NO_FLUSH);
    1318            0 :                 if(err != Z_OK)
    1319              :                 {
    1320            0 :                     stopwriting();
    1321            0 :                     break;
    1322              :                 }
    1323              :             }
    1324            0 :             crc = crc32(crc, static_cast<Bytef *>(const_cast<void *>(buf)), len - zfile.avail_in);
    1325            0 :             return len - zfile.avail_in;
    1326              :         }
    1327              :     private:
    1328              :         enum GzHeader
    1329              :         {
    1330              :             MAGIC1   = 0x1F,
    1331              :             MAGIC2   = 0x8B,
    1332              :             BUFSIZE  = 16384,
    1333              :             OS_UNIX  = 0x03
    1334              :         };
    1335              : 
    1336              :         enum GzFlags
    1337              :         {
    1338              :             F_ASCII    = 0x01,
    1339              :             F_CRC      = 0x02,
    1340              :             F_EXTRA    = 0x04,
    1341              :             F_NAME     = 0x08,
    1342              :             F_COMMENT  = 0x10,
    1343              :             F_RESERVED = 0xE0
    1344              :         };
    1345              : 
    1346              :         stream *file;
    1347              :         z_stream zfile;
    1348              :         uchar *buf;
    1349              :         bool reading, writing, autoclose;
    1350              :         uint crc;
    1351              :         size_t headersize;
    1352              : };
    1353              : 
    1354           11 : stream *openrawfile(const char *filename, const char *mode)
    1355              : {
    1356           11 :     const char *found = findfile(filename, mode);
    1357           11 :     if(!found)
    1358              :     {
    1359            0 :         return nullptr;
    1360              :     }
    1361           11 :     filestream *file = new filestream;
    1362           11 :     if(!file->open(found, mode))
    1363              :     {
    1364            9 :         delete file;
    1365            9 :         return nullptr;
    1366              :     }
    1367            2 :     return file;
    1368              : }
    1369              : 
    1370           11 : stream *openfile(const char *filename, const char *mode)
    1371              : {
    1372           11 :     stream *s = openzipfile(filename, mode);
    1373           11 :     if(s)
    1374              :     {
    1375            0 :         return s;
    1376              :     }
    1377           11 :     return openrawfile(filename, mode);
    1378              : }
    1379              : 
    1380            0 : stream *opengzfile(const char *filename, const char *mode, stream *file, int level)
    1381              : {
    1382            0 :     stream *source = file ? file : openfile(filename, mode);
    1383            0 :     if(!source)
    1384              :     {
    1385            0 :         return nullptr;
    1386              :     }
    1387            0 :     gzstream *gz = new gzstream;
    1388            0 :     if(!gz->open(source, mode, !file, level))
    1389              :     {
    1390            0 :         if(!file)
    1391              :         {
    1392            0 :             delete source;
    1393              :         }
    1394            0 :         delete gz;
    1395            0 :         return nullptr;
    1396              :     }
    1397            0 :     return gz;
    1398              : }
    1399              : 
    1400            7 : char *loadfile(const char *fn, size_t *size, bool utf8)
    1401              : {
    1402            7 :     stream *f = openfile(fn, "rb");
    1403            7 :     if(!f)
    1404              :     {
    1405            7 :         return nullptr;
    1406              :     }
    1407            0 :     stream::offset fsize = f->size();
    1408            0 :     if(fsize <= 0)
    1409              :     {
    1410            0 :         delete f;
    1411            0 :         return nullptr;
    1412              :     }
    1413            0 :     size_t len = fsize;
    1414            0 :     char *buf = new char[len+1];
    1415            0 :     if(!buf)
    1416              :     {
    1417            0 :         delete f;
    1418            0 :         return nullptr;
    1419              :     }
    1420            0 :     size_t offset = 0;
    1421            0 :     if(utf8 && len >= 3)
    1422              :     {
    1423            0 :         if(f->read(buf, 3) != 3)
    1424              :         {
    1425            0 :             delete f;
    1426            0 :             delete[] buf;
    1427            0 :             return nullptr;
    1428              :         }
    1429            0 :         if((reinterpret_cast<uchar*>(buf))[0] == 0xEF &&
    1430            0 :            (reinterpret_cast<uchar*>(buf))[1] == 0xBB &&
    1431            0 :            (reinterpret_cast<uchar*>(buf))[2] == 0xBF)
    1432              :         {
    1433            0 :             len -= 3;
    1434              :         }
    1435              :         else
    1436              :         {
    1437            0 :             offset += 3;
    1438              :         }
    1439              :     }
    1440            0 :     size_t rlen = f->read(&buf[offset], len-offset);
    1441            0 :     delete f;
    1442            0 :     if(rlen != len-offset)
    1443              :     {
    1444            0 :         delete[] buf;
    1445            0 :         return nullptr;
    1446              :     }
    1447            0 :     buf[len] = '\0';
    1448            0 :     if(size!=nullptr)
    1449              :     {
    1450            0 :         *size = len;
    1451              :     }
    1452            0 :     return buf;
    1453              : }
    1454              : 
        

Generated by: LCOV version 2.0-1