Lazarus привлекала своих жертв при помощи документов о вакансиях в таких компаниях, как LockHeed Martin, BAE Systems и Boeing. В данном конкретном случае злоумышленники привлекали жертв, ищущих работу в Boeing, при помощи документа Boeing BDS MSE.docx ( твит ShadowChasing1). Вредоносное ПО извлекает имя хоста, имя пользователя, информацию о сети, список процессов и другую информацию, которая затем должна была передаваться на один из четырёх серверов C2. Предназначенные для передачи данные сжимались, шифровались XOR, кодировались Base64 и отправлялись на сервер C2. Троян реализует четыре действия, в том числе скачивает и исполняет файл .exe или .dll, загружает PE (Portable Executable) в память процессов и исполняет шелл-код.

▍ Технический анализ


SHA256: 803dda6c8dc426f1005acdf765d9ef897dd502cd8a80632eef4738d1d7947269

Файл является DLL, имеющим 7 экспортов. Лишь одна из этих функций выполняет зловредное действие (DllGetFirstChild):


Рисунок 1

Вредоносное ПО вызывает функцию ObtainUserAgentString для получения User Agent. Кроме того, в двоичном файле прописан User Agent «Mozilla / 5.0 (Windows NT 10.0; WOW64; Trident / 7.0; rv:11.0) li», то есть Internet Explorer в Windows 10:


Рисунок 2

Двоичный файл извлекает текущие системные дату и время при помощи GetSystemTimeAsFileTime API:


Рисунок 3

Для получения дескриптора модуля из ntdll.dll используется GetModuleHandleW:


Рисунок 4

Процесс при помощи подпрограммы GetProcAddress получает адрес следующих функций экспорта: "RtlGetCompressionWorkSpaceSize", "RtlCompressBuffer", "RtlDecompressBuffer", "RtlGetVersion". Пример вызова функции показан на рисунке 5:


Рисунок 5

При помощи вызова функции GetComputerNameW получается название локального компьютера в NetBIOS:


Рисунок 6

GetAdaptersInfo API используется для получения информации об адаптере с локальной машины:


Рисунок 7

Извлечённый ранее MAC-адрес записывается в буфер:


Рисунок 8

Файл извлекает string командной строки текущего процесса:


Рисунок 9

CommandLineToArgvW используется для извлечения массива указателей на аргументы командной строки и количества аргументов (аналогично argv и argc):


Рисунок 10

Согласно статье, зловредное ПО должно было запускаться со следующими параметрами:

NTPR P6k+pR6iIKwJpU6oR6ZilgKPL7IxsitJAnpIYSx2KldSSRFFyUIzTBVFAwgzBkI2PS/+EgASBik/GgYBwBbRNy7pP+Xq4uTsxOXU6NPmudaEz7Xy5fLQica6yKHvtu2XkYmnhfeC/4ythf9I6UbAdvxvy1K2Um5ppVrEQY9WiHdxKbolqiKgLMElwSiKJrcWrQ+cMpYy5cnc+s/hufap15LJmsVFwr7MlMWwiLCGgLZPr4uSk5KIqZiadYGOlkS3cml1ZZdiZmyzZVpovmZiVlNPNXJsck4JXzpPIWw2YBcqCRMFCQJBDG4FfchmxkL2fO8V0jbSTeko2u/BI9YA9zGpM6UWoiGsdaVdqAmmIpYHjzWyM7IOSQR6SGE4dilXB0lfRXtCOEwkRTAIMgYWNnsvVRJSEvQp/xryAdsW1Df76fjl3eIb7M7lIujH5vbW7c/e8tTy2on1uuGh+rbml5GJp4X3gv+MrYXwSOFGzHbxb9BSwFLLaaJau0FNVoh3sim4JZYi1Cz1JZYohya0FpEP9TKZMpTJgvqn4e72sdefyZrF4sI=&

