Skip to content

Commit

Permalink
Transfer to QDL (#33)
Browse files Browse the repository at this point in the history
* connect qualcomusb

* finish read write

* sahara setup

* js is painful

* finish sahara

* upload programmer

* fix usblib

* load programmer

* time out

* can reset

* firehose detect partition storage info gpt

* can detect partition

* can write

* can erase

* can get active slot - haven't tested

* setactive slot

* remove releaseinterface

* cleanup

* add sparse

* big cleanup

* cleanup saharadefs

* automate usb packet maxsize

* fix getactiveslot

* import only used from gpt.js

* refactor

* move folder

* integrate

* maxlun = 6

* refactor

* serial

* trailing white space

* fix getactiveslot

* auto download loader

* setactiveslot

* transferout with retry

* add reset userdata

* add semicolon

* update onprogress

* test

* qdl update

* fix sparse

* works

* add loader

* unpack system

* remove sleep at write

* test

* update instructions

* update instructions

* update instructions

* move detach

* cleanup sahara

* update scripts

* update instructions

* update error message

* fasttt

* remove run()

* update eraseuserdata

* auto upload loader

* throw when disconnect + resetuserdata

* delete fb

* cleanup + catch disconnect

* update instruction

* faster

* fix

* zadig_form update

* zadig_create_new_device update to match tint

* update instruction

* remove

* update instruction

* timed out connect

* increase timeout

* update instruction

* update instruction + update detach script

* add copy button

* log setactive successfully

* throw during connecting if error

* error diconnect while connecting

* update throw error

* cleanup

* cleanup

* cleanup

* restructure + cleanup

* fix

* fix style

* style

* fix

* fix

* serial -> int

* cleanup restructure

* cleanup

* cleanup sparse

* cleanup sparse

* change name bytes to num

* clean up path

* consistent var name cmd erase

* cleanup sparse

* write resetuserdata

* remove erase cmd

* cleanup sparse

* update

* update

* cleanup style

* clearer instruction

* cleanup

* update check gpt header consistency

* faster setactiveslot

* move loader into Loaders

* 4x faster setactiveslot

* cleanup

* remove web fastboot

* revert manifest test and image workers

* fix downloadLoader

---------

Co-authored-by: Andrei Radulescu <[email protected]>
  • Loading branch information
2 people authored and adeebshihadeh committed Apr 18, 2024
1 parent 84a3dd7 commit 3bb7105
Show file tree
Hide file tree
Showing 17 changed files with 1,933 additions and 109 deletions.
Binary file modified bun.lockb
Binary file not shown.
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -13,9 +13,9 @@
"node": ">=18.0.0"
},
"dependencies": {
"android-fastboot": "github:commaai/fastboot.js#c3ec6fe3c96a48dab46e23d0c8c861af15b2144a",
"autoprefixer": "10.4.14",
"comlink": "^4.4.1",
"crc-32": "^1.2.2",
"eslint": "8.40.0",
"eslint-config-next": "13.4.1",
"jssha": "^3.3.1",
Expand Down
289 changes: 289 additions & 0 deletions src/QDL/firehose.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,289 @@
import { xmlParser } from "./xmlParser"
import { concatUint8Array, containsBytes, compareStringToBytes, sleep, readBlobAsBuffer } from "./utils"
import * as Sparse from "./sparse";


class response {
constructor(resp=false, data=new Uint8Array(), error="", log=[]) {
this.resp = resp;
this.data = data;
this.error = error;
this.log = log;
}
}


class cfg {
constructor() {
this.ZLPAwareHost = 1;
this.SkipStorageInit = 0;
this.SkipWrite = 0;
this.MaxPayloadSizeToTargetInBytes = 1048576;
this.MaxPayloadSizeFromTargetInBytes = 4096;
this.MaxXMLSizeInBytes = 4096;
this.bit64 = true;
this.SECTOR_SIZE_IN_BYTES = 4096;
this.MemoryName = "UFS";
this.maxlun = 6;
}
}

export class Firehose {
constructor(cdc) {
this.cdc = cdc;
this.xml = new xmlParser();
this.cfg = new cfg();
this.luns = [];
}

getStatus(resp) {
if (resp.hasOwnProperty("value")) {
let value = resp["value"];
return (value === "ACK" || value === "true");
}
return true;
}

async xmlSend(data, wait=true) {
let dataToSend = new TextEncoder().encode(data).slice(0, this.cfg.MaxXMLSizeInBytes);
await this.cdc?.write(dataToSend, null, wait);

let rData = new Uint8Array();
let counter = 0;
let timeout = 3;
while (!(containsBytes("<response value", rData))) {
let tmp = await this.cdc?.read();
if (compareStringToBytes("", tmp)) {
counter += 1;
await sleep(50);
if (counter > timeout) {
break;
}
}
rData = concatUint8Array([rData, tmp]);
}

const resp = this.xml.getReponse(rData);
const status = this.getStatus(resp);
if (resp.hasOwnProperty("rawmode")) {
if (resp["rawmode"] == "false") {
let log = this.xml.getLog(rData);
return new response(status, rData, "", log)
}
} else {
if (status) {
if (containsBytes("log value=", rData)) {
let log = this.xml.getLog(rData);
return new response(status, rData, "", log);
}
return new response(status, rData);
}
}
return new response(true, rData);
}

getLuns() {
return Array.from({length: this.cfg.maxlun}, (x, i) => i)
}

async configure() {
const connectCmd = `<?xml version=\"1.0\" encoding=\"UTF-8\" ?><data>` +
`<configure MemoryName=\"${this.cfg.MemoryName}\" ` +
`Verbose=\"0\" ` +
`AlwaysValidate=\"0\" ` +
`MaxDigestTableSizeInBytes=\"2048\" ` +
`MaxPayloadSizeToTargetInBytes=\"${this.cfg.MaxPayloadSizeToTargetInBytes}\" ` +
`ZLPAwareHost=\"${this.cfg.ZLPAwareHost}\" ` +
`SkipStorageInit=\"${this.cfg.SkipStorageInit}\" ` +
`SkipWrite=\"${this.cfg.SkipWrite}\"/>` +
`</data>`

await this.xmlSend(connectCmd, false);
this.luns = this.getLuns();
return true;
}

async cmdReadBuffer(physicalPartitionNumber, startSector, numPartitionSectors) {
const data = `<?xml version=\"1.0\" ?><data><read SECTOR_SIZE_IN_BYTES=\"${this.cfg.SECTOR_SIZE_IN_BYTES}\"` +
` num_partition_sectors=\"${numPartitionSectors}\"` +
` physical_partition_number=\"${physicalPartitionNumber}\"` +
` start_sector=\"${startSector}\"/>\n</data>`

let rsp = await this.xmlSend(data);
let resData = new Uint8Array();
if (!rsp.resp) {
return rsp;
} else {
let bytesToRead = this.cfg.SECTOR_SIZE_IN_BYTES * numPartitionSectors;
while (bytesToRead > 0) {
let tmp = await this.cdc.read(Math.min(this.cdc.maxSize, bytesToRead));
const size = tmp.length;
bytesToRead -= size;
resData = concatUint8Array([resData, tmp]);
}

const wd = await this.waitForData();
const info = this.xml.getLog(wd);
rsp = this.xml.getReponse(wd);
if (rsp.hasOwnProperty("value")) {
if (rsp["value"] !== "ACK") {
return new response(false, resData, info);
} else if (rsp.hasOwnProperty("rawmode")) {
if (rsp["rawmode"] === "false") {
return new response(true, resData);
}
}
} else {
console.error("Failed read buffer");
return new response(false, resData, rsp[2]);
}
}
let resp = rsp["value"] === "ACK";
return response(resp, resData, rsp[2]);
}

async waitForData() {
let tmp = new Uint8Array();
let timeout = 0;

while (!containsBytes("response value", tmp)) {
let res = await this.cdc.read();
if (compareStringToBytes("", res)) {
timeout += 1;
if (timeout === 4) {
break;
}
await sleep(20);
}
tmp = concatUint8Array([tmp, res]);
}
return tmp;
}

async cmdProgram(physicalPartitionNumber, startSector, blob, onProgress=()=>{}) {
let total = blob.size;
let sparseformat = false;

let sparseHeader = await Sparse.parseFileHeader(blob.slice(0, Sparse.FILE_HEADER_SIZE));
if (sparseHeader !== null) {
sparseformat = true;
total = await Sparse.getSparseRealSize(blob, sparseHeader);
}

let numPartitionSectors = Math.floor(total / this.cfg.SECTOR_SIZE_IN_BYTES);
if (total % this.cfg.SECTOR_SIZE_IN_BYTES !== 0) {
numPartitionSectors += 1;
}

const data = `<?xml version=\"1.0\" ?><data>\n` +
`<program SECTOR_SIZE_IN_BYTES=\"${this.cfg.SECTOR_SIZE_IN_BYTES}\"` +
` num_partition_sectors=\"${numPartitionSectors}\"` +
` physical_partition_number=\"${physicalPartitionNumber}\"` +
` start_sector=\"${startSector}\" />\n</data>`;
let i = 0;
let bytesWritten = 0;
let rsp = await this.xmlSend(data);

if (rsp.resp) {
for await (let split of Sparse.splitBlob(blob)) {
let offset = 0;
let bytesToWriteSplit = split.size;

while (bytesToWriteSplit > 0) {
const wlen = Math.min(bytesToWriteSplit, this.cfg.MaxPayloadSizeToTargetInBytes);
let wdata = new Uint8Array(await readBlobAsBuffer(split.slice(offset, offset + wlen)));
if (wlen % this.cfg.SECTOR_SIZE_IN_BYTES !== 0) {
let fillLen = (Math.floor(wlen/this.cfg.SECTOR_SIZE_IN_BYTES) * this.cfg.SECTOR_SIZE_IN_BYTES) +
this.cfg.SECTOR_SIZE_IN_BYTES;
const fillArray = new Uint8Array(fillLen-wlen).fill(0x00);
wdata = concatUint8Array([wdata, fillArray]);
}
await this.cdc.write(wdata);
await this.cdc.write(new Uint8Array(0), null, true);
offset += wlen;
bytesWritten += wlen;
bytesToWriteSplit -= wlen;

// Need this for sparse image when the data.length < MaxPayloadSizeToTargetInBytes
// Add ~2.4s to total flash time
if (sparseformat && bytesWritten < total) {
await this.cdc.write(new Uint8Array(0), null, true);
}

if (i % 10 === 0) {
onProgress(bytesWritten/total);
}
i += 1;
}
}

const wd = await this.waitForData();
const response = this.xml.getReponse(wd);
if (response.hasOwnProperty("value")) {
if (response["value"] !== "ACK") {
return false;
}
} else {
return false;
}
}

onProgress(1.0);
return true;
}

async cmdErase(physicalPartitionNumber, startSector, numPartitionSectors) {
const data = `<?xml version=\"1.0\" ?><data>\n` +
`<program SECTOR_SIZE_IN_BYTES=\"${this.cfg.SECTOR_SIZE_IN_BYTES}\"` +
` num_partition_sectors=\"${numPartitionSectors}\"` +
` physical_partition_number=\"${physicalPartitionNumber}\"` +
` start_sector=\"${startSector}\" />\n</data>`;
let pos = 0;
let rsp = await this.xmlSend(data)
let bytesToWrite = this.cfg.SECTOR_SIZE_IN_BYTES * numPartitionSectors;
let empty = new Uint8Array(this.cfg.MaxPayloadSizeToTargetInBytes).fill(0);

if (rsp.resp) {
while (bytesToWrite > 0) {
let wlen = Math.min(bytesToWrite, this.cfg.MaxPayloadSizeToTargetInBytes);
await this.cdc.write(empty.slice(0, wlen));
bytesToWrite -= wlen;
pos += wlen;
await this.cdc.write(new Uint8Array(0));
}

const res = await this.waitForData();
const response = this.xml.getReponse(res);
if (response.hasOwnProperty("value")) {
if (response["value"] !== "ACK") {
throw "Failed to erase: NAK";
}
} else {
throw "Failed to erase no return value";
}
}
return true;
}

async cmdSetBootLunId(lun) {
const data = `<?xml version=\"1.0\" ?><data>\n<setbootablestoragedrive value=\"${lun}\" /></data>`
const val = await this.xmlSend(data);
if (val.resp) {
console.log(`Successfully set bootID to lun ${lun}`);
return true;
} else {
throw `Firehose - Failed to set boot lun ${lun}`;
}
}

async cmdReset() {
let data = "<?xml version=\"1.0\" ?><data><power value=\"reset\"/></data>";
let val = await this.xmlSend(data);
if (val.resp) {
console.log("Reset succeeded");
return true;
} else {
throw "Firehose - Reset failed";
}
}
}
Loading

0 comments on commit 3bb7105

Please sign in to comment.