From fdde826976bb790f3d01ed413117b4370869b0d8 Mon Sep 17 00:00:00 2001
From: - <->
Date: Sun, 12 Jan 2014 21:34:53 +0100
Subject: [PATCH] WinMTR (Redux) v1.0 ------------------- + added IPv6 support
- removed Windows 2000 support (kind of... can be added again with ease I
guess) + hop information / properties can still be viewed if state is
"stopped" or "stopping" * added start delay of about 30ms before starting to
ping next hop (should improve performance) ! fixed trace/hop list freeze (one
of them being the use of _beginthread and not _beginthreadex and waiting for
it's handles...) * theme support (now uses OS's default theme instead of
Win98 ! other minor or critical fixes (such as leaks)
---
README.md | 8 +-
src/WinMTR.rc | 188 +++++-----------
src/WinMTRDialog.cpp | 402 ++++++++++++++--------------------
src/WinMTRDialog.h | 11 +-
src/WinMTRGlobal.cpp | 4 +-
src/WinMTRGlobal.h | 11 +-
src/WinMTRMain.cpp | 26 ++-
src/WinMTRNet.cpp | 466 ++++++++++++++++++++++++----------------
src/WinMTRNet.h | 51 +++--
src/WinMTRProperties.h | 2 +-
src/WinMTRStatusBar.cpp | 6 +-
src/resource.h | 2 +-
12 files changed, 573 insertions(+), 604 deletions(-)
diff --git a/README.md b/README.md
index b8a8d8d..6af5ee9 100644
--- a/README.md
+++ b/README.md
@@ -18,4 +18,10 @@ this should improve performance and reduces network load*
- `[ ]` + host history: pressing del key or right mouse will remove selected entry
- `[ ]` * new icon
-~~~~
+### About me / why I decided to create this fork
+There isn't that much to say actually, I've been using IPv6 for a few years now thanks to [**SixXS**](http://sixxs.net/) and it always annoyed me that WinMTR couldn't handle IPv6... finally my ISP got some sort of IPv6 beta test. And that's what I wanted to compare, native vs SixXS with long-term trace routes such as those WinMTR provides.
+Since there wasn't any WinMTR build with IPv6, I decided to do it myself ;) The result can be seen here :P
+
+**If you're looking for an alternative** *(not meant for long-term traces)* there's [**vTrace**](vtrace.pl).
+It's some really interesting piece of Software ;) *(with more then just trace routes)*
+~~~~
\ No newline at end of file
diff --git a/src/WinMTR.rc b/src/WinMTR.rc
index 20399b0..b3cc862 100644
--- a/src/WinMTR.rc
+++ b/src/WinMTR.rc
@@ -70,24 +70,25 @@ IDR_MAINFRAME ICON "WinMTR.ico"
// Dialog
//
-IDD_WINMTR_DIALOG DIALOGEX 0, 0, 390, 193
+IDD_WINMTR_DIALOG DIALOGEX 0, 0, 420, 220
STYLE DS_SETFONT | DS_3DLOOK | DS_CENTER | WS_MINIMIZEBOX | WS_MAXIMIZEBOX | WS_POPUP | WS_VISIBLE | WS_CAPTION | WS_SYSMENU | WS_THICKFRAME
EXSTYLE WS_EX_APPWINDOW
FONT 8, "MS Sans Serif", 0, 0, 0x1
BEGIN
- DEFPUSHBUTTON "&Start",ID_RESTART,203,12,50,14,BS_FLAT
- PUSHBUTTON "&Options",ID_OPTIONS,270,12,51,14,BS_FLAT
- PUSHBUTTON "E&xit",IDCANCEL,328,12,49,14,BS_FLAT
- PUSHBUTTON "&Copy Text to clipboard",ID_CTTC,14,39,89,14,BS_FLAT
- PUSHBUTTON "Co&py HTML to clipboard",ID_CHTC,113,39,89,14,BS_FLAT
- PUSHBUTTON "Export &TEXT",ID_EXPT,270,39,51,14,BS_FLAT
- PUSHBUTTON "Export &HTML",ID_EXPH,328,39,49,14,BS_FLAT
- CONTROL "List1",IDC_LIST_MTR,"SysListView32",LVS_REPORT | LVS_SINGLESEL | LVS_NOSORTHEADER | WS_BORDER | WS_TABSTOP,7,60,376,126
- LTEXT "Host:",IDC_STATIC,13,12,20,13,SS_CENTERIMAGE
- GROUPBOX "",IDC_STATIC,7,3,251,29,BS_FLAT
- GROUPBOX "",IDC_STATICS,263,3,120,29,BS_FLAT
- GROUPBOX "",IDC_STATICJ,7,31,376,26,BS_FLAT
- COMBOBOX IDC_COMBO_HOST,35,12,164,73,CBS_DROPDOWN | CBS_AUTOHSCROLL | WS_VSCROLL | WS_TABSTOP
+ DEFPUSHBUTTON "&Start",ID_RESTART,235,10,50,14,BS_FLAT
+ PUSHBUTTON "&Options",ID_OPTIONS,333,10,41,14,BS_FLAT
+ PUSHBUTTON "E&xit",IDCANCEL,379,10,30,14,BS_FLAT
+ PUSHBUTTON "&Copy Text to clipboard",ID_CTTC,12,37,89,14,BS_FLAT
+ PUSHBUTTON "Co&py HTML to clipboard",ID_CHTC,111,37,89,14,BS_FLAT
+ PUSHBUTTON "Export &TEXT",ID_EXPT,301,37,51,14,BS_FLAT
+ PUSHBUTTON "Export &HTML",ID_EXPH,359,37,49,14,BS_FLAT
+ CONTROL "List1",IDC_LIST_MTR,"SysListView32",LVS_REPORT | LVS_SINGLESEL | LVS_NOSORTHEADER | WS_BORDER | WS_TABSTOP,5,58,409,157
+ LTEXT "Host:",IDC_STATIC,11,10,20,13,SS_LEFT | SS_CENTERIMAGE
+ GROUPBOX "",IDC_STATIC,5,0,287,30,BS_FLAT
+ GROUPBOX "",IDC_STATICS,295,0,120,30,BS_FLAT
+ GROUPBOX "",IDC_STATICJ,5,29,409,26,BS_FLAT
+ COMBOBOX IDC_COMBO_HOST,33,10,198,73,CBS_DROPDOWN | CBS_AUTOHSCROLL | WS_VSCROLL | WS_TABSTOP
+ AUTO3STATE "IPv6",IDC_CHECK_IPV6,301,14,31,8
END
IDD_DIALOG_OPTIONS DIALOGEX 0, 0, 251, 164
@@ -106,11 +107,11 @@ BEGIN
CONTROL "Resolve names",IDC_CHECK_DNS,"Button",BS_AUTOCHECKBOX | WS_TABSTOP,167,119,63,8
GROUPBOX "",IDC_STATIC,7,7,237,82,BS_FLAT
CTEXT "Get Fully Managed Dedicated and Cloud hosting in Europe and USA at killer prices.",IDC_STATIC,31,35,201,17
- CTEXT "WinMTR v0.92 is offered under GPL V2 by Appnor MSP",IDC_STATIC,42,14,194,17
+ CTEXT "WinMTR (Redux) v1.00 is offered under GPLv2",IDC_STATIC,42,14,194,17
PUSHBUTTON "&License and Credits",ID_LICENSE,81,67,94,14,BS_FLAT
LTEXT "Max. hosts in LRU list:",IDC_STATIC,15,119,74,10,NOT WS_GROUP
EDITTEXT IDC_EDIT_MAX_LRU,90,116,34,13,ES_AUTOHSCROLL
- CONTROL "www.appnor.com",IDC_MFCLINK1,"MfcLink",WS_TABSTOP,102,51,60,11
+ LTEXT "www.appnor.com",IDC_STATIC,102,51,60,11,NOT WS_GROUP
END
IDD_DIALOG_LICENSE DIALOGEX 0, 0, 175, 70
@@ -118,40 +119,41 @@ STYLE DS_SETFONT | DS_MODALFRAME | WS_POPUP | WS_CAPTION | WS_SYSMENU
CAPTION "License & Credits"
FONT 8, "MS Sans Serif", 0, 0, 0x0
BEGIN
- DEFPUSHBUTTON "OK",IDOK,59,43,50,14
- LTEXT "Read more on",IDC_STATIC,36,27,46,8
- CONTROL "www.winmtr.net",IDC_MFCLINK1,"MfcLink",WS_TABSTOP,83,26,60,14
- LTEXT "WinMTR is licensed under GPL V2.",IDC_STATIC,33,14,113,8
+ DEFPUSHBUTTON "OK",IDOK,62,36,50,14
+ LTEXT "Read more on",IDC_STATIC,36,21,46,8
+ LTEXT "www.winmtr.net",IDC_STATIC,85,21,60,8
+ CTEXT "WinMTR (Redux) is licensed under GPLv2.",IDC_STATIC,2,9,171,8,SS_CENTER
+ CTEXT "https://github.com/White-Tiger/WinMTR",IDC_STATIC,7,57,161,8,SS_CENTER
END
-IDD_DIALOG_PROPERTIES DIALOG 0, 0, 201, 159
+IDD_DIALOG_PROPERTIES DIALOG 0, 0, 282, 160
STYLE DS_SETFONT | DS_MODALFRAME | WS_POPUP | WS_CAPTION | WS_SYSMENU
CAPTION "Host properties"
FONT 8, "MS Sans Serif"
BEGIN
- DEFPUSHBUTTON "OK",IDOK,71,138,50,14,BS_FLAT
+ DEFPUSHBUTTON "OK",IDOK,116,140,50,14,BS_FLAT
LTEXT "Name:",IDC_STATIC,15,18,24,8
- EDITTEXT IDC_EDIT_PHOST,48,16,136,12,ES_RIGHT | ES_AUTOHSCROLL | ES_READONLY
+ EDITTEXT IDC_EDIT_PHOST,48,16,219,12,ES_RIGHT | ES_AUTOHSCROLL | ES_READONLY
LTEXT "IP Address:",IDC_STATIC,14,32,40,9
- EDITTEXT IDC_EDIT_PIP,58,31,126,12,ES_RIGHT | ES_AUTOHSCROLL | ES_READONLY
- GROUPBOX "Packets",IDC_STATIC,7,74,89,60,BS_FLAT
- GROUPBOX "Ping time",IDC_STATIC,108,74,86,60,BS_FLAT
- GROUPBOX "Host",IDC_STATIC,7,7,187,62,BS_FLAT
- LTEXT "Loss (%):",IDC_STATIC,13,117,31,9
- LTEXT "Received:",IDC_STATIC,13,101,36,9
- LTEXT "Sent:",IDC_STATIC,13,85,21,9
- LTEXT "Last:",IDC_STATIC,114,84,18,9
- LTEXT "Best:",IDC_STATIC,114,96,20,8
- LTEXT "Average:",IDC_STATIC,114,107,32,8
- LTEXT "Worst:",IDC_STATIC,114,118,23,8
- EDITTEXT IDC_EDIT_PSENT,53,82,35,12,ES_RIGHT | ES_AUTOHSCROLL | ES_READONLY | ES_NUMBER
- EDITTEXT IDC_EDIT_PRECV,53,99,35,12,ES_RIGHT | ES_AUTOHSCROLL | ES_READONLY | ES_NUMBER
- EDITTEXT IDC_EDIT_PLOSS,53,116,35,12,ES_RIGHT | ES_AUTOHSCROLL | ES_READONLY | ES_NUMBER
- EDITTEXT IDC_EDIT_PLAST,150,81,34,12,ES_RIGHT | ES_AUTOHSCROLL | ES_READONLY
- EDITTEXT IDC_EDIT_PBEST,150,94,34,12,ES_RIGHT | ES_AUTOHSCROLL | ES_READONLY
- EDITTEXT IDC_EDIT_PAVRG,150,106,34,12,ES_RIGHT | ES_AUTOHSCROLL | ES_READONLY
- EDITTEXT IDC_EDIT_PWORST,150,118,34,12,ES_RIGHT | ES_AUTOHSCROLL | ES_READONLY
- EDITTEXT IDC_EDIT_PCOMMENT,14,50,170,12,ES_AUTOHSCROLL | ES_READONLY
+ EDITTEXT IDC_EDIT_PIP,58,31,209,12,ES_RIGHT | ES_AUTOHSCROLL | ES_READONLY
+ GROUPBOX "Packets",IDC_STATIC,46,74,89,60,BS_FLAT
+ GROUPBOX "Ping time",IDC_STATIC,147,74,86,60,BS_FLAT
+ GROUPBOX "Host",IDC_STATIC,7,7,267,62,BS_FLAT
+ LTEXT "Loss (%):",IDC_STATIC,52,117,31,9
+ LTEXT "Received:",IDC_STATIC,52,101,36,9
+ LTEXT "Sent:",IDC_STATIC,52,85,21,9
+ LTEXT "Last:",IDC_STATIC,153,84,18,9
+ LTEXT "Best:",IDC_STATIC,153,96,20,8
+ LTEXT "Average:",IDC_STATIC,153,107,32,8
+ LTEXT "Worst:",IDC_STATIC,153,118,23,8
+ EDITTEXT IDC_EDIT_PSENT,92,82,35,12,ES_RIGHT | ES_AUTOHSCROLL | ES_READONLY | ES_NUMBER
+ EDITTEXT IDC_EDIT_PRECV,92,99,35,12,ES_RIGHT | ES_AUTOHSCROLL | ES_READONLY | ES_NUMBER
+ EDITTEXT IDC_EDIT_PLOSS,92,116,35,12,ES_RIGHT | ES_AUTOHSCROLL | ES_READONLY | ES_NUMBER
+ EDITTEXT IDC_EDIT_PLAST,189,81,34,12,ES_RIGHT | ES_AUTOHSCROLL | ES_READONLY
+ EDITTEXT IDC_EDIT_PBEST,189,94,34,12,ES_RIGHT | ES_AUTOHSCROLL | ES_READONLY
+ EDITTEXT IDC_EDIT_PAVRG,189,106,34,12,ES_RIGHT | ES_AUTOHSCROLL | ES_READONLY
+ EDITTEXT IDC_EDIT_PWORST,189,118,34,12,ES_RIGHT | ES_AUTOHSCROLL | ES_READONLY
+ EDITTEXT IDC_EDIT_PCOMMENT,14,50,253,12,ES_AUTOHSCROLL | ES_READONLY
END
IDD_DIALOG_HELP DIALOGEX 0, 0, 256, 122
@@ -160,8 +162,8 @@ CAPTION "WinMTR"
FONT 8, "MS Shell Dlg", 400, 0, 0x1
BEGIN
DEFPUSHBUTTON "OK",IDOK,144,101,50,14
- CONTROL "www.appnor.com",IDC_MFCLINK1,"MfcLink",WS_TABSTOP,187,8,60,11
- LTEXT "WinMTR v0.92 is offered under GPL V2 by Appnor MSP",IDC_STATIC,7,9,176,10
+ LTEXT "www.appnor.com",IDC_STATIC,187,9,60,11
+ LTEXT "WinMTR (Redux) v1.00 is offered under GPLv2",IDC_STATIC,7,9,176,10
LTEXT "Usage: WinMTR [options] target_host_name",IDC_STATIC,7,29,144,8
LTEXT "Options:",IDC_STATIC,7,39,28,8
LTEXT " --interval, -i VALUE. Set ping interval.",IDC_STATIC,26,47,131,8
@@ -178,8 +180,8 @@ END
//
VS_VERSION_INFO VERSIONINFO
- FILEVERSION 0,9,0,2
- PRODUCTVERSION 0,9,0,2
+ FILEVERSION 1,0,0,0
+ PRODUCTVERSION 1,0,0,0
FILEFLAGSMASK 0x3fL
#ifdef _DEBUG
FILEFLAGS 0x1L
@@ -194,13 +196,13 @@ BEGIN
BEGIN
BLOCK "040904b0"
BEGIN
- VALUE "FileDescription", "WinMTR by Appnor MSP - www.winmtr.net"
- VALUE "FileVersion", "0.9.0.2"
+ VALUE "FileDescription", "WinMTR Redux (original by Appnor MSP - www.winmtr.net)"
+ VALUE "FileVersion", "1.0.0.0"
VALUE "InternalName", "WinMTR"
VALUE "LegalCopyright", "WinMTR 0.9 (c) 2010-2011 Appnor MSP - Fully Managed Hosting & Cloud Provider www.appnor.com"
- VALUE "OriginalFilename", "WinMTR.EXE"
+ VALUE "OriginalFilename", "WinMTR.exe"
VALUE "ProductName", "WinMTR Application"
- VALUE "ProductVersion", "0.9.0.2"
+ VALUE "ProductVersion", "1.0.0.0"
END
END
BLOCK "VarFileInfo"
@@ -268,67 +270,6 @@ END
#endif // APSTUDIO_INVOKED
-/////////////////////////////////////////////////////////////////////////////
-//
-// Dialog Info
-//
-
-IDD_DIALOG_HELP DLGINIT
-BEGIN
- IDC_MFCLINK1, 0x37c, 238, 0
-0x4d3c, 0x4346, 0x694c, 0x6b6e, 0x555f, 0x6c72, 0x683e, 0x7474, 0x3a70,
-0x2f2f, 0x7777, 0x2e77, 0x7061, 0x6e70, 0x726f, 0x632e, 0x6d6f, 0x3f2f,
-0x7475, 0x5f6d, 0x6f73, 0x7275, 0x6563, 0x773d, 0x6e69, 0x746d, 0x2672,
-0x7475, 0x5f6d, 0x656d, 0x6964, 0x6d75, 0x643d, 0x7365, 0x746b, 0x706f,
-0x7526, 0x6d74, 0x635f, 0x6d61, 0x6170, 0x6769, 0x3d6e, 0x6f73, 0x7466,
-0x6177, 0x6572, 0x2f3c, 0x464d, 0x4c43, 0x6e69, 0x5f6b, 0x7255, 0x3e6c,
-0x4d3c, 0x4346, 0x694c, 0x6b6e, 0x555f, 0x6c72, 0x7250, 0x6665, 0x7869,
-0x3c3e, 0x4d2f, 0x4346, 0x694c, 0x6b6e, 0x555f, 0x6c72, 0x7250, 0x6665,
-0x7869, 0x3c3e, 0x464d, 0x4c43, 0x6e69, 0x5f6b, 0x6f54, 0x6c6f, 0x6974,
-0x3e70, 0x2f3c, 0x464d, 0x4c43, 0x6e69, 0x5f6b, 0x6f54, 0x6c6f, 0x6974,
-0x3e70, 0x4d3c, 0x4346, 0x694c, 0x6b6e, 0x465f, 0x6c75, 0x546c, 0x7865,
-0x5474, 0x6f6f, 0x746c, 0x7069, 0x463e, 0x4c41, 0x4553, 0x2f3c, 0x464d,
-0x4c43, 0x6e69, 0x5f6b, 0x7546, 0x6c6c, 0x6554, 0x7478, 0x6f54, 0x6c6f,
-0x6974, 0x3e70,
- 0
-END
-
-IDD_DIALOG_OPTIONS DLGINIT
-BEGIN
- IDC_MFCLINK1, 0x37c, 238, 0
-0x4d3c, 0x4346, 0x694c, 0x6b6e, 0x555f, 0x6c72, 0x683e, 0x7474, 0x3a70,
-0x2f2f, 0x7777, 0x2e77, 0x7061, 0x6e70, 0x726f, 0x632e, 0x6d6f, 0x3f2f,
-0x7475, 0x5f6d, 0x6f73, 0x7275, 0x6563, 0x773d, 0x6e69, 0x746d, 0x2672,
-0x7475, 0x5f6d, 0x656d, 0x6964, 0x6d75, 0x643d, 0x7365, 0x746b, 0x706f,
-0x7526, 0x6d74, 0x635f, 0x6d61, 0x6170, 0x6769, 0x3d6e, 0x6f73, 0x7466,
-0x6177, 0x6572, 0x2f3c, 0x464d, 0x4c43, 0x6e69, 0x5f6b, 0x7255, 0x3e6c,
-0x4d3c, 0x4346, 0x694c, 0x6b6e, 0x555f, 0x6c72, 0x7250, 0x6665, 0x7869,
-0x3c3e, 0x4d2f, 0x4346, 0x694c, 0x6b6e, 0x555f, 0x6c72, 0x7250, 0x6665,
-0x7869, 0x3c3e, 0x464d, 0x4c43, 0x6e69, 0x5f6b, 0x6f54, 0x6c6f, 0x6974,
-0x3e70, 0x2f3c, 0x464d, 0x4c43, 0x6e69, 0x5f6b, 0x6f54, 0x6c6f, 0x6974,
-0x3e70, 0x4d3c, 0x4346, 0x694c, 0x6b6e, 0x465f, 0x6c75, 0x546c, 0x7865,
-0x5474, 0x6f6f, 0x746c, 0x7069, 0x463e, 0x4c41, 0x4553, 0x2f3c, 0x464d,
-0x4c43, 0x6e69, 0x5f6b, 0x7546, 0x6c6c, 0x6554, 0x7478, 0x6f54, 0x6c6f,
-0x6974, 0x3e70,
- 0
-END
-
-IDD_DIALOG_LICENSE DLGINIT
-BEGIN
- IDC_MFCLINK1, 0x37c, 157, 0
-0x4d3c, 0x4346, 0x694c, 0x6b6e, 0x555f, 0x6c72, 0x3c3e, 0x4d2f, 0x4346,
-0x694c, 0x6b6e, 0x555f, 0x6c72, 0x3c3e, 0x464d, 0x4c43, 0x6e69, 0x5f6b,
-0x7255, 0x506c, 0x6572, 0x6966, 0x3e78, 0x2f3c, 0x464d, 0x4c43, 0x6e69,
-0x5f6b, 0x7255, 0x506c, 0x6572, 0x6966, 0x3e78, 0x4d3c, 0x4346, 0x694c,
-0x6b6e, 0x545f, 0x6f6f, 0x746c, 0x7069, 0x3c3e, 0x4d2f, 0x4346, 0x694c,
-0x6b6e, 0x545f, 0x6f6f, 0x746c, 0x7069, 0x3c3e, 0x464d, 0x4c43, 0x6e69,
-0x5f6b, 0x7546, 0x6c6c, 0x6554, 0x7478, 0x6f54, 0x6c6f, 0x6974, 0x3e70,
-0x4146, 0x534c, 0x3c45, 0x4d2f, 0x4346, 0x694c, 0x6b6e, 0x465f, 0x6c75,
-0x546c, 0x7865, 0x5474, 0x6f6f, 0x746c, 0x7069, "\076"
- 0
-END
-
-
/////////////////////////////////////////////////////////////////////////////
//
// String Table
@@ -337,7 +278,7 @@ END
STRINGTABLE
BEGIN
IDP_SOCKETS_INIT_FAILED "Windows sockets initialization failed."
- IDS_STRING_SB_NAME "WinMTR v0.92 GPL V2 by Appnor MSP - Fully Managed Hosting & Cloud Provider"
+ IDS_STRING_SB_NAME "WinMTR v1.00 GPLv2 (original by Appnor MSP - Fully Managed Hosting & Cloud Provider)"
IDS_STRING_SB_PING "0000000"
IDS_STRING_CLEAR_HISTORY "Clear History"
END
@@ -346,25 +287,8 @@ END
/////////////////////////////////////////////////////////////////////////////
-
-#ifndef APSTUDIO_INVOKED
-/////////////////////////////////////////////////////////////////////////////
//
-// Generated from the TEXTINCLUDE 3 resource.
+// Manifest resources
//
-#define _AFX_NO_SPLITTER_RESOURCES
-#define _AFX_NO_OLE_RESOURCES
-#define _AFX_NO_TRACKER_RESOURCES
-#define _AFX_NO_PROPERTY_RESOURCES
-
-#if !defined(AFX_RESOURCE_DLL) || defined(AFX_TARG_ENU)
-#ifdef _WIN32
-LANGUAGE 9, 1
-#pragma code_page(1252)
-#endif //_WIN32
-#include "afxres.rc" // Standard components
-#endif
-
-/////////////////////////////////////////////////////////////////////////////
-#endif // not APSTUDIO_INVOKED
-
+LANGUAGE LANG_NEUTRAL, SUBLANG_NEUTRAL
+1 RT_MANIFEST ".\\manifest.xml"
diff --git a/src/WinMTRDialog.cpp b/src/WinMTRDialog.cpp
index 9fcc82e..620c368 100644
--- a/src/WinMTRDialog.cpp
+++ b/src/WinMTRDialog.cpp
@@ -11,7 +11,6 @@
#include "WinMTRNet.h"
#include
#include
-#include "afxlinkctrl.h"
#define TRACE_MSG(msg) \
{ \
@@ -76,9 +75,12 @@ WinMTRDialog::WinMTRDialog(CWnd* pParent)
hasPingsizeFromCmdLine = false;
hasMaxLRUFromCmdLine = false;
hasUseDNSFromCmdLine = false;
+ hasUseIPv6FromCmdLine = false;
traceThreadMutex = CreateMutex(NULL, FALSE, NULL);
wmtrnet = new WinMTRNet(this);
+ if(!wmtrnet->hasIPv6) m_checkIPv6.EnableWindow(FALSE);
+ useIPv6=2;
}
WinMTRDialog::~WinMTRDialog()
@@ -99,6 +101,7 @@ void WinMTRDialog::DoDataExchange(CDataExchange* pDX)
DDX_Control(pDX, IDCANCEL, m_buttonExit);
DDX_Control(pDX, ID_RESTART, m_buttonStart);
DDX_Control(pDX, IDC_COMBO_HOST, m_comboHost);
+ DDX_Control(pDX, IDC_CHECK_IPV6, m_checkIPv6);
DDX_Control(pDX, IDC_LIST_MTR, m_listMTR);
DDX_Control(pDX, IDC_STATICS, m_staticS);
DDX_Control(pDX, IDC_STATICJ, m_staticJ);
@@ -115,11 +118,15 @@ void WinMTRDialog::DoDataExchange(CDataExchange* pDX)
BOOL WinMTRDialog::OnInitDialog()
{
CDialog::OnInitDialog();
+ if(!wmtrnet->initialized){
+ EndDialog(-1);
+ return TRUE;
+ }
#ifndef _WIN64
- char caption[] = {"WinMTR v0.92 32 bit by Appnor MSP - www.winmtr.net"};
+ char caption[] = {"WinMTR (Redux) v1.00 32bit"};
#else
- char caption[] = {"WinMTR v0.92 64 bit by Appnor MSP - www.winmtr.net"};
+ char caption[] = {"WinMTR (Redux) v1.00 64bit"};
#endif
SetTimer(1, WINMTR_DIALOG_TIMER, NULL);
@@ -136,22 +143,14 @@ BOOL WinMTRDialog::OnInitDialog()
sbi[0] = IDS_STRING_SB_NAME;
statusBar.SetIndicators( sbi,1);
statusBar.SetPaneInfo(0, statusBar.GetItemID(0),SBPS_STRETCH, NULL );
- { // Add appnor URL
- CMFCLinkCtrl* m_pWndButton = new CMFCLinkCtrl;
- if (!m_pWndButton->Create(_T("www.appnor.com"), WS_CHILD|WS_VISIBLE|WS_TABSTOP, CRect(0,0,0,0), &statusBar, 1234)) {
- TRACE(_T("Failed to create button control.\n"));
- return FALSE;
- }
-
- m_pWndButton->SetURL("http://www.appnor.com/?utm_source=winmtr&utm_medium=desktop&utm_campaign=software");
-
- if(!statusBar.AddPane(1234,1)) {
- AfxMessageBox(_T("Pane index out of range\nor pane with same ID already exists in the status bar"), MB_ICONERROR);
- return FALSE;
+
+ // create Appnor button
+ if(m_buttonAppnor.Create(_T("www.appnor.com"), WS_CHILD|WS_VISIBLE|WS_TABSTOP, CRect(0,0,0,0), &statusBar, 1234)){
+ m_buttonAppnor.SetURL("http://appnor.com/?utm_source=winmtr&utm_medium=desktop&utm_campaign=software");
+ if(statusBar.AddPane(1234,1)){
+ statusBar.SetPaneWidth(statusBar.CommandToIndex(1234),100);
+ statusBar.AddPaneControl(m_buttonAppnor,1234,true);
}
-
- statusBar.SetPaneWidth(statusBar.CommandToIndex(1234), 100);
- statusBar.AddPaneControl(m_pWndButton, 1234, true);
}
for(int i = 0; i< MTR_NR_COLS; i++)
@@ -215,27 +214,7 @@ BOOL WinMTRDialog::InitRegistry()
DWORD res, tmp_dword, value_size;
LONG r;
- r = RegCreateKeyEx( HKEY_CURRENT_USER,
- "Software",
- 0,
- NULL,
- REG_OPTION_NON_VOLATILE,
- KEY_ALL_ACCESS,
- NULL,
- &hKey,
- &res);
- if( r != ERROR_SUCCESS)
- return FALSE;
-
- r = RegCreateKeyEx( hKey,
- "WinMTR",
- 0,
- NULL,
- REG_OPTION_NON_VOLATILE,
- KEY_ALL_ACCESS,
- NULL,
- &hKey,
- &res);
+ r = RegCreateKeyEx(HKEY_CURRENT_USER,"Software\\WinMTR",0,NULL,0,KEY_ALL_ACCESS,NULL,&hKey,&res);
if( r != ERROR_SUCCESS)
return FALSE;
@@ -243,15 +222,7 @@ BOOL WinMTRDialog::InitRegistry()
RegSetValueEx(hKey,"License", 0, REG_SZ, (const unsigned char *)WINMTR_LICENSE, sizeof(WINMTR_LICENSE)+1);
RegSetValueEx(hKey,"HomePage", 0, REG_SZ, (const unsigned char *)WINMTR_HOMEPAGE, sizeof(WINMTR_HOMEPAGE)+1);
- r = RegCreateKeyEx( hKey,
- "Config",
- 0,
- NULL,
- REG_OPTION_NON_VOLATILE,
- KEY_ALL_ACCESS,
- NULL,
- &hKey_v,
- &res);
+ r = RegCreateKeyEx(hKey,"Config",0,NULL,0,KEY_ALL_ACCESS,NULL,&hKey_v,&res);
if( r != ERROR_SUCCESS)
return FALSE;
@@ -259,7 +230,7 @@ BOOL WinMTRDialog::InitRegistry()
tmp_dword = pingsize;
RegSetValueEx(hKey_v,"PingSize", 0, REG_DWORD, (const unsigned char *)&tmp_dword, sizeof(DWORD));
} else {
- if(!hasPingsizeFromCmdLine) pingsize = tmp_dword;
+ if(!hasPingsizeFromCmdLine) pingsize = (WORD)tmp_dword;
}
if(RegQueryValueEx(hKey_v, "MaxLRU", 0, NULL, (unsigned char *)&tmp_dword, &value_size) != ERROR_SUCCESS) {
@@ -275,6 +246,14 @@ BOOL WinMTRDialog::InitRegistry()
} else {
if(!hasUseDNSFromCmdLine) useDNS = (BOOL)tmp_dword;
}
+ if(RegQueryValueEx(hKey_v, "UseIPv6", 0, NULL, (unsigned char *)&tmp_dword, &value_size) != ERROR_SUCCESS) {
+ tmp_dword = useIPv6;
+ RegSetValueEx(hKey_v,"UseIPv6", 0, REG_DWORD, (const unsigned char *)&tmp_dword, sizeof(DWORD));
+ } else {
+ if(!hasUseIPv6FromCmdLine) useIPv6 = (unsigned char)tmp_dword;
+ if(useIPv6>2) useIPv6=1;
+ }
+ m_checkIPv6.SetCheck(useIPv6);
if(RegQueryValueEx(hKey_v, "Interval", 0, NULL, (unsigned char *)&tmp_dword, &value_size) != ERROR_SUCCESS) {
tmp_dword = (DWORD)(interval * 1000);
@@ -283,15 +262,7 @@ BOOL WinMTRDialog::InitRegistry()
if(!hasIntervalFromCmdLine) interval = (float)tmp_dword / 1000.0;
}
- r = RegCreateKeyEx( hKey,
- "LRU",
- 0,
- NULL,
- REG_OPTION_NON_VOLATILE,
- KEY_ALL_ACCESS,
- NULL,
- &hKey_v,
- &res);
+ r = RegCreateKeyEx(hKey,"LRU",0,NULL,0,KEY_ALL_ACCESS,NULL,&hKey_v,&res);
if( r != ERROR_SUCCESS)
return FALSE;
if(RegQueryValueEx(hKey_v, "NrLRU", 0, NULL, (unsigned char *)&tmp_dword, &value_size) != ERROR_SUCCESS) {
@@ -329,10 +300,10 @@ void WinMTRDialog::OnSizing(UINT fwSide, LPRECT pRect)
int iWidth = (pRect->right)-(pRect->left);
int iHeight = (pRect->bottom)-(pRect->top);
- if (iWidth < 600)
- pRect->right = pRect->left + 600;
- if (iHeight <250)
- pRect->bottom = pRect->top + 250;
+ if (iWidth<638)
+ pRect->right = pRect->left+638;
+ if (iHeight<388)
+ pRect->bottom = pRect->top+388;
}
@@ -341,50 +312,41 @@ void WinMTRDialog::OnSizing(UINT fwSide, LPRECT pRect)
//
//
//*****************************************************************************
+/// @todo (White-Tiger#1#): simplify it... use initial positions from "right" to calculate new position (no fix values here)
void WinMTRDialog::OnSize(UINT nType, int cx, int cy)
{
CDialog::OnSize(nType, cx, cy);
- CRect r;
- GetClientRect(&r);
- CRect lb;
+ CRect rct,lb;
+ if(!IsWindow(m_staticS.m_hWnd)) return;
+ GetClientRect(&rct);
+ m_staticS.GetWindowRect(&lb);
+ ScreenToClient(&lb);
+ m_staticS.SetWindowPos(NULL, lb.TopLeft().x, lb.TopLeft().y, rct.Width()-lb.TopLeft().x-8, lb.Height() , SWP_NOMOVE | SWP_NOZORDER);
+
+ m_staticJ.GetWindowRect(&lb);
+ ScreenToClient(&lb);
+ m_staticJ.SetWindowPos(NULL, lb.TopLeft().x, lb.TopLeft().y, rct.Width() - 16, lb.Height(), SWP_NOMOVE | SWP_NOZORDER);
+
+ m_buttonOptions.GetWindowRect(&lb);
+ ScreenToClient(&lb);
+ m_buttonOptions.SetWindowPos(NULL, rct.Width() - lb.Width()-52-16, lb.TopLeft().y, lb.Width(), lb.Height() , SWP_NOSIZE | SWP_NOZORDER);
+ m_buttonExit.GetWindowRect(&lb);
+ ScreenToClient(&lb);
+ m_buttonExit.SetWindowPos(NULL, rct.Width() - lb.Width()-16, lb.TopLeft().y, lb.Width(), lb.Height() , SWP_NOSIZE | SWP_NOZORDER);
- if (::IsWindow(m_staticS.m_hWnd)) {
- m_staticS.GetWindowRect(&lb);
- ScreenToClient(&lb);
- m_staticS.SetWindowPos(NULL, lb.TopLeft().x, lb.TopLeft().y, r.Width()-lb.TopLeft().x-10, lb.Height() , SWP_NOMOVE | SWP_NOZORDER);
- }
-
- if (::IsWindow(m_staticJ.m_hWnd)) {
- m_staticJ.GetWindowRect(&lb);
- ScreenToClient(&lb);
- m_staticJ.SetWindowPos(NULL, lb.TopLeft().x, lb.TopLeft().y, r.Width() - 21, lb.Height(), SWP_NOMOVE | SWP_NOZORDER);
- }
-
- if (::IsWindow(m_buttonExit.m_hWnd)) {
- m_buttonExit.GetWindowRect(&lb);
- ScreenToClient(&lb);
- m_buttonExit.SetWindowPos(NULL, r.Width() - lb.Width()-21, lb.TopLeft().y, lb.Width(), lb.Height() , SWP_NOSIZE | SWP_NOZORDER);
- }
-
- if (::IsWindow(m_buttonExpH.m_hWnd)) {
- m_buttonExpH.GetWindowRect(&lb);
- ScreenToClient(&lb);
- m_buttonExpH.SetWindowPos(NULL, r.Width() - lb.Width()-21, lb.TopLeft().y, lb.Width(), lb.Height() , SWP_NOSIZE | SWP_NOZORDER);
- }
- if (::IsWindow(m_buttonExpT.m_hWnd)) {
- m_buttonExpT.GetWindowRect(&lb);
- ScreenToClient(&lb);
- m_buttonExpT.SetWindowPos(NULL, r.Width() - lb.Width()- 103, lb.TopLeft().y, lb.Width(), lb.Height() , SWP_NOSIZE | SWP_NOZORDER);
- }
+ m_buttonExpH.GetWindowRect(&lb);
+ ScreenToClient(&lb);
+ m_buttonExpH.SetWindowPos(NULL, rct.Width() - lb.Width()-16, lb.TopLeft().y, lb.Width(), lb.Height() , SWP_NOSIZE | SWP_NOZORDER);
+ m_buttonExpT.GetWindowRect(&lb);
+ ScreenToClient(&lb);
+ m_buttonExpT.SetWindowPos(NULL, rct.Width() - lb.Width()- 103, lb.TopLeft().y, lb.Width(), lb.Height() , SWP_NOSIZE | SWP_NOZORDER);
- if (::IsWindow(m_listMTR.m_hWnd)) {
- m_listMTR.GetWindowRect(&lb);
- ScreenToClient(&lb);
- m_listMTR.SetWindowPos(NULL, lb.TopLeft().x, lb.TopLeft().y, r.Width() - 21, r.Height() - lb.top - 25, SWP_NOMOVE | SWP_NOZORDER);
- }
+ m_listMTR.GetWindowRect(&lb);
+ ScreenToClient(&lb);
+ m_listMTR.SetWindowPos(NULL, lb.TopLeft().x, lb.TopLeft().y, rct.Width() - 17, rct.Height() - lb.top - 25, SWP_NOMOVE | SWP_NOZORDER);
RepositionBars(AFX_IDW_CONTROLBAR_FIRST, AFX_IDW_CONTROLBAR_LAST,
- 0, reposQuery, r);
+ 0, reposQuery, rct);
RepositionBars(AFX_IDW_CONTROLBAR_FIRST, AFX_IDW_CONTROLBAR_LAST, 0);
@@ -435,47 +397,39 @@ HCURSOR WinMTRDialog::OnQueryDragIcon()
// WinMTRDialog::OnDblclkList
//
//*****************************************************************************
-void WinMTRDialog::OnDblclkList(NMHDR* pNMHDR, LRESULT* pResult)
+void WinMTRDialog::OnDblclkList(NMHDR* /*pNMHDR*/, LRESULT* pResult)
{
- *pResult = 0;
-
- if(state == TRACING) {
+ *pResult=0;
+ if(state==TRACING || state==IDLE || state==STOPPING) {
POSITION pos = m_listMTR.GetFirstSelectedItemPosition();
if(pos!=NULL) {
int nItem = m_listMTR.GetNextSelectedItem(pos);
WinMTRProperties wmtrprop;
-
- if(wmtrnet->GetAddr(nItem)==0) {
+
+ union{sockaddr* addr;sockaddr_in* addr4;sockaddr_in6* addr6;};
+ addr=wmtrnet->GetAddr(nItem);
+ if(!(addr4->sin_family==AF_INET&&addr4->sin_addr.s_addr) && !(addr6->sin6_family==AF_INET6&&(addr6->sin6_addr.u.Word[0]|addr6->sin6_addr.u.Word[1]|addr6->sin6_addr.u.Word[2]|addr6->sin6_addr.u.Word[3]|addr6->sin6_addr.u.Word[4]|addr6->sin6_addr.u.Word[5]|addr6->sin6_addr.u.Word[6]|addr6->sin6_addr.u.Word[7]))){
strcpy(wmtrprop.host,"");
strcpy(wmtrprop.ip,"");
wmtrnet->GetName(nItem, wmtrprop.comment);
-
- wmtrprop.pck_loss = wmtrprop.pck_sent = wmtrprop.pck_recv = 0;
-
- wmtrprop.ping_avrg = wmtrprop.ping_last = 0.0;
- wmtrprop.ping_best = wmtrprop.ping_worst = 0.0;
- } else {
+ }else{
wmtrnet->GetName(nItem, wmtrprop.host);
- int addr = wmtrnet->GetAddr(nItem);
- sprintf ( wmtrprop.ip , "%d.%d.%d.%d",
- (addr >> 24) & 0xff,
- (addr >> 16) & 0xff,
- (addr >> 8) & 0xff,
- addr & 0xff
- );
- strcpy(wmtrprop.comment , "Host alive.");
-
- wmtrprop.ping_avrg = (float)wmtrnet->GetAvg(nItem);
- wmtrprop.ping_last = (float)wmtrnet->GetLast(nItem);
- wmtrprop.ping_best = (float)wmtrnet->GetBest(nItem);
- wmtrprop.ping_worst = (float)wmtrnet->GetWorst(nItem);
-
- wmtrprop.pck_loss = wmtrnet->GetPercent(nItem);
- wmtrprop.pck_recv = wmtrnet->GetReturned(nItem);
- wmtrprop.pck_sent = wmtrnet->GetXmit(nItem);
+ if(getnameinfo(addr,sizeof(sockaddr_in6),wmtrprop.ip,40,NULL,0,NI_NUMERICHOST)){
+ *wmtrprop.ip='\0';
+ }
+ strcpy(wmtrprop.comment, "Host alive.");
}
+ wmtrprop.ping_avrg = (float)wmtrnet->GetAvg(nItem);
+ wmtrprop.ping_last = (float)wmtrnet->GetLast(nItem);
+ wmtrprop.ping_best = (float)wmtrnet->GetBest(nItem);
+ wmtrprop.ping_worst = (float)wmtrnet->GetWorst(nItem);
+
+ wmtrprop.pck_loss = wmtrnet->GetPercent(nItem);
+ wmtrprop.pck_recv = wmtrnet->GetReturned(nItem);
+ wmtrprop.pck_sent = wmtrnet->GetXmit(nItem);
+
wmtrprop.DoModal();
}
}
@@ -497,7 +451,7 @@ void WinMTRDialog::SetHostName(const char *host)
// WinMTRDialog::SetPingSize
//
//*****************************************************************************
-void WinMTRDialog::SetPingSize(int ps)
+void WinMTRDialog::SetPingSize(WORD ps)
{
pingsize = ps;
}
@@ -547,44 +501,35 @@ void WinMTRDialog::OnRestart()
}
CString sHost;
-
if(state == IDLE) {
m_comboHost.GetWindowText(sHost);
- sHost.TrimLeft();
- sHost.TrimLeft();
-
+ sHost.TrimLeft(); sHost.TrimRight();
if(sHost.IsEmpty()) {
AfxMessageBox("No host specified!");
m_comboHost.SetFocus();
return ;
}
m_listMTR.DeleteAllItems();
- }
-
- if(state == IDLE) {
+ HKEY hKey; DWORD tmp_dword;
+ if(RegCreateKeyEx(HKEY_CURRENT_USER,"Software\\WinMTR\\Config",0,NULL,0,KEY_ALL_ACCESS,NULL,&hKey,&tmp_dword)==ERROR_SUCCESS){
+ tmp_dword=m_checkIPv6.GetCheck();
+ useIPv6=(unsigned char)tmp_dword;
+ RegSetValueEx(hKey,"UseIPv6",0,REG_DWORD,(const unsigned char*)&tmp_dword,sizeof(DWORD));
+ RegCloseKey(hKey);
+ }
if(InitMTRNet()) {
if(m_comboHost.FindString(-1, sHost) == CB_ERR) {
m_comboHost.InsertString(m_comboHost.GetCount() - 1,sHost);
-
- HKEY hKey;
- DWORD tmp_dword;
- LONG r;
char key_name[20];
-
- r = RegOpenKeyEx( HKEY_CURRENT_USER, "Software", 0, KEY_ALL_ACCESS,&hKey);
- r = RegOpenKeyEx( hKey, "WinMTR", 0, KEY_ALL_ACCESS, &hKey);
- r = RegOpenKeyEx( hKey, "LRU", 0, KEY_ALL_ACCESS, &hKey);
-
- if(nrLRU >= maxLRU)
- nrLRU = 0;
-
- nrLRU++;
- sprintf(key_name, "Host%d", nrLRU);
- r = RegSetValueEx(hKey,key_name, 0, REG_SZ, (const unsigned char *)(LPCTSTR)sHost, strlen((LPCTSTR)sHost)+1);
- tmp_dword = nrLRU;
- r = RegSetValueEx(hKey,"NrLRU", 0, REG_DWORD, (const unsigned char *)&tmp_dword, sizeof(DWORD));
- RegCloseKey(hKey);
+ if(RegCreateKeyEx(HKEY_CURRENT_USER,"Software\\WinMTR\\LRU",0,NULL,0,KEY_ALL_ACCESS,NULL,&hKey,&tmp_dword)==ERROR_SUCCESS){
+ if(++nrLRU>maxLRU) nrLRU=0;
+ sprintf(key_name, "Host%d", nrLRU);
+ RegSetValueEx(hKey,key_name, 0, REG_SZ, (const unsigned char*)(LPCTSTR)sHost, (DWORD)strlen((LPCTSTR)sHost)+1);
+ tmp_dword = nrLRU;
+ RegSetValueEx(hKey,"NrLRU", 0, REG_DWORD, (const unsigned char*)&tmp_dword, sizeof(DWORD));
+ RegCloseKey(hKey);
+ }
}
Transit(TRACING);
}
@@ -610,7 +555,7 @@ void WinMTRDialog::OnOptions()
if(IDOK == optDlg.DoModal()) {
- pingsize = optDlg.GetPingSize();
+ pingsize = (WORD)optDlg.GetPingSize();
interval = optDlg.GetInterval();
maxLRU = optDlg.GetMaxLRU();
useDNS = optDlg.GetUseDNS();
@@ -893,10 +838,10 @@ int WinMTRDialog::DisplayRedraw()
int nh = wmtrnet->GetMax();
while( m_listMTR.GetItemCount() > nh ) m_listMTR.DeleteItem(m_listMTR.GetItemCount() - 1);
- for(int i=0;i GetName(i, buf);
- if( strcmp(buf,"")==0 ) strcpy(buf,"No response from host");
+ if(!*buf) strcpy(buf,"No response from host");
sprintf(nr_crt, "%d", i+1);
if(m_listMTR.GetItemCount() <= i )
@@ -941,35 +886,33 @@ int WinMTRDialog::DisplayRedraw()
//*****************************************************************************
int WinMTRDialog::InitMTRNet()
{
- char strtmp[255];
- char *Hostname = strtmp;
+ char hostname[255];
char buf[255];
- struct hostent *host;
- m_comboHost.GetWindowText(strtmp, 255);
-
- if (Hostname == NULL) Hostname = "localhost";
-
- int isIP=1;
- char *t = Hostname;
- while(*t) {
- if(!isdigit(*t) && *t!='.') {
- isIP=0;
- break;
+ m_comboHost.GetWindowText(hostname, 255);
+
+ sprintf(buf, "Resolving host %s...", hostname);
+ statusBar.SetPaneText(0,buf);
+
+ addrinfo nfofilter={0};
+ addrinfo* anfo;
+ if(wmtrnet->hasIPv6){
+ switch(useIPv6){
+ case 0:
+ nfofilter.ai_family=AF_INET;break;
+ case 1:
+ nfofilter.ai_family=AF_INET6;break;
+ default:
+ nfofilter.ai_family=AF_UNSPEC;
}
- t++;
}
-
- if(!isIP) {
- sprintf(buf, "Resolving host %s...", strtmp);
- statusBar.SetPaneText(0,buf);
- host = gethostbyname(Hostname);
- if(host == NULL) {
- statusBar.SetPaneText(0, CString((LPCSTR)IDS_STRING_SB_NAME) );
- AfxMessageBox("Unable to resolve hostname.");
- return 0;
- }
+ nfofilter.ai_socktype=SOCK_RAW;
+ nfofilter.ai_flags=AI_NUMERICSERV|AI_ADDRCONFIG;//|AI_V4MAPPED;
+ if(getaddrinfo(hostname,NULL,&nfofilter,&anfo)||!anfo){
+ statusBar.SetPaneText(0, CString((LPCSTR)IDS_STRING_SB_NAME) );
+ AfxMessageBox("Unable to resolve hostname.");
+ return 0;
}
-
+ freeaddrinfo(anfo);
return 1;
}
@@ -984,44 +927,31 @@ void PingThread(void *p)
WinMTRDialog *wmtrdlg = (WinMTRDialog *)p;
WaitForSingleObject(wmtrdlg->traceThreadMutex, INFINITE);
- struct hostent *host, *lhost;
- char strtmp[255];
- char *Hostname = strtmp;
- int traddr;
- int localaddr;
-
- wmtrdlg->m_comboHost.GetWindowText(strtmp, 255);
-
- if (Hostname == NULL) Hostname = "localhost";
-
- int isIP=1;
- char *t = Hostname;
- while(*t) {
- if(!isdigit(*t) && *t!='.') {
- isIP=0;
- break;
+ char hostname[255];
+ wmtrdlg->m_comboHost.GetWindowText(hostname, 255);
+
+ addrinfo nfofilter={0};
+ addrinfo* anfo;
+ if(wmtrdlg->wmtrnet->hasIPv6){
+ switch(wmtrdlg->useIPv6){
+ case 0:
+ nfofilter.ai_family=AF_INET;break;
+ case 1:
+ nfofilter.ai_family=AF_INET6;break;
+ default:
+ nfofilter.ai_family=AF_UNSPEC;
}
- t++;
}
-
- if(!isIP) {
- host = gethostbyname(Hostname);
- traddr = *(int *)host->h_addr;
- } else
- traddr = inet_addr(Hostname);
-
- lhost = gethostbyname("localhost");
- if(lhost == NULL) {
- AfxMessageBox("Unable to get local IP address.");
- ReleaseMutex(wmtrdlg->traceThreadMutex);
- return;
+ nfofilter.ai_socktype=SOCK_RAW;
+ nfofilter.ai_flags=AI_NUMERICSERV|AI_ADDRCONFIG;//|AI_V4MAPPED;
+ if(getaddrinfo(hostname,NULL,&nfofilter,&anfo)||!anfo){//we use first address returned
+ AfxMessageBox("Unable to resolve hostname. (again)");
+ ReleaseMutex(wmtrdlg->traceThreadMutex);
+ return;
}
- localaddr = *(int *)lhost->h_addr;
-
- wmtrdlg->wmtrnet->DoTrace(traddr);
-
+ wmtrdlg->wmtrnet->DoTrace(anfo->ai_addr);
+ freeaddrinfo(anfo);
ReleaseMutex(wmtrdlg->traceThreadMutex);
- _endthread();
}
@@ -1137,57 +1067,53 @@ void WinMTRDialog::Transit(STATES new_state)
// modify controls according to new state
switch(transition) {
+ case IDLE_TO_IDLE:
+ // nothing to be done
+ break;
case IDLE_TO_TRACING:
m_buttonStart.EnableWindow(FALSE);
m_buttonStart.SetWindowText("Stop");
m_comboHost.EnableWindow(FALSE);
+ m_checkIPv6.EnableWindow(FALSE);
m_buttonOptions.EnableWindow(FALSE);
statusBar.SetPaneText(0, "Double click on host name for more information.");
_beginthread(PingThread, 0 , this);
m_buttonStart.EnableWindow(TRUE);
break;
- case IDLE_TO_IDLE:
- // nothing to be done
+ case IDLE_TO_EXIT:
+ m_buttonStart.EnableWindow(FALSE);
+ m_comboHost.EnableWindow(FALSE);
+ m_buttonOptions.EnableWindow(FALSE);
break;
case STOPPING_TO_IDLE:
+ DisplayRedraw();
m_buttonStart.EnableWindow(TRUE);
statusBar.SetPaneText(0, CString((LPCSTR)IDS_STRING_SB_NAME) );
m_buttonStart.SetWindowText("Start");
m_comboHost.EnableWindow(TRUE);
+ m_checkIPv6.EnableWindow(TRUE);
m_buttonOptions.EnableWindow(TRUE);
m_comboHost.SetFocus();
break;
case STOPPING_TO_STOPPING:
DisplayRedraw();
break;
+ case STOPPING_TO_EXIT:
+ break;
case TRACING_TO_TRACING:
DisplayRedraw();
break;
case TRACING_TO_STOPPING:
m_buttonStart.EnableWindow(FALSE);
- m_comboHost.EnableWindow(FALSE);
- m_buttonOptions.EnableWindow(FALSE);
wmtrnet->StopTrace();
statusBar.SetPaneText(0, "Waiting for last packets in order to stop trace ...");
DisplayRedraw();
break;
- case IDLE_TO_EXIT:
- m_buttonStart.EnableWindow(FALSE);
- m_comboHost.EnableWindow(FALSE);
- m_buttonOptions.EnableWindow(FALSE);
- break;
case TRACING_TO_EXIT:
m_buttonStart.EnableWindow(FALSE);
- m_comboHost.EnableWindow(FALSE);
- m_buttonOptions.EnableWindow(FALSE);
wmtrnet->StopTrace();
statusBar.SetPaneText(0, "Waiting for last packets in order to stop trace ...");
break;
- case STOPPING_TO_EXIT:
- m_buttonStart.EnableWindow(FALSE);
- m_comboHost.EnableWindow(FALSE);
- m_buttonOptions.EnableWindow(FALSE);
- break;
default:
TRACE_MSG("Unknown transition " << transition);
}
@@ -1196,22 +1122,18 @@ void WinMTRDialog::Transit(STATES new_state)
void WinMTRDialog::OnTimer(UINT_PTR nIDEvent)
{
- static unsigned int call_count = 0;
- call_count += 1;
-
+ static unsigned int call_count=0;
if(state == EXIT && WaitForSingleObject(traceThreadMutex, 0) == WAIT_OBJECT_0) {
ReleaseMutex(traceThreadMutex);
OnOK();
}
-
if( WaitForSingleObject(traceThreadMutex, 0) == WAIT_OBJECT_0 ) {
ReleaseMutex(traceThreadMutex);
Transit(IDLE);
- } else if( (call_count % 10 == 0) && (WaitForSingleObject(traceThreadMutex, 0) == WAIT_TIMEOUT) ) {
- ReleaseMutex(traceThreadMutex);
- if( state == TRACING) Transit(TRACING);
- else if( state == STOPPING) Transit(STOPPING);
+ }else if((++call_count&5)==5){
+ if(state==TRACING) Transit(TRACING);
+ else if(state==STOPPING) Transit(STOPPING);
}
CDialog::OnTimer(nIDEvent);
diff --git a/src/WinMTRDialog.h b/src/WinMTRDialog.h
index a214d7b..3ac6534 100644
--- a/src/WinMTRDialog.h
+++ b/src/WinMTRDialog.h
@@ -17,6 +17,7 @@
#include "WinMTRStatusBar.h"
#include "WinMTRNet.h"
+#include "afxlinkctrl.h"
//*****************************************************************************
// CLASS: WinMTRDialog
@@ -59,7 +60,9 @@ class WinMTRDialog : public CDialog
CButton m_buttonExit;
CButton m_buttonStart;
CComboBox m_comboHost;
- CListCtrl m_listMTR;
+ CButton m_checkIPv6;
+ CListCtrl m_listMTR;
+ CMFCLinkCtrl m_buttonAppnor;
CStatic m_staticS;
CStatic m_staticJ;
@@ -77,18 +80,20 @@ class WinMTRDialog : public CDialog
HANDLE traceThreadMutex;
double interval;
bool hasIntervalFromCmdLine;
- int pingsize;
+ WORD pingsize;
bool hasPingsizeFromCmdLine;
int maxLRU;
bool hasMaxLRUFromCmdLine;
int nrLRU;
BOOL useDNS;
bool hasUseDNSFromCmdLine;
+ unsigned char useIPv6;
+ bool hasUseIPv6FromCmdLine;
WinMTRNet* wmtrnet;
void SetHostName(const char *host);
void SetInterval(float i);
- void SetPingSize(int ps);
+ void SetPingSize(WORD ps);
void SetMaxLRU(int mlru);
void SetUseDNS(BOOL udns);
diff --git a/src/WinMTRGlobal.cpp b/src/WinMTRGlobal.cpp
index 0f435c8..053fba1 100644
--- a/src/WinMTRGlobal.cpp
+++ b/src/WinMTRGlobal.cpp
@@ -11,7 +11,7 @@
//
// win32 port of unix gettimeofday
//*****************************************************************************
-int gettimeofday(struct timeval* tv, struct timezone *tz)
+int gettimeofday(struct timeval* tv, struct timezone* /*tz*/)
{
if(!tv)
return -1;
@@ -19,7 +19,7 @@ int gettimeofday(struct timeval* tv, struct timezone *tz)
_ftime(&timebuffer);
- tv->tv_sec = timebuffer.time;
+ tv->tv_sec = (long)timebuffer.time;
tv->tv_usec = timebuffer.millitm * 1000 + 500;
return 0;
}
diff --git a/src/WinMTRGlobal.h b/src/WinMTRGlobal.h
index 53524ee..3f3102e 100644
--- a/src/WinMTRGlobal.h
+++ b/src/WinMTRGlobal.h
@@ -28,6 +28,8 @@
#include
#endif
#include
+#include //IP_OPTION_INFORMATION32
+#include //sockaddr_in6
#include
#include
@@ -46,10 +48,9 @@
#include "resource.h"
-#define WINMTR_VERSION "0.9"
-#define WINMTR_LICENSE "GPL - GNU Public License"
-#define WINMTR_COPYRIGHT "WinMTR 0.9 (c) 2010-2011 Appnor MSP - Fully Managed Hosting & Cloud Provider www.appnor.com"
-#define WINMTR_HOMEPAGE "http://WinMTR.sourceforge.net"
+#define WINMTR_VERSION "1.0"
+#define WINMTR_LICENSE "GPLv2 - GNU General Public License, version 2"
+#define WINMTR_HOMEPAGE "https://github.com/White-Tiger/WinMTR"
#define DEFAULT_PING_SIZE 64
#define DEFAULT_INTERVAL 1.0
@@ -98,7 +99,7 @@ const char MTR_COLS[ MTR_NR_COLS ][10] = {
};
const int MTR_COL_LENGTH[ MTR_NR_COLS ] = {
- 190, 30, 50, 40, 40, 50, 50, 50, 50
+ 249, 30, 50, 40, 40, 50, 50, 50, 50
};
int gettimeofday(struct timeval* tv, struct timezone *tz);
diff --git a/src/WinMTRMain.cpp b/src/WinMTRMain.cpp
index 7e740fe..581583f 100644
--- a/src/WinMTRMain.cpp
+++ b/src/WinMTRMain.cpp
@@ -61,6 +61,8 @@ WinMTRMain::WinMTRMain()
//*****************************************************************************
BOOL WinMTRMain::InitInstance()
{
+ INITCOMMONCONTROLSEX icex={sizeof(INITCOMMONCONTROLSEX),ICC_STANDARD_CLASSES};
+ InitCommonControlsEx(&icex);
if (!AfxSocketInit())
{
AfxMessageBox(IDP_SOCKETS_INIT_FAILED);
@@ -68,10 +70,6 @@ BOOL WinMTRMain::InitInstance()
}
AfxEnableControlContainer();
-
-#ifdef _AFXDLL
- Enable3dControls(); // Call this when using MFC in a shared DLL
-#endif
WinMTRDialog mtrDialog;
@@ -82,7 +80,7 @@ BOOL WinMTRMain::InitInstance()
ParseCommandLineParams(m_lpCmdLine, &mtrDialog);
}
- int nResponse = mtrDialog.DoModal();
+ mtrDialog.DoModal();
return FALSE;
@@ -114,17 +112,25 @@ void WinMTRMain::ParseCommandLineParams(LPTSTR cmd, WinMTRDialog *wmtrdlg)
wmtrdlg->hasIntervalFromCmdLine = true;
}
if(GetParamValue(cmd, "size",'s', value)) {
- wmtrdlg->SetPingSize(atoi(value));
+ wmtrdlg->SetPingSize((WORD)atoi(value));
wmtrdlg->hasPingsizeFromCmdLine = true;
}
if(GetParamValue(cmd, "maxLRU",'m', value)) {
wmtrdlg->SetMaxLRU(atoi(value));
wmtrdlg->hasMaxLRUFromCmdLine = true;
}
- if(GetParamValue(cmd, "numeric",'n', value)) {
+ if(GetParamValue(cmd, "numeric",'n', NULL)) {
wmtrdlg->SetUseDNS(FALSE);
wmtrdlg->hasUseDNSFromCmdLine = true;
}
+ if(GetParamValue(cmd, "ipv6",'6', NULL)) {
+ wmtrdlg->hasUseIPv6FromCmdLine=true;
+ wmtrdlg->useIPv6=1;
+ }
+ if(GetParamValue(cmd, "ipv4",'4', NULL)) {
+ wmtrdlg->hasUseIPv6FromCmdLine=true;
+ wmtrdlg->useIPv6=0;
+ }
}
//*****************************************************************************
@@ -149,7 +155,7 @@ int WinMTRMain::GetParamValue(LPTSTR cmd, char * param, char sparam, char *value
if(p == NULL)
return 0;
- if(strcmp(param,"numeric")==0)
+ if(!value)
return 1;
while(*p && *p!=' ')
@@ -172,7 +178,7 @@ int WinMTRMain::GetParamValue(LPTSTR cmd, char * param, char sparam, char *value
int WinMTRMain::GetHostNameParamValue(LPTSTR cmd, std::string& host_name)
{
// WinMTR -h -i 1 -n google.com
- int size = strlen(cmd);
+ size_t size = strlen(cmd);
std::string name = "";
while(cmd[--size] == ' ');
@@ -200,7 +206,7 @@ int WinMTRMain::GetHostNameParamValue(LPTSTR cmd, std::string& host_name)
possible_argument = cmd[size] + possible_argument;
}
- if(possible_argument.length() && (possible_argument[0] != '-' || possible_argument == "-n" || possible_argument == "--numeric")) {
+ if(possible_argument.length() && (possible_argument[0] != '-' || possible_argument == "-n" || possible_argument == "--numeric" || possible_argument == "-6" || possible_argument == "--ipv6" || possible_argument == "-4" || possible_argument == "--ipv4")) {
host_name = name;
return 1;
}
diff --git a/src/WinMTRNet.cpp b/src/WinMTRNet.cpp
index 7c5e8ae..bcc8a6d 100644
--- a/src/WinMTRNet.cpp
+++ b/src/WinMTRNet.cpp
@@ -19,22 +19,29 @@
#define MAX_HOPS 30
struct trace_thread {
- int address;
- WinMTRNet *winmtr;
+ WinMTRNet* winmtr;
+ in_addr address;
int ttl;
};
+struct trace_thread6 {
+ WinMTRNet* winmtr;
+ sockaddr_in6 address;
+ int ttl;
+};
struct dns_resolver_thread {
+ WinMTRNet* winmtr;
int index;
- WinMTRNet *winmtr;
};
-void TraceThread(void *p);
+unsigned WINAPI TraceThread(void *p);
+unsigned WINAPI TraceThread6(void *p);
void DnsResolverThread(void *p);
WinMTRNet::WinMTRNet(WinMTRDialog *wp) {
ghMutex = CreateMutex(NULL, FALSE, NULL);
+ hasIPv6=true;
tracing=false;
initialized = false;
wmtrdlg = wp;
@@ -44,32 +51,61 @@ WinMTRNet::WinMTRNet(WinMTRDialog *wp) {
AfxMessageBox("Failed initializing windows sockets library!");
return;
}
-
- hICMP_DLL = LoadLibrary(_T("ICMP.DLL"));
- if (hICMP_DLL == 0) {
- AfxMessageBox("Failed: Unable to locate ICMP.DLL!");
- return;
- }
+ OSVERSIONINFOEX osvi={0};
+ osvi.dwOSVersionInfoSize=sizeof(OSVERSIONINFOEX);
+ if(!GetVersionEx((OSVERSIONINFO*) &osvi)){
+ AfxMessageBox("Failed to get Windows version!");
+ return;
+ }
+ if(osvi.dwMajorVersion==5 && osvi.dwMinorVersion==0){//w2k
+ hICMP_DLL=LoadLibrary(_T("ICMP.DLL"));
+ if(!hICMP_DLL){
+ AfxMessageBox("Failed: Unable to locate ICMP.DLL!");
+ return;
+ }
+ }else{
+ hICMP_DLL=LoadLibrary(_T("Iphlpapi.dll"));
+ if(!hICMP_DLL){
+ AfxMessageBox("Failed: Unable to locate Iphlpapi.dll!");
+ return;
+ }
+ }
/*
* Get pointers to ICMP.DLL functions
*/
+ //IPv4
lpfnIcmpCreateFile = (LPFNICMPCREATEFILE)GetProcAddress(hICMP_DLL,"IcmpCreateFile");
lpfnIcmpCloseHandle = (LPFNICMPCLOSEHANDLE)GetProcAddress(hICMP_DLL,"IcmpCloseHandle");
- lpfnIcmpSendEcho = (LPFNICMPSENDECHO)GetProcAddress(hICMP_DLL,"IcmpSendEcho");
- if ((!lpfnIcmpCreateFile) || (!lpfnIcmpCloseHandle) || (!lpfnIcmpSendEcho)) {
- AfxMessageBox("Wrong ICMP.DLL system library !");
+ lpfnIcmpSendEcho2 = (LPFNICMPSENDECHO2)GetProcAddress(hICMP_DLL,"IcmpSendEcho2");
+ if (!lpfnIcmpCreateFile || !lpfnIcmpCloseHandle || !lpfnIcmpSendEcho2) {
+ AfxMessageBox("Wrong ICMP system library !");
return;
}
+ //IPv6
+ lpfnIcmp6CreateFile=(LPFNICMP6CREATEFILE)GetProcAddress(hICMP_DLL,"Icmp6CreateFile");
+ lpfnIcmp6SendEcho2=(LPFNICMP6SENDECHO2)GetProcAddress(hICMP_DLL,"Icmp6SendEcho2");
+ if(!lpfnIcmp6CreateFile || !lpfnIcmp6SendEcho2){
+ hasIPv6=false;
+ AfxMessageBox("IPv6 support not found!");
+ return;//@todo : soft fail
+ }
/*
* IcmpCreateFile() - Open the ping service
*/
hICMP = (HANDLE) lpfnIcmpCreateFile();
if (hICMP == INVALID_HANDLE_VALUE) {
- AfxMessageBox("Error in ICMP.DLL !");
+ AfxMessageBox("Error in ICMP module!");
return;
}
+ if(hasIPv6){
+ hICMP6=(HANDLE)lpfnIcmp6CreateFile();
+ if(hICMP6==INVALID_HANDLE_VALUE){
+ AfxMessageBox("Error in ICMPv6 module!");
+ return;//@todo : soft fail
+ }
+ }
ResetHops();
@@ -83,6 +119,7 @@ WinMTRNet::~WinMTRNet()
/*
* IcmpCloseHandle - Close the ICMP handle
*/
+ if(hasIPv6) lpfnIcmpCloseHandle(hICMP6);
lpfnIcmpCloseHandle(hICMP);
// Shut down...
@@ -96,37 +133,42 @@ WinMTRNet::~WinMTRNet()
void WinMTRNet::ResetHops()
{
- for(int i = 0; i < MaxHost;i++) {
- host[i].addr = 0;
- host[i].xmit = 0;
- host[i].returned = 0;
- host[i].total = 0;
- host[i].last = 0;
- host[i].best = 0;
- host[i].worst = 0;
- memset(host[i].name,0,sizeof(host[i].name));
- }
+ memset(host,0,sizeof(host));
}
-void WinMTRNet::DoTrace(int address)
+void WinMTRNet::DoTrace(sockaddr* sockaddr)
{
HANDLE hThreads[MAX_HOPS];
+ unsigned char hops=0;
tracing = true;
-
ResetHops();
-
- last_remote_addr = address;
-
- // one thread per TTL value
- for(int i = 0; i < MAX_HOPS; i++) {
- trace_thread *current = new trace_thread;
- current->address = address;
- current->winmtr = this;
- current->ttl = i + 1;
- hThreads[i] = (HANDLE)_beginthread(TraceThread, 0 , current);
+ if(sockaddr->sa_family==AF_INET6){
+ host[0].addr6.sin6_family=AF_INET6;
+ last_remote_addr6=((sockaddr_in6*)sockaddr)->sin6_addr;
+ for(; hopsaddress=*(sockaddr_in6*)sockaddr;
+ current->winmtr=this;
+ current->ttl=hops+1;
+ hThreads[hops]=(HANDLE)_beginthreadex(NULL,0,TraceThread6,current,0,NULL);
+ Sleep(30);
+ if(++hops>this->GetMax()) break;
+ }
+ }else{
+ host[0].addr.sin_family=AF_INET;
+ last_remote_addr=((sockaddr_in*)sockaddr)->sin_addr;
+ for(; hopsaddress=((sockaddr_in*)sockaddr)->sin_addr;
+ current->winmtr=this;
+ current->ttl=hops+1;
+ hThreads[hops]=(HANDLE)_beginthreadex(NULL,0,TraceThread,current,0,NULL);
+ Sleep(30);
+ if(++hops>this->GetMax()) break;
+ }
}
-
- WaitForMultipleObjects(MAX_HOPS, hThreads, TRUE, INFINITE);
+ WaitForMultipleObjects(hops, hThreads, TRUE, INFINITE);
+ for(; hops; ) CloseHandle(hThreads[--hops]);
}
void WinMTRNet::StopTrace()
@@ -134,24 +176,27 @@ void WinMTRNet::StopTrace()
tracing = false;
}
-void TraceThread(void *p)
+unsigned WINAPI TraceThread(void *p)
{
trace_thread* current = (trace_thread*)p;
WinMTRNet *wmtrnet = current->winmtr;
- TRACE_MSG("Threaad with TTL=" << current->ttl << " started.");
+ TRACE_MSG("Thread with TTL=" << (int)current->ttl << " started.");
IPINFO stIPInfo, *lpstIPInfo;
DWORD dwReplyCount;
char achReqData[8192];
- int nDataLen = wmtrnet->wmtrdlg->pingsize;
- char achRepData[sizeof(ICMPECHO) + 8192];
+ WORD nDataLen = wmtrnet->wmtrdlg->pingsize;
+ union{
+ ICMP_ECHO_REPLY icmp_echo_reply;
+ char achRepData[sizeof(ICMPECHO)+8192];
+ };
/*
* Init IPInfo structure
*/
lpstIPInfo = &stIPInfo;
- stIPInfo.Ttl = current->ttl;
+ stIPInfo.Ttl = (UCHAR)current->ttl;
stIPInfo.Tos = 0;
stIPInfo.Flags = IPFLAG_DONT_FRAGMENT;
stIPInfo.OptionsSize = 0;
@@ -172,117 +217,107 @@ void TraceThread(void *p)
// - as soon as we get a hop, we start pinging directly that hop, with a greater TTL
// - a drawback would be that, some servers are configured to reply for TTL transit expire, but not to ping requests, so,
// for these servers we'll have 100% loss
- dwReplyCount = wmtrnet->lpfnIcmpSendEcho(wmtrnet->hICMP, current->address, achReqData, nDataLen, lpstIPInfo, achRepData, sizeof(achRepData), ECHO_REPLY_TIMEOUT);
-
- PICMPECHO icmp_echo_reply = (PICMPECHO)achRepData;
+ dwReplyCount = wmtrnet->lpfnIcmpSendEcho2(wmtrnet->hICMP, 0,NULL,NULL, current->address, achReqData, nDataLen, lpstIPInfo, achRepData, sizeof(achRepData), ECHO_REPLY_TIMEOUT);
wmtrnet->AddXmit(current->ttl - 1);
- if (dwReplyCount != 0) {
- TRACE_MSG("TTL " << current->ttl << " reply TTL " << icmp_echo_reply->Options.Ttl << " Status " << icmp_echo_reply->Status << " Reply count " << dwReplyCount);
+ if(dwReplyCount){
+ TRACE_MSG("TTL " << (int)current->ttl << " reply TTL " << (int)icmp_echo_reply.Options.Ttl << " Status " << icmp_echo_reply.Status << " Reply count " << dwReplyCount);
- switch(icmp_echo_reply->Status) {
+ switch(icmp_echo_reply.Status) {
case IP_SUCCESS:
case IP_TTL_EXPIRED_TRANSIT:
- wmtrnet->SetLast(current->ttl - 1, icmp_echo_reply->RoundTripTime);
- wmtrnet->SetBest(current->ttl - 1, icmp_echo_reply->RoundTripTime);
- wmtrnet->SetWorst(current->ttl - 1, icmp_echo_reply->RoundTripTime);
+ wmtrnet->UpdateRTT(current->ttl - 1, icmp_echo_reply.RoundTripTime);
wmtrnet->AddReturned(current->ttl - 1);
- wmtrnet->SetAddr(current->ttl - 1, icmp_echo_reply->Address);
- break;
- case IP_BUF_TOO_SMALL:
- wmtrnet->SetName(current->ttl - 1, "Reply buffer too small.");
- break;
- case IP_DEST_NET_UNREACHABLE:
- wmtrnet->SetName(current->ttl - 1, "Destination network unreachable.");
- break;
- case IP_DEST_HOST_UNREACHABLE:
- wmtrnet->SetName(current->ttl - 1, "Destination host unreachable.");
- break;
- case IP_DEST_PROT_UNREACHABLE:
- wmtrnet->SetName(current->ttl - 1, "Destination protocol unreachable.");
- break;
- case IP_DEST_PORT_UNREACHABLE:
- wmtrnet->SetName(current->ttl - 1, "Destination port unreachable.");
- break;
- case IP_NO_RESOURCES:
- wmtrnet->SetName(current->ttl - 1, "Insufficient IP resources were available.");
- break;
- case IP_BAD_OPTION:
- wmtrnet->SetName(current->ttl - 1, "Bad IP option was specified.");
- break;
- case IP_HW_ERROR:
- wmtrnet->SetName(current->ttl - 1, "Hardware error occurred.");
- break;
- case IP_PACKET_TOO_BIG:
- wmtrnet->SetName(current->ttl - 1, "Packet was too big.");
- break;
- case IP_REQ_TIMED_OUT:
- wmtrnet->SetName(current->ttl - 1, "Request timed out.");
- break;
- case IP_BAD_REQ:
- wmtrnet->SetName(current->ttl - 1, "Bad request.");
- break;
- case IP_BAD_ROUTE:
- wmtrnet->SetName(current->ttl - 1, "Bad route.");
- break;
- case IP_TTL_EXPIRED_REASSEM:
- wmtrnet->SetName(current->ttl - 1, "The time to live expired during fragment reassembly.");
- break;
- case IP_PARAM_PROBLEM:
- wmtrnet->SetName(current->ttl - 1, "Parameter problem.");
- break;
- case IP_SOURCE_QUENCH:
- wmtrnet->SetName(current->ttl - 1, "Datagrams are arriving too fast to be processed and datagrams may have been discarded.");
- break;
- case IP_OPTION_TOO_BIG:
- wmtrnet->SetName(current->ttl - 1, "An IP option was too big.");
- break;
- case IP_BAD_DESTINATION:
- wmtrnet->SetName(current->ttl - 1, "Bad destination.");
- break;
- case IP_GENERAL_FAILURE:
- wmtrnet->SetName(current->ttl - 1, "General failure.");
+ wmtrnet->SetAddr(current->ttl - 1, icmp_echo_reply.Address);
break;
default:
- wmtrnet->SetName(current->ttl - 1, "General failure.");
+ wmtrnet->SetErrorName(current->ttl - 1, icmp_echo_reply.Status);
}
- if(wmtrnet->wmtrdlg->interval * 1000 > icmp_echo_reply->RoundTripTime)
- Sleep(wmtrnet->wmtrdlg->interval * 1000 - icmp_echo_reply->RoundTripTime);
+ if((DWORD)(wmtrnet->wmtrdlg->interval * 1000) > icmp_echo_reply.RoundTripTime)
+ Sleep((DWORD)(wmtrnet->wmtrdlg->interval * 1000) - icmp_echo_reply.RoundTripTime);
+ }else{
+ DWORD err=GetLastError();
+ wmtrnet->SetErrorName(current->ttl - 1, err);
+ switch(err){
+ case IP_REQ_TIMED_OUT: break;
+ default:
+ Sleep((DWORD)(wmtrnet->wmtrdlg->interval * 1000));
+ }
}
-
} /* end ping loop */
- TRACE_MSG("Thread with TTL=" << current->ttl << " stopped.");
+ TRACE_MSG("Thread with TTL=" << (int)current->ttl << " stopped.");
delete p;
- _endthread();
+ return 0;
}
-int WinMTRNet::GetAddr(int at)
+unsigned WINAPI TraceThread6(void* p)
{
- WaitForSingleObject(ghMutex, INFINITE);
- int addr = ntohl(host[at].addr);
- ReleaseMutex(ghMutex);
- return addr;
+ static sockaddr_in6 sockaddrfrom={AF_INET6,0,0,in6addr_any,0};
+ trace_thread6* current = (trace_thread6*)p;
+ WinMTRNet* wmtrnet = current->winmtr;
+ TRACE_MSG("Thread with TTL=" << (int)current->ttl << " started.");
+
+ IPINFO stIPInfo, *lpstIPInfo;
+ DWORD dwReplyCount;
+ char achReqData[8192];
+ WORD nDataLen = wmtrnet->wmtrdlg->pingsize;
+ union{
+ ICMPV6_ECHO_REPLY icmpv6_echo_reply;
+ char achRepData[sizeof(PICMPV6_ECHO_REPLY) + 8192];
+ };
+
+ lpstIPInfo = &stIPInfo;
+ stIPInfo.Ttl = (UCHAR)current->ttl;
+ stIPInfo.Tos = 0;
+ stIPInfo.Flags = IPFLAG_DONT_FRAGMENT;
+ stIPInfo.OptionsSize = 0;
+ stIPInfo.OptionsData = NULL;
+ for(int i=0; itracing){
+ if(current->ttl > wmtrnet->GetMax()) break;
+ dwReplyCount = wmtrnet->lpfnIcmp6SendEcho2(wmtrnet->hICMP6, 0,NULL,NULL, &sockaddrfrom, ¤t->address, achReqData, nDataLen, lpstIPInfo, achRepData, sizeof(achRepData), ECHO_REPLY_TIMEOUT);
+ wmtrnet->AddXmit(current->ttl - 1);
+ if(dwReplyCount) {
+ TRACE_MSG("TTL " << (int)current->ttl << " Status " << icmpv6_echo_reply.Status << " Reply count " << dwReplyCount);
+ switch(icmpv6_echo_reply.Status) {
+ case IP_SUCCESS:
+ case IP_TTL_EXPIRED_TRANSIT:
+ wmtrnet->UpdateRTT(current->ttl - 1, icmpv6_echo_reply.RoundTripTime);
+ wmtrnet->AddReturned(current->ttl - 1);
+ wmtrnet->SetAddr6(current->ttl - 1, icmpv6_echo_reply.Address);
+ break;
+ default:
+ wmtrnet->SetErrorName(current->ttl - 1, icmpv6_echo_reply.Status);
+ }
+ if((DWORD)(wmtrnet->wmtrdlg->interval * 1000) > icmpv6_echo_reply.RoundTripTime)
+ Sleep((DWORD)(wmtrnet->wmtrdlg->interval * 1000) - icmpv6_echo_reply.RoundTripTime);
+ }else{
+ DWORD err=GetLastError();
+ wmtrnet->SetErrorName(current->ttl - 1, err);
+ switch(err){
+ case IP_REQ_TIMED_OUT: break;
+ default:
+ Sleep((DWORD)(wmtrnet->wmtrdlg->interval * 1000));
+ }
+ }
+ }//end loop
+ TRACE_MSG("Thread with TTL=" << (int)current->ttl << " stopped.");
+ delete p;
+ return 0;
+}
+
+sockaddr* WinMTRNet::GetAddr(int at)
+{
+ return (sockaddr*)&host[at].addr;
}
int WinMTRNet::GetName(int at, char *n)
{
WaitForSingleObject(ghMutex, INFINITE);
- if(!strcmp(host[at].name, "")) {
- int addr = GetAddr(at);
- sprintf ( n, "%d.%d.%d.%d",
- (addr >> 24) & 0xff,
- (addr >> 16) & 0xff,
- (addr >> 8) & 0xff,
- addr & 0xff
- );
- if(addr==0)
- strcpy(n,"");
- } else {
- strcpy(n, host[at].name);
- }
+ strcpy(n, host[at].name);
ReleaseMutex(ghMutex);
return 0;
}
@@ -344,112 +379,165 @@ int WinMTRNet::GetXmit(int at)
}
int WinMTRNet::GetMax()
-{
+{// @todo : improve this (last hop guess)
WaitForSingleObject(ghMutex, INFINITE);
- int max = MAX_HOPS;
-
- // first match: traced address responds on ping requests, and the address is in the hosts list
- for(int i = 0; i < MAX_HOPS; i++) {
- if(host[i].addr == last_remote_addr) {
- max = i + 1;
- break;
+ int max=0;//first try to find target, if not found, find best guess (doesn't work actually :P)
+ if(host[0].addr6.sin6_family==AF_INET6){
+ for(; max1 && !memcmp(&host[max-1].addr6.sin6_addr,&host[max-2].addr6.sin6_addr,sizeof(in6_addr)) && (host[max-1].addr6.sin6_addr.u.Word[0]|host[max-1].addr6.sin6_addr.u.Word[1]|host[max-1].addr6.sin6_addr.u.Word[2]|host[max-1].addr6.sin6_addr.u.Word[3]|host[max-1].addr6.sin6_addr.u.Word[4]|host[max-1].addr6.sin6_addr.u.Word[5]|host[max-1].addr6.sin6_addr.u.Word[6]|host[max-1].addr6.sin6_addr.u.Word[7])) --max;
+ }
+ }else{
+ for(; max1 && host[max-1].addr.sin_addr.s_addr==host[max-2].addr.sin_addr.s_addr && host[max-1].addr.sin_addr.s_addr) --max;
}
}
-
- // second match: traced address doesn't responds on ping requests
- if(max == MAX_HOPS) {
- while((max > 1) && (host[max - 1].addr == host[max - 2].addr) && (host[max - 1].addr != 0) ) max--;
- }
-
ReleaseMutex(ghMutex);
return max;
}
-void WinMTRNet::SetAddr(int at, __int32 addr)
+void WinMTRNet::SetAddr(int at, u_long addr)
{
WaitForSingleObject(ghMutex, INFINITE);
- if(host[at].addr == 0 && addr != 0) {
- TRACE_MSG("Start DnsResolverThread for new address " << addr << ". Old addr value was " << host[at].addr);
- host[at].addr = addr;
- dns_resolver_thread *dnt = new dns_resolver_thread;
- dnt->index = at;
- dnt->winmtr = this;
+ if(host[at].addr.sin_addr.s_addr==0){
+ TRACE_MSG("Start DnsResolverThread for new address " << addr << ". Old addr value was " << host[at].addr.sin_addr.s_addr);
+ host[at].addr.sin_family=AF_INET;
+ host[at].addr.sin_addr.s_addr=addr;
+ dns_resolver_thread *dnt=new dns_resolver_thread;
+ dnt->index=at;
+ dnt->winmtr=this;
if(wmtrdlg->useDNS) _beginthread(DnsResolverThread, 0, dnt);
+ else DnsResolverThread(dnt);
}
-
ReleaseMutex(ghMutex);
}
-void WinMTRNet::SetName(int at, char *n)
+void WinMTRNet::SetAddr6(int at, IPV6_ADDRESS_EX addrex)
{
WaitForSingleObject(ghMutex, INFINITE);
- strcpy(host[at].name, n);
+ if(!(host[at].addr6.sin6_addr.u.Word[0]|host[at].addr6.sin6_addr.u.Word[1]|host[at].addr6.sin6_addr.u.Word[2]|host[at].addr6.sin6_addr.u.Word[3]|host[at].addr6.sin6_addr.u.Word[4]|host[at].addr6.sin6_addr.u.Word[5]|host[at].addr6.sin6_addr.u.Word[6]|host[at].addr6.sin6_addr.u.Word[7])){
+ TRACE_MSG("Start DnsResolverThread for new address " << addrex.sin6_addr[0] << ". Old addr value was " << host[at].addr6.sin6_addr.u.Word[0]);
+ host[at].addr6.sin6_family=AF_INET6;
+ host[at].addr6.sin6_addr=*(in6_addr*)&addrex.sin6_addr;
+ dns_resolver_thread* dnt=new dns_resolver_thread;
+ dnt->index=at;
+ dnt->winmtr=this;
+ if(wmtrdlg->useDNS) _beginthread(DnsResolverThread,0,dnt);
+ else DnsResolverThread(dnt);
+ }
ReleaseMutex(ghMutex);
}
-void WinMTRNet::SetBest(int at, int current)
+void WinMTRNet::SetName(int at, char *n)
{
WaitForSingleObject(ghMutex, INFINITE);
- if(host[at].best > current || host[at].xmit == 1) {
- host[at].best = current;
- };
- if(host[at].worst < current) {
- host[at].worst = current;
- }
-
+ strcpy(host[at].name, n);
ReleaseMutex(ghMutex);
}
-void WinMTRNet::SetWorst(int at, int current)
-{
+void WinMTRNet::SetErrorName(int at, DWORD errnum){
+ const char* name;
+ switch(errnum){
+ case IP_BUF_TOO_SMALL:
+ name="Reply buffer too small.";break;
+ case IP_DEST_NET_UNREACHABLE:
+ name="Destination network unreachable.";break;
+ case IP_DEST_HOST_UNREACHABLE:
+ name="Destination host unreachable.";break;
+ case IP_DEST_PROT_UNREACHABLE:
+ name="Destination protocol unreachable.";break;
+ case IP_DEST_PORT_UNREACHABLE:
+ name="Destination port unreachable.";break;
+ case IP_NO_RESOURCES:
+ name="Insufficient IP resources were available.";break;
+ case IP_BAD_OPTION:
+ name="Bad IP option was specified.";break;
+ case IP_HW_ERROR:
+ name="Hardware error occurred.";break;
+ case IP_PACKET_TOO_BIG:
+ name="Packet was too big.";break;
+ case IP_REQ_TIMED_OUT:
+ name="Request timed out.";break;
+ case IP_BAD_REQ:
+ name="Bad request.";break;
+ case IP_BAD_ROUTE:
+ name="Bad route.";break;
+ case IP_TTL_EXPIRED_REASSEM:
+ name="The time to live expired during fragment reassembly.";break;
+ case IP_PARAM_PROBLEM:
+ name="Parameter problem.";break;
+ case IP_SOURCE_QUENCH:
+ name="Datagrams are arriving too fast to be processed and datagrams may have been discarded.";break;
+ case IP_OPTION_TOO_BIG:
+ name="An IP option was too big.";break;
+ case IP_BAD_DESTINATION:
+ name="Bad destination.";break;
+ case IP_GENERAL_FAILURE:
+ name="General failure.";break;
+ default:
+ TRACE_MSG("==UNKNOWN ERROR== " << errnum);
+ name="Unknown error! (please report)";break;
+ }
WaitForSingleObject(ghMutex, INFINITE);
+ if(!*host[at].name)
+ strcpy(host[at].name,name);
ReleaseMutex(ghMutex);
}
-void WinMTRNet::SetLast(int at, int last)
+void WinMTRNet::UpdateRTT(int at, int rtt)
{
WaitForSingleObject(ghMutex, INFINITE);
- host[at].last = last;
- host[at].total += last;
+ host[at].last=rtt;
+ host[at].total+=rtt;
+ if(host[at].best>rtt || host[at].xmit==1)
+ host[at].best=rtt;
+ if(host[at].worstwinmtr;
+ dns_resolver_thread* dnt=(dns_resolver_thread*)p;
+ WinMTRNet* wn=dnt->winmtr;
+ char hostname[NI_MAXHOST];
+ if(!getnameinfo(wn->GetAddr(dnt->index),sizeof(sockaddr_in6),hostname,NI_MAXHOST,NULL,0,NI_NUMERICHOST)){
+ wn->SetName(dnt->index,hostname);
+ }
+ if(wn->wmtrdlg->useDNS){
+ TRACE_MSG("DNS resolver thread started.");
+ if(!getnameinfo(wn->GetAddr(dnt->index),sizeof(sockaddr_in6),hostname,NI_MAXHOST,NULL,0,0)){
+ wn->SetName(dnt->index,hostname);
+ }
+ TRACE_MSG("DNS resolver thread stopped.");
+ }
+ delete p;
+}
+
+
+
+
+
+
+
+
+
- struct hostent *phent ;
- char buf[100];
- int addr = wn->GetAddr(dnt->index);
- sprintf (buf, "%d.%d.%d.%d", (addr >> 24) & 0xff, (addr >> 16) & 0xff, (addr >> 8) & 0xff, addr & 0xff);
- int haddr = htonl(addr);
- phent = gethostbyaddr( (const char*)&haddr, sizeof(int), AF_INET);
- if(phent) {
- wn->SetName(dnt->index, phent->h_name);
- } else {
- wn->SetName(dnt->index, buf);
- }
-
- delete p;
- TRACE_MSG("DNS resolver thread stopped.");
- _endthread();
-}
diff --git a/src/WinMTRNet.h b/src/WinMTRNet.h
index a372a57..d3206d6 100644
--- a/src/WinMTRNet.h
+++ b/src/WinMTRNet.h
@@ -16,18 +16,20 @@
class WinMTRDialog;
-typedef ip_option_information IPINFO, *PIPINFO, FAR *LPIPINFO;
-
+typedef IP_OPTION_INFORMATION IPINFO, *PIPINFO, FAR *LPIPINFO;
#ifdef _WIN64
-typedef icmp_echo_reply32 ICMPECHO, *PICMPECHO, FAR *LPICMPECHO;
+typedef ICMP_ECHO_REPLY32 ICMPECHO, *PICMPECHO, FAR *LPICMPECHO;
#else
-typedef icmp_echo_reply ICMPECHO, *PICMPECHO, FAR *LPICMPECHO;
-#endif
+typedef ICMP_ECHO_REPLY ICMPECHO, *PICMPECHO, FAR *LPICMPECHO;
+#endif // _WIN64
#define ECHO_REPLY_TIMEOUT 5000
struct s_nethost {
- __int32 addr; // IP as a decimal, big endian
+ union{
+ sockaddr_in addr;
+ sockaddr_in6 addr6;
+ };
int xmit; // number of PING packets sent
int returned; // number of ICMP echo replies received
unsigned long total; // total time
@@ -44,19 +46,25 @@ struct s_nethost {
//*****************************************************************************
class WinMTRNet {
+ typedef FARPROC PIO_APC_ROUTINE;//not the best way to do it, but works ;) (we do not use it anyway)
+ //IPv4
typedef HANDLE (WINAPI *LPFNICMPCREATEFILE)(VOID);
typedef BOOL (WINAPI *LPFNICMPCLOSEHANDLE)(HANDLE);
- typedef DWORD (WINAPI *LPFNICMPSENDECHO)(HANDLE, u_long, LPVOID, WORD, LPVOID, LPVOID, DWORD, DWORD);
+ typedef DWORD (WINAPI *LPFNICMPSENDECHO2)(HANDLE IcmpHandle,HANDLE Event,PIO_APC_ROUTINE ApcRoutine,PVOID ApcContext,in_addr DestinationAddress,LPVOID RequestData,WORD RequestSize,PIP_OPTION_INFORMATION RequestOptions,LPVOID ReplyBuffer,DWORD ReplySize,DWORD Timeout);
+ //IPv6
+ typedef HANDLE (WINAPI *LPFNICMP6CREATEFILE)(VOID);
+ typedef BOOL (WINAPI *LPFNICMP6CLOSEHANDLE)(HANDLE);
+ typedef DWORD (WINAPI *LPFNICMP6SENDECHO2)(HANDLE IcmpHandle,HANDLE Event,PIO_APC_ROUTINE ApcRoutine,PVOID ApcContext,sockaddr_in6* SourceAddress,sockaddr_in6* DestinationAddress,LPVOID RequestData,WORD RequestSize,PIP_OPTION_INFORMATION RequestOptions,LPVOID ReplyBuffer,DWORD ReplySize,DWORD Timeout);
public:
WinMTRNet(WinMTRDialog *wp);
~WinMTRNet();
- void DoTrace(int address);
+ void DoTrace(sockaddr* sockaddr);
void ResetHops();
void StopTrace();
- int GetAddr(int at);
+ sockaddr* GetAddr(int at);
int GetName(int at, char *n);
int GetBest(int at);
int GetWorst(int at);
@@ -67,22 +75,31 @@ class WinMTRNet {
int GetXmit(int at);
int GetMax();
- void SetAddr(int at, __int32 addr);
+ void SetAddr(int at, u_long addr);
+ void SetAddr6(int at, IPV6_ADDRESS_EX addrex);
void SetName(int at, char *n);
- void SetBest(int at, int current);
- void SetWorst(int at, int current);
- void SetLast(int at, int last);
+ void SetErrorName(int at,DWORD errnum);
+ void UpdateRTT(int at, int rtt);
void AddReturned(int at);
void AddXmit(int at);
WinMTRDialog *wmtrdlg;
- __int32 last_remote_addr;
+ union{
+ in_addr last_remote_addr;
+ in6_addr last_remote_addr6;
+ };
+ bool hasIPv6;
bool tracing;
bool initialized;
- HANDLE hICMP;
- LPFNICMPCREATEFILE lpfnIcmpCreateFile;
+ HANDLE hICMP;
+ HANDLE hICMP6;
+ //IPv4
+ LPFNICMPCREATEFILE lpfnIcmpCreateFile;
LPFNICMPCLOSEHANDLE lpfnIcmpCloseHandle;
- LPFNICMPSENDECHO lpfnIcmpSendEcho;
+ LPFNICMPSENDECHO2 lpfnIcmpSendEcho2;
+ //IPv6
+ LPFNICMP6CREATEFILE lpfnIcmp6CreateFile;
+ LPFNICMP6SENDECHO2 lpfnIcmp6SendEcho2;
private:
HINSTANCE hICMP_DLL;
diff --git a/src/WinMTRProperties.h b/src/WinMTRProperties.h
index 8a7dafc..8eae2d7 100644
--- a/src/WinMTRProperties.h
+++ b/src/WinMTRProperties.h
@@ -30,7 +30,7 @@ class WinMTRProperties : public CDialog
enum { IDD = IDD_DIALOG_PROPERTIES };
char host[255];
- char ip[16];
+ char ip[40];
char comment[255];
float ping_last;
diff --git a/src/WinMTRStatusBar.cpp b/src/WinMTRStatusBar.cpp
index 5d6ba1d..2903079 100644
--- a/src/WinMTRStatusBar.cpp
+++ b/src/WinMTRStatusBar.cpp
@@ -60,7 +60,7 @@ LRESULT WinMTRStatusBar::WindowProc(UINT message, WPARAM wParam, LPARAM lParam)
void WinMTRStatusBar::RepositionControls()
{
- HDWP _hDWP = ::BeginDeferWindowPos( m_arrPaneControls.GetSize() );
+ HDWP _hDWP = ::BeginDeferWindowPos( (int)m_arrPaneControls.GetSize() );
CRect rcClient;
GetClientRect(&rcClient);
@@ -170,7 +170,7 @@ BOOL WinMTRStatusBar::AddPane(
arrPanesTmp.Add(pNewPane);
}
- int nPanesCount = arrPanesTmp.GetSize();
+ int nPanesCount = (int)arrPanesTmp.GetSize();
UINT* lpIDArray = new UINT[ nPanesCount ];
for (iIndex = 0; iIndex < nPanesCount; iIndex++) {
lpIDArray[iIndex] = arrPanesTmp[iIndex]->nID;
@@ -232,7 +232,7 @@ BOOL WinMTRStatusBar::RemovePane(
}
// set the indicators
- SetIndicators(lpIDArray, arrPanesTmp.GetSize());
+ SetIndicators(lpIDArray, (int)arrPanesTmp.GetSize());
// free memory
for (nIndex = 0; nIndex < arrPanesTmp.GetSize(); nIndex++){
_STATUSBAR_PANE_* pPane = arrPanesTmp[nIndex];
diff --git a/src/resource.h b/src/resource.h
index 1ca45e3..955b8e3 100644
--- a/src/resource.h
+++ b/src/resource.h
@@ -39,7 +39,7 @@
#define IDC_STATICJ 1023
#define IDC_COMBO_HOST 1024
#define IDC_EDIT_MAX_LRU 1025
-#define IDC_MFCLINK1 1026
+#define IDC_CHECK_IPV6 1026
// Next default values for new objects
//