DLL注入技术之输入法注入

 输入法注入原理是利用Windows系统中在切换输入法需要输入字符时,系统就会把这个输入法需要的ime文件装载到当前进程中,而由于这个Ime文件本质上只是个存放在C:\WINDOWS\system32目录下的特殊的DLL文件,因此我们可以利用这个特性,在Ime文件中使用LoadLibrary()函数待注入的DLL文件。
1.编写Ime文件
    输入法的Ime文件其实就是个显式导出19个特殊函数的DLL文件。如下图所示:

    如果想编写输入法程序,那么这19个导出函数都需要仔细的研究,但是对于只想实现注入的我们,现在只需要对ImeInquire()有比较深的认识就可以了。ImeInquire()是启动并初始化当前Ime输入法函数,他的声明如下:
  1. BOOL WINAPI ImeInquire(LPIMEINFO lpIMEInfo,LPTSTR lpszUIClass,LPCTSTR lpszOption)
复制代码
第一个参数lpIMEInfo比较重要,用于输入对Ime输入法初始化的内容结构,如果这个结构填写错误,就会导致输入法不能正常运行。第二个参数是输入一个class类名,我们需要先使用RegisterClassEx()注册出一个窗口类。初始化ImeInquire()主要代码如下所示:

  1. //启动并初始化当前ime输入法
  2. BOOL WINAPI ImeInquire(LPIMEINFO lpIMEInfo,LPTSTR lpszUIClass,LPCTSTR lpszOption)
  3. {
  4.     //输入法初始化过程
  5.     //系统根据它为INPUTCONTEXT.hPrivate分配空间
  6.     lpIMEInfo->dwPrivateDataSize = 0; 
  7.     lpIMEInfo->fdwProperty = IME_PROP_KBD_CHAR_FIRST |
  8.         IME_PROP_IGNORE_UPKEYS |
  9.         IME_PROP_END_UNLOAD; 

  10.     lpIMEInfo->fdwConversionCaps = IME_CMODE_FULLSHAPE |
  11.         IME_CMODE_NATIVE;

  12.     lpIMEInfo->fdwSentenceCaps = IME_SMODE_NONE;
  13.     lpIMEInfo->fdwUICaps = UI_CAP_2700;

  14.     lpIMEInfo->fdwSCSCaps = 0;

  15.     lpIMEInfo->fdwSelectCaps = SELECT_CAP_CONVERSION;

  16.     // 注意该输入法基本窗口类必须注册,否则输入法不能正常运行
  17.     _tcscpy(lpszUIClass,CLSNAME_UI);
  18.     return TRUE;
  19. }
复制代码
注册出一个窗口类的主要代码如下:
  1. BOOL ImeClass_Register(HINSTANCE hInstance)
  2. {
  3.     WNDCLASSEX wc;
  4.     wc.cbSize         = sizeof(WNDCLASSEX);
  5.     wc.style          = CS_VREDRAW | CS_HREDRAW | CS_DBLCLKS | CS_IME;
  6.     wc.lpfnWndProc    = UIWndProc;
  7.     wc.cbClsExtra     = 0;
  8.     wc.cbWndExtra     = 2 * sizeof(LONG);
  9.     wc.hInstance      = hInstance;
  10.     wc.hCursor        = LoadCursor( NULL, IDC_ARROW );
  11.     wc.hIcon          = NULL;
  12.     wc.lpszMenuName   = (LPTSTR)NULL;
  13.     wc.lpszClassName  = CLSNAME_UI;
  14.     wc.hbrBackground  = NULL;
  15.     wc.hIconSm        = NULL;

  16.     if( !RegisterClassEx( (LPWNDCLASSEX)&wc ) )
  17.         return FALSE;

  18.     return TRUE;
  19. }
复制代码
CLSNAME_UI是一个宏定义,如下:
  1. #define CLSNAME_UI _T("DLLCLASSNAME")
复制代码
在DllMain进程加载的过程中注册窗口类,主要代码如下:
  1.     case DLL_PROCESS_ATTACH:
  2.         if(!ImeClass_Register(hinstDLL)) return FALSE;

  3.         //这里填写要load的DLL的路径
  4.         g_hModule = LoadLibrary(_T("D:\\MyDll\\ImeInject\\Debug\\MfcImeInjectDll.dll"));
  5.         
  6.         if (!g_hModule)
  7.         {
  8.             MessageBox(NULL,_T("模块没有load成功"),_T("提示"),MB_OK);
  9.         }
  10.         break;
