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.

883 lines
20 KiB

  1. /*++
  2. Copyright (c) 1995 Microsoft Corporation
  3. Module Name:
  4. misc.c
  5. Abstract:
  6. Miscellaneous management functions
  7. Author:
  8. Stefan Solomon 03/13/1995
  9. Revision History:
  10. --*/
  11. #include "precomp.h"
  12. #pragma hdrstop
  13. UCHAR bcastnode[6] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
  14. VOID
  15. SetAdapterBindingInfo(PIPX_ADAPTER_BINDING_INFO abip,
  16. PACB acbp);
  17. VOID
  18. RMCreateLocalRoute(PICB icbp);
  19. VOID
  20. RMDeleteLocalRoute(PICB icbp);
  21. VOID
  22. ExternalBindInterfaceToAdapter(PICB icbp);
  23. VOID
  24. ExternalUnbindInterfaceFromAdapter(ULONG InterfaceIndex);
  25. /*++
  26. Function: GetTocEntry
  27. Descr: Returns a pointer to the specified table of contents entry
  28. in the interface info block.
  29. --*/
  30. PIPX_TOC_ENTRY
  31. GetTocEntry(PIPX_INFO_BLOCK_HEADER InterfaceInfop,
  32. ULONG InfoEntryType)
  33. {
  34. PIPX_TOC_ENTRY tocep;
  35. UINT i;
  36. for(i=0, tocep = InterfaceInfop->TocEntry;
  37. i<InterfaceInfop->TocEntriesCount;
  38. i++, tocep++) {
  39. if(tocep->InfoType == InfoEntryType) {
  40. return tocep;
  41. }
  42. }
  43. return NULL;
  44. }
  45. /*++
  46. Function: GetInfoEntry
  47. Descr: Returns a pointer to the specified info entry in the interface
  48. control block. If more then one entries, returns a pointer to
  49. the first one.
  50. --*/
  51. LPVOID
  52. GetInfoEntry(PIPX_INFO_BLOCK_HEADER InterfaceInfop,
  53. ULONG InfoEntryType)
  54. {
  55. PIPX_TOC_ENTRY tocep;
  56. if(tocep = GetTocEntry(InterfaceInfop, InfoEntryType)) {
  57. return((LPVOID)((PUCHAR)InterfaceInfop + tocep->Offset));
  58. }
  59. else
  60. {
  61. return NULL;
  62. }
  63. }
  64. /*++
  65. Function: UpdateStaticIfEntries
  66. Descr: Compares the entries in the interface info block with the
  67. stored static entries. Deletes the entries not present
  68. in the interface info blolck and adds the new entries
  69. --*/
  70. DWORD
  71. UpdateStaticIfEntries(
  72. PICB icbp,
  73. HANDLE EnumHandle, // handle for the get next enumeration
  74. ULONG StaticEntrySize,
  75. ULONG NewStaticEntriesCount, // number of new static entries
  76. LPVOID NewStaticEntry, // start of the new entries array
  77. ULONG (*GetNextStaticEntry)(HANDLE EnumHandle, LPVOID entry),
  78. ULONG (*DeleteStaticEntry)(ULONG IfIndex, LPVOID entry),
  79. ULONG (*CreateStaticEntry)(PICB icbp, LPVOID entry))
  80. {
  81. PUCHAR EntryIsNew, nsep, OldStaticEntry;
  82. BOOL found;
  83. UINT i;
  84. // delete non-present entries and add the new entries
  85. // array of flags to mark the new entries
  86. if((EntryIsNew = GlobalAlloc(GPTR, NewStaticEntriesCount)) == NULL) {
  87. return 1;
  88. }
  89. memset(EntryIsNew, 1, NewStaticEntriesCount);
  90. if((OldStaticEntry = GlobalAlloc(GPTR, StaticEntrySize)) == NULL) {
  91. GlobalFree(EntryIsNew);
  92. return 1;
  93. }
  94. if(EnumHandle) {
  95. while(!GetNextStaticEntry(EnumHandle, OldStaticEntry))
  96. {
  97. // compare it with each new static static entry until we find a match
  98. found = FALSE;
  99. for(i = 0, nsep = NewStaticEntry;
  100. i<NewStaticEntriesCount;
  101. i++, nsep+= StaticEntrySize) {
  102. if(!memcmp(OldStaticEntry, nsep, StaticEntrySize)) {
  103. // match - set the flags to OLD
  104. EntryIsNew[i] = 0;
  105. found = TRUE;
  106. break;
  107. }
  108. }
  109. if(!found) {
  110. // non present old entry -> delete it
  111. DeleteStaticEntry(icbp->InterfaceIndex, OldStaticEntry);
  112. }
  113. }
  114. }
  115. // all compared and old non-present ones deleted
  116. // now, add all the new ones
  117. for(i=0, nsep = NewStaticEntry;
  118. i<NewStaticEntriesCount;
  119. i++, nsep+= StaticEntrySize) {
  120. if(EntryIsNew[i]) {
  121. CreateStaticEntry(icbp, nsep);
  122. }
  123. }
  124. GlobalFree(EntryIsNew);
  125. return 0;
  126. }
  127. /*++
  128. Function: GetInterfaceAnsiName
  129. Arguments:
  130. AnsiInterfaceNameBuffer - buffer of IPX_INTERFACE_ANSI_NAME_LEN
  131. UnicodeInterfaceNameBuffer -
  132. Descr:
  133. --*/
  134. VOID
  135. GetInterfaceAnsiName(PUCHAR AnsiInterfaceNameBuffer,
  136. PWSTR UnicodeInterfaceNameBuffer)
  137. {
  138. UNICODE_STRING UnicodeInterfaceName;
  139. ANSI_STRING AnsiInterfaceName;
  140. NTSTATUS ntStatus;
  141. // init a unicode string with the interface name string
  142. RtlInitUnicodeString(&UnicodeInterfaceName, UnicodeInterfaceNameBuffer);
  143. // make the interface name unicode string an ansi string
  144. // in an rtl allocated buffer
  145. ntStatus = RtlUnicodeStringToAnsiString(&AnsiInterfaceName,
  146. &UnicodeInterfaceName,
  147. TRUE // allocate the ansi buffer
  148. );
  149. if (ntStatus != STATUS_SUCCESS)
  150. {
  151. return;
  152. }
  153. // copy the interface name into the supplied buffer, up to the
  154. // argument buffer max size
  155. memcpy(AnsiInterfaceNameBuffer,
  156. AnsiInterfaceName.Buffer,
  157. min(AnsiInterfaceName.MaximumLength, IPX_INTERFACE_ANSI_NAME_LEN));
  158. // free the rtl allocated buffer
  159. RtlFreeAnsiString(&AnsiInterfaceName);
  160. }
  161. /*++
  162. Function: BindInterfaceToAdapter
  163. Descr: Binds the interface to the adapter in the router manager and in
  164. all the other modules and creates a local route for the interface
  165. in the RTM
  166. --*/
  167. VOID
  168. BindInterfaceToAdapter(PICB icbp,
  169. PACB acbp)
  170. {
  171. DWORD rc;
  172. Trace(BIND_TRACE, "BindInterfaceToAdapter: Bind interface # %d to adapter # %d",
  173. icbp->InterfaceIndex,
  174. acbp->AdapterIndex);
  175. if(icbp->acbp != NULL)
  176. {
  177. Trace(BIND_TRACE, "BindInterfaceToAdapter: interface # %d already bound !!!",
  178. icbp->InterfaceIndex);
  179. //SS_ASSERT(FALSE);
  180. return;
  181. }
  182. // Make sure that the adapter is not currently claimed by any
  183. // interface either.
  184. //
  185. if ((acbp->icbp) && (acbp->icbp->acbp == acbp))
  186. {
  187. Trace(
  188. BIND_TRACE,
  189. "BindInterfaceToAdapter: adapter # %d already bound to int # %d!!",
  190. acbp->AdapterIndex,
  191. acbp->icbp->InterfaceIndex);
  192. return;
  193. }
  194. // internal bind the adapter control block and the interface control block
  195. icbp->acbp = acbp;
  196. acbp->icbp = icbp;
  197. // if a connection was requested on this if, mark that it has been done
  198. if(icbp->ConnectionRequestPending) {
  199. icbp->ConnectionRequestPending = FALSE;
  200. }
  201. if(!icbp->InterfaceReachable) {
  202. // we should never hit this code path in normal operation
  203. // However, should anybody atempt and succeed a dial (manually?) on an
  204. // interface marked unreachable, we should reset our state
  205. // icbp->InterfaceReachable = TRUE;
  206. if(icbp->AdminState == ADMIN_STATE_ENABLED) {
  207. // enable all static routes for this interface
  208. EnableStaticRoutes(icbp->InterfaceIndex);
  209. // enable external interfaces. Implicitly, this will enable static services
  210. // bound to this interface to be advertised
  211. ExternalEnableInterface(icbp->InterfaceIndex);
  212. }
  213. }
  214. if (icbp->AdminState==ADMIN_STATE_ENABLED) {
  215. icbp->OperState = OPER_STATE_UP;
  216. // create a local route entry in RTM for the connected interface
  217. RMCreateLocalRoute(icbp);
  218. }
  219. ExternalBindInterfaceToAdapter(icbp);
  220. // if the interface is a local client type (i.e. host doing manual dial from
  221. // the local machine, try to update the internal routing table
  222. if(icbp->MIBInterfaceType == IF_TYPE_ROUTER_WORKSTATION_DIALOUT) {
  223. if((rc = RtProtRequestRoutesUpdate(icbp->InterfaceIndex)) == NO_ERROR) {
  224. icbp->UpdateReq.RoutesReqStatus = UPDATE_PENDING;
  225. }
  226. else
  227. {
  228. Trace(UPDATE_TRACE, "BindInterfaceToAdapter: Routing Update is Disabled");
  229. }
  230. if((rc = RtProtRequestServicesUpdate(icbp->InterfaceIndex)) == NO_ERROR) {
  231. icbp->UpdateReq.ServicesReqStatus = UPDATE_PENDING;
  232. }
  233. else
  234. {
  235. Trace(UPDATE_TRACE, "BindInterfaceToAdapter: Services Update is Disabled");
  236. }
  237. }
  238. }
  239. /*++
  240. Function: UnbindInterfaceFromAdapter
  241. Descr: Unbind the Rip, Sap and Forwarder interfaces with this index from
  242. the respective adapter
  243. --*/
  244. VOID
  245. UnbindInterfaceFromAdapter(PICB icbp)
  246. {
  247. PACB acbp;
  248. ULONG new_if_oper_state;
  249. acbp = icbp->acbp;
  250. if (acbp==NULL) {
  251. Trace(BIND_TRACE, "UnbindInterfaceFromAdapter:Interface # %d is not bound to any adapter",
  252. icbp->InterfaceIndex);
  253. return;
  254. }
  255. Trace(BIND_TRACE, "UnbindInterfaceFromAdapter: Unbind interface # %d from adapter # %d",
  256. icbp->InterfaceIndex,
  257. acbp->AdapterIndex);
  258. switch(icbp->MIBInterfaceType) {
  259. case IF_TYPE_PERSONAL_WAN_ROUTER:
  260. case IF_TYPE_WAN_WORKSTATION:
  261. case IF_TYPE_WAN_ROUTER:
  262. case IF_TYPE_ROUTER_WORKSTATION_DIALOUT:
  263. if (icbp->AdminState==ADMIN_STATE_ENABLED) {
  264. icbp->OperState = OPER_STATE_SLEEPING;
  265. break;
  266. }
  267. default:
  268. icbp->OperState = OPER_STATE_DOWN;
  269. break;
  270. }
  271. if (icbp->AdminState==ADMIN_STATE_ENABLED) {
  272. // delete local route and ext unbind
  273. RMDeleteLocalRoute(icbp);
  274. }
  275. ExternalUnbindInterfaceFromAdapter(icbp->InterfaceIndex);
  276. // if there were updates going on they will get cancelled automatically
  277. // by the respective routing protocols.
  278. // We just have to reset the update state in the ICB
  279. ResetUpdateRequest(icbp);
  280. // Now we can unbind the adapter from interface
  281. acbp->icbp = NULL;
  282. icbp->acbp = NULL;
  283. }
  284. /*++
  285. Function: GetNextInterfaceIndex
  286. Descr: Returns the next available interface index. There are a number of
  287. policies to consider here. The one we'll use is to keep the
  288. interface index a small number and to return the first unused
  289. interface index between 1 and MAX_INTERFACE_INDEX.
  290. Note: Called with database locked
  291. --*/
  292. ULONG
  293. GetNextInterfaceIndex(VOID)
  294. {
  295. PICB icbp;
  296. PLIST_ENTRY lep;
  297. ULONG i;
  298. if((icbp = GetInterfaceByIndex(1)) == NULL) {
  299. return 1;
  300. }
  301. lep = icbp->IndexListLinkage.Flink;
  302. i = 2;
  303. while(lep != &IndexIfList)
  304. {
  305. icbp = CONTAINING_RECORD(lep, ICB, IndexListLinkage);
  306. if(i < icbp->InterfaceIndex) {
  307. return i;
  308. }
  309. i = icbp->InterfaceIndex + 1;
  310. if(i == MAX_INTERFACE_INDEX) {
  311. // abort
  312. SS_ASSERT(FALSE);
  313. return i;
  314. }
  315. lep = icbp->IndexListLinkage.Flink;
  316. }
  317. SS_ASSERT(i < MAX_INTERFACE_INDEX);
  318. return i;
  319. }
  320. VOID
  321. SetAdapterBindingInfo(PIPX_ADAPTER_BINDING_INFO abip,
  322. PACB acbp)
  323. {
  324. abip->AdapterIndex = acbp->AdapterIndex;
  325. memcpy(abip->Network, acbp->AdapterInfo.Network, 4);
  326. memcpy(abip->LocalNode, acbp->AdapterInfo.LocalNode, 6);
  327. if(acbp->AdapterInfo.NdisMedium != NdisMediumWan) {
  328. memcpy(abip->RemoteNode, bcastnode, 6);
  329. }
  330. else
  331. {
  332. memcpy(abip->RemoteNode, acbp->AdapterInfo.RemoteNode, 6);
  333. }
  334. abip->MaxPacketSize = acbp->AdapterInfo.MaxPacketSize;
  335. abip->LinkSpeed = acbp->AdapterInfo.LinkSpeed;
  336. }
  337. VOID
  338. ExternalBindInterfaceToAdapter(PICB icbp)
  339. {
  340. PACB acbp;
  341. IPX_ADAPTER_BINDING_INFO abi;
  342. acbp = icbp->acbp;
  343. SetAdapterBindingInfo(&abi, acbp);
  344. FwBindFwInterfaceToAdapter(icbp->InterfaceIndex, &abi);
  345. BindRoutingProtocolsIfsToAdapter(icbp->InterfaceIndex, &abi);
  346. }
  347. VOID
  348. ExternalUnbindInterfaceFromAdapter(ULONG InterfaceIndex)
  349. {
  350. UnbindRoutingProtocolsIfsFromAdapter(InterfaceIndex);
  351. FwUnbindFwInterfaceFromAdapter(InterfaceIndex);
  352. }
  353. VOID
  354. ExternalEnableInterface(ULONG InterfaceIndex)
  355. {
  356. RoutingProtocolsEnableIpxInterface(InterfaceIndex);
  357. FwEnableFwInterface(InterfaceIndex);
  358. }
  359. VOID
  360. ExternalDisableInterface(ULONG InterfaceIndex)
  361. {
  362. FwDisableFwInterface(InterfaceIndex);
  363. RoutingProtocolsDisableIpxInterface(InterfaceIndex);
  364. }
  365. VOID
  366. RMCreateLocalRoute(PICB icbp)
  367. {
  368. PADAPTER_INFO aip;
  369. // check if a network number has been assigned to this interface
  370. aip = &(icbp->acbp->AdapterInfo);
  371. if(!memcmp(aip->Network, nullnet, 4)) {
  372. // no net number
  373. return;
  374. }
  375. // if the interface is a remote workstation and global wan net exists,
  376. // we are done.
  377. if((icbp->MIBInterfaceType == IF_TYPE_WAN_WORKSTATION) &&
  378. EnableGlobalWanNet &&
  379. !LanOnlyMode) {
  380. SS_ASSERT(!memcmp(aip->Network, GlobalWanNet, 4));
  381. return;
  382. }
  383. CreateLocalRoute(icbp);
  384. }
  385. VOID
  386. RMDeleteLocalRoute(PICB icbp)
  387. {
  388. PADAPTER_INFO aip;
  389. // check if a network number has been assigned to this interface
  390. aip = &(icbp->acbp->AdapterInfo);
  391. if(!memcmp(aip->Network, nullnet, 4)) {
  392. // no net number
  393. return;
  394. }
  395. // if the interface is a remote workstation and global wan net exists,
  396. // we are done.
  397. if((icbp->MIBInterfaceType == IF_TYPE_WAN_WORKSTATION) &&
  398. EnableGlobalWanNet &&
  399. !LanOnlyMode) {
  400. SS_ASSERT(!memcmp(aip->Network, GlobalWanNet, 4));
  401. return;
  402. }
  403. DeleteLocalRoute(icbp);
  404. }
  405. VOID
  406. AdminEnable(PICB icbp)
  407. {
  408. PACB acbp;
  409. IPX_ADAPTER_BINDING_INFO aii;
  410. if(icbp->AdminState == ADMIN_STATE_ENABLED) {
  411. return;
  412. }
  413. icbp->AdminState = ADMIN_STATE_ENABLED;
  414. // XP 497242. AdminEnable and AdminDisable are always called with the
  415. // database lock held. But locking is order is supposed to be from
  416. // DDM to the router managers. So when we callback to DDM, we must
  417. // first release our lock.
  418. //
  419. RELEASE_DATABASE_LOCK;
  420. InterfaceEnabled (icbp->hDIMInterface, PID_IPX, TRUE);
  421. ACQUIRE_DATABASE_LOCK;
  422. if(icbp->acbp != NULL) {
  423. // bound to adapter
  424. icbp->OperState = OPER_STATE_UP;
  425. RMCreateLocalRoute(icbp);
  426. }
  427. else {
  428. switch(icbp->MIBInterfaceType) {
  429. case IF_TYPE_PERSONAL_WAN_ROUTER:
  430. case IF_TYPE_WAN_WORKSTATION:
  431. case IF_TYPE_WAN_ROUTER:
  432. case IF_TYPE_ROUTER_WORKSTATION_DIALOUT:
  433. icbp->OperState = OPER_STATE_SLEEPING;
  434. break;
  435. default:
  436. icbp->OperState = OPER_STATE_DOWN;
  437. break;
  438. }
  439. }
  440. // if REACHABLE, resume advertising routes and services
  441. if(icbp->InterfaceReachable) {
  442. // enable all static routes for this interface
  443. EnableStaticRoutes(icbp->InterfaceIndex);
  444. // enable external interfaces. Implicitly, this will enable static services
  445. // bound to this interface to be advertised
  446. ExternalEnableInterface(icbp->InterfaceIndex);
  447. }
  448. }
  449. VOID
  450. AdminDisable(PICB icbp)
  451. {
  452. if(icbp->AdminState == ADMIN_STATE_DISABLED) {
  453. return;
  454. }
  455. icbp->AdminState = ADMIN_STATE_DISABLED;
  456. // XP 497242. AdminEnable and AdminDisable are always called with the
  457. // database lock held. But locking is order is supposed to be from
  458. // DDM to the router managers. So when we callback to DDM, we must
  459. // first release our lock.
  460. //
  461. RELEASE_DATABASE_LOCK;
  462. InterfaceEnabled (icbp->hDIMInterface, PID_IPX, FALSE);
  463. ACQUIRE_DATABASE_LOCK;
  464. icbp->OperState = OPER_STATE_DOWN;
  465. if(icbp->acbp != NULL)
  466. RMDeleteLocalRoute(icbp);
  467. // disable all static routes for this interface
  468. DisableStaticRoutes(icbp->InterfaceIndex);
  469. // disable external interfaces. Implicitly, static services bound to this
  470. // interface will stop being advertised.
  471. ExternalDisableInterface(icbp->InterfaceIndex);
  472. }
  473. NET_INTERFACE_TYPE
  474. MapIpxToNetInterfaceType(PICB icbp)
  475. {
  476. NET_INTERFACE_TYPE NetInterfaceType;
  477. switch(icbp->MIBInterfaceType) {
  478. case IF_TYPE_WAN_ROUTER:
  479. case IF_TYPE_PERSONAL_WAN_ROUTER:
  480. NetInterfaceType = DEMAND_DIAL;
  481. break;
  482. case IF_TYPE_ROUTER_WORKSTATION_DIALOUT:
  483. NetInterfaceType = LOCAL_WORKSTATION_DIAL;
  484. break;
  485. case IF_TYPE_WAN_WORKSTATION:
  486. NetInterfaceType = REMOTE_WORKSTATION_DIAL;
  487. break;
  488. default:
  489. NetInterfaceType = PERMANENT;
  490. break;
  491. }
  492. return NetInterfaceType;
  493. }
  494. /*++
  495. Function: I_SetFilters
  496. Descr: Internal parses the traffic filter info block and sets the filter
  497. driver info.
  498. --*/
  499. /*
  500. DWORD
  501. I_SetFilters(ULONG InterfaceIndex,
  502. ULONG FilterMode, // inbound or outbound
  503. LPVOID FilterInfop)
  504. {
  505. PIPX_TRAFFIC_FILTER_GLOBAL_INFO gip;
  506. PIPX_TOC_ENTRY tocep;
  507. LPVOID FilterDriverInfop;
  508. ULONG FilterDriverInfoSize;
  509. DWORD rc;
  510. if(FilterInfop == NULL) {
  511. // remove all filters
  512. rc = SetFilters(InterfaceIndex,
  513. FilterMode, // in or outbound,
  514. 0,
  515. 0,
  516. NULL,
  517. 0);
  518. return rc;
  519. }
  520. gip = GetInfoEntry((PIPX_INFO_BLOCK_HEADER)FilterInfop,
  521. IPX_TRAFFIC_FILTER_GLOBAL_INFO_TYPE);
  522. if(gip == NULL) {
  523. return ERROR_CAN_NOT_COMPLETE;
  524. }
  525. FilterDriverInfop = GetInfoEntry((PIPX_INFO_BLOCK_HEADER)FilterInfop,
  526. IPX_TRAFFIC_FILTER_INFO_TYPE);
  527. if(FilterDriverInfop == NULL) {
  528. rc = SetFilters(InterfaceIndex,
  529. FilterMode, // in or outbound,
  530. 0, // pass or don't pass
  531. 0, // filter size
  532. NULL,
  533. 0);
  534. return rc;
  535. }
  536. tocep = GetTocEntry((PIPX_INFO_BLOCK_HEADER)FilterInfop,
  537. IPX_TRAFFIC_FILTER_INFO_TYPE);
  538. FilterDriverInfoSize = tocep->Count * tocep->InfoSize;
  539. rc = SetFilters(InterfaceIndex,
  540. FilterMode, // in or outbound,
  541. gip->FilterAction, // pass or don't pass
  542. tocep->InfoSize, // filter size
  543. FilterDriverInfop,
  544. FilterDriverInfoSize);
  545. return rc;
  546. }
  547. */
  548. /*++
  549. Function: I_GetFilters
  550. Descr: Internal builds the traffic filters info block from the filter driver
  551. information.
  552. --*/
  553. /*
  554. typedef struct _FILTERS_INFO_HEADER {
  555. IPX_INFO_BLOCK_HEADER Header;
  556. IPX_TOC_ENTRY TocEntry;
  557. IPX_TRAFFIC_FILTER_GLOBAL_INFO GlobalInfo;
  558. } FILTERS_INFO_HEADER, *PFILTERS_INFO_HEADER;
  559. DWORD
  560. I_GetFilters(ULONG InterfaceIndex,
  561. ULONG FilterMode,
  562. LPVOID FilterInfop,
  563. PULONG FilterInfoSize)
  564. {
  565. DWORD rc;
  566. ULONG FilterAction;
  567. ULONG FilterSize;
  568. PFILTERS_INFO_HEADER fhp;
  569. LPVOID FilterDriverInfop;
  570. ULONG FilterDriverInfoSize = 0;
  571. PIPX_TOC_ENTRY tocep;
  572. if((FilterInfop == NULL) || (*FilterInfoSize == 0)) {
  573. // we are asked for size
  574. rc = GetFilters(InterfaceIndex,
  575. FilterMode,
  576. &FilterAction,
  577. &FilterSize,
  578. NULL,
  579. &FilterDriverInfoSize);
  580. if((rc != NO_ERROR) && (rc != ERROR_INSUFFICIENT_BUFFER)) {
  581. return ERROR_CAN_NOT_COMPLETE;
  582. }
  583. if(FilterDriverInfoSize) {
  584. // there are filters
  585. *FilterInfoSize = sizeof(FILTERS_INFO_HEADER) + FilterDriverInfoSize;
  586. return ERROR_INSUFFICIENT_BUFFER;
  587. }
  588. else
  589. {
  590. // NO filters exist
  591. *FilterInfoSize = 0;
  592. return NO_ERROR;
  593. }
  594. }
  595. if(*FilterInfoSize <= sizeof(FILTERS_INFO_HEADER)) {
  596. return ERROR_INSUFFICIENT_BUFFER;
  597. }
  598. FilterDriverInfoSize = *FilterInfoSize - sizeof(FILTERS_INFO_HEADER);
  599. FilterDriverInfop = (LPVOID)((PUCHAR)FilterInfop + sizeof(FILTERS_INFO_HEADER));
  600. rc = GetFilters(InterfaceIndex,
  601. FilterMode,
  602. &FilterAction,
  603. &FilterSize,
  604. FilterDriverInfop,
  605. &FilterDriverInfoSize);
  606. if(rc != NO_ERROR) {
  607. if(rc == ERROR_MORE_DATA) {
  608. *FilterInfoSize = sizeof(FILTERS_INFO_HEADER) + FilterDriverInfoSize;
  609. return ERROR_INSUFFICIENT_BUFFER;
  610. }
  611. else
  612. {
  613. return ERROR_CAN_NOT_COMPLETE;
  614. }
  615. }
  616. // got it
  617. fhp = (PFILTERS_INFO_HEADER)FilterInfop;
  618. fhp->Header.Version = IPX_ROUTER_VERSION_1;
  619. fhp->Header.Size = *FilterInfoSize;
  620. fhp->Header.TocEntriesCount = 2;
  621. tocep = fhp->Header.TocEntry;
  622. tocep->InfoType = IPX_TRAFFIC_FILTER_GLOBAL_INFO_TYPE;
  623. tocep->InfoSize = sizeof(IPX_TRAFFIC_FILTER_GLOBAL_INFO);
  624. tocep->Count = 1;
  625. tocep->Offset = (ULONG)((PUCHAR)&(fhp->GlobalInfo) - (PUCHAR)FilterInfop);
  626. tocep++;
  627. tocep->InfoType = IPX_TRAFFIC_FILTER_INFO_TYPE;
  628. tocep->InfoSize = FilterSize;
  629. tocep->Count = FilterDriverInfoSize / FilterSize;
  630. tocep->Offset = sizeof(FILTERS_INFO_HEADER);
  631. fhp->GlobalInfo.FilterAction = FilterAction;
  632. return NO_ERROR;
  633. }
  634. */