feat: add field-range map for hex ↔ field highlighting

This commit is contained in:
kerms 2026-03-10 13:17:41 +01:00
parent 9b572bb9d3
commit 0e21d6cb99
Signed by: kerms
GPG Key ID: 5432C10DDCF8DAD5
2 changed files with 155 additions and 0 deletions

View File

@ -18,3 +18,7 @@ export {
// Parser
export { parseAppImage } from './parser';
// Ranges (field ↔ hex highlighting)
export { computeFieldRanges } from './ranges';
export type { FieldDef, FieldGroup } from './ranges';

151
lib/app-image/ranges.ts Normal file
View File

@ -0,0 +1,151 @@
import { readU32 } from '../shared/binary-reader';
import {
IMAGE_HEADER_SIZE, EXTENDED_HEADER_SIZE, SEGMENT_HEADER_SIZE,
APP_DESC_MAGIC, APP_DESC_SIZE,
} from './constants';
import {
SPI_FLASH_MODE_NAMES, SPI_FLASH_SPEED_NAMES, SPI_FLASH_SIZE_NAMES,
} from './types';
import type { AppImageInfo } from './types';
export interface FieldDef {
label: string;
start: number;
end: number; // exclusive
value: string;
}
export interface FieldGroup {
label: string;
start: number;
end: number; // exclusive
fields: FieldDef[];
}
function h(n: number, pad = 2): string {
return '0x' + n.toString(16).toUpperCase().padStart(pad, '0');
}
export function computeFieldRanges(data: Uint8Array, info: AppImageInfo): FieldGroup[] {
const groups: FieldGroup[] = [];
// ── Image Header (bytes 07) ──
const spiSpeedSize = data[3];
groups.push({
label: '镜像头 (Image Header)',
start: 0,
end: IMAGE_HEADER_SIZE,
fields: [
{ label: 'magic', start: 0, end: 1, value: h(info.header.magic) },
{ label: 'segmentCount', start: 1, end: 2, value: String(info.header.segmentCount) },
{ label: 'spiMode', start: 2, end: 3, value: SPI_FLASH_MODE_NAMES[info.header.spiMode] ?? h(info.header.spiMode) },
{
label: 'spiSpeed / spiSize',
start: 3,
end: 4,
value: `${SPI_FLASH_SPEED_NAMES[info.header.spiSpeed] ?? h(spiSpeedSize & 0x0F)} / ${SPI_FLASH_SIZE_NAMES[info.header.spiSize] ?? h((spiSpeedSize >> 4) & 0x0F)}`,
},
{ label: 'entryPoint', start: 4, end: 8, value: h(info.header.entryPoint, 8) },
],
});
// ── Extended Header (bytes 823) ──
const extOff = IMAGE_HEADER_SIZE;
groups.push({
label: '扩展头 (Extended Header)',
start: extOff,
end: extOff + EXTENDED_HEADER_SIZE,
fields: [
{ label: 'wpPin', start: extOff, end: extOff + 1, value: h(info.extendedHeader.wpPin) },
{ label: 'spiPinDrv[0]', start: extOff + 1, end: extOff + 2, value: h(info.extendedHeader.spiPinDrv[0]) },
{ label: 'spiPinDrv[1]', start: extOff + 2, end: extOff + 3, value: h(info.extendedHeader.spiPinDrv[1]) },
{ label: 'spiPinDrv[2]', start: extOff + 3, end: extOff + 4, value: h(info.extendedHeader.spiPinDrv[2]) },
{ label: 'chipId', start: extOff + 4, end: extOff + 6, value: `${h(info.extendedHeader.chipId, 4)} (${info.chipName})` },
{ label: 'minChipRev', start: extOff + 6, end: extOff + 7, value: h(info.extendedHeader.minChipRev) },
{ label: 'minChipRevFull',start: extOff + 7, end: extOff + 9, value: String(info.extendedHeader.minChipRevFull / 100) },
{ label: 'maxChipRevFull',start: extOff + 9, end: extOff + 11, value: info.extendedHeader.maxChipRevFull === 0xFFFF ? '不限' : String(info.extendedHeader.maxChipRevFull / 100) },
{ label: 'reserved', start: extOff + 11, end: extOff + 15, value: '(reserved)' },
{ label: 'hashAppended', start: extOff + 15, end: extOff + 16, value: info.extendedHeader.hashAppended ? '是' : '否' },
],
});
// ── Segments ──
let segOff = IMAGE_HEADER_SIZE + EXTENDED_HEADER_SIZE;
for (let i = 0; i < info.segments.length; i++) {
const seg = info.segments[i];
const hdrEnd = segOff + SEGMENT_HEADER_SIZE;
const dataEnd = hdrEnd + seg.dataLen;
groups.push({
label: `${i} 头 (Segment ${i} Header)`,
start: segOff,
end: hdrEnd,
fields: [
{ label: 'loadAddr', start: segOff, end: segOff + 4, value: h(seg.loadAddr, 8) },
{ label: 'dataLen', start: segOff + 4, end: hdrEnd, value: `${seg.dataLen} 字节` },
],
});
groups.push({
label: `${i} 数据 (Segment ${i} Data)`,
start: hdrEnd,
end: dataEnd,
fields: [],
});
segOff = dataEnd;
}
// ── App Description — walk segments to find the offset ──
if (info.appDescription) {
let appDescOff: number | null = null;
let scanOff = IMAGE_HEADER_SIZE + EXTENDED_HEADER_SIZE;
for (const seg of info.segments) {
const dataOff = scanOff + SEGMENT_HEADER_SIZE;
if (dataOff + 4 <= data.length && readU32(data, dataOff) === APP_DESC_MAGIC) {
appDescOff = dataOff;
break;
}
scanOff = dataOff + seg.dataLen;
}
if (appDescOff !== null) {
const o = appDescOff;
const d = info.appDescription;
groups.push({
label: '应用信息 (App Description)',
start: o,
end: o + APP_DESC_SIZE,
fields: [
{ label: 'magicWord', start: o, end: o + 4, value: h(d.magicWord, 8) },
{ label: 'secureVersion', start: o + 4, end: o + 8, value: String(d.secureVersion) },
{ label: 'version', start: o + 16, end: o + 48, value: d.version },
{ label: 'projectName', start: o + 48, end: o + 80, value: d.projectName },
{ label: 'compileTime', start: o + 80, end: o + 96, value: d.compileTime },
{ label: 'compileDate', start: o + 96, end: o + 112, value: d.compileDate },
{ label: 'idfVersion', start: o + 112, end: o + 144, value: d.idfVersion },
{ label: 'appElfSha256', start: o + 144, end: o + 176, value: Array.from(d.appElfSha256).map(b => b.toString(16).padStart(2, '0')).join('') },
],
});
}
}
// ── Appended SHA256 ──
if (info.extendedHeader.hashAppended && data.length >= 32) {
groups.push({
label: '附加 SHA256 (Appended Hash)',
start: data.length - 32,
end: data.length,
fields: [
{
label: 'sha256',
start: data.length - 32,
end: data.length,
value: Array.from(data.slice(-32)).map(b => b.toString(16).padStart(2, '0')).join(''),
},
],
});
}
return groups;
}