195 lines
4.7 KiB
C++
195 lines
4.7 KiB
C++
// NOTE: For unknown reason functions in this module use __stdcall instead
|
|
// of regular __usercall.
|
|
|
|
#include "file_utils.h"
|
|
#include "platform_compat.h"
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <zlib.h>
|
|
|
|
#include <vector>
|
|
|
|
// 0x452740
|
|
int fileCopyDecompressed(const char* existingFilePath, const char* newFilePath)
|
|
{
|
|
FILE* stream = compat_fopen(existingFilePath, "rb");
|
|
if (stream == NULL) {
|
|
return -1;
|
|
}
|
|
|
|
int magic[2];
|
|
magic[0] = fgetc(stream);
|
|
magic[1] = fgetc(stream);
|
|
fclose(stream);
|
|
|
|
if (magic[0] == 0x1F && magic[1] == 0x8B) {
|
|
gzFile inStream = compat_gzopen(existingFilePath, "rb");
|
|
FILE* outStream = compat_fopen(newFilePath, "wb");
|
|
|
|
if (inStream != NULL && outStream != NULL) {
|
|
for (;;) {
|
|
int ch = gzgetc(inStream);
|
|
if (ch == -1) {
|
|
break;
|
|
}
|
|
|
|
fputc(ch, outStream);
|
|
}
|
|
|
|
gzclose(inStream);
|
|
fclose(outStream);
|
|
} else {
|
|
if (inStream != NULL) {
|
|
gzclose(inStream);
|
|
}
|
|
|
|
if (outStream != NULL) {
|
|
fclose(outStream);
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
} else {
|
|
fileCopy(existingFilePath, newFilePath, true);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
// 0x452804
|
|
int fileCopyCompressed(const char* existingFilePath, const char* newFilePath)
|
|
{
|
|
FILE* inStream = compat_fopen(existingFilePath, "rb");
|
|
if (inStream == NULL) {
|
|
return -1;
|
|
}
|
|
|
|
int magic[2];
|
|
magic[0] = fgetc(inStream);
|
|
magic[1] = fgetc(inStream);
|
|
rewind(inStream);
|
|
|
|
if (magic[0] == 0x1F && magic[1] == 0x8B) {
|
|
// Source file is already gzipped, there is no need to do anything
|
|
// besides copying.
|
|
fclose(inStream);
|
|
fileCopy(existingFilePath, newFilePath, true);
|
|
} else {
|
|
gzFile outStream = compat_gzopen(newFilePath, "wb");
|
|
if (outStream == NULL) {
|
|
fclose(inStream);
|
|
return -1;
|
|
}
|
|
|
|
// Copy byte-by-byte.
|
|
for (;;) {
|
|
int ch = fgetc(inStream);
|
|
if (ch == -1) {
|
|
break;
|
|
}
|
|
|
|
gzputc(outStream, ch);
|
|
}
|
|
|
|
fclose(inStream);
|
|
gzclose(outStream);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
// TODO: Check, implementation looks odd.
|
|
int _gzdecompress_file(const char* existingFilePath, const char* newFilePath)
|
|
{
|
|
FILE* stream = compat_fopen(existingFilePath, "rb");
|
|
if (stream == NULL) {
|
|
return -1;
|
|
}
|
|
|
|
int magic[2];
|
|
magic[0] = fgetc(stream);
|
|
magic[1] = fgetc(stream);
|
|
fclose(stream);
|
|
|
|
// TODO: Is it broken?
|
|
if (magic[0] != 0x1F || magic[1] != 0x8B) {
|
|
gzFile gzstream = compat_gzopen(existingFilePath, "rb");
|
|
if (gzstream == NULL) {
|
|
return -1;
|
|
}
|
|
|
|
stream = compat_fopen(newFilePath, "wb");
|
|
if (stream == NULL) {
|
|
gzclose(gzstream);
|
|
return -1;
|
|
}
|
|
|
|
while (1) {
|
|
int ch = gzgetc(gzstream);
|
|
if (ch == -1) {
|
|
break;
|
|
}
|
|
|
|
fputc(ch, stream);
|
|
}
|
|
|
|
gzclose(gzstream);
|
|
fclose(stream);
|
|
} else {
|
|
fileCopy(existingFilePath, newFilePath, true);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
// Modelled as replacement for `CopyFileA`, but `overwrite` is the opposite to
|
|
// `bFailIfExists` param. Update callers accordingly.
|
|
void fileCopy(const char* existingFilePath, const char* newFilePath, bool overwrite)
|
|
{
|
|
char nativeExistingFilePath[COMPAT_MAX_PATH];
|
|
strcpy(nativeExistingFilePath, existingFilePath);
|
|
compat_windows_path_to_native(nativeExistingFilePath);
|
|
|
|
char nativeNewFilePath[COMPAT_MAX_PATH];
|
|
strcpy(nativeNewFilePath, newFilePath);
|
|
compat_windows_path_to_native(nativeNewFilePath);
|
|
|
|
char outMode[4];
|
|
outMode[0] = 'w';
|
|
outMode[1] = 'b';
|
|
|
|
if (!overwrite) {
|
|
outMode[2] = 'x';
|
|
}
|
|
|
|
FILE* in = fopen(nativeExistingFilePath, "rb");
|
|
FILE* out = fopen(nativeNewFilePath, outMode);
|
|
if (in != NULL && out != NULL) {
|
|
std::vector<unsigned char> buffer(0xFFFF);
|
|
|
|
size_t bytesRead;
|
|
while ((bytesRead = fread(buffer.data(), sizeof(*buffer.data()), buffer.size(), in)) > 0) {
|
|
size_t bytesWritten;
|
|
size_t offset = 0;
|
|
while ((bytesWritten = fwrite(buffer.data() + offset, sizeof(*buffer.data()), bytesRead, out)) > 0) {
|
|
bytesRead -= bytesWritten;
|
|
offset += bytesWritten;
|
|
}
|
|
|
|
if (bytesWritten < 0) {
|
|
bytesRead = -1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (in != NULL) {
|
|
fclose(in);
|
|
}
|
|
|
|
if (out != NULL) {
|
|
fclose(out);
|
|
}
|
|
}
|