/*++ 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; }