Support APS (N64) patches
This commit is contained in:
parent
5cd9f099a7
commit
48e7040ba2
10 changed files with 324 additions and 4 deletions
|
@ -3,7 +3,7 @@
|
|||
UniPatcher
|
||||
----------
|
||||
|
||||
UniPatcher is a ROM patcher for Android that supports IPS, UPS, BPS, PPF, DPS, EBP and XDelta3 patch types.
|
||||
UniPatcher is a ROM patcher for Android that supports IPS, UPS, BPS, APS (N64), PPF, DPS, EBP and XDelta3 patch types.
|
||||
|
||||
### Additional features:
|
||||
|
||||
|
|
|
@ -34,6 +34,7 @@
|
|||
|
||||
<data android:scheme="file" />
|
||||
<data android:mimeType="*/*" />
|
||||
<data android:pathPattern="/.*\\.aps" />
|
||||
<data android:pathPattern="/.*\\.ips" />
|
||||
<data android:pathPattern="/.*\\.ups" />
|
||||
<data android:pathPattern="/.*\\.bps" />
|
||||
|
|
|
@ -29,6 +29,7 @@ import android.support.v4.app.NotificationCompat;
|
|||
|
||||
import org.apache.commons.io.FileUtils;
|
||||
import org.apache.commons.io.FilenameUtils;
|
||||
import org.emunix.unipatcher.patch.APS;
|
||||
import org.emunix.unipatcher.patch.BPS;
|
||||
import org.emunix.unipatcher.patch.DPS;
|
||||
import org.emunix.unipatcher.patch.EBP;
|
||||
|
@ -119,6 +120,8 @@ public class WorkerService extends IntentService {
|
|||
patcher = new BPS(this, patchFile, romFile, outputFile);
|
||||
else if ("ppf".equals(ext))
|
||||
patcher = new PPF(this, patchFile, romFile, outputFile);
|
||||
else if ("aps".equals(ext))
|
||||
patcher = new APS(this, patchFile, romFile, outputFile);
|
||||
else if ("ebp".equals(ext))
|
||||
patcher = new EBP(this, patchFile, romFile, outputFile);
|
||||
else if ("dps".equals(ext))
|
||||
|
|
76
app/src/main/java/org/emunix/unipatcher/patch/APS.java
Normal file
76
app/src/main/java/org/emunix/unipatcher/patch/APS.java
Normal file
|
@ -0,0 +1,76 @@
|
|||
/*
|
||||
Copyright (C) 2017 Boris Timofeev
|
||||
|
||||
This file is part of UniPatcher.
|
||||
|
||||
UniPatcher is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
UniPatcher is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with UniPatcher. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
package org.emunix.unipatcher.patch;
|
||||
|
||||
import android.content.Context;
|
||||
|
||||
import org.apache.commons.io.IOUtils;
|
||||
import org.emunix.unipatcher.R;
|
||||
|
||||
import java.io.File;
|
||||
import java.io.FileInputStream;
|
||||
import java.io.IOException;
|
||||
import java.util.Arrays;
|
||||
|
||||
public class APS extends Patch {
|
||||
|
||||
public static final int NOT_APS_PATCH = 0;
|
||||
public static final int APS_N64_PATCH = 1;
|
||||
public static final int APS_GBA_PATCH = 2;
|
||||
|
||||
private static final byte[] APS_N64_MAGIC = {0x41, 0x50, 0x53, 0x31, 0x30}; // APS10
|
||||
private static final byte[] APS_GBA_MAGIC = {0x41, 0x50, 0x53, 0x31, 0x00}; // APS1
|
||||
|
||||
public APS(Context context, File patch, File rom, File output) {
|
||||
super(context, patch, rom, output);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void apply() throws PatchException, IOException {
|
||||
Patch aps = null;
|
||||
switch (checkAPS(patchFile)) {
|
||||
case APS_N64_PATCH:
|
||||
aps = new APS_N64(context, patchFile, romFile, outputFile); break;
|
||||
case APS_GBA_PATCH:
|
||||
throw new PatchException(context.getString(R.string.notify_error_aps_gba));
|
||||
case NOT_APS_PATCH:
|
||||
throw new PatchException(context.getString(R.string.notify_error_not_aps_patch));
|
||||
}
|
||||
|
||||
aps.apply();
|
||||
}
|
||||
|
||||
public int checkAPS(File file) throws IOException {
|
||||
FileInputStream stream = null;
|
||||
try {
|
||||
stream = new FileInputStream(file);
|
||||
byte[] magic = new byte[5];
|
||||
stream.read(magic);
|
||||
if (Arrays.equals(magic, APS_N64_MAGIC)) {
|
||||
return APS_N64_PATCH;
|
||||
} else if (Arrays.equals(magic, APS_GBA_MAGIC)) {
|
||||
return APS_GBA_PATCH;
|
||||
}
|
||||
} finally {
|
||||
IOUtils.closeQuietly(stream);
|
||||
}
|
||||
return NOT_APS_PATCH;
|
||||
}
|
||||
}
|
232
app/src/main/java/org/emunix/unipatcher/patch/APS_N64.java
Normal file
232
app/src/main/java/org/emunix/unipatcher/patch/APS_N64.java
Normal file
|
@ -0,0 +1,232 @@
|
|||
/*
|
||||
Copyright (C) 2017 Boris Timofeev
|
||||
|
||||
This file is part of UniPatcher.
|
||||
|
||||
UniPatcher is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
UniPatcher is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with UniPatcher. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
package org.emunix.unipatcher.patch;
|
||||
|
||||
import android.content.Context;
|
||||
|
||||
import org.apache.commons.io.IOUtils;
|
||||
import org.emunix.unipatcher.R;
|
||||
import org.emunix.unipatcher.Utils;
|
||||
|
||||
import java.io.BufferedInputStream;
|
||||
import java.io.BufferedOutputStream;
|
||||
import java.io.File;
|
||||
import java.io.FileInputStream;
|
||||
import java.io.FileOutputStream;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
import java.io.RandomAccessFile;
|
||||
import java.util.Arrays;
|
||||
|
||||
public class APS_N64 extends Patch {
|
||||
|
||||
private static final byte[] MAGIC_NUMBER = {0x41, 0x50, 0x53, 0x31, 0x30}; // APS10
|
||||
private static final int TYPE_SIMPLE_PATCH = 0;
|
||||
private static final int TYPE_N64_PATCH = 1;
|
||||
private static final int ENCODING_SIMPLE = 0;
|
||||
|
||||
public APS_N64(Context context, File patch, File rom, File output) {
|
||||
super(context, patch, rom, output);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void apply() throws PatchException, IOException {
|
||||
BufferedInputStream romStream = null;
|
||||
BufferedInputStream patchStream = null;
|
||||
BufferedOutputStream outputStream = null;
|
||||
|
||||
try {
|
||||
patchStream = new BufferedInputStream(new FileInputStream(patchFile));
|
||||
|
||||
long patchSize = patchFile.length();
|
||||
long romSize = romFile.length();
|
||||
long outSize;
|
||||
int romPos = 0;
|
||||
int outPos = 0;
|
||||
int patchPos = 0;
|
||||
long offset, size;
|
||||
|
||||
// check magic string
|
||||
byte[] magic = new byte[5];
|
||||
size = patchStream.read(magic);
|
||||
if (size != 5 || !Arrays.equals(magic, MAGIC_NUMBER))
|
||||
throw new PatchException(context.getString(R.string.notify_error_not_aps_patch));
|
||||
patchPos += 5;
|
||||
|
||||
// read and check type of the patch
|
||||
int patchType = patchStream.read();
|
||||
if ((patchType != TYPE_SIMPLE_PATCH) && (patchType != TYPE_N64_PATCH))
|
||||
throw new PatchException(context.getString(R.string.notify_error_not_aps_patch));
|
||||
patchPos++;
|
||||
|
||||
// check encoding method
|
||||
int encoding = patchStream.read();
|
||||
if (encoding != ENCODING_SIMPLE)
|
||||
throw new PatchException(context.getString(R.string.notify_error_not_aps_patch));
|
||||
patchPos++;
|
||||
|
||||
// skip description
|
||||
byte[] description = new byte[50];
|
||||
size = patchStream.read(description);
|
||||
if (size < 50)
|
||||
throw new PatchException(context.getString(R.string.notify_error_not_aps_patch));
|
||||
patchPos += 50;
|
||||
|
||||
// validate ROM
|
||||
if (patchType == TYPE_N64_PATCH) {
|
||||
int endianness = patchStream.read();
|
||||
int cardID = ((patchStream.read() & 0xff) << 8) + (patchStream.read() & 0xff);
|
||||
int country = patchStream.read();
|
||||
byte[] crc = new byte[8];
|
||||
patchStream.read(crc);
|
||||
if (!validateROM(endianness, cardID, country, crc))
|
||||
throw new PatchException(context.getString(R.string.notify_error_rom_not_compatible_with_patch));
|
||||
// skip bytes for future expansion
|
||||
byte[] skip = new byte[5];
|
||||
patchStream.read(skip);
|
||||
patchPos += 17;
|
||||
}
|
||||
|
||||
// read size of destination image.
|
||||
outSize = readLELong(patchStream);
|
||||
patchPos += 4;
|
||||
|
||||
romStream = new BufferedInputStream(new FileInputStream(romFile));
|
||||
outputStream = new BufferedOutputStream(new FileOutputStream(outputFile));
|
||||
|
||||
// apply patch
|
||||
while (patchPos < patchSize) {
|
||||
offset = readLELong(patchStream);
|
||||
if (offset < 0)
|
||||
throw new PatchException(context.getString(R.string.notify_error_patch_corrupted));
|
||||
patchPos += 4;
|
||||
|
||||
// copy data from rom to out
|
||||
if (offset <= romSize) {
|
||||
if (outPos < offset) {
|
||||
size = offset - outPos;
|
||||
Utils.copy(romStream, outputStream, size);
|
||||
romPos += size;
|
||||
outPos += size;
|
||||
}
|
||||
} else {
|
||||
if (outPos < romSize) {
|
||||
size = (int) romSize - outPos;
|
||||
Utils.copy(romStream, outputStream, size);
|
||||
romPos += size;
|
||||
outPos += size;
|
||||
}
|
||||
if (outPos < offset) {
|
||||
size = offset - outPos;
|
||||
Utils.copy(size, (byte) 0x0, outputStream);
|
||||
outPos += size;
|
||||
}
|
||||
}
|
||||
|
||||
// copy data from patch to out
|
||||
size = patchStream.read();
|
||||
patchPos++;
|
||||
if (size != 0) {
|
||||
byte[] data = new byte[(int)size];
|
||||
patchStream.read(data);
|
||||
patchPos += size;
|
||||
outputStream.write(data);
|
||||
outPos += size;
|
||||
} else { // RLE
|
||||
byte val = (byte) patchStream.read();
|
||||
size = patchStream.read();
|
||||
patchPos += 2;
|
||||
byte[] data = new byte[(int)size];
|
||||
Arrays.fill(data, val);
|
||||
outputStream.write(data);
|
||||
outPos += size;
|
||||
}
|
||||
|
||||
// skip rom data
|
||||
if (offset <= romSize) {
|
||||
if (romPos + size > romSize) {
|
||||
romPos = (int) romSize;
|
||||
} else {
|
||||
byte[] buf = new byte[(int)size];
|
||||
romStream.read(buf);
|
||||
romPos += size;
|
||||
}
|
||||
}
|
||||
}
|
||||
// write rom tail and trim
|
||||
Utils.copy(romStream, outputStream, outSize - outPos);
|
||||
} finally {
|
||||
IOUtils.closeQuietly(romStream);
|
||||
IOUtils.closeQuietly(patchStream);
|
||||
IOUtils.closeQuietly(outputStream);
|
||||
}
|
||||
}
|
||||
|
||||
private boolean validateROM(int endianness, int cartID, int country, byte[] crc) throws IOException {
|
||||
RandomAccessFile rom = new RandomAccessFile(romFile, "r");
|
||||
int val;
|
||||
try {
|
||||
// check endianness
|
||||
val = rom.read();
|
||||
if ((endianness == 1 && val != 0x80) || (endianness == 0 && val != 0x37))
|
||||
return false;
|
||||
|
||||
// check cartID
|
||||
rom.seek(0x3c);
|
||||
if (endianness == 1) {
|
||||
val = ((rom.read() & 0xff) << 8) + (rom.read() & 0xff);
|
||||
} else {
|
||||
val = (rom.read() & 0xff) + ((rom.read() & 0xff) << 8);
|
||||
}
|
||||
if (cartID != val)
|
||||
return false;
|
||||
|
||||
// check country
|
||||
val = rom.read();
|
||||
if (endianness == 0)
|
||||
val = rom.read();
|
||||
if (country != val)
|
||||
return false;
|
||||
|
||||
// check crc
|
||||
byte[] buf = new byte[8];
|
||||
rom.seek(0x10);
|
||||
rom.read(buf);
|
||||
if (endianness == 0) {
|
||||
byte tmp;
|
||||
for (int i = 0; i < buf.length; i += 2) {
|
||||
tmp = buf[i];
|
||||
buf[i] = buf[i + 1];
|
||||
buf[i + 1] = tmp;
|
||||
}
|
||||
}
|
||||
if (!Arrays.equals(crc, buf))
|
||||
return false;
|
||||
} finally {
|
||||
IOUtils.closeQuietly(rom);
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
private long readLELong(InputStream stream) throws IOException {
|
||||
return (stream.read() & 0xff) + ((stream.read() & 0xff) << 8)
|
||||
+ ((stream.read() & 0xff) << 16) + ((stream.read() & 0xff) << 24);
|
||||
}
|
||||
}
|
|
@ -1,5 +1,5 @@
|
|||
<body>
|
||||
<p>UniPatcher is a ROM patcher that supports IPS, UPS, BPS, PPF, DPS, EBP and XDelta3 patch types.</p>
|
||||
<p>UniPatcher is a ROM patcher that supports IPS, UPS, BPS, APS (N64), PPF, DPS, EBP and XDelta3 patch types.</p>
|
||||
<p>Additional features:</p>
|
||||
<ul>
|
||||
<li>Fix checksum in Sega Mega Drive ROMs</li>
|
||||
|
|
|
@ -1,5 +1,10 @@
|
|||
<body>
|
||||
|
||||
<h4>0.12 (January 15, 2017)</h4>
|
||||
<ul>
|
||||
<li>Support APS patches (Nintendo 64 only)</li>
|
||||
</ul>
|
||||
|
||||
<h4>0.11 (December 25, 2016)</h4>
|
||||
<ul>
|
||||
<li>Support EBP patches (for EarthBound game)</li>
|
||||
|
|
|
@ -6,7 +6,7 @@
|
|||
<p>UniPatcher is an Android tool for applying patches to ROM images of various video game consoles.</p>
|
||||
|
||||
<h4>What patch formats are supported?</h4>
|
||||
<p>The app supports IPS, UPS, BPS, PPF, DPS, EBP and XDelta3 patches.</p>
|
||||
<p>The app supports IPS, UPS, BPS, APS (N64), PPF, DPS, EBP and XDelta3 patches.</p>
|
||||
|
||||
<h4>Can I hack or crack Android game using this app?</h4>
|
||||
<p>No. UniPatcher is not designed to hack the Android games.</p>
|
||||
|
|
|
@ -64,6 +64,8 @@
|
|||
<string name="notify_error_not_ups_patch">Not a valid UPS patch</string>
|
||||
<string name="notify_error_not_bps_patch">Not a valid BPS patch</string>
|
||||
<string name="notify_error_not_ppf_patch">Not a valid PPF patch</string>
|
||||
<string name="notify_error_not_aps_patch">Not a valid APS patch</string>
|
||||
<string name="notify_error_aps_gba">APS for GBA not supported yet</string>
|
||||
<string name="notify_error_not_ebp_patch">Not a valid EBP patch</string>
|
||||
<string name="notify_error_not_dps_patch">Not a valid DPS patch</string>
|
||||
<string name="notify_error_not_xdelta3_patch">Not a valid XDelta3 patch</string>
|
||||
|
@ -124,7 +126,7 @@
|
|||
<string name="send_feedback_error_no_email_apps">There are no email clients installed</string>
|
||||
|
||||
<string name="share_dialog_title">Share UniPatcher</string>
|
||||
<string name="share_text">Download the best ROM patcher for Android. It supports IPS, UPS, BPS, PPF, DPS, EBP and XDelta3 patch types.\n\n</string>
|
||||
<string name="share_text">Download the best ROM patcher for Android. It supports IPS, UPS, BPS, APS (N64), PPF, DPS, EBP and XDelta3 patch types.\n\n</string>
|
||||
|
||||
<!-- Navigation list -->
|
||||
<string name="nav_apply_patch">Apply patch</string>
|
||||
|
|
|
@ -10,6 +10,7 @@ Utility to apply patches to game rom.
|
|||
• IPS
|
||||
• UPS
|
||||
• BPS
|
||||
• APS (N64)
|
||||
• PPF
|
||||
• DPS
|
||||
• EBP
|
||||
|
|
Loading…
Add table
Reference in a new issue