« Поставить закладку » « Сделать стартовой »

« Форумы » « Блоги » « Статьи » « Новости » « Файлы » « Realcoding IRC » « Site map » « Поиск »


Главная Главная
Анонсы Анонсы
Форумы Форумы
Каталог Каталог
Поиск Поиск
Опросы Опросы
Книжный магазин Книжный магазин
Реклама на сайте
Публикации Публикации
Партнеры Партнеры
Карта Карта сайта
Рассылки Рассылки
RSS экспорт
Настройки Настройки
О нас пишут О нас пишут
Контакты Контакты
Гостевая книга Гостевая книга

Создание сайта за 3999 руб.!

ПнВтСрЧтПтСбВс
        1 2 3
4 5 6 7 8 9 10
11 12 13 14 15 16 17
18 19 20 21 22 23 24
25 26 27 28 29 30 31
    Популярное
Описание функций C (Си) / C++ - strstr

DataSet (Набор данных) и XML

Функция AccessResource

Создание псевдонима с указанием пути к папке с базой

Что такое Fluke?

Решение задач бизнеса с помощью SQL Server 2005 Analysis Services

Протоколы OSI. Основы технологии, Доступ к среде, Сетевой уровень, Услуги без установления соединения, Услуги с установлением соединения, Адресация, Транспортный уровень, Протоколы высших уровней, Сеансовый уровень, Представительный уровень, Прикладной ур

Использование справочной системы в приложениях

Использование самостоятельно созданного атрибута

Функция BeginPaint




    Архив файлов



    Сообщества



    Документация

Статьи:: C/С++ :: Visual C++ :: Скрытие программ в Visual C++


отправить ссылку другу версия для печати  Обсудить на форуме

Скрытие программ в Visual C++



Часто бывает так, что нужно скрыть информацию так, чтобы не только не дать чужим людям прочитать ее, но и скрыть сам факт передачи/хранения информации. Такое скрытие называется стеганографией. Она бывает полезна, например, когда в универе или на работе вредный админ иногда контролирует содержимое пользовательских папок и грозит серьезными санкциями.

Стеганография не решает всех проблем. Кроме того, чтобы хранить файл с закодированными данными, надо, кроме того, хранить программу для кодирования и раскодирования. Ее тоже необходимо хранить скрытно, чтобы не вызывать подозрений. Ясен пень, что хранить ее в закодированном виде бессмысленно, так как сама себя она не раскодирует. Стало быть, задача состоит в том, чтобы хранить программу так, чтобы она запускалась без каких-либо специальных средств, но при невыполнении заданных условий не давала понять свое истинное предназначение и имитировала работу какой-либо другой программы.
Метод первый. Пусть имеется сишный исходник скрываемой программы и исходник какой-нибудь ерунды, которую можно использовать как программу-носитель, т. е. программу, в которую мы будем прятать нашу программу. Пусть также обе программы оконные и имеют в качестве точки входа функцию WinMain. Переделать этот пример для консольных программ легко, заменив WinMain на main.
Объединим оба исходника в один проект и переименуем WinMain скрываемой-программы в WinMain1, а WinMain программы-носителя в WinMain2. Теперь напишем новую WinMain такого содержания:

 


int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
{
if (Condition())
return WinMain1(hInstance, hPrevInstance, lpCmdLine, nCmdShow); else
return WinMain2(hInstance, hPrevInstance, lpCmdLine, nCmdShow);
}
 


Как видите, идея простая: проверяем условие и запускаем либо одну, либо другую функцию, передавая ей параметры, с которыми была запущена наша программа. Это код еще даже можно сократить:

 

int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
{
return (Condition() ? WinMain1 : WinMain2)(hInstance, hPrevInstance, lpCmdLine, nCmdShow);
}
 


