Windows2003-3790/base/ntos/rtl/add2strt.h
2020-09-30 16:53:55 +02:00

365 lines
9.1 KiB
C

/*++
Copyright (c) 2001 Microsoft Corporation
Module Name:
add2strt.h
Abstract:
Code for IP address-to-string translation routines.
Author:
Dave Thaler (dthaler) 3-28-2001
Revision History:
IPv6 conversion code originally by Rich Draves (richdr)
--*/
struct in6_addr {
union {
UCHAR Byte[16];
USHORT Word[8];
} u;
};
#define s6_bytes u.Byte
#define s6_words u.Word
struct in_addr {
union {
struct { UCHAR s_b1,s_b2,s_b3,s_b4; } S_un_b;
struct { USHORT s_w1,s_w2; } S_un_w;
ULONG S_addr;
} S_un;
};
#define s_addr S_un.S_addr
#define AF_INET 2
#define AF_INET6 23
#define INET_ADDRSTRLEN 22
#define INET6_ADDRSTRLEN 65
LPTSTR
RtlIpv6AddressToStringT(
IN const struct in6_addr *Addr,
OUT LPTSTR S
)
/*++
Routine Description:
Generates an IPv6 string literal corresponding to the address Addr.
The shortened canonical forms are used (RFC 1884 etc).
The basic string representation consists of 8 hex numbers
separated by colons, with a couple embellishments:
- a string of zero numbers (at most one) is replaced
with a double-colon.
- the last 32 bits are represented in IPv4-style dotted-octet notation
if the address is a v4-compatible or ISATAP address.
For example,
::
::1
::157.56.138.30
::ffff:156.56.136.75
ff01::
ff02::2
0:1:2:3:4:5:6:7
Arguments:
S - Receives a pointer to the buffer in which to place the
string literal.
Addr - Receives the IPv6 address.
Return Value:
Pointer to the null byte at the end of the string inserted.
This can be used by the caller to easily append more information.
--*/
{
int maxFirst, maxLast;
int curFirst, curLast;
int i;
int endHex = 8;
// Check for IPv6-compatible, IPv4-mapped, and IPv4-translated
// addresses
if ((Addr->s6_words[0] == 0) && (Addr->s6_words[1] == 0) &&
(Addr->s6_words[2] == 0) && (Addr->s6_words[3] == 0) &&
(Addr->s6_words[6] != 0)) {
if ((Addr->s6_words[4] == 0) &&
((Addr->s6_words[5] == 0) || (Addr->s6_words[5] == 0xffff)))
{
// compatible or mapped
S += _stprintf(S, _T("::%hs%u.%u.%u.%u"),
Addr->s6_words[5] == 0 ? "" : "ffff:",
Addr->s6_bytes[12], Addr->s6_bytes[13],
Addr->s6_bytes[14], Addr->s6_bytes[15]);
return S;
}
else if ((Addr->s6_words[4] == 0xffff) && (Addr->s6_words[5] == 0)) {
// translated
S += _stprintf(S, _T("::ffff:0:%u.%u.%u.%u"),
Addr->s6_bytes[12], Addr->s6_bytes[13],
Addr->s6_bytes[14], Addr->s6_bytes[15]);
return S;
}
}
// Find largest contiguous substring of zeroes
// A substring is [First, Last), so it's empty if First == Last.
maxFirst = maxLast = 0;
curFirst = curLast = 0;
// ISATAP EUI64 starts with 00005EFE (or 02005EFE)...
if (((Addr->s6_words[4] & 0xfffd) == 0) && (Addr->s6_words[5] == 0xfe5e)) {
endHex = 6;
}
for (i = 0; i < endHex; i++) {
if (Addr->s6_words[i] == 0) {
// Extend current substring
curLast = i+1;
// Check if current is now largest
if (curLast - curFirst > maxLast - maxFirst) {
maxFirst = curFirst;
maxLast = curLast;
}
}
else {
// Start a new substring
curFirst = curLast = i+1;
}
}
// Ignore a substring of length 1.
if (maxLast - maxFirst <= 1)
maxFirst = maxLast = 0;
// Write colon-separated words.
// A double-colon takes the place of the longest string of zeroes.
// All zeroes is just "::".
for (i = 0; i < endHex; i++) {
// Skip over string of zeroes
if ((maxFirst <= i) && (i < maxLast)) {
S += _stprintf(S, _T("::"));
i = maxLast-1;
continue;
}
// Need colon separator if not at beginning
if ((i != 0) && (i != maxLast))
S += _stprintf(S, _T(":"));
S += _stprintf(S, _T("%x"), RtlUshortByteSwap(Addr->s6_words[i]));
}
if (endHex < 8) {
S += _stprintf(S, _T(":%u.%u.%u.%u"),
Addr->s6_bytes[12], Addr->s6_bytes[13],
Addr->s6_bytes[14], Addr->s6_bytes[15]);
}
return S;
}
NTSTATUS
RtlIpv6AddressToStringExT(
IN const struct in6_addr *Address,
IN ULONG ScopeId,
IN USHORT Port,
OUT LPTSTR AddressString,
IN OUT PULONG AddressStringLength
)
/*++
Routine Description:
This is the extension routine which handles a full address conversion
including address, scopeid and port (scopeid and port are optional).
Arguments:
Address - The address part to be translated.
ScopeId - The Scope ID of the address (optional).
Port - The port number of the address (optional).
Port is in network byte order.
AddressString - Pointer to output buffer where we will fill in address string.
AddressStringLength - For input, it is the length of the input buffer; for
output it is the length we actual returned.
Return Value:
STATUS_SUCCESS if the operation is successful, error code otherwise.
--*/
{
TCHAR String[INET6_ADDRSTRLEN];
LPTSTR S;
ULONG Length;
if ((Address == NULL) ||
(AddressString == NULL) ||
(AddressStringLength == NULL)) {
return STATUS_INVALID_PARAMETER;
}
S = String;
if (Port) {
S += _stprintf(S, _T("["));
}
//
// Now translate this address.
//
S = RtlIpv6AddressToStringT(Address, S);
if (ScopeId != 0) {
S += _stprintf(S, _T("%%%u"), ScopeId);
}
if (Port != 0) {
S += _stprintf(S, _T("]:%u"), RtlUshortByteSwap(Port));
}
Length = (ULONG)(S - String + 1);
if (*AddressStringLength < Length) {
//
// Before return, tell the caller how big
// the buffer we need.
//
*AddressStringLength = Length;
return STATUS_INVALID_PARAMETER;
}
*AddressStringLength = Length;
RtlCopyMemory(AddressString, String, Length * sizeof(TCHAR));
return STATUS_SUCCESS;
}
LPTSTR
RtlIpv4AddressToStringT(
IN const struct in_addr *Addr,
OUT LPTSTR S
)
/*++
Routine Description:
Generates an IPv4 string literal corresponding to the address Addr.
Arguments:
S - Receives a pointer to the buffer in which to place the
string literal.
Addr - Receives the IPv4 address.
Return Value:
Pointer to the null byte at the end of the string inserted.
This can be used by the caller to easily append more information.
--*/
{
S += _stprintf(S, _T("%u.%u.%u.%u"),
( Addr->s_addr >> 0 ) & 0xFF,
( Addr->s_addr >> 8 ) & 0xFF,
( Addr->s_addr >> 16 ) & 0xFF,
( Addr->s_addr >> 24 ) & 0xFF );
return S;
}
NTSTATUS
RtlIpv4AddressToStringExT(
IN const struct in_addr *Address,
IN USHORT Port,
OUT LPTSTR AddressString,
IN OUT PULONG AddressStringLength
)
/*++
Routine Description:
This is the extension routine which handles a full address conversion
including address and port (port is optional).
Arguments:
Address - The address part to translate.
Port - Port number if there is any, otherwise 0. Port is in network
byte order.
AddressString - Receives the formatted address string.
AddressStringLength - On input, contains the length of AddressString.
On output, contains the number of characters actually written
to AddressString.
Return Value:
STATUS_SUCCESS if the operation is successful, error code otherwise.
--*/
{
TCHAR String[INET_ADDRSTRLEN];
LPTSTR S;
ULONG Length;
//
// Quick sanity checks.
//
if ((Address == NULL) ||
(AddressString == NULL) ||
(AddressStringLength == NULL)) {
return STATUS_INVALID_PARAMETER;
}
S = String;
//
// Now translate this address.
//
S = RtlIpv4AddressToStringT(Address, S);
if (Port != 0) {
S += _stprintf(S, _T(":%u"), RtlUshortByteSwap(Port));
}
Length = (ULONG)(S - String + 1);
if (*AddressStringLength < Length) {
//
// Before return, tell the caller how big
// the buffer we need.
//
*AddressStringLength = Length;
return STATUS_INVALID_PARAMETER;
}
RtlCopyMemory(AddressString, String, Length * sizeof(TCHAR));
*AddressStringLength = Length;
return STATUS_SUCCESS;
}