IcmpSendEcho2异步回调无法正常工作(c ++,vs2010,win7)

时间:2014-06-24 05:24:49

标签: c++ callback icmp icmpsendecho2

这是测试代码,我试图了解IcmpSendEcho2的回调;我的回调不会触发。 我试图将icmp函数移动到其他线程(link),但没有结果。 我尝试使用XP回调样式(这里我对win7什么都没有感到惊讶)

#include <WinSock2.h>  //includes Windows.h
#include <Ws2tcpip.h>

/**
    struct from WinDDK Wdm.h (include Wdm.h, Ntddk.h, or Ntifs.h)
*/
typedef struct _IO_STATUS_BLOCK {
  union {
    NTSTATUS Status;
    PVOID    Pointer;
  };
  ULONG_PTR Information;
} IO_STATUS_BLOCK, *PIO_STATUS_BLOCK;

/**
    typedef for PIO_APC_ROUTINE from MSDN
*/
typedef
VOID
(NTAPI *PIO_APC_ROUTINE) (
    IN PVOID ApcContext,
    IN PIO_STATUS_BLOCK IoStatusBlock,
    IN ULONG Reserved
    );
//defined
#define PIO_APC_ROUTINE_DEFINED

#include <iphlpapi.h>
#include <icmpapi.h>
#include <stdio.h>

#include <string>
using namespace std;

#pragma comment(lib, "Ws2_32.lib")
#pragma comment(lib, "iphlpapi.lib")

bool    icmpLoop = false;

VOID NTAPI callbackVistaPlus(PVOID ApcContext,PIO_STATUS_BLOCK IoStatusBlock,ULONG Reserved)
{
    MessageBoxA(0,"wow!","msg",MB_OK);
}

DWORD WINAPI IcmpThread(PVOID Parameter)
{
    unsigned long _serverAddr = inet_addr("192.168.1.2"); //my linux machine
    HANDLE _hIcmp = IcmpCreateFile();
    bool init = true;
    if (_hIcmp == INVALID_HANDLE_VALUE)
    {
        printf("cant icmpcreatefile\n");
        return 0;
    }
    //for test i create large buffer. it's testing code, not real
    char * _replysBuffer = (char*)malloc(sizeof(ICMP_ECHO_REPLY)*1000+8); 
    if (!_replysBuffer)
    {
        printf("cant buffer\n");
        return 0;
    }
    char * data = "hi all"; //my testing data to see with tcpdump at linux machine
//  bool loop = true;
    DWORD ret = IcmpSendEcho2(
            _hIcmp,
            0,
            callbackVistaPlus,
            0,
            _serverAddr,
            data,
            strlen(data),
            NULL,
            _replysBuffer,
            sizeof(ICMP_ECHO_REPLY)*1000+8,
            1000);
    /**/
    printf("ping sended\n");
    getchar();
//  while (icmpLoop){Sleep(100);}
    printf("exit\n");
    IcmpCloseHandle(_hIcmp);
}

int main(int argc, char* argv[])
{
    icmpLoop = true;
//  HANDLE hThread = CreateThread(NULL,0,IcmpThread,NULL,0,NULL);
    IcmpThread(0);
//  getchar();
//  icmpLoop = false;
    getchar();
    return 0;
}

我完全确定replyBuffer的正确存在,并且IcmpCloseHandle在我得到回复之前没有关闭(循环为线程样式,getchar()“暂停”为非线程)

我在linux机器上检查tcpdump(我用它来发送回声),我看到传入和传出的ICMP数据包与我的数据 - 它正在工作! 我调试已编译的代码,并在replyBuffer之后查看IcmpSendEcho2对我的数据的更改。 所以.. ICMP确实有效,但不是回调。

我不想转向事件风格,因为它会给我的代码带来问题

我对下一个情况感到惊讶: MSDN:

When called asynchronously, the IcmpSendEcho2 function 
returns ERROR_IO_PENDING to indicate the operation is in progress. 

IcmpSendEcho2返回0,GetLastError()返回ERROR_IO_PENDING 好吗?

所以,如果有人能帮助我 - 那就太棒了。

1 个答案:

答案 0 :(得分:0)

只有在调用线程处于“alertable state”时才会调用回调。 基本上,您需要使用alertable flag = true调用其中一个扩展等待函数,例如WaitForSingleObjectEx