新书推介:《语义网技术体系》
作者:瞿裕忠,胡伟,程龚
   XML论坛     W3CHINA.ORG讨论区     计算机科学论坛     SOAChina论坛     Blog     开放翻译计划     新浪微博  
 
  • 首页
  • 登录
  • 注册
  • 软件下载
  • 资料下载
  • 核心成员
  • 帮助
  •   Add to Google

    >> 本版讨论高级C/C++编程、代码重构(Refactoring)、极限编程(XP)、泛型编程等话题
    [返回] 中文XML论坛 - 专业的XML技术讨论区计算机技术与应用『 C/C++编程思想 』 → 驱动中实现模拟键盘按键 查看新帖用户列表

      发表一个新主题  发表一个新投票  回复主题  (订阅本版) 您是本帖的第 7963 个阅读者  浏览上一篇主题  刷新本主题   树形显示贴子 浏览下一篇主题
     * 贴子主题: 驱动中实现模拟键盘按键 举报  打印  推荐  IE收藏夹 
       本主题类别:     
     卷积内核 帅哥哟,离线,有人找我吗?
      
      
      威望:8
      头衔:总统
      等级:博士二年级(版主)
      文章:3942
      积分:27590
      门派:XML.ORG.CN
      注册:2004/7/21

    姓名:(无权查看)
    城市:(无权查看)
    院校:(无权查看)
    给卷积内核发送一个短消息 把卷积内核加入好友 查看卷积内核的个人资料 搜索卷积内核在『 C/C++编程思想 』的所有贴子 访问卷积内核的主页 引用回复这个贴子 回复这个贴子 查看卷积内核的博客楼主
    发贴心情 驱动中实现模拟键盘按键

    在ring3中实现模拟键盘按键有N^N种方式,比如SendInput()、keybd_event()……但在驱动中要怎么模拟呢?

    1、写端口大法
    #define defI8042_DATA_PORT ((PUCHAR)0x60)
    #define defI8042_CTRL_PORT ((PUCHAR)0x64)
    #define defOBUFFER_FULL 0x01

    BOOLEAN
    WaitForKeyboardWrite(VOID)
    {
        INT i;
        UCHAR c;

        for (i = 0; i < 1000; ++i)
        {
            KeStallExecutionProcessor(50);
            c = READ_PORT_UCHAR(defI8042_CTRL_PORT);
            if ((c & defOBUFFER_FULL) == defOBUFFER_FULL)
                break;
        }

        return i ? TRUE : FALSE;
    }

    VOID PressKeyByScanCode(
        IN CONST BYTE ScanCode
    )
    {
        WRITE_PORT_UCHAR(defI8042_CTRL_PORT, 0xd2);

        WaitForKeyboardWrite();

        WRITE_PORT_UCHAR(defI8042_DATA_PORT, ScanCode);
    }

    搞定,简单易行!缺点是只对PS/2键盘有效,USB的就一边凉快去吧。

    2、构造IRP大法
    这个比较麻烦,而且由于某些原因,略……

    3、调用KeyboardClassServiceCallback()大法
    kbdclass驱动是在I8042prt和hidkbd这两个驱动之上的,所以直接调用它的KeyboardClassServiceCallback()是能对PS/2和USB同时生效的,由于某些原因偶也不想给出具体的代码来。大致指点一下方向:

    KeyboaredClassServiceCallback函数的原型如下:

    typedef VOID (* PFN_KeyboardClassServiceCallback)(
        IN PDEVICE_OBJECT DeviceObject,
        IN PKEYBOARD_INPUT_DATA InputDataStart,
        IN PKEYBOARD_INPUT_DATA InputDataEnd,
        IN OUT PULONG InputDataConsumed
        );

    第一个参数DeviceObject是kbdclass的DeviceObject,怎么得到它呢?思路:先找到kbdclass的DriverObject,然后从DriverObject得到DeviceObject。
    第二和第三个参数是KEYBOARD_INPUT_DATA,自己填充,想模拟什么按键就写什么MakeCode和Flags吧。注意第三个参数应该为第二个参数的指针地址值+1。
    第四个参数,随便弄个ULONG型的变量给它就行了,反正是可以丢弃的。(I8042prt里面用到了它,但我们自己的模拟按键不用也行)

    缺点:每个系统的kbdclass.sys中的KeyboardClassServiceCallback的RVA都不相同,要自己hardcode一下,不过写驱动本来就是那么麻烦,习惯了就好。


       收藏   分享  
    顶(0)
      





    关闭广告显示

    ----------------------------------------------
    事业是国家的,荣誉是单位的,成绩是领导的,工资是老婆的,财产是孩子的,错误是自己的。

    点击查看用户来源及管理<br>发贴IP:*.*.*.* 2007/8/30 16:33:00
     
     卷积内核 帅哥哟,离线,有人找我吗?
      
      
      威望:8
      头衔:总统
      等级:博士二年级(版主)
      文章:3942
      积分:27590
      门派:XML.ORG.CN
      注册:2004/7/21

    姓名:(无权查看)
    城市:(无权查看)
    院校:(无权查看)
    给卷积内核发送一个短消息 把卷积内核加入好友 查看卷积内核的个人资料 搜索卷积内核在『 C/C++编程思想 』的所有贴子 访问卷积内核的主页 引用回复这个贴子 回复这个贴子 查看卷积内核的博客2
    发贴心情 
    操作硬件板卡方式
    WinCE一般有两种方式访问硬件
    1、内存隐射
    代码如下:
    #include <Pkfuncs.h>

                    LPVOID reg = VirtualAlloc(0, 32, MEM_RESERVE, PAGE_NOACCESS);
                    if (reg)
                    {
                            if ( !VirtualCopy(reg, (LPVOID)(0X0DA000>>8), 32, PAGE_PHYSICAL | PAGE_READWRITE | PAGE_NOCACHE) )
                            {
                                    DWORD dwErr = GetLastError();
                                    VirtualFree(reg, 32, MEM_RELEASE);
                                    reg = NULL;
                            }
                    }
    //reg 就是访问硬件的地址

    VirtualFree((PVOID)reg , 32, MEM_RELEASE);  //用完了要记得释放哦


    2、端口
    #include <ceddk.h>
    READ_PORT_UCHAR
    WRITE_PORT_UCHAR
    API可以使用,代码略

    ----------------------------------------------
    事业是国家的,荣誉是单位的,成绩是领导的,工资是老婆的,财产是孩子的,错误是自己的。

    点击查看用户来源及管理<br>发贴IP:*.*.*.* 2007/8/30 16:33:00
     
     GoogleAdSense
      
      
      等级:大一新生
      文章:1
      积分:50
      门派:无门无派
      院校:未填写
      注册:2007-01-01
    给Google AdSense发送一个短消息 把Google AdSense加入好友 查看Google AdSense的个人资料 搜索Google AdSense在『 C/C++编程思想 』的所有贴子 访问Google AdSense的主页 引用回复这个贴子 回复这个贴子 查看Google AdSense的博客广告
    2025/7/16 13:18:59

    本主题贴数2,分页: [1]

    管理选项修改tag | 锁定 | 解锁 | 提升 | 删除 | 移动 | 固顶 | 总固顶 | 奖励 | 惩罚 | 发布公告
    W3C Contributing Supporter! W 3 C h i n a ( since 2003 ) 旗 下 站 点
    苏ICP备05006046号《全国人大常委会关于维护互联网安全的决定》《计算机信息网络国际联网安全保护管理办法》
    218.750ms