UNPKG

igir

Version:

🕹 A zero-setup ROM collection manager that sorts, filters, extracts or archives, patches, and reports on collections of any size on any OS.

110 lines (109 loc) • 5.45 kB
import FsPoly from '../../polyfill/fsPoly.js'; import IOFile from '../../polyfill/ioFile.js'; import IgirException from '../exceptions/igirException.js'; import FileChecksums, { ChecksumBitmask } from '../files/fileChecksums.js'; import Patch from './patch.js'; const BPSAction = { SOURCE_READ: 0, TARGET_READ: 1, SOURCE_COPY: 2, TARGET_COPY: 3, }; /** * @see https://github.com/blakesmith/rombp/blob/master/docs/bps_spec.md * @see https://github.com/btimofeev/UniPatcher/wiki/BPS */ export default class BPSPatch extends Patch { static SUPPORTED_EXTENSIONS = ['.bps']; static FILE_SIGNATURE = Buffer.from('BPS1'); static async patchFrom(file) { let crcBefore = ''; let crcAfter = ''; let targetSize = 0; await file.extractToTempIOFile('r', async (patchFile) => { patchFile.seek(BPSPatch.FILE_SIGNATURE.length); await Patch.readUpsUint(patchFile); // source size targetSize = await Patch.readUpsUint(patchFile); patchFile.seek(patchFile.getSize() - 12); crcBefore = (await patchFile.readNext(4)).reverse().toString('hex'); crcAfter = (await patchFile.readNext(4)).reverse().toString('hex'); // Validate the patch contents const patchChecksumExpected = (await patchFile.readNext(4)).reverse().toString('hex'); patchFile.seek(0); const patchData = await patchFile.readNext(patchFile.getSize() - 4); const patchChecksumsActual = await FileChecksums.hashData(patchData, ChecksumBitmask.CRC32); if (patchChecksumsActual.crc32 !== patchChecksumExpected) { throw new IgirException(`BPS patch is invalid, CRC of contents (${patchChecksumsActual.crc32}) doesn't match expected (${patchChecksumExpected}): ${file.toString()}`); } }); if (crcBefore.length !== 8 || crcAfter.length !== 8) { throw new IgirException(`couldn't parse base file CRC for patch: ${file.toString()}`); } return new BPSPatch(file, crcBefore, crcAfter, targetSize); } async createPatchedFile(inputRomFile, outputRomPath) { return this.getFile().extractToTempIOFile('r', async (patchFile) => { const header = await patchFile.readNext(4); if (!header.equals(BPSPatch.FILE_SIGNATURE)) { throw new IgirException(`BPS patch header is invalid: ${this.getFile().toString()}`); } const sourceSize = await Patch.readUpsUint(patchFile); if (inputRomFile.getSize() !== sourceSize) { throw new IgirException(`BPS patch expected ROM size of ${FsPoly.sizeReadable(sourceSize)}: ${this.getFile().toString()}`); } await Patch.readUpsUint(patchFile); // target size const metadataSize = await Patch.readUpsUint(patchFile); if (metadataSize) { patchFile.skipNext(metadataSize); } return this.writeOutputFile(inputRomFile, outputRomPath, patchFile); }); } async writeOutputFile(inputRomFile, outputRomPath, patchFile) { return inputRomFile.extractToTempIOFile('r', async (inputRomIOFile) => { const targetFile = await IOFile.fileOfSize(outputRomPath, 'r+', this.getSizeAfter()); try { await BPSPatch.applyPatch(patchFile, inputRomIOFile, targetFile); } finally { await targetFile.close(); } }); } static async applyPatch(patchFile, sourceFile, targetFile) { let sourceRelativeOffset = 0; let targetRelativeOffset = 0; while (patchFile.getPosition() < patchFile.getSize() - 12) { const blockHeader = await Patch.readUpsUint(patchFile); const action = (blockHeader & 3); const length = (blockHeader >> 2) + 1; if (action === BPSAction.SOURCE_READ) { await targetFile.write(await sourceFile.readAt(targetFile.getPosition(), length)); } else if (action === BPSAction.TARGET_READ) { const data = await patchFile.readNext(length); await targetFile.write(data); } else if (action === BPSAction.SOURCE_COPY) { const offset = await Patch.readUpsUint(patchFile); sourceRelativeOffset += (offset & 1 ? -1 : +1) * (offset >> 1); await targetFile.write(await sourceFile.readAt(sourceRelativeOffset, length)); sourceRelativeOffset += length; } else if (action === BPSAction.TARGET_COPY) { const offset = await Patch.readUpsUint(patchFile); targetRelativeOffset += (offset & 1 ? -1 : +1) * (offset >> 1); // WARN: you explicitly can't read the target file all at once, you have to read byte by // byte, because later iterations of the loop may need to read data that was changed by // earlier iterations of the loop. for (let i = 0; i < length; i += 1) { await targetFile.write(await targetFile.readAt(targetRelativeOffset, 1)); targetRelativeOffset += 1; } } else { throw new IgirException(`BPS action ${action} isn't supported`); } } } }