Solved

system wide key/mouse hook does NOT work on NT4.0 if a background process

Posted on 2003-12-11
7
384 Views
Last Modified: 2013-12-03
Based on the prior expert's advice (thanks) I implemented system-wide (I wish) hook in C on NT 4.0.
It has two components: a hook DLL and the main test program. When I run the main test program in DOS console it does capture keyboard/mouse events.

However, if I start the test program in the background (EITHER via a home-grown network messaging mechanism where I issue system("start /B c:\path\main.exe") from an already running background program OR via a remote cmd console (using psexec)); it does not work.

Visual C++ source code is below (Main starts up, waits for an event and when detects one, it exits).

What am I doing wrong ?


//HOOK DLL
#include <windows.h>
#include <stdio.h>

HINSTANCE instance_handle;

#pragma data_seg (".hookdat")
HHOOK g_hHookKbd = NULL;
HHOOK g_hHookMse = NULL;
DWORD g_dwLastInputTick = 0;
#pragma data_seg()
#pragma comment (linker,"/section:.hookdat,rws")


// Get tick count of last keyboard or mouse event
__declspec( dllexport ) DWORD IdleCPUGetLastInputTime()
{
     return g_dwLastInputTick;
}

// Keyboard hook: record tick count
LRESULT CALLBACK MyKbdHook(int code, WPARAM wParam, LPARAM lParam)
{
    if (code==HC_ACTION)
     {
          g_dwLastInputTick = GetTickCount();
    }
    return CallNextHookEx(g_hHookKbd, code, wParam, lParam);
}
 
// Mouse hook: record tick count
LRESULT CALLBACK MyMouseHook(int code, WPARAM wParam, LPARAM lParam)
{
    if (code==HC_ACTION)
     {
       g_dwLastInputTick = GetTickCount();
    }
    return CallNextHookEx(g_hHookMse, code, wParam, lParam);
}
 
// Initialize DLL: install kbd/mouse hooks.
__declspec( dllexport ) bool IdleCPUInit()
{
   if (g_hHookKbd == NULL)
   {
          g_hHookKbd = SetWindowsHookEx(WH_KEYBOARD, (HOOKPROC) MyKbdHook, instance_handle, 0);
          g_hHookMse = SetWindowsHookEx(WH_MOUSE,    (HOOKPROC) MyMouseHook, instance_handle, 0);
          g_dwLastInputTick = GetTickCount(); // init count
    }
    return TRUE;
}
 
// Terminate DLL: remove hooks.
__declspec( dllexport ) void IdleCPUTerm()
{
    BOOL bRet1 = UnhookWindowsHookEx(g_hHookKbd);
    BOOL bRet2 = UnhookWindowsHookEx(g_hHookMse);
}

int WINAPI DllMain(HINSTANCE hinst, unsigned long reason, void* lpReserved)
{
     switch(reason) {
     case DLL_PROCESS_ATTACH:
          DisableThreadLibraryCalls(hinst);
          instance_handle = hinst;
          return TRUE;
     case DLL_PROCESS_DETACH:
          if ((g_hHookKbd != NULL) || (g_hHookMse != NULL))
          {
               IdleCPUTerm();
          }
          return TRUE;
     default:
          return TRUE;
     }
}


//MAIN TEST STUB
//COMMENT: I am using ordinal numbers for hook dll lookup because had some problems with passing the name
#include <stdio.h>
#include <windows.h>

typedef BOOL (*INITPROC) ();
typedef DWORD (*TIMEPROC) ();
typedef VOID (*TERMPROC) ();

int main(int argc,char *argv[]) {
    HINSTANCE hinstLib;
    INITPROC ProcInit = NULL;
    TIMEPROC ProcTime = 0;
    TERMPROC ProcTerm;
    BOOL fFreeResult;
    long last_input_time, time_running, time_started, time_diff;
    BOOL no_interrupt;

    hinstLib = LoadLibrary("hook");
    if (hinstLib != NULL) {
        ProcInit = (INITPROC) GetProcAddress(hinstLib, 2);
        if (ProcInit)  {
         fFreeResult=ProcInit ();
        }
    }
// initial time tick
    time_started = (long) GetTickCount();
    ProcTime = (TIMEPROC) GetProcAddress(hinstLib, 1);
    if (ProcTime) {
        no_interrupt=TRUE;
        while (no_interrupt) {
            last_input_time = (long) ProcTime();
            time_running = (long)((GetTickCount() - time_started)/1000);
            time_diff = (long)((GetTickCount() - last_input_time)/1000);
            if ((time_diff + 4) < time_running) {
                no_interrupt=FALSE;
            }
            if (no_interrupt==TRUE) {
                Sleep (5000);
            }
        }
    }
    ProcTerm = (TERMPROC) GetProcAddress(hinstLib, 3);
    if (NULL != ProcTerm) {
        (ProcTerm) ();
    }
    fFreeResult = FreeLibrary(hinstLib);
}