Двоичный файл расшифровывает этот параметр при помощи алгоритма, показанного на рисунке 11. Список получившихся строк содержит несколько серверов C2:


Рисунок 11


Рисунок 12

В процессе реверс-инжиниринга расшифрованы следующие URL:

  • https[:]//mante.li/images/draw.php
  • https[:]//bmanal.com/images/draw.php
  • https[:]//shopandtravelusa.com/vendor/monolog/monolog/src/Monolog/monolog.php
  • https[:]//industryinfostructure.com/templates/worldgroup/view.php

Подпрограмма GetNetworkParams используется для получения сетевых параметров локального компьютера:


Рисунок 13

Зловредный процесс извлекает имя DNS-домена, назначенного локальному хосту (0x2 = ComputerNameDnsDomain:


Рисунок 14

Во временный буфер записывается следующая сетевая информация:


Рисунок 15


Рисунок 16

Вызывая функцию GetUserNameW, процесс получает имя пользователя, связанное с текущим потоком:


Рисунок 17

Двоичный файл делает снэпшот всех процессов в системе при помощи CreateToolhelp32Snapshot API (0x2 = TH32CS_SNAPPROCESS):


Рисунок 18

Файл извлекает информацию о первом процессе из снэпшота при помощи вызова Process32FirstW:


Рисунок 19

Затем двоичный файл открывает объект процесса при помощи подпрограммы OpenProcess (0x410 = PROCESS_QUERY_INFORMATION | PROCESS_VM_READ):


Рисунок 20

Если у файла не хватает прав для открытия процесса, он копирует во временный буфер «Unknown» с именем процесса.

Далее он при помощи CreateToolhelp32Snapshot API (0x8 = TH32CS_SNAPMODULE) делает снэпшот текущего процесса со всеми его модулями:


Рисунок 21

Для получения информации о первом модуле, связанном с текущим процессом, используется Module32FirstW:


Рисунок 22

Зловредный DLL получает информацию о следующем процессе, записанном в снэпшот:


Рисунок 23

Для открытия связанного с процессом токена доступа используется подпрограмма OpenProcessToken (0x8 = TOKEN_QUERY):


Рисунок 24

Для извлечения аккаунта пользователя токена используется GetTokenInformation (0x1 = TokenUser):


Рисунок 25

При помощи вызова функции LookupAccountSidW процесс получает имя аккаунта для SID и имя первого домена, в котором найден SID:


Рисунок 26

Для извлечения идентификатора связанной с токеном сессии Terminal Services используется GetTokenInformation (0xC = TokenSessionId):


Рисунок 27

Чтобы определить нужный размер буфера WorkSpace для функции RtlCompressBuffer используется RtlGetCompressionWorkSpaceSize API (0x102 = COMPRESSION_FORMAT_LZNT1 | COMPRESSION_ENGINE_MAXIMUM):


Рисунок 28

При помощи функции RtlCompressBuffer процесс сжимает буферы, показанные на рисунках 15 и 16 (0x102 = COMPRESSION_FORMAT_LZNT1 | COMPRESSION_ENGINE_MAXIMUM):


Рисунок 29

DLL случайным образом выбирает один из четырёх серверов C2 в списке, а потом при помощи вызова InternetOpenW инициализирует использование приложением функций WinINet:


Рисунок 30

Для приведения URL в каноничный вид используется InternetCanonicalizeUrlW:


Рисунок 31

При помощи вызова InternetCrackUrlW API зловред разбивает URL на компоненты:


Рисунок 32

При помощи подпрограммы InternetSetOptionW устанавливаются таймауты подключения, отправки и получения на 150 с (0x2 = INTERNET_OPTION_CONNECT_TIMEOUT, 0x5 = INTERNET_OPTION_SEND_TIMEOUT, 0x6 = INTERNET_OPTION_RECEIVE_TIMEOUT):


Рисунок 33


Рисунок 34


Рисунок 35

DLL открывает HTTP-сессию с сервером C2 через порт 443 (0x3 = INTERNET_SERVICE_HTTP):


Рисунок 36

Двоичный файл создаёт дескриптор POST-запроса к URI, извлечённому из указанного URL:


Рисунок 37

Флаги безопасности для дескриптора задаются при помощи InternetSetOptionW API (0x1F = INTERNET_OPTION_SECURITY_FLAGS, 0xF180 = SECURITY_FLAG_IGNORE_REVOCATION | SECURITY_FLAG_IGNORE_UNKNOWN_CA | SECURITY_FLAG_IGNORE_CERT_CN_INVALID | SECURITY_FLAG_IGNORE_CERT_DATE_INVALID | SECURITY_FLAG_IGNORE_REDIRECT_TO_HTTP | SECURITY_FLAG_IGNORE_REDIRECT_TO_HTTPS):


Рисунок 38

Ранее сжатый буфер (конкатенация двух буферов) шифруется при помощи XOR (ключ = 32-байтный массив):


Рисунок 39


Рисунок 40

Далее зашифрованный буфер кодируется при помощи Base64:


Рисунок 41


Рисунок 42

Двоичный файл создаёт следующие параметры: «search=YOIPOUP&ei=6128&oq=<Base64-encoded buffer>»:


Рисунок 43

Извлечённый ранее User Agent добавляется к дескриптору HTTP-запроса при помощи подпрограммы HttpAddRequestHeadersW (0xA0000000 = HTTP_ADDREQ_FLAG_REPLACE | HTTP_ADDREQ_FLAG_ADD):


Рисунок 44

Для передачи данных на сервер C2 используется HttpSendRequestW:


Рисунок 45

Стоит отметить, что на момент выполнения нашего анализа все серверы C2 уже были отключены. Мы эмулировали сетевые соединения при помощи FakeNet.

Размер ответа C2 получается при помощи вызова подпрограммы HttpQueryInfoW (0x5 = HTTP_QUERY_CONTENT_LENGTH):


Рисунок 46

Двоичный файл копирует ответ C2 в буфер при помощи вызова функции InternetReadFile:


Рисунок 47

Зловредный процес парсит данные между тэгами <html></html> и <div></div>:


Рисунок 48

Зловредное ПО выполняет похожий POST-запрос с другими значениями параметров: «search=DOWPANY&ei=6128»:


Рисунок 49

Ответ C2 декодируется при помощи Base64, а затем расшифровывается XOR. Зловредный код реализует четыре разных действия на основании значения регистра EAX:


Рисунок 50

EAX = 0 — загрузка PE в память текущего процесса

Для получения информации о текущей системе используется GetNativeSystemInfo:


Рисунок 51

DLL выполняет несколько вызовов функции VirtualAlloc, выделяющих память под новый исполняемый файл (0x3000 = MEM_COMMIT | MEM_RESERVE, 0x4 = PAGE_READWRITE):


Рисунок 52

Зловредный код в зависимости от сегмента меняет защиту памяти (например, защита памяти для сегмента кода установлена на 0x20 = PAGE_EXECUTE_READ):


Рисунок 53

Спустя ещё несколько операций процесс передаёт поток управления новому PE.

EAX = 1 — скачать и исполнить файл .exe

Двоичный файл получает путь к папке AppData, вызывая подпрограмму SHGetFolderPathW (0x1c = CSIDL_LOCAL_APPDATA):


Рисунок 54

Для получения количества секунд после запуска системы используется GetTickCount:


Рисунок 55

Зловредный код создаёт файл на основании указанного выше значения (0x40000000 = GENERIC_WRITE, 0x1 = FILE_SHARE_READ, 0x2 = CREATE_ALWAYS, 0x80 = FILE_ATTRIBUTE_NORMAL):


Рисунок 56

Созданный файл заполняется содержимым, который должен быть передан на сервер C2:


Рисунок 57

Зловредный двоичный код исполняет файл, вызывая CreateProcessW API:


Рисунок 58

EAX = 2 — скачать и исполнить файл .dll

Поток исполнения схож с показанным выше, поэтому мы покажем только различия. Для исполнения файла DLL используется Rundll32.exe (также в командной строке может быть задана функция экспорта):


Рисунок 59

EAX = 3 — копировать и исполнить шелл-код

Процесс выделяет память при помощи подпрограммы VirtualAlloc (0x1000 = MEM_COMMIT, 0x40 = PAGE_EXECUTE_READWRITE):


Рисунок 60

DLL реализует проверку на антианализ. Она вызывает isProcessorFeaturePresent API, чтобы определить, доступен ли _fastfail(). Если эта функция не поддерживается, текущий процесс прекращается вызовом функций GetCurrentProcess и TerminateProcess (0x17 = PF_FASTFAIL_AVAILABLE):


Рисунок 61

Зловредный код переходит к шелл-коду, а затем освобождает выделенную ранее область памяти:


Рисунок 62

Как говорилось в начале нашего анализа, злоумышленник добавил рассмотренную выше функцию экспорта, а все остальные вполне безобидны.

Мы изучили подлинное расширение оболочки Notepad++ (SHA256: f3e2e6f9e7aa065e89040a0c16d1f948489b3751e5eb5efac8106d5f7d65d98d 64-bit) и сравнили функции экспорта двух файлов. Как видно ниже, функции очень схожи:


Рисунок 63


Рисунок 64

Справочные материалы


  1. MSDN
  2. Fakenet
  3. VirusTotal
  4. MalwareBazaar

Признаки компрометации

Домены C2:
  • mante.li
  • bmanal.com
  • shopandtravelusa.com
  • industryinfostructure.com

SHA256: 803dda6c8dc426f1005acdf765d9ef897dd502cd8a80632eef4738d1d7947269
URL:
  • https[:]//mante.li/images/draw.php
  • https[:]//bmanal.com/images/draw.php
  • https[:]//shopandtravelusa.com/vendor/monolog/monolog/src/Monolog/monolog.php
  • https[:]//industryinfostructure.com/templates/worldgroup/view.php

Комментарии (7)


  1. vkovalchuk
    08.06.2022 18:32
    +25

    я ничего не понял. Какое именно расширение notepad++? Как оно относится к docx файлу? В какой момент notepad++ входит на сцену?


    1. domix32
      08.06.2022 20:39
      +13

      Ответ прост - да.


    1. asaks
      09.06.2022 13:57

      Видимо имелось ввиду не конкретное расширение notepad++, а то что, вредонос мимикрирует под такое расширение, используя похожие функции экспорта, как если бы это делал настоящий notepad++


    1. praeivis
      09.06.2022 14:46
      +1

      Вплоть до недавнего времени для интеграции с виндовс меню нотепад++ использовал NppShell_06.dll и видимо docx каким то способом мог переписать этот NppShell_06.dll.

      Хотя спросит некого так как это перевод статьи от января месяца и я не нашел чтобы где-то, кто-то смогли разобраться что хотел донести автор.


  1. vanyas
    08.06.2022 19:29
    +23

    В заголовке notepad++, далее пишется что распространялся файл docs после чего незапно говорится по какую-то dll, как одно вытекает из другого не ясно абсолютно


  1. ECCOsea
    09.06.2022 13:52

    Как вообще docx может запустить exe или dll? Если только какое то динамическое содержимое встроено? Любая программа обычно об этом предупреждает, спрашивает разрешать ли его, и рассказывает о последствиях. Или здесь открыл docx просто и попал? Совсем не понятно(


  1. DrGluck07
    10.06.2022 15:43

    Этому посту определенно не хватает вводной части, где будет кратко рассказано кто на ком стоял.