Теперь о функции Condition(), т. е. о том, какое может быть условие раскрытия скрытой программы и как его проверять.
В качестве примера пусть наша программа будет запускаться при нажатии определенной комбинации клавиш. Проверить, нажата ли клавиша, можно с помощью функции GetKeyState или GetAsyncKeyState. Разница между ними в том, что кроме проверки состояния клавиши в момент вызова функции GetKeyState() проверяет, включен ли CapsLock, NumLock или ScrollLock (если передать код одной из этих клавиш), а GetAsyncKeyState проверяет, была ли клавиша нажата с момента предыдущего вызова. Так что если нужно использовать состояния NumLock или CapsLock, то только GetKeyState(), иначе можно выбрать любую из этих функций.
Каждая из этих функций принимает в качестве параметра код виртуальной клавиши (virtual key code) и возвращает значение типа SHORT, старший бит которого указывает, что клавиша нажата сейчас, а младший бит указывает для GetKeyState(), включен ли CapsLock, NumLock или ScrollLock, а для GetAsyncKeyState, была ли клавиша нажата с момента предыдущего вызова.
Необходимо учесть, что пользователю нужно время для того, чтобы нажать на клавиши после запуска программы. Удерживать их во время запуска не получится в стандартной оболочке и наиболее распространенных ее альтернативах. Так что перед проверкой условия вызовем Sleep(1000), чтобы дать юзеру одну секунду.
Пусть нашей секретной комбинацией будет SHIFT+CTRL+A+D. Итак, получается вот такой код:

 

BOOL Condition()
{
SHORT ks;

/* Ждем секунду, чтобы юзер успел нажать комбинацию клавиш */
Sleep(1000);

/* Проверяем, нажаты ли SHIFT+CTRL+A+D */
ks = GetKeyState(VK_SHIFT) &
GetKeyState(VK_CONTROL) &
GetKeyState(′A′) &
GetKeyState(′D′);

/* О нажатии клавиш в данный момент говорит старший бит */
return (ks & 0x8000);
}
 


В исходниках эта программа лежит в папке cprog.

Метод второй, более общий. Не всегда есть исходники нужного софта. Это не мешает нам спрятать один ехешник внутрь другого. По большому счету, для этого мы должны должны написать полноценный ехе-джойнер, с той только разницей, что запуск программ должен происходить в зависимости от нажатых клавиш. Может быть, в одной из следующих статей я расскажу о структуре ехе-файлов и о написании джойнеров, а сейчас рассмотрим более простую вещь, в которой, тем не менее присутствуют другие важные для нашей темы моменты.
Наша скрытая программа будет написана на асме, и из нее мы сделаем сырой бинарник, в котором не будет ни релокаций, ни импорта, ни секций, а только код, точка входа которого находится в начале и вызывается как функция, в параметрах которой передаются указатели на функции LoadLibrary, FreeLibrary и GetProcAddress (надеюсь, не надо объяснять, почему этих функций достаточно для любой программы). Иными словами, указатель на точку входа бинарника можно определить так:

 

/* Прототипы функций системы */
typedef HMODULE (WINAPI * pLoadLibrary) (LPCTSTR);
typedef FARPROC (WINAPI * pGetProcAddress) (HMODULE, LPCTSTR);
typedef BOOL (WINAPI * pFreeLibrary) (HMODULE);

/* Прототип точки входа нашего бинарника */
typedef int (WINAPI * pBinEntry) (pLoadLibrary, pGetProcAddress, pFreeLibrary);
 


Еще одна фишка, которую мы на этом примере продемонстрируем: зашифруем бинарник. Может потребоваться скрывать программу не только от глаз админа, но и от антивируса. Я вовсе не говорю о чем-то незаконном. Просто некоторые производители антивирусов добавляют в свои базы программы, которые сами по себе не вредят тому компу, на котором находятся, но могут не понравиться аднинам. Шифрование скроет код от антивируса на случай, если вы собираетесь работать не у себя дома. Будем использовать массив флагов состояний клавиш в качестве ключа к шифру, расшифровывать бинарник и проверять его хеш. Если хеш совпадает, передадим управление, а иначе запустим фальшивку. Такой ключ, конечно, слабый, но антивирусы никогда не ломают шифры перебором, а защищать код от человека, который обнаружит факт его скрытия, мы не будем - у нас другая цель.
Сам зашифрованный бинарник я поместил в ресурс. Код программы от этого буде больше, зато для замены бинарника надо просто перезаписать файл в проекте.
Итак, это может выглядеть так:

 