0
Comment
Question by:jerzyb
  • 3
  • 2
7 Comments
 
LVL 86

Expert Comment

by:jkr
ID: 9922809
>>However, if I start the test program in the background (EITHER via a home-grown network
>>messaging mechanism where I issue system("start /B c:\path\main.exe") from an already running
>>background program OR via a remote cmd console (using psexec)); it does not work.

That's not surprising, as there is a different desktop assiciated with the process than the one of the logged on user. You should put that in the autostart group for every user.
0
 

Author Comment

by:jerzyb
ID: 9922973
There are lots of users.
The job of the background process is to do heavy calculations (which have nothing to do with the logged in user) and suspend itself if a keyboard/mouse activity detected (to minimize any effects on the user).

The process is already LOW priority but sometimes its not enough.

The screensavers somehow do it so, there must be a different way.

Jerzy
0
 
LVL 86

Expert Comment

by:jkr
ID: 9922995
>>The screensavers somehow do it

Yes, but they are launched in the user's context...
0
MS Dynamics Made Instantly Simpler

Make Your Microsoft Dynamics Investment Count  & Drastically Decrease Training Time by Providing Intuitive Step-By-Step WalkThru Tutorials.

 

Author Comment

by:jerzyb
ID: 9923590
OK, since my program runs with admin priviledges, can I somehow start my key/mouse detection program in the context of the logged-in user?
That is, the program will monitor who is logged in and restart appropriately (but how?).
0
 
LVL 86

Accepted Solution

by:
jkr earned 500 total points
ID: 9923704
>>can I somehow start my key/mouse detection program in the context of the logged-in user?

Yes, you can, but it ain't that simple :o)

You have to obtain a token from a process the user is running (explorer.exe is a good choice <s>). You'll need the PID to call 'OpenProcess()', then 'OpenProcessToken()' with 'TOKEN_IMPERSONATE' access. See http://support.microsoft.com/default.aspx?scid=kb;EN-US;q175030 ("HOWTO: Enumerate Applications Using Win32 APIs (Q175030)") on how to get that PID - it comes with sample code. Then, once you have the token, you have to call 'ImpersonateLoggedOnUser()', et voilà, you are there - either launch the program or do that inside the program before initializing the hooks.
0
 
LVL 8

Expert Comment

by:adg080898
ID: 9953760
You are using #pragma data_seg()

You must use #pragma bss_seg because the shared variables are uninitialized. You are initializing them to zero, but zero doesn't count as initializing. Instead of #pragma data_seg use #pragma bss_seg, or even both (haven't tried both).

Sections:
text_seg: program code
data_seg: NONZERO initialized variables
bss_seg: zero initialized variables

The idea is that the linker does not neet to write the bss_seg content to the output file because it is understood to be all-zeros, reducing executable size.

Use DUMPBIN in the msvc tools directory to verify that the output file does contain a .hookdat section.
0

Featured Post

MS Dynamics Made Instantly Simpler

Make Your Microsoft Dynamics Investment Count  & Drastically Decrease Training Time by Providing Intuitive Step-By-Step WalkThru Tutorials.

Question has a verified solution.

If you are experiencing a similar issue, please ask a related question

Suggested Solutions

Title # Comments Views Activity
Clean Uninstall of VS Professional 2015 5 59
Need to create an object factory 2 49
Strange - Deleted Resource Info appears in Azure Portal 6 95
Best book to learn C++ 4 87
This article describes how to programmatically preset the "Pages per Sheet" option that's available with most printer drivers.   This setting lets you do "n-Up" printing, where two, four, or more pages are printed on each sheet of paper. If your …
This article describes how to add a user-defined command button to the Windows 7 Explorer toolbar.  In the previous article (http://www.experts-exchange.com/A_2172.html), we saw how to put the Delete button back there where it belongs.  "Delete" is …
This is Part 3 in a 3-part series on Experts Exchange to discuss error handling in VBA code written for Excel. Part 1 of this series discussed basic error handling code using VBA. http://www.experts-exchange.com/videos/1478/Excel-Error-Handlin…

713 members asked questions and received personalized solutions in the past 7 days.

Join the community of 500,000 technology professionals and ask your questions.

Join & Ask a Question