Compare commits

...

4 commits

Author SHA1 Message Date
104fdb1f95 update 2024-07-01 22:55:55 +08:00
725b939278 update 2024-07-01 22:38:27 +08:00
c6f95f4725 update 2024-07-01 22:30:38 +08:00
dbb1c3aeb3 update 2024-07-01 22:23:30 +08:00
2 changed files with 103 additions and 12 deletions

View file

@ -953,6 +953,74 @@ async function readPacket(port, expectedData, timeout = 1000) {
} }
/**
* Waits for a packet from the radio. The packet data is returned as a Uint8Array.
* @param {SerialPort} port - The serial port to read from.
* @param {number} timeout - The timeout in milliseconds.
* @returns {Promise<Uint8Array>} - A promise that resolves with the received packet or gets rejected on timeout.
*/
async function readPacketNoVerify(port, timeout = 1000) {
// Create a reader to read data from the serial port
const reader = port.readable.getReader();
let buffer = new Uint8Array();
let timeoutId; // Store the timeout ID to clear it later
try {
return await new Promise((resolve, reject) => {
// Event listener to handle incoming data
function handleData({ value, done }) {
if (done) {
// If `done` is true, then the reader has been cancelled
reject('Reader has been cancelled.');
console.log('Reader has been cancelled. Current Buffer:', buffer, uint8ArrayToHexString(buffer));
return;
}
// Append the new data to the buffer
buffer = new Uint8Array([...buffer, ...value]);
// Strip the beginning of the buffer until the first 0xAB byte
// This is done to ensure that the buffer does not contain any incomplete packets
while (buffer.length > 0 && buffer.indexOf(0xAB) != -1 && buffer.indexOf(0xCD) != -1) {
resolve(true);
return;
}
// Continue reading data
reader.read().then(handleData).catch(error => {
console.error('Error reading data from the serial port:', error);
reject(error);
return;
});
}
// Subscribe to the data event to start listening for incoming data
reader.read().then(handleData).catch(error_1 => {
console.error('Error reading data from the serial port:', error_1);
reject(error_1);
return;
});
// Set the timeout to reject the Promise if the packet is not received within the specified time
timeoutId = setTimeout(() => {
reader.cancel().then(() => {
reject('Timeout: Packet not received within the specified time.');
return;
}).catch(error_2 => {
console.error('Error cancelling reader:', error_2);
reject(error_2);
return;
});
}, timeout);
});
} finally {
// Clear the timeout when the promise is settled (resolved or rejected)
clearTimeout(timeoutId);
// Release the reader in the finally block to ensure it is always released
reader.releaseLock();
}
}
/** /**
* Sends a packet to the radio. * Sends a packet to the radio.
* @param {SerialPort} port - The serial port to write to. * @param {SerialPort} port - The serial port to write to.
@ -1427,5 +1495,6 @@ export {
flash_flashFirmware, flash_flashFirmware,
flash_generateCommand, flash_generateCommand,
unpackVersion, unpackVersion,
unpack unpack,
readPacketNoVerify
} }

View file

@ -4,10 +4,20 @@
<a-row :gutter="20" align="stretch"> <a-row :gutter="20" align="stretch">
<a-col :span="24"> <a-col :span="24">
<a-card class="general-card" :title="$t('menu.flash') + $t('global.onBoot')"> <a-card class="general-card" :title="$t('menu.flash') + $t('global.onBoot')">
<div style="display: flex; justify-content: space-between; align-items: center;">
<div>
<a-space> <a-space>
<a-button @click="selectFile">{{ state.binaryFile ? state.binaryName : $t('tool.selectFirmware') }}</a-button> <a-button @click="selectFile">{{ state.binaryFile ? state.binaryName : $t('tool.selectFirmware') }}</a-button>
<a-button type="primary" :disabled="!state.binaryFile" @click="flashIt">{{ $t('tool.flash') }}</a-button> <a-button type="primary" :disabled="!state.binaryFile" @click="flashIt">{{ $t('tool.flash') }}</a-button>
</a-space> </a-space>
</div>
<div>
<a-radio-group type="button" size="mini" v-model="state.protocol">
<a-radio value="Official">Official</a-radio>
<a-radio value="Losehu">Losehu</a-radio>
</a-radio-group>
</div>
</div>
<a-divider /> <a-divider />
<div id="statusArea" style="height: 20em; background-color: azure; color: silver; overflow: auto; padding: 20px" <div id="statusArea" style="height: 20em; background-color: azure; color: silver; overflow: auto; padding: 20px"
v-html="state.status"></div> v-html="state.status"></div>
@ -21,18 +31,20 @@
import { reactive, nextTick, onMounted } from 'vue'; import { reactive, nextTick, onMounted } from 'vue';
import { useRoute } from 'vue-router'; import { useRoute } from 'vue-router';
import { useAppStore } from '@/store'; import { useAppStore } from '@/store';
import { disconnect, connect, readPacket, sendPacket, unpackVersion, unpack, flash_generateCommand } from '@/utils/serial.js'; import { disconnect, connect, readPacket, sendPacket, unpackVersion, unpack, flash_generateCommand, readPacketNoVerify } from '@/utils/serial.js';
const appStore = useAppStore(); const appStore = useAppStore();
const state : { const state : {
status: any, status: any,
binaryName: any, binaryName: any,
binaryFile: any binaryFile: any,
protocol: string
} = reactive({ } = reactive({
status: "点击更新按钮更新固件到设备<br/><br/>", status: "点击更新按钮更新固件到设备<br/><br/>",
binaryFile: undefined, binaryFile: undefined,
binaryName: '' binaryName: '',
protocol: 'Official'
}) })
const route = useRoute(); const route = useRoute();
@ -78,11 +90,17 @@ const flashIt = async () => {
await disconnect(appStore.connectPort); await disconnect(appStore.connectPort);
} }
let _connect = await connect(); let _connect = await connect();
if(state.protocol == 'Official'){
await readPacket(_connect, 0x18, 1000); await readPacket(_connect, 0x18, 1000);
}
const rawVersion = unpackVersion(state.binaryFile); const rawVersion = unpackVersion(state.binaryFile);
const _data = new Uint8Array([0x30, 0x5, rawVersion.length, 0x0, ...rawVersion]); const _data = new Uint8Array([0x30, 0x5, rawVersion.length, 0x0, ...rawVersion]);
if(state.protocol == 'Official'){
await sendPacket(_connect, _data); await sendPacket(_connect, _data);
await readPacket(_connect, 0x18) await readPacket(_connect, 0x18);
}
const firmware = unpack(state.binaryFile); const firmware = unpack(state.binaryFile);
if (firmware.length > 0xefff) throw new Error('Last resort boundary check failed. Whoever touched the code is an idiot.'); if (firmware.length > 0xefff) throw new Error('Last resort boundary check failed. Whoever touched the code is an idiot.');
@ -93,7 +111,11 @@ const flashIt = async () => {
try { try {
await sendPacket(_connect, command); await sendPacket(_connect, command);
if(state.protocol == 'Official'){
await readPacket(_connect, 0x1a); await readPacket(_connect, 0x1a);
}else{
await readPacketNoVerify(_connect);
}
} catch (e) { } catch (e) {
console.log('Flash command rejected. Aborting.'); console.log('Flash command rejected. Aborting.');
return Promise.reject(e); return Promise.reject(e);