Total Commander Forum Index Total Commander
Форум поддержки пользователей Total Commander
Сайты: Все о Total Commander | Totalcmd.net | Ghisler.com | RU.TCKB
 
 RulesRules   SearchSearch   FAQFAQ   MemberlistMemberlist   UsergroupsUsergroups   RegisterRegister 
 ProfileProfile   Log in to check your private messagesLog in to check your private messages   Log inLog in 

Single Post  Topic: Как идентифицировать бинарник .net ? 
Author Message
Serge Yolkin



PostPosted: Thu Sep 19, 2013 14:28    Post subject: Reply with quote

Несмотря на то, что актуальная проблема была решена благодаря MVV и ProgMan13 в другой ветке, идея разбора бинарника из скрипта показалась интересной (и не такой сложной, как на первый взгляд)
Code:
var
 b=new ActiveXObject('Microsoft.XMLDOM').createElement('binObj'),
 d=false,   // DLL?
 f=new ActiveXObject('Scripting.FileSystemObject'),
 n,      // имя файла (аргумент)
 s=new ActiveXObject('ADODB.Stream'),
 t;
b.dataType='bin.hex';
s.type=1;
s.open();

if(WScript.Arguments.length&&f.fileExists(WScript.Arguments(0)))n=WScript.Arguments(0);
else fa('Файл не найден!');

if(f.getFile(n).size<256)fa(n+'\nНе исполняемый');

s.loadFromFile(n);
b.nodeTypedValue=s.read(64);      // читаем заголовок (64 байта)
// b.text — hex строка

t=b.text.substr(0,4);         // берём первые 2 байта (сигнатура)

if(t=='5a4d')fa(n+'\nDOS 16 ZM(exe)');   // ZM — не актуально?
if(t!='4d5a')fa(n+'\nНе исполняемый');   // не MZ

// теперь у нас есть файл, и он, вроде как, исполняемый

t=b.text.substr(48,2);         // берём байт по смещению 0x18 (24)
if('0x'+t<64)fa(n+'\nDOS 16 MZ(exe)');   // адрес relocation table

t=fb(60);            // берём 4 байта по смещению 0x3C (60)
// здесь находится адрес заголовка PE
if(f.getFile(n).size<256+t)fa(n+'\nНеизвестный формат');

s.Position=t;
b.nodeTypedValue=s.read(24);      // читаем заголовок PE (24 байта)

t=b.text.substr(0,4);         // берём первые 2 байта (сигнатура)
if(t=='4e45')fa(n+'\nWin 16 NE(exe)');   // NE
if(t=='4c45')fa(n+'\nWin 16 LE(VxD)');   // LE — не актуально?
if(t=='4c58')fa(n+'\nOS2 16 LX(VxD)');   // LX — не актуально?
if(t=='5733')fa(n+'\nWin 16 W3(exe)');   // W3 — не актуально?
if(t=='5734')fa(n+'\nWin 32 W4(VMM)');   // W4 — не актуально?
if(t!='5045')fa(n+'\nНеизвестный формат');         // не PE
if(b.text.substr(4,4)!='0000')fa(n+'\nНеизвестный формат');   // неправильный PE?

// теперь, похоже, наш файл ещё и PE

t=1*('0x'+b.text.substr(46,2));      // берём последний байт (Characteristics)
if(t&32)d=true;            // да, это DLL

// дополнительный заголовок PE:
// заголовок COFF      28 байт?
// заголовок NT         64 байта
// массив директорий данных   128 байт?

b.nodeTypedValue=s.read(20);      // читаем 20 байт дополнительного заголовка PE
if(b.text.substr(2,2)=='02'){      // если 64 бита (PE32+), на этом и закончили
  if(d)fa(n+'\nWin 64 (DLL)');
  else fa(n+'\nWin 64 (exe)');
}

// теперь мы знаем, что наш файл PE и он 32 бита
// осталось проверить на дотнетность

s.Position+=(208-20);
// перешли на смещение D0 (208) от начала дополнительного заголовка
b.nodeTypedValue=s.read(8);
// здесь должен быть The CLR header address and size

t=fb(4);            // берём 4 байта по смещению 0x04 (4)

if(t){               // если размер CLR заголовка > 0 значит, .Net
  if(d)fa(n+'\nDotNet DLL');
  else fa(n+'\nDotNet exe');
}else{               // иначе — Win32PE
  if(d)fa(n+'\nWin 32 DLL');
  else fa(n+'\nWin 32 exe');
}

s.close();

function fa(e){s.close();WScript.echo(e);WScript.quit();}   // Выход с сообщением
function fb(e){            // вычисление двордов (DWORD)
  return(
   1*('0x'+b.text.substr(2*e,2))+
   256*('0x'+b.text.substr(2*e+2,2))+
   65536*('0x'+b.text.substr(2*e+4,2))+
   16777216*('0x'+b.text.substr(2*e+6,2))
  );
}
View user's profile Send private message


Powered by phpBB © 2001, 2005 phpBB Group