void GKS(PBYTE pb)
{
int i;

for (i = 1; i < ′Z′; i++) {
if (GetKeyState(i) & 0x8000) pb[i ]++;
}
}

int DecyptCheckAndRun()
{
int rv;
const void* pData;
void* pBuffer;
HRSRC hResource;
DWORD dwSize;
DWORD dwRSize;
HMODULE hThis;
RC6_WORD key[20*2+4];
BYTE ks[256];
SHA1Context ctx;
uint8_t hash[SHA1HashSize];
uint8_t RealHash[SHA1HashSize] = {195, 183, 15, 151, 251, 143, 113, 170, 15, 176, 25, 156, 18, 166, 233, 151, 155, 162, 96, 128};

rv = 0;
hThis = GetModuleHandle(NULL);
hResource = FindResource(hThis, MAKEINTRESOURCE(IDR_BINARY), MAKEINTRESOURCE(RT_RCDATA));
if (hResource) {
dwSize = SizeofResource(hThis, hResource);
pData = LockResource(LoadResource(hThis, hResource));
if (pData) {
dwRSize = (dwSize + 4095) & 0xFFFFF000; /* Round up to page */
pBuffer = VirtualAlloc(NULL, dwRSize, MEM_COMMIT, PAGE_EXECUTE_READWRITE);

memset(pBuffer, 0, dwRSize);
memcpy(pBuffer, pData, dwSize);

/* Получим состояния всех клавиш */
memset(ks, 0, 256);
GKS(ks);

/* Сформируем на его основе ключ шифра */
rc6ks(key, ks, 20, 256);

/* Расшифруем код */
rc6decrypt((LPRC6_WORD)pBuffer, key, 20, dwSize / RC6_BLOCK_SIZE);

/* Посчитаем хеш */
SHA1Reset(&ctx);
SHA1Input(&ctx, (uint8_t*)pBuffer, dwSize);
SHA1Result(&ctx, hash);

/* Сравним значения хешей */
if (memcmp(hash, RealHash, SHA1HashSize) == 0) {

/* Вызовем код бинарника */
(*(pBinEntry)pBuffer) (&LoadLibrary, &GetProcAddress, &FreeLibrary);

/* Возвращаем флаг, указывающий, что не надо запускать носитель */
rv = 1;
}

VirtualFree(pBuffer, dwSize, MEM_FREE);
}
}
return rv;
}

int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
{
/* Ждем */
Sleep(1000);

/* Расшифруем и выполним код и возвратим управление, при успехе */
if (DecyptCheckAndRun()) return 0;

// ...
}
 


В этой программе используется моя реализация алгоритма шифрования RC6 и реализация алгоритма хеширования sha1 из RFC3174.
Для получения состояния всех клавиш я написал функцию GKS. Вообще-то в винде есть функция GetKeyboardState, но среди возвращаемой ей информации есть не только текущие состояния клавиш, и при ее использовании условия запуска скрытой программы не ограничивались бы комбинацией клавиш.
В исходниках есть тестовый бинарник (rawbinary), программа для его шифрования (encryptbin) и программа, в которой он зашифрован с ключевой комбинацией CTRL+SHIFT+S, которую надо нажать втечение секунды (hiddenbin)



Рубрика: Visual C++




ASP.NET и немного поисковой оптимизации.

Советы

С тех пор, как появилась Альтависта, большинство посетителей стали приходить на сайты из поисковых машин. И головной боли стало больше. Теперь мало написать сайт, нужно сделать его совместимым с поисковыми ботами. Одна из тех задач, которую приходится решать — быстрая переиндексация обновлений на сайте. Поисковые роботы чаще посещают те сайты, которые чаще меняются.


Подробнее... | Рубрика: Советы | Добавлено: 30.07.2008

Protocol buffers: библиотека обмена данными для C++, Java, Python от Google.

