这是测试代码,我试图了解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
好吗?
所以,如果有人能帮助我 - 那就太棒了。
答案 0 :(得分:0)
只有在调用线程处于“alertable state”时才会调用回调。 基本上,您需要使用alertable flag = true调用其中一个扩展等待函数,例如WaitForSingleObjectEx。