Util: Improve speed of UPS patch loading

This commit is contained in:
Vicki Pfau 2021-05-02 23:18:13 -07:00
parent 0fc331e8ce
commit df0bb5cd48
2 changed files with 47 additions and 19 deletions

View File

@ -6,6 +6,8 @@ Other fixes:
- Qt: Fix infrequent deadlock when using sync to video - Qt: Fix infrequent deadlock when using sync to video
- Qt: Fix applying savetype-only overrides - Qt: Fix applying savetype-only overrides
- Util: Fix loading UPS patches that affect the last byte of the file - Util: Fix loading UPS patches that affect the last byte of the file
Misc:
- Util: Improve speed of UPS patch loading
0.9.1: (2021-04-18) 0.9.1: (2021-04-18)
Emulation fixes: Emulation fixes:

View File

@ -5,6 +5,7 @@
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
#include <mgba-util/patch/ips.h> #include <mgba-util/patch/ips.h>
#include <mgba-util/circle-buffer.h>
#include <mgba-util/crc32.h> #include <mgba-util/crc32.h>
#include <mgba-util/patch.h> #include <mgba-util/patch.h>
#include <mgba-util/vfs.h> #include <mgba-util/vfs.h>
@ -13,6 +14,8 @@ enum {
IN_CHECKSUM = -12, IN_CHECKSUM = -12,
OUT_CHECKSUM = -8, OUT_CHECKSUM = -8,
PATCH_CHECKSUM = -4, PATCH_CHECKSUM = -4,
BUFFER_SIZE = 128
}; };
static size_t _UPSOutputSize(struct Patch* patch, size_t inSize); static size_t _UPSOutputSize(struct Patch* patch, size_t inSize);
@ -20,7 +23,7 @@ static size_t _UPSOutputSize(struct Patch* patch, size_t inSize);
static bool _UPSApplyPatch(struct Patch* patch, const void* in, size_t inSize, void* out, size_t outSize); static bool _UPSApplyPatch(struct Patch* patch, const void* in, size_t inSize, void* out, size_t outSize);
static bool _BPSApplyPatch(struct Patch* patch, const void* in, size_t inSize, void* out, size_t outSize); static bool _BPSApplyPatch(struct Patch* patch, const void* in, size_t inSize, void* out, size_t outSize);
static size_t _decodeLength(struct VFile* vf); static size_t _decodeLength(struct VFile* vf, struct CircleBuffer* buffer);
bool loadPatchUPS(struct Patch* patch) { bool loadPatchUPS(struct Patch* patch) {
patch->vf->seek(patch->vf, 0, SEEK_SET); patch->vf->seek(patch->vf, 0, SEEK_SET);
@ -58,10 +61,10 @@ bool loadPatchUPS(struct Patch* patch) {
size_t _UPSOutputSize(struct Patch* patch, size_t inSize) { size_t _UPSOutputSize(struct Patch* patch, size_t inSize) {
UNUSED(inSize); UNUSED(inSize);
patch->vf->seek(patch->vf, 4, SEEK_SET); patch->vf->seek(patch->vf, 4, SEEK_SET);
if (_decodeLength(patch->vf) != inSize) { if (_decodeLength(patch->vf, NULL) != inSize) {
return 0; return 0;
} }
return _decodeLength(patch->vf); return _decodeLength(patch->vf, NULL);
} }
bool _UPSApplyPatch(struct Patch* patch, const void* in, size_t inSize, void* out, size_t outSize) { bool _UPSApplyPatch(struct Patch* patch, const void* in, size_t inSize, void* out, size_t outSize) {
@ -69,36 +72,47 @@ bool _UPSApplyPatch(struct Patch* patch, const void* in, size_t inSize, void* ou
size_t filesize = patch->vf->size(patch->vf); size_t filesize = patch->vf->size(patch->vf);
patch->vf->seek(patch->vf, 4, SEEK_SET); patch->vf->seek(patch->vf, 4, SEEK_SET);
_decodeLength(patch->vf); // Discard input size _decodeLength(patch->vf, NULL); // Discard input size
if (_decodeLength(patch->vf) != outSize) { if (_decodeLength(patch->vf, NULL) != outSize) {
return false; return false;
} }
struct CircleBuffer buffer;
memcpy(out, in, inSize > outSize ? outSize : inSize); memcpy(out, in, inSize > outSize ? outSize : inSize);
size_t offset = 0; size_t offset = 0;
size_t alreadyRead = 0; size_t alreadyRead = 0;
uint8_t* buf = out; uint8_t* buf = out;
CircleBufferInit(&buffer, BUFFER_SIZE);
while (alreadyRead < filesize + IN_CHECKSUM) { while (alreadyRead < filesize + IN_CHECKSUM) {
offset += _decodeLength(patch->vf); offset += _decodeLength(patch->vf, &buffer);
uint8_t byte; int8_t byte;
while (true) { while (true) {
if (patch->vf->read(patch->vf, &byte, 1) != 1) { if (!CircleBufferSize(&buffer)) {
return false; uint8_t block[BUFFER_SIZE];
ssize_t read = patch->vf->read(patch->vf, block, sizeof(block));
if (read < 1) {
CircleBufferDeinit(&buffer);
return false;
}
CircleBufferWrite(&buffer, block, read);
} }
CircleBufferRead8(&buffer, &byte);
if (!byte) { if (!byte) {
break; break;
} }
if (offset >= outSize) { if (offset >= outSize) {
CircleBufferDeinit(&buffer);
return false; return false;
} }
buf[offset] ^= byte; buf[offset] ^= byte;
++offset; ++offset;
} }
++offset; ++offset;
alreadyRead = patch->vf->seek(patch->vf, 0, SEEK_CUR); alreadyRead = patch->vf->seek(patch->vf, 0, SEEK_CUR) - CircleBufferSize(&buffer);
} }
CircleBufferDeinit(&buffer);
uint32_t goodCrc32; uint32_t goodCrc32;
patch->vf->seek(patch->vf, OUT_CHECKSUM, SEEK_END); patch->vf->seek(patch->vf, OUT_CHECKSUM, SEEK_END);
@ -129,14 +143,14 @@ bool _BPSApplyPatch(struct Patch* patch, const void* in, size_t inSize, void* ou
ssize_t filesize = patch->vf->size(patch->vf); ssize_t filesize = patch->vf->size(patch->vf);
patch->vf->seek(patch->vf, 4, SEEK_SET); patch->vf->seek(patch->vf, 4, SEEK_SET);
_decodeLength(patch->vf); // Discard input size _decodeLength(patch->vf, NULL); // Discard input size
if (_decodeLength(patch->vf) != outSize) { if (_decodeLength(patch->vf, NULL) != outSize) {
return false; return false;
} }
if (inSize > SSIZE_MAX || outSize > SSIZE_MAX) { if (inSize > SSIZE_MAX || outSize > SSIZE_MAX) {
return false; return false;
} }
size_t metadataLength = _decodeLength(patch->vf); size_t metadataLength = _decodeLength(patch->vf, NULL);
patch->vf->seek(patch->vf, metadataLength, SEEK_CUR); // Skip metadata patch->vf->seek(patch->vf, metadataLength, SEEK_CUR); // Skip metadata
size_t writeLocation = 0; size_t writeLocation = 0;
ssize_t readSourceLocation = 0; ssize_t readSourceLocation = 0;
@ -145,7 +159,7 @@ bool _BPSApplyPatch(struct Patch* patch, const void* in, size_t inSize, void* ou
uint8_t* writeBuffer = out; uint8_t* writeBuffer = out;
const uint8_t* readBuffer = in; const uint8_t* readBuffer = in;
while (patch->vf->seek(patch->vf, 0, SEEK_CUR) < filesize + IN_CHECKSUM) { while (patch->vf->seek(patch->vf, 0, SEEK_CUR) < filesize + IN_CHECKSUM) {
size_t command = _decodeLength(patch->vf); size_t command = _decodeLength(patch->vf, NULL);
size_t length = (command >> 2) + 1; size_t length = (command >> 2) + 1;
if (writeLocation + length > outSize) { if (writeLocation + length > outSize) {
return false; return false;
@ -168,7 +182,7 @@ bool _BPSApplyPatch(struct Patch* patch, const void* in, size_t inSize, void* ou
break; break;
case 0x2: case 0x2:
// SourceCopy // SourceCopy
readOffset = _decodeLength(patch->vf); readOffset = _decodeLength(patch->vf, NULL);
if (readOffset & 1) { if (readOffset & 1) {
readSourceLocation -= readOffset >> 1; readSourceLocation -= readOffset >> 1;
} else { } else {
@ -184,7 +198,7 @@ bool _BPSApplyPatch(struct Patch* patch, const void* in, size_t inSize, void* ou
break; break;
case 0x3: case 0x3:
// TargetCopy // TargetCopy
readOffset = _decodeLength(patch->vf); readOffset = _decodeLength(patch->vf, NULL);
if (readOffset & 1) { if (readOffset & 1) {
readTargetLocation -= readOffset >> 1; readTargetLocation -= readOffset >> 1;
} else { } else {
@ -209,13 +223,25 @@ bool _BPSApplyPatch(struct Patch* patch, const void* in, size_t inSize, void* ou
return true; return true;
} }
size_t _decodeLength(struct VFile* vf) { size_t _decodeLength(struct VFile* vf, struct CircleBuffer* buffer) {
size_t shift = 1; size_t shift = 1;
size_t value = 0; size_t value = 0;
uint8_t byte; uint8_t byte;
while (true) { while (true) {
if (vf->read(vf, &byte, 1) != 1) { if (buffer) {
break; if (!CircleBufferSize(buffer)) {
uint8_t block[BUFFER_SIZE];
ssize_t read = vf->read(vf, block, sizeof(block));
if (read < 1) {
return false;
}
CircleBufferWrite(buffer, block, read);
}
CircleBufferRead8(buffer, (int8_t*) &byte);
} else {
if (vf->read(vf, &byte, 1) != 1) {
break;
}
} }
value += (byte & 0x7f) * shift; value += (byte & 0x7f) * shift;
if (byte & 0x80) { if (byte & 0x80) {