windows-nt/Source/XPSP1/NT/net/ias/mmc/proxy/resolver.cpp
2020-09-26 16:20:57 +08:00

339 lines
7.3 KiB
C++

///////////////////////////////////////////////////////////////////////////////
//
// Copyright (c) 2000, Microsoft Corp. All rights reserved.
//
// FILE
//
// resolver.cpp
//
// SYNOPSIS
//
// Defines the class Resolver.
//
// MODIFICATION HISTORY
//
// 02/28/2000 Original version.
// 05/11/2000 Make OK button the default if name resolution succeeds.
//
///////////////////////////////////////////////////////////////////////////////
#include <proxypch.h>
#include <resolver.h>
#include <winsock2.h>
#include <svcguid.h>
/////////
// Unicode version of gethostbyname. The caller must free the returned hostent
// struct by calling LocalFree.
/////////
PHOSTENT
WINAPI
IASGetHostByName(
IN PCWSTR name
)
{
// We put these at function scope, so we can clean them up on the way out.
DWORD error = NO_ERROR;
HANDLE lookup = NULL;
union
{
WSAQUERYSETW querySet;
BYTE buffer[512];
};
PWSAQUERYSETW result = NULL;
PHOSTENT retval = NULL;
do
{
//////////
// Create the query set
//////////
GUID hostAddrByNameGuid = SVCID_INET_HOSTADDRBYNAME;
AFPROTOCOLS protocols[2] =
{
{ AF_INET, IPPROTO_UDP },
{ AF_INET, IPPROTO_TCP }
};
memset(&querySet, 0, sizeof(querySet));
querySet.dwSize = sizeof(querySet);
querySet.lpszServiceInstanceName = (PWSTR)name;
querySet.lpServiceClassId = &hostAddrByNameGuid;
querySet.dwNameSpace = NS_ALL;
querySet.dwNumberOfProtocols = 2;
querySet.lpafpProtocols = protocols;
//////////
// Execute the query.
//////////
error = WSALookupServiceBeginW(
&querySet,
LUP_RETURN_ADDR,
&lookup
);
if (error)
{
error = WSAGetLastError();
break;
}
//////////
// How much space do we need for the result?
//////////
DWORD length = sizeof(buffer);
error = WSALookupServiceNextW(
lookup,
0,
&length,
&querySet
);
if (!error)
{
result = &querySet;
}
else
{
error = WSAGetLastError();
if (error != WSAEFAULT)
{
break;
}
/////////
// Allocate memory to hold the result.
/////////
result = (PWSAQUERYSETW)LocalAlloc(0, length);
if (!result)
{
error = WSA_NOT_ENOUGH_MEMORY;
break;
}
/////////
// Get the result.
/////////
error = WSALookupServiceNextW(
lookup,
0,
&length,
result
);
if (error)
{
error = WSAGetLastError();
break;
}
}
if (result->dwNumberOfCsAddrs == 0)
{
error = WSANO_DATA;
break;
}
///////
// Allocate memory to hold the hostent struct
///////
DWORD naddr = result->dwNumberOfCsAddrs;
SIZE_T nbyte = sizeof(hostent) +
(naddr + 1) * sizeof(char*) +
naddr * sizeof(in_addr);
retval = (PHOSTENT)LocalAlloc(0, nbyte);
if (!retval)
{
error = WSA_NOT_ENOUGH_MEMORY;
break;
}
///////
// Initialize the hostent struct.
///////
retval->h_name = NULL;
retval->h_aliases = NULL;
retval->h_addrtype = AF_INET;
retval->h_length = sizeof(in_addr);
retval->h_addr_list = (char**)(retval + 1);
///////
// Store the addresses.
///////
u_long* nextAddr = (u_long*)(retval->h_addr_list + naddr + 1);
for (DWORD i = 0; i < naddr; ++i)
{
sockaddr_in* sin = (sockaddr_in*)
result->lpcsaBuffer[i].RemoteAddr.lpSockaddr;
retval->h_addr_list[i] = (char*)nextAddr;
*nextAddr++ = sin->sin_addr.S_un.S_addr;
}
///////
// NULL terminate the address list.
///////
retval->h_addr_list[i] = NULL;
} while (FALSE);
//////////
// Clean up and return.
//////////
if (result && result != &querySet) { LocalFree(result); }
if (lookup) { WSALookupServiceEnd(lookup); }
if (error)
{
if (error == WSASERVICE_NOT_FOUND) { error = WSAHOST_NOT_FOUND; }
WSASetLastError(error);
}
return retval;
}
Resolver::Resolver(PCWSTR dnsName, CWnd* pParent)
: CHelpDialog(IDD_RESOLVE_ADDRESS, pParent),
name(dnsName),
choice(name)
{
WSADATA wsaData;
WSAStartup(MAKEWORD(2, 0), &wsaData);
}
Resolver::~Resolver()
{
WSACleanup();
}
BOOL Resolver::OnInitDialog()
{
/////////
// Subclass the list control.
/////////
if (!results.SubclassWindow(::GetDlgItem(m_hWnd, IDC_LIST_IPADDRS)))
{
AfxThrowNotSupportedException();
}
/////////
// Set the column header.
/////////
RECT rect;
results.GetClientRect(&rect);
LONG width = rect.right - rect.left;
ResourceString addrsCol(IDS_RESOLVER_COLUMN_ADDRS);
results.InsertColumn(0, addrsCol, LVCFMT_LEFT, width);
results.SetExtendedStyle(results.GetExtendedStyle() | LVS_EX_FULLROWSELECT);
return CHelpDialog::OnInitDialog();
}
void Resolver::DoDataExchange(CDataExchange* pDX)
{
CHelpDialog::DoDataExchange(pDX);
DDX_Text(pDX, IDC_EDIT_NAME, name);
if (pDX->m_bSaveAndValidate)
{
int item = results.GetNextItem(-1, LVNI_SELECTED);
choice = (item >= 0) ? results.GetItemText(item, 0) : name;
}
}
void Resolver::OnResolve()
{
// Remove the existing result set.
results.DeleteAllItems();
// Get the name to resolve.
UpdateData();
// Change the cursor to busy signal since this will block.
BeginWaitCursor();
// Resolve the hostname.
PHOSTENT he = IASGetHostByName(name);
// The blocking part is over, so restore the cursor.
EndWaitCursor();
if (he)
{
// Add the IP addresses to the combo box.
for (ULONG i = 0; he->h_addr_list[i] && i < 8; ++i)
{
PBYTE p = (PBYTE)he->h_addr_list[i];
WCHAR szAddr[16];
wsprintfW(szAddr, L"%u.%u.%u.%u", p[0], p[1], p[2], p[3]);
results.InsertItem(i, szAddr);
}
// Free the results.
LocalFree(he);
// Make the OK button the default ...
setButtonStyle(IDOK, BS_DEFPUSHBUTTON, true);
setButtonStyle(IDC_BUTTON_RESOLVE, BS_DEFPUSHBUTTON, false);
// ... and give it the focus.
setFocusControl(IDOK);
}
else
{
ResourceString text(IDS_SERVER_E_NO_RESOLVE);
ResourceString caption(IDS_SERVER_E_CAPTION);
MessageBox(text, caption, MB_ICONWARNING);
}
}
BEGIN_MESSAGE_MAP(Resolver, CHelpDialog)
ON_BN_CLICKED(IDC_BUTTON_RESOLVE, OnResolve)
END_MESSAGE_MAP()
void Resolver::setButtonStyle(int controlId, long flags, bool set)
{
// Get the button handle.
HWND button = ::GetDlgItem(m_hWnd, controlId);
// Retrieve the current style.
long style = ::GetWindowLong(button, GWL_STYLE);
// Update the flags.
if (set)
{
style |= flags;
}
else
{
style &= ~flags;
}
// Set the new style.
::SendMessage(button, BM_SETSTYLE, LOWORD(style), MAKELPARAM(1,0));
}
void Resolver::setFocusControl(int controlId)
{
::SetFocus(::GetDlgItem(m_hWnd, controlId));
}