Программирование для Web на C

Наконец-то я могу поделиться тем, чему уже давно радуюсь сам: Google открывает исходники Protocol Buffers! Что это и почему надо радоваться? Это простой и удобный способ обмена данными. Можно сказать, альтернатива XML, но гораздо менее амбициозная и (поэтому) более быстрая и компактная. Далее - перевод отрывка анонса в блоге Open Source at Google и пример использования.


Подробнее... | Рубрика: Программирование для Web на C | Добавлено: 30.07.2008

Описание VivaVisualCode.

VivaCore

В данной статье рассмотрена программа VivaVisualCode, демонстрирующая использование библиотеки VivaCore. Программа VivaVisualCode графически отображает дерево разбора для вводимого исходного кода на языке Си++.


Подробнее... | Рубрика: VivaCore | Добавлено: 29.07.2008

Остальные статьи:

JQuery: Пара сотен плагинов в одной заметке
Касание сетки
Разработка элементов управления ASP.NET на примере навигационной панели
Сохранения параметров приложения в .Net
Custom cursors в .Net
Бегун убегает от хозяина
"Битрикс" выпустил седьмую версию CMS
Выбираем систему управления сайтом
Рынок CMS в Европе или впечатления с CeBIT 2008
32 подводных камня OpenMP при программировании на Си++
Проeкт - шифровка
А что если..? (операторы if...else в C#)
Измерение скорости работы скрипта
О том как разработчики пьют кофе
Работаем с LINQ to XML
XmlSerializer - Assembly Leak без спроса


Цитата дня (все,добавить):



    Рубрикатор

Программирование

C/С++
Обучение
Windows API
XAML
Моделирование
Паттерны
Visual Basic 7 .NET
WxWidgets
Функции WinApi
Функции С++
Разработка под Mac OS
Eiffel
Visual Studio 2008
UI дизайн
Алгоритмы
Конкурсные статьи
Turbo Pascal
Visual Studio
CASE-средства
Visual Studio 2005
Без VCL
Delphi
Тех. документация
Тестирование
Software Testing
ООП
TCP/IP
Google Android
Windows Installer
.NET Framework
Драйвера
C# C Sharp
Справка
Проектирование
Информ. системы
Visual Basic
Assembler
Оптимизация кода
Gtk+
Компоненты
Реинжиниринг
Управление проектами
Extreeme programming
Lotus Notes
Алгебраическое проектирование

Интернет технологии

PHP
Perl
ASP
WAP
Cookies
SSI
CGI
Web Servers
VB Script
DNS
CSS
XML
Html
Java Script
Java2ME
Firewall
Flash
.htaccess
Apache
VRML
Протоколы
Поисковые системы
Технология JAVA
Учебник по PHP
Учебник по JavaScript
Учебник по XML
Java Q&A
AJAX
DHTML
XHTML
Dreamweaver
Web 2.0
Python
Вебмастеру
Cisco
Ruby on Rails
Silverlight

Базы данных

Access
InterBase
MySQL
Oracle
ADO .NET
Основы SQL
Учебник по Access 2002
MS
Microsoft FoxPro
Доступ к данным
XML в MS SQL Server 2000
ODBC и MyODBC
Обучение
Caché
DB2
PostgresSQL
Sybase
Теория
Хранилища данных
Безопасность
Реляционные данные
MySQL и mSQL

Остальное:

Разное
Обзоры книг
Безопасность
Графика и дизайн
Юмор
Linux
Фракталы
Microsoft Axapta
Многоядерность
Сети
Microsoft Office
Работа
MS-DOS
Криптография
Графика и игроделание
Новости SDK
Системы защиты
Учебник по AutoCad
CVS
Windows XP
Windows Server 2003
Windows Vista
Windows 7
Мероприятия

    Кто на сайте
Вы не зарегистрированы.
Имя:

Пароль:

Запомнить

Регистрация позволит Вам пользоваться дополнительными сервисами.
Сейчас на сайте:
Гостей: 99
Пользователей: 1
  • l_kator