398 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			398 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
// FAT filesystem implementation (FAT12/16 and FAT32)
 | 
						||
 | 
						||
import { BaseFilesystem } from './base.js';
 | 
						||
import { checkDependencies, updateResultItem } from '../utils.js';
 | 
						||
 | 
						||
export class FATFilesystem extends BaseFilesystem {
 | 
						||
    constructor() {
 | 
						||
        super('FAT', [
 | 
						||
            {
 | 
						||
                id: 'fat1216',
 | 
						||
                name: 'FAT12/16',
 | 
						||
                constants: [
 | 
						||
                    { id: 'baseOffset1216', label: 'Basis-Offset', unit: 'Bytes', default: '0x0' },
 | 
						||
                    { id: 'sectorSize1216', label: 'Sektorgröße', unit: 'Bytes', default: '0x200' }
 | 
						||
                ],
 | 
						||
                inputs: [
 | 
						||
                    { id: 'reservedSectors1216', label: 'Reservierte Sektoren', unit: 'Anzahl', placeholder: '0x1' },
 | 
						||
                    { id: 'numFATs1216', label: 'Anzahl FATs', unit: 'Anzahl', placeholder: '0x2' },
 | 
						||
                    { id: 'fatSizeSectors1216', label: 'FAT-Größe', unit: 'Sektoren', placeholder: '0x4000' },
 | 
						||
                    { id: 'maxRootEntries1216', label: 'Max. Root-Einträge', unit: 'Anzahl', placeholder: '0x200' },
 | 
						||
                    { id: 'partitionSizeInSectors1216', label: 'Partitionsgröße', unit: 'Sektoren', placeholder: '0x4015' },
 | 
						||
                    { id: 'clusterSizeSectors1216', label: 'Clustergröße', unit: 'Sektoren', placeholder: '0x2' },
 | 
						||
                    { id: 'clusterNumber1216', label: 'Cluster-Nummer', unit: 'Nummer', placeholder: '0x2' }
 | 
						||
                ],
 | 
						||
                resultGroups: [
 | 
						||
                    {
 | 
						||
                        name: 'FAT-Struktur',
 | 
						||
                        results: [
 | 
						||
                            { id: 'fatStart1216', label: 'FAT-Bereich Anfang (Bytes)', dependencies: ['reservedSectors1216', 'sectorSize1216', 'baseOffset1216'], formula: 'reservedSectors × sectorSize + baseOffset' },
 | 
						||
                            { id: 'fatStartSector1216', label: 'FAT-Bereich Anfang (Sektor)', dependencies: ['reservedSectors1216', 'sectorSize1216', 'baseOffset1216'], formula: 'fatStart ÷ sectorSize' },
 | 
						||
                            { id: 'fatSize1216', label: 'FAT-Bereich Größe (Bytes)', dependencies: ['numFATs1216', 'fatSizeSectors1216', 'sectorSize1216'], formula: 'numFATs × fatSizeSectors × sectorSize' },
 | 
						||
                            { id: 'fat2Start1216', label: 'FAT2 Anfang (Bytes)', dependencies: ['reservedSectors1216', 'sectorSize1216', 'baseOffset1216', 'fatSizeSectors1216'], formula: 'fatStart + (fatSizeSectors × sectorSize)' },
 | 
						||
                            { id: 'fatEnd1216', label: 'FAT-Bereich Ende (Bytes)', dependencies: ['reservedSectors1216', 'sectorSize1216', 'baseOffset1216', 'numFATs1216', 'fatSizeSectors1216'], formula: 'fatStart + fatSize' }
 | 
						||
                        ]
 | 
						||
                    },
 | 
						||
                    {
 | 
						||
                        name: 'Root-Directory',
 | 
						||
                        results: [
 | 
						||
                            { id: 'rootDirStart1216', label: 'Root-Directory Anfang (Bytes)', dependencies: ['reservedSectors1216', 'sectorSize1216', 'baseOffset1216', 'numFATs1216', 'fatSizeSectors1216'], formula: 'fatEnd' },
 | 
						||
                            { id: 'rootDirSize1216', label: 'Root-Directory Größe (Bytes)', dependencies: ['maxRootEntries1216'], formula: 'maxRootEntries × 0x20' }
 | 
						||
                        ]
 | 
						||
                    },
 | 
						||
                    {
 | 
						||
                        name: 'Daten-Bereich',
 | 
						||
                        results: [
 | 
						||
                            { id: 'dataStart1216', label: 'Daten-Bereich Anfang (Bytes)', dependencies: ['reservedSectors1216', 'sectorSize1216', 'baseOffset1216', 'numFATs1216', 'fatSizeSectors1216', 'maxRootEntries1216'], formula: 'fatEnd + rootDirSize' },
 | 
						||
                            { id: 'dataStartSector1216', label: 'Daten-Bereich Anfang (Sektor)', dependencies: ['reservedSectors1216', 'sectorSize1216', 'baseOffset1216', 'numFATs1216', 'fatSizeSectors1216', 'maxRootEntries1216'], formula: 'dataStart ÷ sectorSize' },
 | 
						||
                            { id: 'dataSize1216', label: 'Daten-Bereich Größe (Bytes)', dependencies: ['partitionSizeInSectors1216', 'sectorSize1216', 'reservedSectors1216', 'baseOffset1216', 'numFATs1216', 'fatSizeSectors1216', 'maxRootEntries1216'], formula: '(partitionSizeInSectors × sectorSize) - (dataStart - baseOffset)' },
 | 
						||
                            { id: 'clusterSize1216', label: 'Cluster-Größe (Bytes)', dependencies: ['sectorSize1216', 'clusterSizeSectors1216'], formula: 'sectorSize × clusterSizeSectors' },
 | 
						||
                            { id: 'numClusters1216', label: 'Anzahl Cluster', dependencies: ['partitionSizeInSectors1216', 'sectorSize1216', 'reservedSectors1216', 'baseOffset1216', 'numFATs1216', 'fatSizeSectors1216', 'maxRootEntries1216', 'clusterSizeSectors1216'], formula: 'dataSize ÷ clusterSize' }
 | 
						||
                        ]
 | 
						||
                    },
 | 
						||
                    {
 | 
						||
                        name: 'Spezifischer Cluster',
 | 
						||
                        results: [
 | 
						||
                            { id: 'clusterStart1216', label: 'Cluster Anfang (Bytes)', dependencies: ['reservedSectors1216', 'sectorSize1216', 'baseOffset1216', 'numFATs1216', 'fatSizeSectors1216', 'maxRootEntries1216', 'clusterSizeSectors1216', 'clusterNumber1216'], formula: 'dataStart + (clusterNumber - 0x2) × clusterSize' },
 | 
						||
                            { id: 'clusterStartSector1216', label: 'Cluster Anfang (Sektor)', dependencies: ['reservedSectors1216', 'sectorSize1216', 'baseOffset1216', 'numFATs1216', 'fatSizeSectors1216', 'maxRootEntries1216', 'clusterSizeSectors1216', 'clusterNumber1216'], formula: 'clusterStart ÷ sectorSize' },
 | 
						||
                            { id: 'fatEntryPos1216', label: 'FAT-Eintrag Position (Bytes, nur FAT16)', dependencies: ['reservedSectors1216', 'sectorSize1216', 'baseOffset1216', 'clusterNumber1216'], formula: 'fatStart + (clusterNumber × 2) für FAT16' }
 | 
						||
                        ]
 | 
						||
                    }
 | 
						||
                ],
 | 
						||
                formulas: [
 | 
						||
                    { name: 'FAT-Bereich Anfang', expression: 'Reservierte Sektoren × Sektorgröße + Basis-Offset' },
 | 
						||
                    { name: 'FAT-Bereich Größe', expression: 'Anzahl FATs × FAT-Größe in Sektoren × Sektorgröße' },
 | 
						||
                    { name: 'FAT2 Anfang', expression: 'FAT-Bereich Anfang + FAT-Größe in Bytes' },
 | 
						||
                    { name: 'FAT-Bereich Ende', expression: 'FAT-Bereich Anfang + FAT-Bereich Größe' },
 | 
						||
                    { name: 'Root-Directory Größe', expression: 'Max. Root-Einträge × 0x20' },
 | 
						||
                    { name: 'Daten-Bereich Anfang', expression: 'FAT-Bereich Ende + Root-Directory Größe' },
 | 
						||
                    { name: 'Daten-Bereich Größe', expression: '(Partitionsgröße × Sektorgröße) - (Daten-Bereich Anfang - Basis-Offset)' },
 | 
						||
                    { name: 'Cluster-Größe', expression: 'Sektorgröße × Cluster-Größe in Sektoren' },
 | 
						||
                    { name: 'Anzahl Cluster', expression: 'Daten-Bereich Größe ÷ Cluster-Größe' },
 | 
						||
                    { name: 'Cluster Anfang', expression: 'Daten-Bereich Anfang + (Cluster-Nummer - 0x2) × Cluster-Größe' },
 | 
						||
                    { name: 'FAT-Eintrag Position', expression: 'FAT-Bereich Anfang + (Cluster-Nummer × Eintraggröße)' }
 | 
						||
                ]
 | 
						||
            },
 | 
						||
            {
 | 
						||
                id: 'fat32',
 | 
						||
                name: 'FAT32',
 | 
						||
                constants: [
 | 
						||
                    { id: 'baseOffset32', label: 'Basis-Offset', unit: 'Bytes', default: '0x0' },
 | 
						||
                    { id: 'sectorSize32', label: 'Sektorgröße', unit: 'Bytes', default: '0x200' }
 | 
						||
                ],
 | 
						||
                inputs: [
 | 
						||
                    { id: 'reservedSectors32', label: 'Reservierte Sektoren', unit: 'Anzahl', placeholder: '0x20' },
 | 
						||
                    { id: 'numFATs32', label: 'Anzahl FATs', unit: 'Anzahl', placeholder: '0x2' },
 | 
						||
                    { id: 'fatSizeSectors32', label: 'FAT-Größe', unit: 'Sektoren', placeholder: '0x20000' },
 | 
						||
                    { id: 'rootDirCluster32', label: 'Root-Directory Cluster', unit: 'Nummer', placeholder: '0x2' },
 | 
						||
                    { id: 'partitionSizeInSectors32', label: 'Partitionsgröße', unit: 'Sektoren', placeholder: '0x100000' },
 | 
						||
                    { id: 'clusterSizeSectors32', label: 'Clustergröße', unit: 'Sektoren', placeholder: '0x8' },
 | 
						||
                    { id: 'clusterNumber32', label: 'Cluster-Nummer', unit: 'Nummer', placeholder: '0x2' }
 | 
						||
                ],
 | 
						||
                resultGroups: [
 | 
						||
                    {
 | 
						||
                        name: 'FAT-Struktur',
 | 
						||
                        results: [
 | 
						||
                            { id: 'fatStart32', label: 'FAT-Bereich Anfang (Bytes)', dependencies: ['reservedSectors32', 'sectorSize32', 'baseOffset32'], formula: 'reservedSectors × sectorSize + baseOffset' },
 | 
						||
                            { id: 'fatStartSector32', label: 'FAT-Bereich Anfang (Sektor)', dependencies: ['reservedSectors32', 'sectorSize32', 'baseOffset32'], formula: 'fatStart ÷ sectorSize' },
 | 
						||
                            { id: 'fatSize32', label: 'FAT-Bereich Größe (Bytes)', dependencies: ['numFATs32', 'fatSizeSectors32', 'sectorSize32'], formula: 'numFATs × fatSizeSectors × sectorSize' },
 | 
						||
                            { id: 'fat2Start32', label: 'FAT2 Anfang (Bytes)', dependencies: ['reservedSectors32', 'sectorSize32', 'baseOffset32', 'fatSizeSectors32'], formula: 'fatStart + (fatSizeSectors × sectorSize)' },
 | 
						||
                            { id: 'fatEnd32', label: 'FAT-Bereich Ende (Bytes)', dependencies: ['reservedSectors32', 'sectorSize32', 'baseOffset32', 'numFATs32', 'fatSizeSectors32'], formula: 'fatStart + fatSize' }
 | 
						||
                        ]
 | 
						||
                    },
 | 
						||
                    {
 | 
						||
                        name: 'Daten-Bereich',
 | 
						||
                        results: [
 | 
						||
                            { id: 'dataStart32', label: 'Daten-Bereich Anfang (Bytes)', dependencies: ['reservedSectors32', 'sectorSize32', 'baseOffset32', 'numFATs32', 'fatSizeSectors32'], formula: 'fatEnd' },
 | 
						||
                            { id: 'dataStartSector32', label: 'Daten-Bereich Anfang (Sektor)', dependencies: ['reservedSectors32', 'sectorSize32', 'baseOffset32', 'numFATs32', 'fatSizeSectors32'], formula: 'dataStart ÷ sectorSize' },
 | 
						||
                            { id: 'dataSize32', label: 'Daten-Bereich Größe (Bytes)', dependencies: ['partitionSizeInSectors32', 'sectorSize32', 'reservedSectors32', 'baseOffset32', 'numFATs32', 'fatSizeSectors32'], formula: '(partitionSizeInSectors × sectorSize) - (dataStart - baseOffset)' },
 | 
						||
                            { id: 'clusterSize32', label: 'Cluster-Größe (Bytes)', dependencies: ['sectorSize32', 'clusterSizeSectors32'], formula: 'sectorSize × clusterSizeSectors' },
 | 
						||
                            { id: 'numClusters32', label: 'Anzahl Cluster', dependencies: ['partitionSizeInSectors32', 'sectorSize32', 'reservedSectors32', 'baseOffset32', 'numFATs32', 'fatSizeSectors32', 'clusterSizeSectors32'], formula: 'dataSize ÷ clusterSize' }
 | 
						||
                        ]
 | 
						||
                    },
 | 
						||
                    {
 | 
						||
                        name: 'Spezifischer Cluster',
 | 
						||
                        results: [
 | 
						||
                            { id: 'clusterStart32', label: 'Cluster Anfang (Bytes)', dependencies: ['reservedSectors32', 'sectorSize32', 'baseOffset32', 'numFATs32', 'fatSizeSectors32', 'clusterSizeSectors32', 'clusterNumber32'], formula: 'dataStart + (clusterNumber - 0x2) × clusterSize' },
 | 
						||
                            { id: 'clusterStartSector32', label: 'Cluster Anfang (Sektor)', dependencies: ['reservedSectors32', 'sectorSize32', 'baseOffset32', 'numFATs32', 'fatSizeSectors32', 'clusterSizeSectors32', 'clusterNumber32'], formula: 'clusterStart ÷ sectorSize' },
 | 
						||
                            { id: 'fatEntryPos32', label: 'FAT-Eintrag Position (Bytes)', dependencies: ['reservedSectors32', 'sectorSize32', 'baseOffset32', 'clusterNumber32'], formula: 'fatStart + (clusterNumber × 4)' },
 | 
						||
                            { id: 'rootDirPos32', label: 'Root-Directory Position (Bytes)', dependencies: ['reservedSectors32', 'sectorSize32', 'baseOffset32', 'numFATs32', 'fatSizeSectors32', 'clusterSizeSectors32', 'rootDirCluster32'], formula: 'dataStart + (rootDirCluster - 0x2) × clusterSize' }
 | 
						||
                        ]
 | 
						||
                    }
 | 
						||
                ],
 | 
						||
                formulas: [
 | 
						||
                    { name: 'FAT-Bereich Anfang', expression: 'Reservierte Sektoren × Sektorgröße + Basis-Offset' },
 | 
						||
                    { name: 'FAT-Bereich Größe', expression: 'Anzahl FATs × FAT-Größe in Sektoren × Sektorgröße' },
 | 
						||
                    { name: 'FAT2 Anfang', expression: 'FAT-Bereich Anfang + FAT-Größe in Bytes' },
 | 
						||
                    { name: 'FAT-Bereich Ende', expression: 'FAT-Bereich Anfang + FAT-Bereich Größe' },
 | 
						||
                    { name: 'Daten-Bereich Anfang', expression: 'FAT-Bereich Ende (kein festes Root Directory)' },
 | 
						||
                    { name: 'Daten-Bereich Größe', expression: '(Partitionsgröße × Sektorgröße) - (Daten-Bereich Anfang - Basis-Offset)' },
 | 
						||
                    { name: 'Cluster-Größe', expression: 'Sektorgröße × Cluster-Größe in Sektoren' },
 | 
						||
                    { name: 'Anzahl Cluster', expression: 'Daten-Bereich Größe ÷ Cluster-Größe' },
 | 
						||
                    { name: 'Cluster Anfang', expression: 'Daten-Bereich Anfang + (Cluster-Nummer - 0x2) × Cluster-Größe' },
 | 
						||
                    { name: 'FAT-Eintrag Position', expression: 'FAT-Bereich Anfang + (Cluster-Nummer × 4)' },
 | 
						||
                    { name: 'Root-Directory Position', expression: 'Daten-Bereich Anfang + (Root-Directory Cluster - 0x2) × Cluster-Größe' }
 | 
						||
                ]
 | 
						||
            }
 | 
						||
        ]);
 | 
						||
    }
 | 
						||
 | 
						||
    calculate(variantId) {
 | 
						||
        const values = this.getInputValues(variantId);
 | 
						||
        const results = {};
 | 
						||
 | 
						||
        if (variantId === 'fat1216') {
 | 
						||
            this.calculateFAT1216(values, results);
 | 
						||
        } else if (variantId === 'fat32') {
 | 
						||
            this.calculateFAT32(values, results);
 | 
						||
        }
 | 
						||
    }
 | 
						||
 | 
						||
    calculateFAT1216(values, results) {
 | 
						||
        const suffix = '1216';
 | 
						||
 | 
						||
        // FAT Start
 | 
						||
        if (checkDependencies([`reservedSectors${suffix}`, `sectorSize${suffix}`, `baseOffset${suffix}`])) {
 | 
						||
            results.fatStart = values.reservedSectors1216 * values.sectorSize1216 + values.baseOffset1216;
 | 
						||
            results.fatStartSector = Math.floor(results.fatStart / values.sectorSize1216);
 | 
						||
            updateResultItem(`fatStart${suffix}`, results.fatStart, true, 
 | 
						||
                `0x${values.reservedSectors1216.toString(16)} × 0x${values.sectorSize1216.toString(16)} + 0x${values.baseOffset1216.toString(16)} = 0x${results.fatStart.toString(16)}`);
 | 
						||
            updateResultItem(`fatStartSector${suffix}`, results.fatStartSector, true,
 | 
						||
                `0x${results.fatStart.toString(16)} ÷ 0x${values.sectorSize1216.toString(16)} = 0x${results.fatStartSector.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem(`fatStart${suffix}`, 0, false);
 | 
						||
            updateResultItem(`fatStartSector${suffix}`, 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // FAT Size
 | 
						||
        if (checkDependencies([`numFATs${suffix}`, `fatSizeSectors${suffix}`, `sectorSize${suffix}`])) {
 | 
						||
            results.fatSize = values.numFATs1216 * values.fatSizeSectors1216 * values.sectorSize1216;
 | 
						||
            updateResultItem(`fatSize${suffix}`, results.fatSize, true,
 | 
						||
                `0x${values.numFATs1216.toString(16)} × 0x${values.fatSizeSectors1216.toString(16)} × 0x${values.sectorSize1216.toString(16)} = 0x${results.fatSize.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem(`fatSize${suffix}`, 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // FAT2 Start
 | 
						||
        if (checkDependencies([`reservedSectors${suffix}`, `sectorSize${suffix}`, `baseOffset${suffix}`, `fatSizeSectors${suffix}`]) && results.fatStart !== undefined) {
 | 
						||
            results.fat2Start = results.fatStart + (values.fatSizeSectors1216 * values.sectorSize1216);
 | 
						||
            updateResultItem(`fat2Start${suffix}`, results.fat2Start, true,
 | 
						||
                `0x${results.fatStart.toString(16)} + 0x${(values.fatSizeSectors1216 * values.sectorSize1216).toString(16)} = 0x${results.fat2Start.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem(`fat2Start${suffix}`, 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // FAT End
 | 
						||
        if (results.fatStart !== undefined && results.fatSize !== undefined) {
 | 
						||
            results.fatEnd = results.fatStart + results.fatSize;
 | 
						||
            updateResultItem(`fatEnd${suffix}`, results.fatEnd, true,
 | 
						||
                `0x${results.fatStart.toString(16)} + 0x${results.fatSize.toString(16)} = 0x${results.fatEnd.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem(`fatEnd${suffix}`, 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // Root Directory Size
 | 
						||
        if (checkDependencies(['maxRootEntries1216'])) {
 | 
						||
            results.rootDirSize = values.maxRootEntries1216 * 0x20;
 | 
						||
            updateResultItem('rootDirSize1216', results.rootDirSize, true,
 | 
						||
                `0x${values.maxRootEntries1216.toString(16)} × 0x20 = 0x${results.rootDirSize.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem('rootDirSize1216', 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // Root Directory Start
 | 
						||
        if (results.fatEnd !== undefined) {
 | 
						||
            results.rootDirStart = results.fatEnd;
 | 
						||
            updateResultItem('rootDirStart1216', results.rootDirStart, true,
 | 
						||
                `0x${results.fatEnd.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem('rootDirStart1216', 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // Data Start
 | 
						||
        if (results.fatEnd !== undefined && results.rootDirSize !== undefined) {
 | 
						||
            results.dataStart = results.fatEnd + results.rootDirSize;
 | 
						||
            results.dataStartSector = Math.floor(results.dataStart / values.sectorSize1216);
 | 
						||
            updateResultItem('dataStart1216', results.dataStart, true,
 | 
						||
                `0x${results.fatEnd.toString(16)} + 0x${results.rootDirSize.toString(16)} = 0x${results.dataStart.toString(16)}`);
 | 
						||
            updateResultItem('dataStartSector1216', results.dataStartSector, true,
 | 
						||
                `0x${results.dataStart.toString(16)} ÷ 0x${values.sectorSize1216.toString(16)} = 0x${results.dataStartSector.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem('dataStart1216', 0, false);
 | 
						||
            updateResultItem('dataStartSector1216', 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // Data Size
 | 
						||
        const requiredForDataSize = [`partitionSizeInSectors${suffix}`, `sectorSize1216`, `reservedSectors${suffix}`, `baseOffset1216`, `numFATs${suffix}`, `fatSizeSectors${suffix}`, 'maxRootEntries1216'];
 | 
						||
        if (checkDependencies(requiredForDataSize) && results.dataStart !== undefined) {
 | 
						||
            const partitionSizeBytes = values.partitionSizeInSectors1216 * values.sectorSize1216;
 | 
						||
            results.dataSize = partitionSizeBytes - (results.dataStart - values.baseOffset1216);
 | 
						||
            updateResultItem(`dataSize${suffix}`, results.dataSize, true,
 | 
						||
                `(0x${values.partitionSizeInSectors1216.toString(16)} × 0x${values.sectorSize1216.toString(16)}) - (0x${results.dataStart.toString(16)} - 0x${values.baseOffset1216.toString(16)}) = 0x${results.dataSize.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem(`dataSize${suffix}`, 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // Cluster Size
 | 
						||
        if (checkDependencies([`sectorSize1216`, `clusterSizeSectors${suffix}`])) {
 | 
						||
            results.clusterSize = values.sectorSize1216 * values.clusterSizeSectors1216;
 | 
						||
            updateResultItem(`clusterSize${suffix}`, results.clusterSize, true,
 | 
						||
                `0x${values.sectorSize1216.toString(16)} × 0x${values.clusterSizeSectors1216.toString(16)} = 0x${results.clusterSize.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem(`clusterSize${suffix}`, 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // Number of Clusters
 | 
						||
        if (results.dataSize !== undefined && results.clusterSize !== undefined && results.clusterSize > 0) {
 | 
						||
            results.numClusters = Math.floor(results.dataSize / results.clusterSize);
 | 
						||
            updateResultItem(`numClusters${suffix}`, results.numClusters, true,
 | 
						||
                `0x${results.dataSize.toString(16)} ÷ 0x${results.clusterSize.toString(16)} = 0x${results.numClusters.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem(`numClusters${suffix}`, 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // Specific Cluster Start
 | 
						||
        const requiredForCluster = [`reservedSectors${suffix}`, `sectorSize1216`, `baseOffset1216`, `numFATs${suffix}`, `fatSizeSectors${suffix}`, 'maxRootEntries1216', `clusterSizeSectors${suffix}`, `clusterNumber${suffix}`];
 | 
						||
        if (checkDependencies(requiredForCluster) && results.dataStart !== undefined && results.clusterSize !== undefined) {
 | 
						||
            results.clusterStart = results.dataStart + (values.clusterNumber1216 - 0x2) * results.clusterSize;
 | 
						||
            results.clusterStartSector = Math.floor(results.clusterStart / values.sectorSize1216);
 | 
						||
            updateResultItem(`clusterStart${suffix}`, results.clusterStart, true,
 | 
						||
                `0x${results.dataStart.toString(16)} + (0x${values.clusterNumber1216.toString(16)} - 0x2) × 0x${results.clusterSize.toString(16)} = 0x${results.clusterStart.toString(16)}`);
 | 
						||
            updateResultItem(`clusterStartSector${suffix}`, results.clusterStartSector, true,
 | 
						||
                `0x${results.clusterStart.toString(16)} ÷ 0x${values.sectorSize1216.toString(16)} = 0x${results.clusterStartSector.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem(`clusterStart${suffix}`, 0, false);
 | 
						||
            updateResultItem(`clusterStartSector${suffix}`, 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // FAT Entry Position
 | 
						||
        if (checkDependencies([`reservedSectors${suffix}`, `sectorSize1216`, `baseOffset1216`, `clusterNumber${suffix}`]) && results.fatStart !== undefined) {
 | 
						||
            const entrySize = 2; // FAT16 uses 2 bytes per entry
 | 
						||
            results.fatEntryPos = results.fatStart + (values.clusterNumber1216 * entrySize);
 | 
						||
            updateResultItem(`fatEntryPos${suffix}`, results.fatEntryPos, true,
 | 
						||
                `0x${results.fatStart.toString(16)} + (0x${values.clusterNumber1216.toString(16)} × ${entrySize}) = 0x${results.fatEntryPos.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem(`fatEntryPos${suffix}`, 0, false);
 | 
						||
        }
 | 
						||
    }
 | 
						||
 | 
						||
    calculateFAT32(values, results) {
 | 
						||
        const suffix = '32';
 | 
						||
 | 
						||
        // FAT Start
 | 
						||
        if (checkDependencies([`reservedSectors${suffix}`, `sectorSize${suffix}`, `baseOffset${suffix}`])) {
 | 
						||
            results.fatStart = values.reservedSectors32 * values.sectorSize32 + values.baseOffset32;
 | 
						||
            results.fatStartSector = Math.floor(results.fatStart / values.sectorSize32);
 | 
						||
            updateResultItem(`fatStart${suffix}`, results.fatStart, true, 
 | 
						||
                `0x${values.reservedSectors32.toString(16)} × 0x${values.sectorSize32.toString(16)} + 0x${values.baseOffset32.toString(16)} = 0x${results.fatStart.toString(16)}`);
 | 
						||
            updateResultItem(`fatStartSector${suffix}`, results.fatStartSector, true,
 | 
						||
                `0x${results.fatStart.toString(16)} ÷ 0x${values.sectorSize32.toString(16)} = 0x${results.fatStartSector.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem(`fatStart${suffix}`, 0, false);
 | 
						||
            updateResultItem(`fatStartSector${suffix}`, 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // FAT Size
 | 
						||
        if (checkDependencies([`numFATs${suffix}`, `fatSizeSectors${suffix}`, `sectorSize${suffix}`])) {
 | 
						||
            results.fatSize = values.numFATs32 * values.fatSizeSectors32 * values.sectorSize32;
 | 
						||
            updateResultItem(`fatSize${suffix}`, results.fatSize, true,
 | 
						||
                `0x${values.numFATs32.toString(16)} × 0x${values.fatSizeSectors32.toString(16)} × 0x${values.sectorSize32.toString(16)} = 0x${results.fatSize.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem(`fatSize${suffix}`, 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // FAT2 Start
 | 
						||
        if (checkDependencies([`reservedSectors${suffix}`, `sectorSize${suffix}`, `baseOffset${suffix}`, `fatSizeSectors${suffix}`]) && results.fatStart !== undefined) {
 | 
						||
            results.fat2Start = results.fatStart + (values.fatSizeSectors32 * values.sectorSize32);
 | 
						||
            updateResultItem(`fat2Start${suffix}`, results.fat2Start, true,
 | 
						||
                `0x${results.fatStart.toString(16)} + 0x${(values.fatSizeSectors32 * values.sectorSize32).toString(16)} = 0x${results.fat2Start.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem(`fat2Start${suffix}`, 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // FAT End
 | 
						||
        if (results.fatStart !== undefined && results.fatSize !== undefined) {
 | 
						||
            results.fatEnd = results.fatStart + results.fatSize;
 | 
						||
            updateResultItem(`fatEnd${suffix}`, results.fatEnd, true,
 | 
						||
                `0x${results.fatStart.toString(16)} + 0x${results.fatSize.toString(16)} = 0x${results.fatEnd.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem(`fatEnd${suffix}`, 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // Data Start (FAT32 has no fixed root directory)
 | 
						||
        if (results.fatEnd !== undefined) {
 | 
						||
            results.dataStart = results.fatEnd;
 | 
						||
            results.dataStartSector = Math.floor(results.dataStart / values.sectorSize32);
 | 
						||
            updateResultItem('dataStart32', results.dataStart, true,
 | 
						||
                `0x${results.fatEnd.toString(16)}`);
 | 
						||
            updateResultItem('dataStartSector32', results.dataStartSector, true,
 | 
						||
                `0x${results.dataStart.toString(16)} ÷ 0x${values.sectorSize32.toString(16)} = 0x${results.dataStartSector.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem('dataStart32', 0, false);
 | 
						||
            updateResultItem('dataStartSector32', 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // Data Size
 | 
						||
        const requiredForDataSize = [`partitionSizeInSectors${suffix}`, `sectorSize32`, `reservedSectors${suffix}`, `baseOffset32`, `numFATs${suffix}`, `fatSizeSectors${suffix}`];
 | 
						||
        if (checkDependencies(requiredForDataSize) && results.dataStart !== undefined) {
 | 
						||
            const partitionSizeBytes = values.partitionSizeInSectors32 * values.sectorSize32;
 | 
						||
            results.dataSize = partitionSizeBytes - (results.dataStart - values.baseOffset32);
 | 
						||
            updateResultItem(`dataSize${suffix}`, results.dataSize, true,
 | 
						||
                `(0x${values.partitionSizeInSectors32.toString(16)} × 0x${values.sectorSize32.toString(16)}) - (0x${results.dataStart.toString(16)} - 0x${values.baseOffset32.toString(16)}) = 0x${results.dataSize.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem(`dataSize${suffix}`, 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // Cluster Size
 | 
						||
        if (checkDependencies([`sectorSize32`, `clusterSizeSectors${suffix}`])) {
 | 
						||
            results.clusterSize = values.sectorSize32 * values.clusterSizeSectors32;
 | 
						||
            updateResultItem(`clusterSize${suffix}`, results.clusterSize, true,
 | 
						||
                `0x${values.sectorSize32.toString(16)} × 0x${values.clusterSizeSectors32.toString(16)} = 0x${results.clusterSize.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem(`clusterSize${suffix}`, 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // Number of Clusters
 | 
						||
        if (results.dataSize !== undefined && results.clusterSize !== undefined && results.clusterSize > 0) {
 | 
						||
            results.numClusters = Math.floor(results.dataSize / results.clusterSize);
 | 
						||
            updateResultItem(`numClusters${suffix}`, results.numClusters, true,
 | 
						||
                `0x${results.dataSize.toString(16)} ÷ 0x${results.clusterSize.toString(16)} = 0x${results.numClusters.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem(`numClusters${suffix}`, 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // Specific Cluster Start
 | 
						||
        const requiredForCluster = [`reservedSectors${suffix}`, `sectorSize32`, `baseOffset32`, `numFATs${suffix}`, `fatSizeSectors${suffix}`, `clusterSizeSectors${suffix}`, `clusterNumber${suffix}`];
 | 
						||
        if (checkDependencies(requiredForCluster) && results.dataStart !== undefined && results.clusterSize !== undefined) {
 | 
						||
            results.clusterStart = results.dataStart + (values.clusterNumber32 - 0x2) * results.clusterSize;
 | 
						||
            results.clusterStartSector = Math.floor(results.clusterStart / values.sectorSize32);
 | 
						||
            updateResultItem(`clusterStart${suffix}`, results.clusterStart, true,
 | 
						||
                `0x${results.dataStart.toString(16)} + (0x${values.clusterNumber32.toString(16)} - 0x2) × 0x${results.clusterSize.toString(16)} = 0x${results.clusterStart.toString(16)}`);
 | 
						||
            updateResultItem(`clusterStartSector${suffix}`, results.clusterStartSector, true,
 | 
						||
                `0x${results.clusterStart.toString(16)} ÷ 0x${values.sectorSize32.toString(16)} = 0x${results.clusterStartSector.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem(`clusterStart${suffix}`, 0, false);
 | 
						||
            updateResultItem(`clusterStartSector${suffix}`, 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // FAT Entry Position
 | 
						||
        if (checkDependencies([`reservedSectors${suffix}`, `sectorSize32`, `baseOffset32`, `clusterNumber${suffix}`]) && results.fatStart !== undefined) {
 | 
						||
            const entrySize = 4; // FAT32 uses 4 bytes per entry
 | 
						||
            results.fatEntryPos = results.fatStart + (values.clusterNumber32 * entrySize);
 | 
						||
            updateResultItem(`fatEntryPos${suffix}`, results.fatEntryPos, true,
 | 
						||
                `0x${results.fatStart.toString(16)} + (0x${values.clusterNumber32.toString(16)} × ${entrySize}) = 0x${results.fatEntryPos.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem(`fatEntryPos${suffix}`, 0, false);
 | 
						||
        }
 | 
						||
 | 
						||
        // Root Directory Position (FAT32 specific)
 | 
						||
        const requiredForRootPos = [`reservedSectors${suffix}`, `sectorSize32`, `baseOffset32`, `numFATs${suffix}`, `fatSizeSectors${suffix}`, `clusterSizeSectors${suffix}`, `rootDirCluster32`];
 | 
						||
        if (checkDependencies(requiredForRootPos) && results.dataStart !== undefined && results.clusterSize !== undefined) {
 | 
						||
            results.rootDirPos = results.dataStart + (values.rootDirCluster32 - 0x2) * results.clusterSize;
 | 
						||
            updateResultItem('rootDirPos32', results.rootDirPos, true,
 | 
						||
                `0x${results.dataStart.toString(16)} + (0x${values.rootDirCluster32.toString(16)} - 0x2) × 0x${results.clusterSize.toString(16)} = 0x${results.rootDirPos.toString(16)}`);
 | 
						||
        } else {
 | 
						||
            updateResultItem('rootDirPos32', 0, false);
 | 
						||
        }
 | 
						||
    }
 | 
						||
} |