// -*- mode: C++; tab-width: 4; indent-tabs-mode: nil -*- (for GNU Emacs) // // Copyright (c) 1985-2000 Microsoft Corporation // // This file is part of the Microsoft Research IPv6 Network Protocol Stack. // You should have received a copy of the Microsoft End-User License Agreement // for this software along with this release; see the file "license.txt". // If not, please see http://www.research.microsoft.com/msripv6/license.htm, // or write to Microsoft Research, One Microsoft Way, Redmond, WA 98052-6399. // // Abstract: // // IPv6 private definitions. // // This file contains all of the definitions for IPv6 that // are not visible to outside layers. // #ifndef IPv6DEF_INCLUDED #define IPv6DEF_INCLUDED 1 #pragma warning(push) #pragma warning(disable:4200) // zero-sized array in struct/union typedef struct NeighborCacheEntry NeighborCacheEntry; typedef struct AddressEntry AddressEntry; typedef struct MulticastAddressEntry MulticastAddressEntry; typedef struct AnycastAddressEntry AnycastAddressEntry; typedef struct NetTableEntryOrInterface NetTableEntryOrInterface; typedef struct NetTableEntry NetTableEntry; typedef struct Interface Interface; typedef struct IPSecProc IPSecProc; // REVIEW: Added so the build will work. typedef unsigned long IPAddr; #define INADDR_ANY 0 #ifndef ABORTMSG #if DBG #define ABORT() \ RtlAssert( "FALSE", __FILE__, __LINE__, NULL ) #define ABORTMSG(msg) \ RtlAssert( "FALSE", __FILE__, __LINE__, (msg) ) #else #define ABORT() #define ABORTMSG(msg) #endif // DBG #endif // ABORTMSG // // Per-neighbor information. We keep address translation and unreachability // detection info for each of our neighbors that we're in communication with. // // A non-zero reference count prevents the NCE from being reclaimed. // An NCE with zero references may be kept cached. // A per-interface lock protects all NCEs for that interface. // // NCEs with a non-zero reference count hold a reference for their interface. // NCEs with a zero reference count do not hold a reference. // This means if you hold a reference for an NCE, // you can always safely access and dereference NCE->IF. // // The Next/Prev fields link NCEs into a circular doubly-linked list. // They must be first and must match the IF->FirstNCE/LastNCE fields // to make the casting work out. // // The list of NCEs is kept sorted, from most-recently-used to least. // struct NeighborCacheEntry { // a.k.a. NCE NeighborCacheEntry *Next; // Next entry on I/F neighbor list. NeighborCacheEntry *Prev; // Previous entry on I/F neighbor list. IPv6Addr NeighborAddress; // Address of I/F on neighboring node. void *LinkAddress; // Media address corresponding to above. // NB: LinkAddressLength field not needed - use IF->LinkAddressLength. ushort IsRouter:1, // Is the neighbor a router? IsUnreachable:1, // Does ND indicate unreachability? // DoRoundRobin is only meaningful if IsUnreachable is TRUE. DoRoundRobin:1, // Should FindNextHop do round-robin? IsLoopback:1; // Do we loopback to this neighbor // in software? ushort NDState; // Neighbor Discovery Protocol state. uint LastReachability; // Timestamp (IPv6Timer ticks). ushort NSTimer; // In IPv6Timer ticks (see IPv6Timeout). uchar NSCount; // Number of solicits sent so far. uchar NSLimit; // Total number of solicits to send. Interface *IF; // Interface on media with neighbor. NDIS_PACKET *WaitQueue; // Queue of packets waiting on ND. long RefCnt; // Reference count - interlocked. }; // // The caller must already have a reference for the NCE. // The interface need not be locked. // __inline void AddRefNCE(NeighborCacheEntry *NCE) { long RefCnt = InterlockedIncrement(&NCE->RefCnt); ASSERT(RefCnt != 1); UNREFERENCED_PARAMETER(RefCnt); } extern void AddRefNCEInCache(NeighborCacheEntry *NCE); extern void ReleaseNCE(NeighborCacheEntry *NCE); // // Values for "NDState" above. See RFC 1970, section 7.3.2 for details. // Note: only state names are documented, we chose the values used here. // // In the INCOMPLETE state, the LinkAddress is not valid. // In all other states, LinkAddress may be used to send packets. // WaitQueue is usually only non-NULL in the INCOMPLETE state, // but sometimes a packet is left queued for NeighborCacheTimeout. // // The INCOMPLETE state has two flavors, dormant and active. If // EventTimer and EventCount are both zero, then we are not actively // trying to solicit the link address. If someone tries to send to // this neighbor, then we start soliciting the link address. If the // solicitation fails (or if we enter the PROBE state and then fail to // confirm reachability), then any waiting packets are discarded and // we reset to INCOMPLETE with zero EventTimer/EventCount. (So with // the next use of this neighbor, we start soliciting again from scratch.) // // The DELAY state is not used internally. Instead we use the PROBE state // with zero NSCount and non-zero NSTimer to indicate that we are delaying // the start of probing. However link-layer lip_cvaddr functions can // return ND_STATE_DELAY and IoctlQueryNeighborCache returns ND_STATE_DELAY. // // The IsUnreachable flag tracks separately whether the neighbor is // *known* to be unreachable. For example, a new NCE will be in in the // INCOMPLETE state, but IsUnreachable is FALSE because we don't know // yet whether the neighbor is reachable. Because FindNextHop uses // IsUnreachable, code paths that change this flag must call // InvalidateRouteCache. // // These definitions are also in llip6if.h and ntddip6.w. // #define ND_STATE_INCOMPLETE 0 #define ND_STATE_PROBE 1 #define ND_STATE_DELAY 2 // Not used internally. #define ND_STATE_STALE 3 #define ND_STATE_REACHABLE 4 #define ND_STATE_PERMANENT 5 // // There are a few places in the implementation where we need // to pass a pointer which is either a NetTableEntry or an Interface. // NetTableEntries and Interfaces share this structure as their // first element. With Interfaces, the IF field points back // at the Interface itself. // struct NetTableEntryOrInterface { // a.k.a. NTEorIF Interface *IF; }; __inline int IsNTE(NetTableEntryOrInterface *NTEorIF) { return (NetTableEntryOrInterface *)NTEorIF->IF != NTEorIF; } __inline NetTableEntry * CastToNTE(NetTableEntryOrInterface *NTEorIF) { ASSERT(IsNTE(NTEorIF)); return (NetTableEntry *) NTEorIF; } __inline NetTableEntryOrInterface * CastFromNTE(NetTableEntry *NTE) { return (NetTableEntryOrInterface *) NTE; } __inline int IsIF(NetTableEntryOrInterface *NTEorIF) { return (NetTableEntryOrInterface *)NTEorIF->IF == NTEorIF; } __inline Interface * CastToIF(NetTableEntryOrInterface *NTEorIF) { ASSERT(IsIF(NTEorIF)); return (Interface *) NTEorIF; } __inline NetTableEntryOrInterface * CastFromIF(Interface *IF) { return (NetTableEntryOrInterface *) IF; } // // Local address information. Each interface keeps track of the addresses // assigned to the interface. Depending on the type of address, each // ADE structure is the first element of a larger NTE, MAE, or AAE structure. // The address information is protected by the interface's lock. // // The NTEorIF field must be first. In NTEs, it points to the interface // and holds a reference for the interface. In MAEs and AAEs, it can // point to the interface or to one of the NTEs on the interface but in // either case it does NOT hold a reference. // struct AddressEntry { // a.k.a. ADE union { Interface *IF; NetTableEntry *NTE; NetTableEntryOrInterface *NTEorIF; }; AddressEntry *Next; // Linkage on chain. IPv6Addr Address; // Address identifying this entry. ushort Type; // Address type (unicast, multicast, etc). ushort Scope; // Address scope (link, site, global, etc). }; // // Values for address Type. // #define ADE_UNICAST 0x00 #define ADE_ANYCAST 0x01 #define ADE_MULTICAST 0x02 #define ADE_NONE ((ushort)-1) // Indicates absence of an ADE. // // Values for address Scope. // #define ADE_SMALLEST_SCOPE 0x00 #define ADE_INTERFACE_LOCAL 0x01 #define ADE_LINK_LOCAL 0x02 #define ADE_SUBNET_LOCAL 0x03 #define ADE_ADMIN_LOCAL 0x04 #define ADE_SITE_LOCAL 0x05 #define ADE_ORG_LOCAL 0x08 #define ADE_GLOBAL 0x0e #define ADE_LARGEST_SCOPE 0x0f #define ADE_NUM_SCOPES (ADE_LARGEST_SCOPE - ADE_SMALLEST_SCOPE + 1) // // Multicast ADEs are really MAEs. // // MAEs can be a separate global QueryList. // If an MAE on an Interface has a non-zero MCastTimer value, // then it is on the QueryList. // // An MAE can be on the QueryList with a zero MCastTimer value // only when it is not on any interface and it just needs // a Done message sent before it can be deleted. // When it is in this state (but not otherwise), MAE->IF // holds a reference for the interface. // struct MulticastAddressEntry { // a.k.a. MAE AddressEntry; // Inherit the ADE fields. uint MCastRefCount; // Sockets/etc receiving from this group. // // The fields below are protected by the QueryList lock. // ushort MCastFlags:4, // Necessary info about a group. MCastCount:4; // Count of initial reports left to send. ushort MCastTimer; // Ticks until a membership report is sent. MulticastAddressEntry *NextQL; // For the QueryList. }; // // Bit values for MCastFlags. // #define MAE_REPORTABLE 0x01 // We should send Reports. #define MAE_LAST_REPORTER 0x02 // We should send Done. // // Anycast ADEs are really AAEs. // Currently an AAE has no additional fields. // struct AnycastAddressEntry { // a.k.a. AAE AddressEntry; // Inherit the ADE fields. }; // // Unicast ADEs are really NTEs. // There is one NTE for each source (unicast) address // assigned to an interface. // // NTEs hold a reference for their interface, // so if you have a reference for an NTE // you can always safely access and dereference NTE->IF. // // Most NTE fields are either read-only or are protected // by the interface lock. The interface WorkerLock protects // the TdiRegistrationHandle field. // // Temporary addresses (AddrConf == ADDR_CONF_TEMPORARY) // have extra fields - see TempNetTableEntry. // struct NetTableEntry { // a.k.a. NTE AddressEntry; // Inherit the ADE fields. NetTableEntry *NextOnNTL; // Next NTE on NetTableList. NetTableEntry **PrevOnNTL; // Previous Next pointer in NetTableList. HANDLE TdiRegistrationHandle; // Opaque token for TDI De/notification. uint TdiRegistrationScopeId; // Scope Id when registered with TDI. long RefCnt; // Reference count - interlocked. uint ValidLifetime; // In IPv6Timer ticks (see IPv6Timeout). uint PreferredLifetime; // In IPv6Timer ticks (see IPv6Timeout). uchar AddrConf; // Address configuration status. uchar DADState; // Address configuration state. ushort DADCount; // How many DAD solicits left to send. ushort DADTimer; // In IPv6Timer ticks (see IPv6Timeout). }; __inline void AddRefNTE(NetTableEntry *NTE) { InterlockedIncrement(&NTE->RefCnt); } __inline void ReleaseNTE(NetTableEntry *NTE) { InterlockedDecrement(&NTE->RefCnt); } struct AddrConfEntry { union { uchar Value; // Address configuration status. struct { uchar InterfaceIdConf : 4; uchar PrefixConf : 4; }; }; }; // // Values for PrefixConf - must fit in 4 bits. // These must match the values in ntddip6.h, as well as the // IP_PREFIX_ORIGIN values in iptypes.h. // #define PREFIX_CONF_OTHER 0 // None of the ones below. #define PREFIX_CONF_MANUAL 1 // From a user or administrator. #define PREFIX_CONF_WELLKNOWN 2 // IANA-assigned. #define PREFIX_CONF_DHCP 3 // Configured via DHCP. #define PREFIX_CONF_RA 4 // From a Router Advertisement. // // Values for InterfaceIdConf - must fit in 4 bits. // These must match the values in ntddip6.h, as well as the // IP_SUFFIX_ORIGIN values in iptypes.h. // #define IID_CONF_OTHER 0 // None of the ones below. #define IID_CONF_MANUAL 1 // From a user or administrator. #define IID_CONF_WELLKNOWN 2 // IANA-assigned. #define IID_CONF_DHCP 3 // Configured via DHCP. #define IID_CONF_LL_ADDRESS 4 // Derived from the link-layer address. #define IID_CONF_RANDOM 5 // Random, e.g. temporary address. // // Values for AddrConf - must fit in 8 bits. // #define ADDR_CONF_MANUAL ((PREFIX_CONF_MANUAL << 4) | IID_CONF_MANUAL) #define ADDR_CONF_PUBLIC ((PREFIX_CONF_RA << 4) | IID_CONF_LL_ADDRESS) #define ADDR_CONF_TEMPORARY ((PREFIX_CONF_RA << 4) | IID_CONF_RANDOM) #define ADDR_CONF_DHCP ((PREFIX_CONF_DHCP << 4) | IID_CONF_DHCP) #define ADDR_CONF_WELLKNOWN ((PREFIX_CONF_WELLKNOWN << 4) | IID_CONF_WELLKNOWN) #define ADDR_CONF_LINK ((PREFIX_CONF_WELLKNOWN << 4) | IID_CONF_LL_ADDRESS) __inline int IsValidPrefixConfValue(uint PrefixConf) { return PrefixConf < (1 << 4); } __inline int IsValidInterfaceIdConfValue(uint InterfaceIdConf) { return InterfaceIdConf < (1 << 4); } __inline int IsStatelessAutoConfNTE(NetTableEntry *NTE) { return ((struct AddrConfEntry *)&NTE->AddrConf)->PrefixConf == PREFIX_CONF_RA; } // // Values for DADState. // // The "deprecated" and "preferred" states are valid, // meaning that addresses in those two states can be // used as a source address, can receive packets, etc. // The invalid states mean that the address is // not actually assigned to the interface, // using the terminology of RFC 2462. // // Valid<->invalid and deprecated<->preferred transitions // must call InvalidateRouteCache because they affect // source address selection. // // Among valid states, bigger is better // for source address selection. // #define DAD_STATE_INVALID 0 #define DAD_STATE_TENTATIVE 1 #define DAD_STATE_DUPLICATE 2 #define DAD_STATE_DEPRECATED 3 #define DAD_STATE_PREFERRED 4 __inline int IsValidNTE(NetTableEntry *NTE) { return (NTE->DADState >= DAD_STATE_DEPRECATED); } __inline int IsTentativeNTE(NetTableEntry *NTE) { return (NTE->DADState == DAD_STATE_TENTATIVE); } // // We use this infinite lifetime value for prefix lifetimes, // router lifetimes, address lifetimes, etc. // #define INFINITE_LIFETIME 0xffffffff // // Temporary addresses have extra fields. // typedef struct TempNetTableEntry { NetTableEntry; // Inherit the NTE fields. NetTableEntry *Public; // Does not hold a reference. uint CreationTime; // In ticks (see IPv6TickCount). } TempNetTableEntry; // // Each interface keeps track of which link-layer multicast addresses // are currently enabled for receive. A reference count is required because // multiple IPv6 multicast addresses can map to a single link-layer // multicast address. The low bit of RefCntAndFlags is a flag that, if set, // indicates the link-layer address has been registered with the link. // typedef struct LinkLayerMulticastAddress { uint RefCntAndFlags; uchar LinkAddress[]; // The link-layer address follows in memory. // Padded to provide alignment. } LinkLayerMulticastAddress; #define LLMA_FLAG_REGISTERED 0x1 __inline void AddRefLLMA(LinkLayerMulticastAddress *LLMA) { LLMA->RefCntAndFlags += (LLMA_FLAG_REGISTERED << 1); } __inline void ReleaseLLMA(LinkLayerMulticastAddress *LLMA) { LLMA->RefCntAndFlags -= (LLMA_FLAG_REGISTERED << 1); } __inline int IsLLMAReferenced(LinkLayerMulticastAddress *LLMA) { return LLMA->RefCntAndFlags > LLMA_FLAG_REGISTERED; } // // Information about IPv6 interfaces. There can be multiple NTEs for each // interface, but there is exactly one interface per NTE. // struct Interface { // a.k.a. IF NetTableEntryOrInterface; // For NTEorIF. Points to self. Interface *Next; // Next interface on chain. long RefCnt; // Reference count - interlocked. // // Interface to the link layer. The functions all take // the LinkContext as their first argument. See comments // in llip6if.h. // void *LinkContext; // Link layer context. void (*CreateToken)(void *Context, IPv6Addr *Address); const void *(*ReadLLOpt)(void *Context, const uchar *OptionData); void (*WriteLLOpt)(void *Context, uchar *OptionData, const void *LinkAddress); ushort (*ConvertAddr)(void *Context, const IPv6Addr *Address, void *LinkAddress); NTSTATUS (*SetRouterLLAddress)(void *Context, const void *TokenLinkAddress, const void *RouterLinkAddress); void (*Transmit)(void *Context, PNDIS_PACKET Packet, uint Offset, const void *LinkAddress); NDIS_STATUS (*SetMCastAddrList)(void *Context, const void *LinkAddresses, uint NumKeep, uint NumAdd, uint NumDel); void (*Close)(void *Context); void (*Cleanup)(void *Context); uint Index; // Node unique index of this I/F. uint Type; // Values in ntddip6.h. uint Flags; // Changes require lock, reads don't. uint DefaultPreference; // Read-only. uint Preference; // For routing. // // ZoneIndices[0] (ADE_SMALLEST_SCOPE) and // ZoneIndices[1] (ADE_INTERFACE_LOCAL) must be Index. // ZoneIndices[14] (ADE_GLOBAL) and // ZoneIndices[15] (ADE_LARGEST_SCOPE) must be one. // ZoneIndices must respect zone containment: // If two interfaces have the same value for ZoneIndices[N], // then they must have the same value for ZoneIndices[N+1]. // To ensure consistency, modifying ZoneIndices requires // the global ZoneUpdateLock. // uint ZoneIndices[ADE_NUM_SCOPES]; // Changes require lock, reads don't. AddressEntry *ADE; // List of ADEs on this I/F. NetTableEntry *LinkLocalNTE; // Primary link-local address. KSPIN_LOCK LockNC; // Neighbor cache lock. NeighborCacheEntry *FirstNCE; // List of active neighbors on I/F. NeighborCacheEntry *LastNCE; // Last NCE in the list. uint NCENumUnused; // Number of unused NCEs - interlocked. NDIS_PACKET *PacketList; // List of packets to be completed. uint TrueLinkMTU; // Read-only, true maximum MTU. uint DefaultLinkMTU; // Read-only, default for LinkMTU. uint LinkMTU; // Manually configured or received from RAs. uint CurHopLimit; // Default Hop Limit for unicast. uint BaseReachableTime; // Base for random ReachableTime (in ms). uint ReachableTime; // Reachable timeout (in IPv6Timer ticks). uint RetransTimer; // NS timeout (in IPv6Timer ticks). uint DefaultDupAddrDetectTransmits; // Read-only. uint DupAddrDetectTransmits; // Number of solicits during DAD. uint DupAddrDetects; // Number of consecutive DAD detects. uint DefSitePrefixLength; // Default Site Prefix Length for RAs. uint TempStateAge; // Age of the temporary state. IPv6Addr TempState; // State for generating temporary addresses. uint RSCount; // Number of Router Solicits sent. uint RSTimer; // RS timeout (in IPv6Timer ticks). uint RACount; // Number of "fast" RAs left to send. uint RATimer; // RA timeout (in IPv6Timer ticks). uint RALast; // Time of last RA (in IPv6Timer ticks). uint LinkAddressLength; // Length of I/F link-level address. uchar *LinkAddress; // Pointer to link-level address. uint LinkHeaderSize; // Length of link-level header. KSPIN_LOCK Lock; // Main interface lock. KMUTEX WorkerLock; // Serializes worker thread operations. LinkLayerMulticastAddress *MCastAddresses; // Current addresses. uint MCastAddrNum; // Number of link-layer mcast addresses. uint TcpInitialRTT; // InitialRTT that TCP connections should use // on this interface. HANDLE TdiRegistrationHandle; // Opaque token for TDI De/notification. GUID Guid; NDIS_STRING DeviceName; // IPV6_EXPORT_STRING_PREFIX + string Guid. }; __inline NeighborCacheEntry * SentinelNCE(Interface *IF) { return (NeighborCacheEntry *) &IF->FirstNCE; } __inline uint SizeofLinkLayerMulticastAddress(Interface *IF) { uint RawSize = (sizeof(struct LinkLayerMulticastAddress) + IF->LinkAddressLength); uint Align = __builtin_alignof(struct LinkLayerMulticastAddress) - 1; return (RawSize + Align) &~ Align; } // // These values should agree with definitions also // found in llip6if.h and ntddip6.h. // #define IF_TYPE_LOOPBACK 0 #define IF_TYPE_ETHERNET 1 #define IF_TYPE_FDDI 2 #define IF_TYPE_TUNNEL_AUTO 3 #define IF_TYPE_TUNNEL_6OVER4 4 #define IF_TYPE_TUNNEL_V6V4 5 #define IF_TYPE_TUNNEL_6TO4 6 #define IF_TYPE_TUNNEL_TEREDO 7 #define IF_TYPE_MIPV6 8 // Holds a mobile node's home addresses. __inline int IsIPv4TunnelIF(Interface *IF) { return ((IF_TYPE_TUNNEL_AUTO <= IF->Type) && (IF->Type <= IF_TYPE_TUNNEL_6TO4)); } // // These values should agree with definitions also // found in llip6if.h and ntddip6.h. // #define IF_FLAG_PSEUDO 0x00000001 #define IF_FLAG_P2P 0x00000002 #define IF_FLAG_NEIGHBOR_DISCOVERS 0x00000004 #define IF_FLAG_FORWARDS 0x00000008 #define IF_FLAG_ADVERTISES 0x00000010 #define IF_FLAG_MULTICAST 0x00000020 #define IF_FLAG_ROUTER_DISCOVERS 0x00000040 #define IF_FLAG_PERIODICMLD 0x00000080 #define IF_FLAG_FIREWALL_ENABLED 0x00000100 #define IF_FLAG_MEDIA_DISCONNECTED 0x00001000 #define IF_FLAGS_DISCOVERS \ (IF_FLAG_NEIGHBOR_DISCOVERS|IF_FLAG_ROUTER_DISCOVERS) #define IF_FLAGS_BINDINFO 0x0000ffff #define IF_FLAG_DISABLED 0x00010000 #define IF_FLAG_MCAST_SYNC 0x00020000 #define IF_FLAG_OTHER_STATEFUL_CONFIG 0x00040000 // // The DISCONNECTED and RECONNECTED flags should not both be set. // RECONNECTED indicates that the host interface was recently reconnected; // it is cleared upon receiving a Router Advertisement. // #define IF_FLAG_MEDIA_RECONNECTED 0x00080000 // // This function should be used after taking the interface lock // or interface list lock, to check if the interface is disabled. // __inline int IsDisabledIF(Interface *IF) { return IF->Flags & IF_FLAG_DISABLED; } // // Called with the interface lock held. // __inline int IsMCastSyncNeeded(Interface *IF) { return IF->Flags & IF_FLAG_MCAST_SYNC; } // // Active interfaces hold a reference to themselves. // NTEs hold a reference to their interface. // NCEs that have a non-zero ref count hold a reference. // MAEs and AAEs do not hold a reference for their NTE or IF. // __inline void AddRefIF(Interface *IF) { // // A stronger assertion would be !IsDisabledIF(IF), // which is mostly true, but that assertion would // imply that AddRefIF could be used only while // holding the interface list lock or the interface lock, // which is an undesirable restriction. // ASSERT(IF->RefCnt > 0); InterlockedIncrement(&IF->RefCnt); } __inline void ReleaseIF(Interface *IF) { InterlockedDecrement(&IF->RefCnt); } // // We have a periodic timer (IPv6Timer) that causes our IPv6Timeout // routine to be called IPv6_TICKS_SECOND times per second. Most of the // timers and timeouts in this implementation are driven off this routine. // // There is a trade-off here between timer granularity/resolution // and overhead. The resolution should be subsecond because // RETRANS_TIMER is only one second. // extern uint IPv6TickCount; #define IPv6_TICKS_SECOND 2 // Two ticks per second. #define IPv6_TIMEOUT (1000 / IPv6_TICKS_SECOND) // In milliseconds. #define IPv6TimerTicks(seconds) ((seconds) * IPv6_TICKS_SECOND) // // ConvertSecondsToTicks and ConvertTicksToSeconds // both leave the value INFINITE_LIFETIME unchanged. // extern uint ConvertSecondsToTicks(uint Seconds); extern uint ConvertTicksToSeconds(uint Ticks); // // ConvertMillisToTicks and ConvertTicksToMillis // do not have an infinite value. // extern uint ConvertMillisToTicks(uint Millis); __inline uint ConvertTicksToMillis(uint Ticks) { return Ticks * IPv6_TIMEOUT; } // // REVIEW: Hack to handle those few remaining places where we still need // REVIEW: to allocate space for a link-level header before we know the // REVIEW: outgoing inteface (and thus know how big said header will be). // REVIEW: When these places have all been fixed, we won't need this. // #define MAX_LINK_HEADER_SIZE 32 // // Various constants from the IPv6 RFCs... // // REVIEW: Some of these should be per link-layer type. // REVIEW: Put them in the Interface structure? // #define MAX_INITIAL_RTR_ADVERT_INTERVAL IPv6TimerTicks(16) #define MAX_INITIAL_RTR_ADVERTISEMENTS 3 // Produces 4 quick RAs. #define MAX_FINAL_RTR_ADVERTISEMENTS 3 #define MIN_DELAY_BETWEEN_RAS IPv6TimerTicks(3) #define MAX_RA_DELAY_TIME 1 // 0.5 seconds #define MaxRtrAdvInterval IPv6TimerTicks(600) #define MinRtrAdvInterval IPv6TimerTicks(200) // MAX_RTR_SOLICITATION_DELAY IPv6_TIMEOUT is used instead. #define RTR_SOLICITATION_INTERVAL IPv6TimerTicks(4) // 4 seconds. #define SLOW_RTR_SOLICITATION_INTERVAL IPv6TimerTicks(15 * 60) // 15 minutes. #define MAX_RTR_SOLICITATIONS 3 #define MAX_MULTICAST_SOLICIT 3 // Total transmissions before giving up. #define MAX_UNICAST_SOLICIT 3 // Total transmissions before giving up. #define MAX_UNREACH_SOLICIT 1 // Total transmissions before giving up. #define UNREACH_SOLICIT_INTERVAL IPv6TimerTicks(60) // 1 minute. #define MAX_ANYCAST_DELAY_TIME 1 // seconds. #define REACHABLE_TIME (30 * 1000) // 30 seconds in milliseconds. #define MAX_REACHABLE_TIME (60 * 60 * 1000) // 1 hour in milliseconds. #define ICMP_MIN_ERROR_INTERVAL 1 // Ticks - a half second. #define RETRANS_TIMER IPv6TimerTicks(1) // 1 second. #define DELAY_FIRST_PROBE_TIME IPv6TimerTicks(5) // 5 seconds. #define MIN_RANDOM_FACTOR 50 // Percentage of base value. #define MAX_RANDOM_FACTOR 150 // Percentage of base value. #define PREFIX_LIFETIME_SAFETY IPv6TimerTicks(2 * 60 * 60) // 2 hours. #define RECALC_REACHABLE_INTERVAL IPv6TimerTicks(3 * 60 * 60) // 3 hours. #define PATH_MTU_RETRY_TIME IPv6TimerTicks(10 * 60) // 10 minutes. #define MLD_UNSOLICITED_REPORT_INTERVAL IPv6TimerTicks(10) // 10 seconds. #define MLD_QUERY_INTERVAL IPv6TimerTicks(125) // 125 seconds. #define MLD_NUM_INITIAL_REPORTS 2 #define MAX_TEMP_DAD_ATTEMPTS 5 #define MAX_TEMP_PREFERRED_LIFETIME (24 * 60 * 60) // 1 day. #define MAX_TEMP_VALID_LIFETIME (7 * MAX_TEMP_PREFERRED_LIFETIME) #define TEMP_REGENERATE_TIME 5 // 5 seconds. #define MAX_TEMP_RANDOM_TIME (10 * 60) // 10 minutes. #define DEFAULT_CUR_HOP_LIMIT 0x80 #define DEFAULT_SITE_PREFIX_LENGTH 48 // // Various implementation constants. // #define NEIGHBOR_CACHE_LIMIT 256 #define ROUTE_CACHE_LIMIT 32 #define BINDING_CACHE_LIMIT 32 #define SMALL_POOL 10000 #define MEDIUM_POOL 30000 #define LARGE_POOL 60000 // // Under NT, we use the assembly language version of the common core checksum // routine instead of the C language version. // ULONG tcpxsum(IN ULONG Checksum, IN PUCHAR Source, IN ULONG Length); #define Cksum(Buffer, Length) ((ushort)tcpxsum(0, (PUCHAR)(Buffer), (Length))) // // Protocol Receive Procedures ("Next Header" handlers) have this prototype. // typedef uchar ProtoRecvProc(IPv6Packet *Packet); typedef struct StatusArg { IP_STATUS Status; unsigned long Arg; IPv6Header UNALIGNED *IP; } StatusArg; // // Protocol Control Receive Procedures have this prototype. // These receive handlers are called for ICMP errors. // typedef uchar ProtoControlRecvProc(IPv6Packet *Packet, StatusArg *Arg); typedef struct ProtocolSwitch { ProtoRecvProc *DataReceive; ProtoControlRecvProc *ControlReceive; } ProtocolSwitch; extern ProtoRecvProc IPv6HeaderReceive; extern ProtoRecvProc ICMPv6Receive; extern ProtoRecvProc FragmentReceive; extern ProtoRecvProc DestinationOptionsReceive; extern ProtoRecvProc RoutingReceive; extern ProtoRecvProc EncapsulatingSecurityPayloadReceive; extern ProtoRecvProc AuthenticationHeaderReceive; extern ProtoControlRecvProc ICMPv6ControlReceive; extern ProtoControlRecvProc ExtHdrControlReceive; // // Hop-by-Hop Options use a special receive handler. // This is because they are processed even when a // a packet is being forwarded instead of received. // Note that they are only processed when immediately // following an IPv6 header. // extern int HopByHopOptionsReceive(IPv6Packet *Packet); // // The Raw Receive handler supports external protocol handlers. // extern int RawReceive(IPv6Packet *Packet, uchar Protocol); // // The actual definition of a reassembly structure // can be found in fragment.h. // typedef struct Reassembly Reassembly; #define USE_TEMP_NO 0 // Don't use temporary addresses. #define USE_TEMP_YES 1 // Use them. #define USE_TEMP_ALWAYS 2 // Always generating random numbers. #define USE_TEMP_COUNTER 3 // Use them with per-interface counter. // // Prototypes for global variables. // extern uint DefaultCurHopLimit; extern uint MaxTempDADAttempts; extern uint MaxTempPreferredLifetime; // Ticks. extern uint MaxTempValidLifetime; // Ticks. extern uint TempRegenerateTime; // Ticks. extern uint UseTemporaryAddresses; // See values above. extern uint MaxTempRandomTime; // Ticks. extern uint TempRandomTime; // Ticks. extern ProtocolSwitch ProtocolSwitchTable[]; extern KSPIN_LOCK NetTableListLock; extern NetTableEntry *NetTableList; // Pointer to the net table. extern KSPIN_LOCK IFListLock; extern Interface *IFList; // List of all interfaces on the system. extern KSPIN_LOCK ZoneUpdateLock; extern struct EchoControl *ICMPv6OutstandingEchos; extern LIST_ENTRY PendingEchoList; // def needed for initialization. extern Interface *LoopInterface; extern IPv6Addr UnspecifiedAddr; extern IPv6Addr LoopbackAddr; extern IPv6Addr AllNodesOnNodeAddr; extern IPv6Addr AllNodesOnLinkAddr; extern IPv6Addr AllRoutersOnLinkAddr; extern IPv6Addr LinkLocalPrefix; extern IPv6Addr SiteLocalPrefix; extern IPv6Addr SixToFourPrefix; extern IPv6Addr V4MappedPrefix; extern PDEVICE_OBJECT IPDeviceObject; extern HANDLE IPv6ProviderHandle; // // Some handy functions for working with IPv6 addresses. // __inline IPv6Addr * AlignAddr(IPv6Addr UNALIGNED *Addr) { // // IPv6 addresses only have char & short members, // so they need 2-byte alignment. // In practice addresses in headers are always // appropriately aligned. // ASSERT(((UINT_PTR)Addr % __builtin_alignof(IPv6Addr)) == 0); return (IPv6Addr *) Addr; } __inline int IsUnspecified(const IPv6Addr *Addr) { return IP6_ADDR_EQUAL(Addr, &UnspecifiedAddr); } __inline int IsLoopback(const IPv6Addr *Addr) { return IP6_ADDR_EQUAL(Addr, &LoopbackAddr); } __inline int IsGlobal(const IPv6Addr *Addr) { // // Check the format prefix and exclude addresses // whose high 4 bits are all zero or all one. // This is a cheap way of excluding v4-compatible, // v4-mapped, loopback, multicast, link-local, site-local. // uint High = (Addr->s6_bytes[0] & 0xf0); return (High != 0) && (High != 0xf0); } __inline int IsMulticast(const IPv6Addr *Addr) { return Addr->s6_bytes[0] == 0xff; } __inline int IsLinkLocal(const IPv6Addr *Addr) { return ((Addr->s6_bytes[0] == 0xfe) && ((Addr->s6_bytes[1] & 0xc0) == 0x80)); } __inline int IsLinkLocalMulticast(const IPv6Addr *Addr) { return IsMulticast(Addr) && ((Addr->s6_bytes[1] & 0xf) == ADE_LINK_LOCAL); } __inline int IsInterfaceLocalMulticast(const IPv6Addr *Addr) { return (IsMulticast(Addr) && ((Addr->s6_bytes[1] & 0xf) == ADE_INTERFACE_LOCAL)); } extern int IsSolicitedNodeMulticast(const IPv6Addr *Addr); __inline int IsSiteLocal(const IPv6Addr *Addr) { return ((Addr->s6_bytes[0] == 0xfe) && ((Addr->s6_bytes[1] & 0xc0) == 0xc0)); } __inline int IsSiteLocalMulticast(const IPv6Addr *Addr) { return IsMulticast(Addr) && ((Addr->s6_bytes[1] & 0xf) == ADE_SITE_LOCAL); } extern int IP6_ADDR_LTEQ(const IPv6Addr *A, const IPv6Addr *B); extern int IsEUI64Address(const IPv6Addr *Addr); extern int IsKnownAnycast(const IPv6Addr *Addr); extern int IsSubnetRouterAnycast(const IPv6Addr *Addr); extern int IsSubnetReservedAnycast(const IPv6Addr *Addr); extern int IsInvalidSourceAddress(const IPv6Addr *Addr); extern int IsNotManualAddress(const IPv6Addr *Addr); extern int IsV4Compatible(const IPv6Addr *Addr); extern void CreateV4Compatible(IPv6Addr *Addr, IPAddr V4Addr); extern int IsV4Mapped(const IPv6Addr *Addr); extern void CreateV4Mapped(IPv6Addr *Addr, IPAddr V4Addr); __inline IPAddr ExtractV4Address(const IPv6Addr *Addr) { return * (IPAddr UNALIGNED *) &Addr->s6_bytes[12]; } __inline int Is6to4(const IPv6Addr *Addr) { return Addr->s6_words[0] == 0x0220; } __inline IPAddr Extract6to4Address(const IPv6Addr *Addr) { return * (IPAddr UNALIGNED *) &Addr->s6_bytes[2]; } __inline int IsISATAP(const IPv6Addr *Addr) { return (((Addr->s6_words[4] & 0xFFFD) == 0x0000) && (Addr->s6_words[5] == 0xfe5e)); } __inline int IsV4Multicast(IPAddr Addr) { return (Addr & 0x000000f0) == 0x000000e0; } __inline int IsV4Broadcast(IPAddr Addr) { return Addr == 0xffffffff; } __inline int IsV4Loopback(IPAddr Addr) { return (Addr & 0x000000ff) == 0x0000007f; } __inline int IsV4Unspecified(IPAddr Addr) { return (Addr & 0x000000ff) == 0x00000000; } __inline ushort MulticastAddressScope(const IPv6Addr *Addr) { return Addr->s6_bytes[1] & 0xf; } extern ushort UnicastAddressScope(const IPv6Addr *Addr); extern ushort AddressScope(const IPv6Addr *Addr); extern ushort V4AddressScope(IPAddr Addr); extern uint DetermineScopeId(const IPv6Addr *Addr, Interface *IF); extern void CreateSolicitedNodeMulticastAddress(const IPv6Addr *Addr, IPv6Addr *MCastAddr); extern int HasPrefix(const IPv6Addr *Addr, const IPv6Addr *Prefix, uint PrefixLength); extern void CopyPrefix(IPv6Addr *Addr, const IPv6Addr *Prefix, uint PrefixLength); extern uint CommonPrefixLength(const IPv6Addr *Addr, const IPv6Addr *Addr2); extern int IntersectPrefix(const IPv6Addr *Prefix1, uint Prefix1Length, const IPv6Addr *Prefix2, uint Prefix2Length); // // Function prototypes. // extern int GetSystemRandomBits(uchar *Buffer, uint Length); extern void SeedRandom(const uchar *Seed, uint Length); extern uint Random(void); extern uint RandomNumber(uint Min, uint Max); // // Taken from ws2tcpip.h - unfortunately we can not include that file here. // These numbers include space for a port number, which we do not need, // but that's OK. // #define INET_ADDRSTRLEN 22 #define INET6_ADDRSTRLEN 65 __inline int ParseV6Address(const WCHAR *Sz, const WCHAR **Terminator, IPv6Addr *Addr) { return NT_SUCCESS(RtlIpv6StringToAddressW(Sz, Terminator, Addr)); } __inline void FormatV6AddressWorker(char *Sz, const IPv6Addr *Addr) { (void) RtlIpv6AddressToStringA(Addr, Sz); } extern char * FormatV6Address(const IPv6Addr *Addr); __inline int ParseV4Address(const WCHAR *Sz, const WCHAR **Terminator, IPAddr *Addr) { return NT_SUCCESS(RtlIpv4StringToAddressW(Sz, TRUE, Terminator, (struct in_addr *)Addr)); } __inline void FormatV4AddressWorker(char *Sz, IPAddr Addr) { (void) RtlIpv4AddressToStringA((struct in_addr *)&Addr, Sz); } extern char * FormatV4Address(IPAddr Addr); extern ushort ChecksumPacket(PNDIS_PACKET Packet, uint Offset, uchar *Data, uint Length, const IPv6Addr *Source, const IPv6Addr *Dest, uchar NextHeader); extern void LoopQueueTransmit(PNDIS_PACKET Packet); extern NDIS_STATUS IPv6SendLater(LARGE_INTEGER Time, Interface *IF, PNDIS_PACKET Packet, uint Offset, const void *LinkAddress); extern void IPv6SendLL(Interface *IF, PNDIS_PACKET Packet, uint Offset, const void *LinkAddress); extern void IPv6SendND(PNDIS_PACKET Packet, uint Offset, NeighborCacheEntry *NCE, const IPv6Addr *DiscoveryAddress); #define SEND_FLAG_BYPASS_BINDING_CACHE 0x00000001 extern void IPv6Send(PNDIS_PACKET Packet, uint Offset, IPv6Header UNALIGNED *IP, uint PayloadLength, RouteCacheEntry *RCE, uint Flags, ushort TransportProtocol, ushort SourcePort, ushort DestPort); extern void IPv6Forward(NetTableEntryOrInterface *RecvNTEorIF, PNDIS_PACKET Packet, uint Offset, IPv6Header UNALIGNED *IP, uint PayloadLength, int Redirect, IPSecProc *IPSecToDo, RouteCacheEntry *RCE); extern void IPv6SendAbort(NetTableEntryOrInterface *NTEorIF, PNDIS_PACKET Packet, uint Offset, uchar ICMPType, uchar ICMPCode, ulong ErrorParameter, int MulticastOverride); extern void ICMPv6EchoTimeout(void); extern void IPULUnloadNotify(void); extern Interface * FindInterfaceFromIndex(uint Index); extern Interface * FindInterfaceFromGuid(const GUID *Guid); extern Interface * FindNextInterface(Interface *IF); extern Interface * FindInterfaceFromZone(Interface *OrigIF, uint Scope, uint Index); extern uint FindNewZoneIndex(uint Scope); extern void InitZoneIndices(uint *ZoneIndices, uint Index); extern void UpdateZoneIndices(Interface *IF, uint *ZoneIndices); extern Interface * FindDefaultInterfaceForZone(uint Scope, uint ScopeId); extern void IPv6Timeout(PKDPC MyDpcObject, void *Context, void *Unused1, void *Unused2); extern int MapNdisBuffers(NDIS_BUFFER *Buffer); extern uchar * GetDataFromNdis(PNDIS_BUFFER SrcBuffer, uint SrcOffset, uint Length, uchar *DataBuffer); extern IPv6Header UNALIGNED * GetIPv6Header(PNDIS_PACKET Packet, uint Offset, IPv6Header *HdrBuffer); extern int CheckLinkLayerMulticastAddress(Interface *IF, const void *LinkAddress); extern void AddNTEToInterface(Interface *IF, NetTableEntry *NTE); extern uint InterfaceIndex(void); extern void AddInterface(Interface *IF); extern void UpdateLinkMTU(Interface *IF, uint MTU); extern NetTableEntry * CreateNTE(Interface *IF, const IPv6Addr *Address, uint AddrConf, uint ValidLifetime, uint PreferredLifetime); extern MulticastAddressEntry * FindOrCreateMAE(Interface *IF, const IPv6Addr *Addr, NetTableEntry *NTE); extern MulticastAddressEntry * FindAndReleaseMAE(Interface *IF, const IPv6Addr *Addr); extern int FindOrCreateAAE(Interface *IF, const IPv6Addr *Addr, NetTableEntryOrInterface *NTEorIF); extern int FindAndDeleteAAE(Interface *IF, const IPv6Addr *Addr); extern void DestroyADEs(Interface *IF, NetTableEntry *NTE); extern void DestroyNTE(Interface *IF, NetTableEntry *NTE); extern void EnlivenNTE(Interface *IF, NetTableEntry *NTE); extern void DestroyIF(Interface *IF); extern AddressEntry ** FindADE(Interface *IF, const IPv6Addr *Addr); extern NetTableEntryOrInterface * FindAddressOnInterface(Interface *IF, const IPv6Addr *Addr, ushort *AddrType); extern NetTableEntry * GetLinkLocalNTE(Interface *IF); extern int GetLinkLocalAddress(Interface *IF, IPv6Addr *Addr); extern void DeferRegisterNetAddress(NetTableEntry *NTE); extern void DeferSynchronizeMulticastAddresses(Interface *IF); extern int IPInit(void); extern int Unloading; extern void IPUnload(void); extern void AddrConfUpdate(Interface *IF, const IPv6Addr *Prefix, uint ValidLifetime, uint PreferredLifetime, int Authenticated, NetTableEntry **pNTE); extern int FindOrCreateNTE(Interface *IF, const IPv6Addr *Addr, uint AddrConf, uint ValidLifetime, uint PreferredLifetime); extern void AddrConfDuplicate(Interface *IF, NetTableEntry *NTE); extern void AddrConfNotDuplicate(Interface *IF, NetTableEntry *NTE); extern void AddrConfResetAutoConfig(Interface *IF, uint MaxLifetime); extern void AddrConfTimeout(NetTableEntry *NTE); extern void NetTableTimeout(void); extern void NetTableCleanup(void); extern void InterfaceTimeout(void); extern void InterfaceCleanup(void); extern void InterfaceReset(void); extern NTSTATUS UpdateInterface(Interface *IF, int Advertises, int Forwards); extern void ReconnectInterface(Interface *IF); extern void InterfaceResetAutoConfig(Interface *IF); extern int LanInit(void); extern void LanUnload(void); extern int LoopbackInit(void); extern void ProtoTabInit(void); extern void ICMPv6Init(void); extern int IPSecInit(void); extern void IPSecUnload(void); extern int TunnelInit(void); extern void TunnelUnload(void); extern NTSTATUS TunnelCreateTunnel(IPAddr SrcAddr, IPAddr DstAddr, uint Flags, Interface **ReturnIF); extern int TunnelGetSourceAddress(IPAddr Dest, IPAddr *Source); extern ulong NewFragmentId(void); extern void ReassemblyInit(void); extern void ReassemblyUnload(void); extern void ReassemblyRemove(Interface *IF); extern void CreateGUIDFromName(const char *Name, GUID *Guid); extern void ConfigureGlobalParameters(void); extern void ConfigureInterface(Interface *IF); extern void ConfigurePrefixPolicies(void); extern void ConfigurePersistentInterfaces(void); #pragma warning(pop) #endif // IPv6DEF_INCLUDED