复制代码
此刻默认输出的DLL文件扩展名是.dll,与扩展名是.ime不符,可以通过下图设置将其扩展名改为.ime:

2.编写装载输入法程序:
    装载输入法的基本逻辑就是将他们编写的输入法设置为默认输入法,这样只要系统中所有进程都会默认加载他们的恶意输入法程序。
    黑客们首先需要得到系统当前的默认的输入法,以便恢复时使用。然后需要将ime文件拷贝到C:\WINDOWS\system32目录下,最后将装载成功后将我们的输入法设置成为默认输入法,主要代码如下:
  1. void CMfcImeInjectDlg::OnBnClickedAttach()
  2. {
  3.     // TODO: 在此添加控件通知处理程序代码
  4.     //得到默认的输入法句柄并保存
  5.     ::SystemParametersInfo(
  6.         SPI_GETDEFAULTINPUTLANG,
  7.         0,
  8.         &m_retV,
  9.         0);

  10.     //拷贝到system目录,只有ime文件在system32下才能装载成功
  11.     CopyFile(
  12.         _T("D:\\MyDll\\ImeInject\\Debug\\MyImeDll.ime"),
  13.         _T("C:\\WINDOWS\\system32\\MyImeDll.ime"),
  14.         FALSE);

  15.     //装载输入法
  16.     m_hImeFile = ImmInstallIME(
  17.         _T("MyImeDll.ime"), 
  18.         _T("我的输入法"));

  19.     if( ImmIsIME(m_hImeFile) )
  20.     {
  21.         //设置为默认输入法
  22.         SystemParametersInfo(
  23.             SPI_SETDEFAULTINPUTLANG,
  24.             0,
  25.             &m_hImeFile,
  26.             SPIF_SENDWININICHANGE);
  27.         MessageBox(_T("安装输入法成功"));
  28.     }
  29. }
复制代码
3.编写卸载输入法:
    当新建进程不再需要注入时,我们就需要卸载输入法。卸载输入法时需要先判定系统当前的输入法不是其原有默认输入法,确认无误后将系统的默认输入法恢复后,再将恶意输入法卸载即可,主要代码如下:
  1. void CMfcImeInjectDlg::OnBnClickedDettach()
  2. {
  3.     // TODO: 在此添加控件通知处理程序代码
  4.     ::SystemParametersInfo(
  5.         SPI_SETDEFAULTINPUTLANG,
  6.         0,
  7.         &m_retV,
  8.         SPIF_SENDWININICHANGE);
  9.     if (UnloadKeyboardLayout(m_hImeFile))
  10.     {
  11.         MessageBox(_T("输入法卸载成功"));
  12.     }
  13. }
  14.    
复制代码
输入法注入的实现需要对输入法IME文件的生成有所了解,API使用较多,所以实现起来比较难,但是由于系统存在多个输入法,被注入进程很难判别当前是可信赖输入法还是用于注入的恶意输入法,所以难以阻止,大大提高了注入的几率
  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Delphi是一种编程语言,而DLL(Dynamic-Link Library)是一种模块化的文件格式,用于存储代码和数据,可以被多个应用程序共享。DLL注入是一种技术,它允许将DLL文件加载到正在运行的进程中,并使得该进程能够调用DLL中的函数和使用其中的数据。 在Delphi中实现DLL注入的方法有很多种。一种常见的方法是使用Windows API函数LoadLibrary和GetProcAddress。通过调用LoadLibrary函数,将DLL文件加载到进程的虚拟地址空间中。然后使用GetProcAddress函数获取DLL中导出函数的地址,并将其传递给需要调用的函数。通过这种方式,可以在运行时将DLL注入到目标进程中,并且通过调用DLL中的函数来扩展进程的功能。 DLL注入在实际应用中有多种用途。例如,可以使用DLL注入来为某个程序添加额外的功能或修改程序的行为。DLL注入还可以用于实现一些调试和监控的功能。通过注入DLL,可以截获程序的输入和输出,或者在程序执行某些指定的操作时进行额外的处理。 在Delphi中实现DLL注入需要一定的编程知识和技巧。需要考虑目标进程的架构和权限限制,以及如何管理注入DLL的生命周期和资源管理。同时,还需要处理一些安全性和稳定性方面的问题,以确保注入过程不会对目标进程造成损害或崩溃。 总之,Delphi可以通过调用Windows API函数来实现DLL注入,从而扩展和修改进程的功能。但在实际应用中,需要考虑各种方面的问题,并且遵守相关的法律和规定,以确保注入操作的安全性和合法性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值