mirror of
https://github.com/Ylianst/MeshCentral.git
synced 2024-12-26 07:05:52 -05:00
authenticode.js improved encoding of the sections and header.
This commit is contained in:
parent
ad5e95f4ea
commit
e0817fd354
@ -1661,20 +1661,34 @@ function createAuthenticodeHandler(path) {
|
||||
var fullHeaderLen = obj.header.SectionHeadersPtr + (obj.header.coff.numberOfSections * 40);
|
||||
var fullHeader = readFileSlice(written, fullHeaderLen);
|
||||
|
||||
// Compute the size of the resource segment
|
||||
//const resSizes = { tables: 0, items: 0, names: 0, data: 0 };
|
||||
//getResourceSectionSize(obj.resources, resSizes);
|
||||
// Create the resource section and pad to next 512 byte boundry
|
||||
var rsrcSection = generateResourceSection(obj.resources);
|
||||
var rsrcSectionVirtualSize = rsrcSection.length;
|
||||
var x = (rsrcSection.length % 512);
|
||||
if (x != 0) { rsrcSection = Buffer.concat([rsrcSection, Buffer.alloc(512 - x)]); }
|
||||
var rsrcSectionRawSize = rsrcSection.length;
|
||||
|
||||
// Calculate the location and original and new size of the resource segment
|
||||
var fileAlign = obj.header.peWindows.fileAlignment
|
||||
var resPtr = obj.header.sections['.rsrc'].rawAddr;
|
||||
var oldResSize = obj.header.sections['.rsrc'].rawSize;
|
||||
var newResSize = obj.header.sections['.rsrc'].rawSize; // TODO: resSizes.data;
|
||||
var newResSize = rsrcSection.length;
|
||||
var resDeltaSize = newResSize - oldResSize;
|
||||
|
||||
// Compute the sizeOfInitializedData
|
||||
var sizeOfInitializedData = 0;
|
||||
for (var i in obj.header.sections) {
|
||||
if (i != '.text') {
|
||||
if (i == '.rsrc') {
|
||||
sizeOfInitializedData += rsrcSectionRawSize;
|
||||
} else {
|
||||
sizeOfInitializedData += obj.header.sections[i].rawSize;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Change PE optional header sizeOfInitializedData standard field
|
||||
fullHeader.writeUInt32LE(obj.header.peStandard.sizeOfInitializedData + resDeltaSize, obj.header.peOptionalHeaderLocation + 8);
|
||||
fullHeader.writeUInt32LE(obj.header.peWindows.sizeOfImage, obj.header.peOptionalHeaderLocation + 56); // TODO: resDeltaSize
|
||||
fullHeader.writeUInt32LE(sizeOfInitializedData, obj.header.peOptionalHeaderLocation + 8);
|
||||
|
||||
// Update the checksum to zero
|
||||
fullHeader.writeUInt32LE(0, obj.header.peOptionalHeaderLocation + 64);
|
||||
@ -1698,19 +1712,34 @@ function createAuthenticodeHandler(path) {
|
||||
if (obj.header.dataDirectories.clrRuntimeHeader.addr > resPtr) { fullHeader.writeUInt32LE(obj.header.dataDirectories.clrRuntimeHeader.addr + resDeltaSize, obj.header.peOptionalHeaderLocation + 208 + pePlusOffset); }
|
||||
|
||||
// Make changes to the segments table
|
||||
var virtualAddress = 4096;
|
||||
for (var i in obj.header.sections) {
|
||||
const section = obj.header.sections[i];
|
||||
if (i == '.rsrc') {
|
||||
// Change the size of the resource section
|
||||
fullHeader.writeUInt32LE(section.rawSize + resDeltaSize, section.ptr + 8); // virtualSize (TODO)
|
||||
fullHeader.writeUInt32LE(section.rawSize + resDeltaSize, section.ptr + 16); // rawSize
|
||||
fullHeader.writeUInt32LE(rsrcSectionVirtualSize, section.ptr + 8); // virtualSize
|
||||
fullHeader.writeUInt32LE(rsrcSectionRawSize, section.ptr + 16); // rawSize
|
||||
|
||||
// Set the virtual address of the section
|
||||
fullHeader.writeUInt32LE(virtualAddress, section.ptr + 12); // Virtual address
|
||||
var virtualAddressPadding = (rsrcSectionVirtualSize % 4096);
|
||||
virtualAddress += rsrcSectionVirtualSize;
|
||||
if (virtualAddressPadding != 0) { virtualAddress += (4096 - virtualAddressPadding); }
|
||||
} else {
|
||||
// Change the location of any other section if located after the resource section
|
||||
if (section.virtualAddr > resPtr) { fullHeader.writeUInt32LE(section.virtualAddr + resDeltaSize, section.ptr + 12); }
|
||||
if (section.rawAddr > resPtr) { fullHeader.writeUInt32LE(section.rawAddr + resDeltaSize, section.ptr + 20); }
|
||||
|
||||
// Set the virtual address of the section
|
||||
fullHeader.writeUInt32LE(virtualAddress, section.ptr + 12); // Virtual address
|
||||
var virtualAddressPadding = (section.virtualSize % 4096);
|
||||
virtualAddress += section.virtualSize;
|
||||
if (virtualAddressPadding != 0) { virtualAddress += (4096 - virtualAddressPadding); }
|
||||
}
|
||||
}
|
||||
|
||||
// Write size of image. We put the next virtual address.
|
||||
fullHeader.writeUInt32LE(virtualAddress, obj.header.peOptionalHeaderLocation + 56); // sizeOfImage
|
||||
|
||||
// Write the entire header to the destination file
|
||||
//console.log('Write header', fullHeader.length, written);
|
||||
fs.writeSync(output, fullHeader);
|
||||
@ -1726,7 +1755,6 @@ function createAuthenticodeHandler(path) {
|
||||
}
|
||||
|
||||
// Write the new resource section
|
||||
var rsrcSection = generateResourceSection(obj.resources);
|
||||
fs.writeSync(output, rsrcSection);
|
||||
written += rsrcSection.length;
|
||||
//console.log('Write res', rsrcSection.length, written);
|
||||
@ -1989,7 +2017,7 @@ function start() {
|
||||
}
|
||||
|
||||
// Check that a valid command is passed in
|
||||
if (['info', 'sign', 'unsign', 'createcert', 'icons', 'saveicon', 'saveicons', 'header', 'timestamp', 'signblock'].indexOf(process.argv[2].toLowerCase()) == -1) {
|
||||
if (['info', 'sign', 'unsign', 'createcert', 'icons', 'saveicon', 'saveicons', 'header', 'sections', 'timestamp', 'signblock'].indexOf(process.argv[2].toLowerCase()) == -1) {
|
||||
console.log("Invalid command: " + process.argv[2]);
|
||||
console.log("Valid commands are: info, sign, unsign, createcert, timestamp");
|
||||
return;
|
||||
@ -2103,6 +2131,26 @@ function start() {
|
||||
if (command == 'header') { // Display the full executable header in JSON format
|
||||
if (exe == null) { console.log("Missing --exe [filename]"); return; }
|
||||
console.log(exe.header);
|
||||
// Check that the header is valid
|
||||
var ptr = 1024, sizeOfCode = 0, sizeOfInitializedData = 0;
|
||||
for (var i in exe.header.sections) {
|
||||
if (i == '.text') { sizeOfCode += exe.header.sections[i].rawSize; } else { sizeOfInitializedData += exe.header.sections[i].rawSize; }
|
||||
if (exe.header.sections[i].rawAddr != ptr) { console.log('WARNING: ' + i + ' section should have a rawAddr or ' + ptr + ', but has ' + exe.header.sections[i].rawAddr + ' instead.'); }
|
||||
ptr += exe.header.sections[i].rawSize;
|
||||
}
|
||||
if (exe.header.peStandard.sizeOfCode != sizeOfCode) { console.log('WARNING: Size of code is ' + exe.header.peStandard.sizeOfCode + ', should be ' + sizeOfCode + '.'); }
|
||||
if (exe.header.peStandard.sizeOfInitializedData != sizeOfInitializedData) { console.log('WARNING: Size of initialized data is ' + exe.header.peStandard.sizeOfInitializedData + ', should be ' + sizeOfInitializedData + '.'); }
|
||||
}
|
||||
if (command == 'sections') { // Display sections in CSV format
|
||||
if (exe == null) { console.log("Missing --exe [filename]"); return; }
|
||||
var csvHeader = 'section';
|
||||
for (var i in exe.header.sections['.text']) { csvHeader += ',' + i; }
|
||||
console.log(csvHeader);
|
||||
for (var i in exe.header.sections) {
|
||||
var csvData = i;
|
||||
for (var j in exe.header.sections[i]) { csvData += ',' + exe.header.sections[i][j]; }
|
||||
console.log(csvData);
|
||||
}
|
||||
}
|
||||
if (command == 'sign') { // Sign an executable
|
||||
if (typeof args.exe != 'string') { console.log("Missing --exe [filename]"); return; }
|
||||
|
Loading…
Reference in New Issue
Block a user