Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

571 lines
13 KiB

  1. /*++
  2. Copyright (c) 1991 Microsoft Corporation
  3. Module Name:
  4. tcpip\ip\ipmcast.h
  5. Abstract:
  6. Defines and internal structures for IP Multicasting
  7. Author:
  8. Amritansh Raghav
  9. Revision History:
  10. AmritanR Created
  11. Notes:
  12. --*/
  13. #ifndef __IPMCAST_H__
  14. #define __IPMCAST_H__
  15. typedef unsigned long DWORD, *PDWORD;
  16. typedef unsigned char BYTE, *PBYTE;
  17. #include "iproute.h"
  18. #include "iprtdef.h"
  19. #include "debug.h"
  20. #include "ipmlock.h"
  21. #include "ddipmcst.h"
  22. #define is ==
  23. #define isnot !=
  24. #define or ||
  25. #define and &&
  26. //
  27. // Symbolic link into DOS space
  28. //
  29. #define WIN32_IPMCAST_SYMBOLIC_LINK L"\\DosDevices\\IPMULTICAST"
  30. //
  31. // Nifty macro for printing IP Addresses
  32. //
  33. #define PRINT_IPADDR(x) \
  34. ((x)&0x000000FF),(((x)&0x0000FF00)>>8),(((x)&0x00FF0000)>>16),(((x)&0xFF000000)>>24)
  35. //
  36. // We use lookaside lists for a lot of stuff. The following #defines are for
  37. // the depths of the lists
  38. //
  39. #define GROUP_LOOKASIDE_DEPTH 16
  40. #define SOURCE_LOOKASIDE_DEPTH 64
  41. #define OIF_LOOKASIDE_DEPTH 128
  42. #define MSG_LOOKASIDE_DEPTH 8
  43. //
  44. // The number of packets pending per (S,G) entry when queuing is being done
  45. //
  46. #define MAX_PENDING 4
  47. //
  48. // The multicast state
  49. // MCAST_STARTED is again defined in iproute.c and MUST be kept in synch with
  50. // this #define
  51. //
  52. #define MCAST_STOPPED 0
  53. #define MCAST_STARTED 1
  54. //
  55. // The groups are kept in a hash tableof size GROUP_TABLE_SIZE
  56. // GROUP_HASH is the hash function
  57. //
  58. //
  59. // TODO: Need to refine the hash function
  60. //
  61. #define GROUP_TABLE_SIZE 127
  62. #define GROUP_HASH(addr) (addr % GROUP_TABLE_SIZE)
  63. //
  64. // The number of seconds of inactivity after which an SOURCE is deleted
  65. //
  66. #define INACTIVITY_PERIOD (10 * 60)
  67. //
  68. // The default timeout when a MFE is created
  69. //
  70. #define DEFAULT_LIFETIME (1 * 60)
  71. //
  72. // Number of seconds before which another wrong i/f upcall can be generated
  73. //
  74. #define UPCALL_PERIOD (3 * 60)
  75. //
  76. // Some #defines for time/timeticks conversions
  77. //
  78. #define TIMER_IN_MILLISECS (60 * 1000)
  79. #define SYS_UNITS_IN_ONE_MILLISEC (1000 * 10)
  80. #define MILLISECS_TO_TICKS(ms) \
  81. ((LONGLONG)(ms) * SYS_UNITS_IN_ONE_MILLISEC / KeQueryTimeIncrement())
  82. #define SECS_TO_TICKS(s) \
  83. ((LONGLONG)MILLISECS_TO_TICKS((s) * 1000))
  84. //
  85. // We walk only BUCKETS_PER_QUANTUM number of buckets each time the Timer DPC
  86. // fires. This way we do not hog up too much CPU. So currently we walk enough
  87. // so that we need to fire 5 times every INACTIVITY_PERIOD
  88. //
  89. #define BUCKETS_PER_QUANTUM ((GROUP_TABLE_SIZE/5) + 1)
  90. //
  91. // All IOCTLs are handled by functions with the prototype below. This allows
  92. // us to build a table of pointers and call out to them instead of doing
  93. // a switch
  94. //
  95. typedef
  96. NTSTATUS
  97. (*PFN_IOCTL_HNDLR)(
  98. PIRP pIrp,
  99. ULONG ulInLength,
  100. ULONG ulOutLength
  101. );
  102. //
  103. // If an IRP is not available, the forwarder queues the notification message
  104. // onto a list. Then next time an IRP is posted, a message is pulled of the
  105. // list, copied to the IRP and the IRP is then completed.
  106. //
  107. typedef struct _NOTIFICATION_MSG
  108. {
  109. LIST_ENTRY leMsgLink;
  110. IPMCAST_NOTIFICATION inMessage;
  111. }NOTIFICATION_MSG, *PNOTIFICATION_MSG;
  112. //
  113. // The information for the GROUP entry
  114. //
  115. typedef struct _GROUP
  116. {
  117. //
  118. // Link to the hash bucket
  119. //
  120. LIST_ENTRY leHashLink;
  121. //
  122. // Class D IP Address of the Group
  123. //
  124. DWORD dwGroup;
  125. //
  126. // The number of sources on this group. Not really used for anything
  127. // right now.
  128. //
  129. ULONG ulNumSources;
  130. //
  131. // Linked list of sources active on the group. We should make this
  132. // a singly linked list.
  133. //
  134. LIST_ENTRY leSrcHead;
  135. }GROUP, *PGROUP;
  136. typedef struct _OUT_IF OUT_IF, *POUT_IF;
  137. //
  138. // The information for each outgoing interface
  139. //
  140. struct _OUT_IF
  141. {
  142. //
  143. // Link to the list of OIFs hanging off a source
  144. //
  145. POUT_IF pNextOutIf;
  146. //
  147. // Pointer to IP's Interface structure for the correspongind interface
  148. // If DemandDial, then it points to DummyInterface, when disconnected
  149. //
  150. Interface *pIpIf;
  151. //
  152. // The Interface Index.
  153. //
  154. DWORD dwIfIndex;
  155. //
  156. // The NextHopAddr is either the IP Address of the receiver, for RAS client
  157. // and NBMA type interfaces, or the Group Address.
  158. //
  159. DWORD dwNextHopAddr;
  160. //
  161. // The context to dial out with in case of Demand Dial interfaces
  162. //
  163. DWORD dwDialContext;
  164. //
  165. // The following fields are statistics kept for the OIF
  166. //
  167. ULONG ulTtlTooLow;
  168. ULONG ulFragNeeded;
  169. ULONG ulOutPackets;
  170. ULONG ulOutDiscards;
  171. };
  172. typedef struct _EXCEPT_IF EXCEPT_IF, *PEXCEPT_IF;
  173. struct _EXCEPT_IF
  174. {
  175. //
  176. // Link to the list of i/fs that are exceptions to the wrong i/f bit
  177. //
  178. PEXCEPT_IF pNextExceptIf;
  179. //
  180. // We just store the index - it makes a lot of pnp issues go away
  181. //
  182. DWORD dwIfIndex;
  183. };
  184. //
  185. // Information about an active source
  186. //
  187. typedef struct _SOURCE
  188. {
  189. //
  190. // Link on the list of sources hanging off a group
  191. //
  192. LIST_ENTRY leGroupLink;
  193. //
  194. // IP Address of the source
  195. //
  196. DWORD dwSource;
  197. //
  198. // Mask associated with the source. Not used. Must be 0xFFFFFFFF
  199. //
  200. DWORD dwSrcMask;
  201. //
  202. // The index of the correct incoming interface
  203. //
  204. DWORD dwInIfIndex;
  205. //
  206. // The lock for the structure
  207. //
  208. RT_LOCK mlLock;
  209. //
  210. // Pointer to the IP Interface corresponding to the incoming interface
  211. //
  212. Interface *pInIpIf;
  213. //
  214. // The number of outgoing interfaces
  215. //
  216. ULONG ulNumOutIf;
  217. //
  218. // Singly linked list of OIFs
  219. //
  220. POUT_IF pFirstOutIf;
  221. //
  222. // Singly linked list of wrong i/f exception interfaces
  223. //
  224. PEXCEPT_IF pFirstExceptIf;
  225. //
  226. // Number of packets queued
  227. //
  228. ULONG ulNumPending;
  229. //
  230. // The list of queued packets
  231. //
  232. FWQ fwqPending;
  233. //
  234. // Used to refcount the structure
  235. //
  236. LONG lRefCount;
  237. //
  238. // Some stats pertaining to this source
  239. //
  240. ULONG ulInPkts;
  241. ULONG ulInOctets;
  242. ULONG ulPktsDifferentIf;
  243. ULONG ulQueueOverflow;
  244. ULONG ulUninitMfe;
  245. ULONG ulNegativeMfe;
  246. ULONG ulInDiscards;
  247. ULONG ulInHdrErrors;
  248. ULONG ulTotalOutPackets;
  249. //
  250. // The KeQueryTickCount() value the last time this structure was used
  251. //
  252. LONGLONG llLastActivity;
  253. //
  254. // User supplied timeout. If 0, then the source is timed out on the basis
  255. // of inactivity after INACTIVITY_PERIOD
  256. //
  257. LONGLONG llTimeOut;
  258. //
  259. // The time the structure was created
  260. //
  261. LONGLONG llCreateTime;
  262. //
  263. // The state of the source
  264. //
  265. BYTE byState;
  266. }SOURCE, *PSOURCE;
  267. #pragma warning(push)
  268. #pragma warning(disable:4127) // conditional expression is constant
  269. _inline
  270. VOID
  271. UpdateActivityTime(PSOURCE pSource)
  272. {
  273. KeQueryTickCount((PLARGE_INTEGER)&(((pSource)->llLastActivity)));
  274. }
  275. #pragma warning(pop)
  276. //
  277. // The states of the MFE
  278. //
  279. #define MFE_UNINIT 0x0
  280. #define MFE_NEGATIVE 0x1
  281. #define MFE_QUEUE 0x2
  282. #define MFE_INIT 0x3
  283. //
  284. // The structure of a hash bucket
  285. //
  286. typedef struct _GROUP_ENTRY
  287. {
  288. //
  289. // List of Groups that fall in the bucket
  290. //
  291. LIST_ENTRY leHashHead;
  292. #if DBG
  293. //
  294. // Current number of groups
  295. //
  296. ULONG ulGroupCount;
  297. ULONG ulCacheHits;
  298. ULONG ulCacheMisses;
  299. #endif
  300. //
  301. // One deep cache
  302. //
  303. PGROUP pGroup;
  304. RW_LOCK rwlLock;
  305. }GROUP_ENTRY, *PGROUP_ENTRY;
  306. //
  307. // The LIST_ENTRY macros from ntrtl.h modified to work on FWQ
  308. //
  309. #define InsertTailFwq(ListHead, Entry) \
  310. { \
  311. FWQ *_EX_Blink; \
  312. FWQ *_EX_ListHead; \
  313. _EX_ListHead = (ListHead); \
  314. _EX_Blink = _EX_ListHead->fq_prev; \
  315. (Entry)->fq_next = _EX_ListHead; \
  316. (Entry)->fq_prev = _EX_Blink; \
  317. _EX_Blink->fq_next = (Entry); \
  318. _EX_ListHead->fq_prev = (Entry); \
  319. }
  320. #define RemoveEntryFwq(Entry) \
  321. { \
  322. FWQ *_EX_Blink; \
  323. FWQ *_EX_Flink; \
  324. _EX_Flink = (Entry)->fq_next; \
  325. _EX_Blink = (Entry)->fq_prev; \
  326. _EX_Blink->fq_next = _EX_Flink; \
  327. _EX_Flink->fq_prev = _EX_Blink; \
  328. }
  329. #define RemoveHeadFwq(ListHead) \
  330. (ListHead)->fq_next; \
  331. {RemoveEntryFwq((ListHead)->fq_next)}
  332. #define IsFwqEmpty(ListHead) \
  333. ((ListHead)->fq_next == (ListHead))
  334. #define InitializeFwq(ListHead) \
  335. { \
  336. (ListHead)->fq_next = (ListHead)->fq_prev = (ListHead); \
  337. }
  338. #define CopyFwq(Dest, Source) \
  339. { \
  340. *(Dest) = *(Source); \
  341. (Source)->fq_next->fq_prev = (Dest); \
  342. (Source)->fq_prev->fq_next = (Dest); \
  343. }
  344. //
  345. // The ref count for a SOURCE is set to 2, once because a pointer is saved in
  346. // the group list and once because the function that creates the SOURCE will
  347. // deref it once
  348. //
  349. #define InitRefCount(pSource) \
  350. (pSource)->lRefCount = 2
  351. #define ReferenceSource(pSource) \
  352. InterlockedIncrement(&((pSource)->lRefCount))
  353. #define DereferenceSource(pSource) \
  354. { \
  355. if(InterlockedDecrement(&((pSource)->lRefCount)) == 0) \
  356. { \
  357. DeleteSource((pSource)); \
  358. } \
  359. }
  360. //
  361. // #defines to keep track of number of threads of execution in our code
  362. // This is needed for us to stop cleanly
  363. //
  364. //
  365. // EnterDriver returns if the driver is stopping
  366. //
  367. #define EnterDriver() EnterDriverWithStatus(NOTHING)
  368. #define EnterDriverWithStatus(_Status) \
  369. { \
  370. RtAcquireSpinLockAtDpcLevel(&g_rlStateLock); \
  371. if(g_dwMcastState is MCAST_STOPPED) \
  372. { \
  373. RtReleaseSpinLockFromDpcLevel(&g_rlStateLock); \
  374. return _Status; \
  375. } \
  376. g_dwNumThreads++; \
  377. RtReleaseSpinLockFromDpcLevel(&g_rlStateLock); \
  378. }
  379. #define ExitDriver() \
  380. { \
  381. RtAcquireSpinLockAtDpcLevel(&g_rlStateLock); \
  382. g_dwNumThreads--; \
  383. if((g_dwMcastState is MCAST_STOPPED) and \
  384. (g_dwNumThreads is 0)) \
  385. { \
  386. KeSetEvent(&g_keStateEvent, \
  387. 0, \
  388. FALSE); \
  389. } \
  390. RtReleaseSpinLockFromDpcLevel(&g_rlStateLock); \
  391. }
  392. #if MCAST_REF
  393. #define RefMIF(p) \
  394. { \
  395. InterlockedIncrement(&((p)->if_mfecount)); \
  396. (p)->if_refcount++; \
  397. }
  398. #define DerefMIF(p) \
  399. { \
  400. InterlockedDecrement(&((p)->if_mfecount)); \
  401. DerefIF((p)); \
  402. }
  403. #else
  404. #define RefMIF(p) \
  405. { \
  406. (p)->if_refcount++; \
  407. }
  408. #define DerefMIF(p) \
  409. { \
  410. DerefIF((p)); \
  411. }
  412. #endif // MCAST_REF
  413. #endif // __IPMCAST_H__