?? listendlg.cpp
字號:
// listenDlg.cpp : implementation file
//
#include "stdafx.h"
#include "listen.h"
#include "listenDlg.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
/////////////////////////////////////////////////////////////////////////////
// CListenDlg dialog
BOOL b_IsRun=FALSE;
UINT RecvIpPro(LPVOID lpParam);
char* GetIp();
CString GetNowTime();
BOOL CreateLogFile(char *strInfo,char *strFileName="log.txt");
BOOL CALLBACK DialogProc(HWND hwndDlg,UINT uMsg,
WPARAM wParam,LPARAM lParam);
CListenDlg::CListenDlg(CWnd* pParent /*=NULL*/)
: CDialog(CListenDlg::IDD, pParent)
{
//{{AFX_DATA_INIT(CListenDlg)
m_iWprot = 0;
//}}AFX_DATA_INIT
// Note that LoadIcon does not require a subsequent DestroyIcon in Win32
m_IsAllPort=TRUE;
m_IsAllIp=TRUE;
m_IsAllProto=TRUE;
m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}
void CListenDlg::DoDataExchange(CDataExchange* pDX)
{
CDialog::DoDataExchange(pDX);
//{{AFX_DATA_MAP(CListenDlg)
DDX_Control(pDX, IDC_LIST1, m_NewList);
DDX_Text(pDX, IDC_EDIT1, m_iWprot);
DDV_MinMaxInt(pDX, m_iWprot, 0, 10000);
//}}AFX_DATA_MAP
}
BEGIN_MESSAGE_MAP(CListenDlg, CDialog)
//{{AFX_MSG_MAP(CListenDlg)
ON_WM_PAINT()
ON_WM_QUERYDRAGICON()
ON_BN_CLICKED(IDC_BUTTON1, OnButton1)
ON_BN_CLICKED(IDC_BUTTON3, OnButton3)
ON_BN_CLICKED(IDC_RADIO2, OnRadio2)
ON_BN_CLICKED(IDC_RADIO3, OnRadio3)
ON_BN_CLICKED(IDC_RADIO1, OnRadio1)
ON_BN_CLICKED(IDC_RADIO4, OnRadio4)
ON_BN_CLICKED(IDC_RADIO5, OnRadio5)
ON_BN_CLICKED(IDC_RADIO6, OnRadio6)
ON_BN_CLICKED(IDC_RADIO7, OnRadio7)
ON_BN_CLICKED(IDC_RADIO8, OnRadio8)
ON_COMMAND(ID_COPY_SIP, OnCopySip)
ON_COMMAND(ID_COPY_DIP, OnCopyDip)
ON_WM_SYSCOMMAND()
ON_LBN_SELCHANGE(IDC_LIST1, OnSelchangeList1)
ON_WM_DESTROY()
//}}AFX_MSG_MAP
END_MESSAGE_MAP()
/////////////////////////////////////////////////////////////////////////////
// CListenDlg message handlers
BOOL CListenDlg::OnInitDialog()
{
CDialog::OnInitDialog();
// Set the icon for this dialog. The framework does this automatically
// when the application's main window is not a dialog
SetIcon(m_hIcon, TRUE); // Set big icon
SetIcon(m_hIcon, FALSE); // Set small icon
// TODO: Add extra initialization here
if(!InitSock())
return FALSE;
//添加系統菜單
CMenu *pMenu=GetSystemMenu(FALSE);
AppendMenu(pMenu->m_hMenu,MF_SEPARATOR,0,NULL);
AppendMenu(pMenu->m_hMenu,MF_STRING,IDM_SYSMENU,"關于...");
//初始化控件
((CButton *)GetDlgItem(IDC_RADIO2))->SetCheck(TRUE);
((CButton *)GetDlgItem(IDC_RADIO1))->SetCheck(TRUE);
((CButton *)GetDlgItem(IDC_RADIO5))->SetCheck(TRUE);
//初始化臨界區對象
InitializeCriticalSection(&m_ls);
//建立顯示IP包窗口
m_IpShowDlg.Create(IDD_IPDATA,this);
m_IpShowDlg.ShowWindow(SW_SHOW);
CRect rect;
GetClientRect(&rect);
// ClientToScreen(&rect);
m_IpShowDlg.MoveWindow(10,265,rect.right-20,100);
return TRUE; // return TRUE unless you set the focus to a control
}
// If you add a minimize button to your dialog, you will need the code below
// to draw the icon. For MFC applications using the document/view model,
// this is automatically done for you by the framework.
void CListenDlg::OnPaint()
{
if (IsIconic())
{
CPaintDC dc(this); // device context for painting
SendMessage(WM_ICONERASEBKGND, (WPARAM) dc.GetSafeHdc(), 0);
// Center icon in client rectangle
int cxIcon = GetSystemMetrics(SM_CXICON);
int cyIcon = GetSystemMetrics(SM_CYICON);
CRect rect;
GetClientRect(&rect);
int x = (rect.Width() - cxIcon + 1) / 2;
int y = (rect.Height() - cyIcon + 1) / 2;
// Draw the icon
dc.DrawIcon(x, y, m_hIcon);
}
else
{
CDialog::OnPaint();
}
}
// The system calls this to obtain the cursor to display while the user drags
// the minimized window.
HCURSOR CListenDlg::OnQueryDragIcon()
{
return (HCURSOR) m_hIcon;
}
void CListenDlg::OnButton1()
{
// TODO: Add your control notification handler code here
CWinThread *pThread=NULL;
CButton* pBtn=(CButton *)GetDlgItem(IDC_BUTTON1);
int n=pBtn->GetCheck();
m_HostIp=GetIp(); //得到本機IP
if(1==n) //開始監聽
{
if(CreateSock()!=0) //建立
{
AfxMessageBox("WinSock設置失敗");
DestroyWindow();
}
else
{
b_IsRun=TRUE;
pThread=AfxBeginThread(RecvIpPro,this);
}
pBtn->SetWindowText("停止監聽");
}
else //停止監聽
{
b_IsRun=FALSE;
if(pThread)
{
TerminateThread(pThread->m_hThread, 0);
CloseHandle(pThread->m_hThread);
}
if(m_RawSock)
closesocket(m_RawSock);
pBtn->SetWindowText("開始監聽");
}
}
//在線程沖中不斷接受IP包
UINT RecvIpPro(LPVOID lpParam)
{
int err=0;
int recvLen;
CListenDlg* pDlg=(CListenDlg*)lpParam;
char recvBuf[MAX_PACK_LEN]={0};
while(b_IsRun)
{
recvLen=recv(pDlg->m_RawSock,recvBuf,MAX_PACK_LEN,0);//接受數據
if(err==SOCKET_ERROR)
{
if(WSAGetLastError()==WSAEWOULDBLOCK) //如果錯誤為阻塞那么將繼續接受
continue;
break;
}
//處理已經接受到的IP包
EnterCriticalSection(&pDlg->m_ls);
pDlg->SplitIpPack(recvBuf,recvLen); //分解IP包
pDlg->ShowIpInfo(recvBuf,recvLen); //顯示數據
LeaveCriticalSection(&pDlg->m_ls);
}
return 0;
}
//加載SOCK
BOOL CListenDlg::InitSock()
{
WSADATA wsaData;
int err;
if((err=WSAStartup(MAKEWORD(2,2),&wsaData))!=0) //加載失敗
return FALSE;
if (LOBYTE(wsaData.wVersion)!=2 || HIBYTE(wsaData.wVersion) != 2)
{
WSACleanup();//未能要求到所需的版本
return FALSE;
}
return TRUE;
}
//建立原始套接字
int CListenDlg::CreateSock()
{
int err;
char name[10];
hostent *pHostent;
int port=8310;
SOCKADDR_IN sockaddr;
m_RawSock=socket(AF_INET,SOCK_RAW,IPPROTO_IP);
if(m_RawSock==INVALID_SOCKET)
{
return WSAGetLastError(); //socket創建失敗
}
err=gethostname(name,10);
if(err==SOCKET_ERROR)
{
return WSAGetLastError(); //取本機用戶名失敗
}
pHostent=gethostbyname (name);
sockaddr.sin_family=AF_INET;
sockaddr.sin_port=htons(port);
memcpy(&sockaddr.sin_addr.s_addr,pHostent->h_addr_list[0],
pHostent->h_length);
//free(pHostent);
err=bind(m_RawSock,(SOCKADDR *)&sockaddr,sizeof(sockaddr));//綁定
if(err==SOCKET_ERROR)
{
return WSAGetLastError(); //取本機用戶名失敗
}
BOOL bOptval=TRUE;
//設置套節字選項
setsockopt(m_RawSock,SOL_SOCKET,SO_REUSEADDR,(char*)&bOptval,sizeof(bOptval));
err=setsockopt(m_RawSock,IPPROTO_IP,IP_HDRINCL,(char*)&bOptval, //IP_HDRINCL該選項使之能操作IP頭
sizeof(bOptval));
if(err==SOCKET_ERROR)
{
return WSAGetLastError(); //設置套節字選項失敗
}
//把網卡置于混雜模式。獲取IO操作的相關信息
DWORD dwBufferLen[10] ;
DWORD dwBufferInLen = 1 ;
DWORD dwBytesReturned = 0 ;
err=WSAIoctl (
m_RawSock,
SIO_RCVALL, //設置SOCK_RAW為SIO_RCVALL,以便接收所有的IP包
&dwBufferInLen,
sizeof(dwBufferInLen),
&dwBufferLen,
sizeof(dwBufferLen),
&dwBytesReturned,
NULL,
NULL
);//最后兩參數設置非阻塞
if(err==SOCKET_ERROR)
{
return WSAGetLastError(); //設置套節字選項失敗
}
return 0;
}
//分解IP包
void CListenDlg::SplitIpPack(char *pData, int len)
{
IP_HEADER* pIpHead;
pIpHead=(IP_HEADER*)pData; //轉為IP頭結構
SOCKADDR_IN saSource,saDest; //中間變量分別為源IP地址結構和目標IP地址結構
//得到包中協議
int iProtocol;
iProtocol=pIpHead->proto;
//得到協議字符形式表示m_szProtocol
strncpy(m_szProtocol,CheckProtocol(iProtocol),MAX_PROTO_TEXT_LEN);
//得到源IP
saSource.sin_addr.s_addr=pIpHead->sourceIP;
strncpy(m_szSourceIP,inet_ntoa(saSource.sin_addr),MAX_ADDR_LEN);
//得到目標IP
saDest.sin_addr.s_addr=pIpHead->destIP;
strncpy(m_szDestIP,inet_ntoa(saDest.sin_addr),MAX_ADDR_LEN);
//得到TTL
int ttl;
ttl=pIpHead->ttl;
wsprintf(m_szTTL,"%d",ttl);
//得到頭長度
int ihLen=sizeof(unsigned long)*(pIpHead->h_lenver & 0xf);
m_ihLen=ihLen;
unsigned short srcPort=0,destPort=0;
switch(iProtocol) {
case IPPROTO_TCP:
TCP_HEADER* pTcpHead;
pTcpHead=(TCP_HEADER*)(pData+ihLen);
srcPort=ntohs(pTcpHead->th_sport);
destPort=ntohs(pTcpHead->th_dport);
wsprintf(m_szSourcePort,"%d",srcPort);
wsprintf(m_szDestPort,"%d",destPort);
break;
case IPPROTO_UDP:
UDP_HEADER* pUdpHead;
pUdpHead=(UDP_HEADER*)(pData+ihLen);
srcPort=ntohs(pUdpHead->uh_sport);
destPort=ntohs(pUdpHead->uh_dport);
wsprintf(m_szSourcePort,"%d",srcPort);
wsprintf(m_szDestPort,"%d",destPort);
break;
case IPPROTO_ICMP:
ICMP_HEADER* pIcmpHead;
pIcmpHead=(ICMP_HEADER*)(pData+ihLen);
wsprintf(m_szSourcePort,"%s","");
wsprintf(m_szDestPort,"%s","");
break;
default:
wsprintf(m_szSourcePort,"%s","");
wsprintf(m_szDestPort,"%s","");
}
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -