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.

933 lines
33 KiB

  1. /**********************************************************************/
  2. /** Microsoft Windows NT **/
  3. /** Copyright(c) Microsoft Corp., 1993 **/
  4. /**********************************************************************/
  5. /*
  6. perfftp.cxx
  7. This file implements the Extensible Performance Objects for
  8. the FTP Server service.
  9. FILE HISTORY:
  10. KeithMo 07-Jun-1993 Created, based on RussBl's sample code.
  11. MuraliK 16-Nov-1995 Modified dependencies and removed NetApi
  12. SophiaC 06-Nov-1996 Supported mutlitiple instances
  13. */
  14. #define INITGUID
  15. #include <windows.h>
  16. #include <winperf.h>
  17. #include <lm.h>
  18. #include <string.h>
  19. #include <stdlib.h>
  20. #include <ole2.h>
  21. #include "iis64.h"
  22. #include "dbgutil.h"
  23. #include "iisinfo.h"
  24. #include "ftpd.h"
  25. #include "ftpctrs.h"
  26. #include "ftpmsg.h"
  27. #include "iadm.h"
  28. extern "C" {
  29. #include "perfutil.h"
  30. #include "apiutil.h"
  31. #include "ftpdata.h"
  32. } // extern "C"
  33. #define APP_NAME (TEXT("FTPCtrs"))
  34. #define MAX_SIZEOF_INSTANCE_NAME METADATA_MAX_NAME_LEN
  35. #define TOTAL_INSTANCE_NAME L"_Total"
  36. //
  37. // Private globals.
  38. //
  39. DWORD cOpens = 0; // Active "opens" reference count.
  40. BOOL fInitOK = FALSE; // TRUE if DLL initialized OK.
  41. HANDLE hEventLog = NULL; // event log handle
  42. //
  43. // Public prototypes.
  44. //
  45. PM_OPEN_PROC OpenFtpPerformanceData;
  46. PM_COLLECT_PROC CollectFtpPerformanceData;
  47. PM_CLOSE_PROC CloseFtpPerformanceData;
  48. //
  49. // Private prototypes.
  50. //
  51. VOID
  52. CopyStatisticsData(
  53. IN FTP_STATISTICS_0 * pFTPStats,
  54. OUT FTPD_COUNTER_BLOCK * pCounterBlock
  55. );
  56. VOID
  57. Update_TotalStatisticsData(
  58. IN FTPD_COUNTER_BLOCK * pCounterBlock,
  59. OUT FTPD_COUNTER_BLOCK * pTotal
  60. );
  61. //
  62. // Public functions.
  63. //
  64. /*******************************************************************
  65. NAME: OpenFtpPerformanceData
  66. SYNOPSIS: Initializes the data structures used to communicate
  67. performance counters with the registry.
  68. ENTRY: lpDeviceNames - Poitner to object ID of each device
  69. to be opened.
  70. RETURNS: DWORD - Win32 status code.
  71. HISTORY:
  72. KeithMo 07-Jun-1993 Created.
  73. ********************************************************************/
  74. DWORD OpenFtpPerformanceData( LPWSTR lpDeviceNames )
  75. {
  76. DWORD err = NO_ERROR;
  77. HKEY hkey = NULL;
  78. DWORD size;
  79. DWORD type;
  80. DWORD dwFirstCounter;
  81. DWORD dwFirstHelp;
  82. PERF_COUNTER_DEFINITION * pctr;
  83. FTPD_COUNTER_BLOCK ftpc;
  84. DWORD i;
  85. //
  86. // Since WINLOGON is multi-threaded and will call this routine in
  87. // order to service remote performance queries, this library
  88. // must keep track of how many times it has been opened (i.e.
  89. // how many threads have accessed it). The registry routines will
  90. // limit access to the initialization routine to only one thread
  91. // at a time so synchronization (i.e. reentrancy) should not be
  92. // a problem.
  93. //
  94. if( !fInitOK )
  95. {
  96. //
  97. // This is the *first* open.
  98. //
  99. // open event log interface
  100. if (hEventLog == NULL){
  101. hEventLog = RegisterEventSource ((LPTSTR)NULL, // Use Local Machine
  102. APP_NAME); // event log app name to find in registry
  103. if (hEventLog == NULL)
  104. {
  105. return GetLastError();
  106. }
  107. }
  108. //
  109. // Open the FTP Server service's Performance key.
  110. //
  111. err = RegOpenKeyEx( HKEY_LOCAL_MACHINE,
  112. FTPD_PERFORMANCE_KEY,
  113. 0,
  114. KEY_READ,
  115. &hkey );
  116. if( err == NO_ERROR )
  117. {
  118. //
  119. // Read the first counter DWORD.
  120. //
  121. size = sizeof(DWORD);
  122. err = RegQueryValueEx( hkey,
  123. "First Counter",
  124. NULL,
  125. &type,
  126. (LPBYTE)&dwFirstCounter,
  127. &size );
  128. if( err == NO_ERROR )
  129. {
  130. //
  131. // Read the first help DWORD.
  132. //
  133. size = sizeof(DWORD);
  134. err = RegQueryValueEx( hkey,
  135. "First Help",
  136. NULL,
  137. &type,
  138. (LPBYTE)&dwFirstHelp,
  139. &size );
  140. if ( err == NO_ERROR )
  141. {
  142. //
  143. // Update the object & counter name & help indicies.
  144. //
  145. FtpdDataDefinition.FtpdObjectType.ObjectNameTitleIndex
  146. += dwFirstCounter;
  147. FtpdDataDefinition.FtpdObjectType.ObjectHelpTitleIndex
  148. += dwFirstHelp;
  149. FtpdDataDefinition.FtpdBytesSent.CounterNameTitleIndex
  150. += dwFirstCounter;
  151. FtpdDataDefinition.FtpdBytesSent.CounterHelpTitleIndex
  152. += dwFirstHelp;
  153. FtpdDataDefinition.FtpdBytesSent.CounterOffset =
  154. (DWORD)((LPBYTE)&ftpc.BytesSent - (LPBYTE)&ftpc);
  155. FtpdDataDefinition.FtpdBytesReceived.CounterNameTitleIndex
  156. += dwFirstCounter;
  157. FtpdDataDefinition.FtpdBytesReceived.CounterHelpTitleIndex
  158. += dwFirstHelp;
  159. FtpdDataDefinition.FtpdBytesReceived.CounterOffset =
  160. (DWORD)((LPBYTE)&ftpc.BytesReceived - (LPBYTE)&ftpc);
  161. FtpdDataDefinition.FtpdBytesTotal.CounterNameTitleIndex
  162. += dwFirstCounter;
  163. FtpdDataDefinition.FtpdBytesTotal.CounterHelpTitleIndex
  164. += dwFirstHelp;
  165. FtpdDataDefinition.FtpdBytesTotal.CounterOffset =
  166. (DWORD)((LPBYTE)&ftpc.BytesTotal - (LPBYTE)&ftpc);
  167. FtpdDataDefinition.FtpdFilesSent.CounterNameTitleIndex
  168. += dwFirstCounter;
  169. FtpdDataDefinition.FtpdFilesSent.CounterHelpTitleIndex
  170. += dwFirstHelp;
  171. FtpdDataDefinition.FtpdFilesSent.CounterOffset =
  172. (DWORD)((LPBYTE)&ftpc.FilesSent - (LPBYTE)&ftpc);
  173. FtpdDataDefinition.FtpdFilesReceived.CounterNameTitleIndex
  174. += dwFirstCounter;
  175. FtpdDataDefinition.FtpdFilesReceived.CounterHelpTitleIndex
  176. += dwFirstHelp;
  177. FtpdDataDefinition.FtpdFilesReceived.CounterOffset =
  178. (DWORD)((LPBYTE)&ftpc.FilesReceived - (LPBYTE)&ftpc);
  179. FtpdDataDefinition.FtpdFilesTotal.CounterNameTitleIndex
  180. += dwFirstCounter;
  181. FtpdDataDefinition.FtpdFilesTotal.CounterHelpTitleIndex
  182. += dwFirstHelp;
  183. FtpdDataDefinition.FtpdFilesTotal.CounterOffset =
  184. (DWORD)((LPBYTE)&ftpc.FilesTotal - (LPBYTE)&ftpc);
  185. FtpdDataDefinition.FtpdCurrentAnonymous.CounterNameTitleIndex
  186. += dwFirstCounter;
  187. FtpdDataDefinition.FtpdCurrentAnonymous.CounterHelpTitleIndex
  188. += dwFirstHelp;
  189. FtpdDataDefinition.FtpdCurrentAnonymous.CounterOffset =
  190. (DWORD)((LPBYTE)&ftpc.CurrentAnonymous - (LPBYTE)&ftpc);
  191. FtpdDataDefinition.FtpdCurrentNonAnonymous.CounterNameTitleIndex
  192. += dwFirstCounter;
  193. FtpdDataDefinition.FtpdCurrentNonAnonymous.CounterHelpTitleIndex
  194. += dwFirstHelp;
  195. FtpdDataDefinition.FtpdCurrentNonAnonymous.CounterOffset =
  196. (DWORD)((LPBYTE)&ftpc.CurrentNonAnonymous - (LPBYTE)&ftpc);
  197. FtpdDataDefinition.FtpdTotalAnonymous.CounterNameTitleIndex
  198. += dwFirstCounter;
  199. FtpdDataDefinition.FtpdTotalAnonymous.CounterHelpTitleIndex
  200. += dwFirstHelp;
  201. FtpdDataDefinition.FtpdTotalAnonymous.CounterOffset =
  202. (DWORD)((LPBYTE)&ftpc.TotalAnonymous - (LPBYTE)&ftpc);
  203. FtpdDataDefinition.FtpdTotalNonAnonymous.CounterNameTitleIndex
  204. += dwFirstCounter;
  205. FtpdDataDefinition.FtpdTotalNonAnonymous.CounterHelpTitleIndex
  206. += dwFirstHelp;
  207. FtpdDataDefinition.FtpdTotalNonAnonymous.CounterOffset =
  208. (DWORD)((LPBYTE)&ftpc.TotalNonAnonymous - (LPBYTE)&ftpc);
  209. FtpdDataDefinition.FtpdMaxAnonymous.CounterNameTitleIndex
  210. += dwFirstCounter;
  211. FtpdDataDefinition.FtpdMaxAnonymous.CounterHelpTitleIndex
  212. += dwFirstHelp;
  213. FtpdDataDefinition.FtpdMaxAnonymous.CounterOffset =
  214. (DWORD)((LPBYTE)&ftpc.MaxAnonymous - (LPBYTE)&ftpc);
  215. FtpdDataDefinition.FtpdMaxNonAnonymous.CounterNameTitleIndex
  216. += dwFirstCounter;
  217. FtpdDataDefinition.FtpdMaxNonAnonymous.CounterHelpTitleIndex
  218. += dwFirstHelp;
  219. FtpdDataDefinition.FtpdMaxNonAnonymous.CounterOffset =
  220. (DWORD)((LPBYTE)&ftpc.MaxNonAnonymous - (LPBYTE)&ftpc);
  221. FtpdDataDefinition.FtpdCurrentConnections.CounterNameTitleIndex
  222. += dwFirstCounter;
  223. FtpdDataDefinition.FtpdCurrentConnections.CounterHelpTitleIndex
  224. += dwFirstHelp;
  225. FtpdDataDefinition.FtpdCurrentConnections.CounterOffset =
  226. (DWORD)((LPBYTE)&ftpc.CurrentConnections - (LPBYTE)&ftpc);
  227. FtpdDataDefinition.FtpdMaxConnections.CounterNameTitleIndex
  228. += dwFirstCounter;
  229. FtpdDataDefinition.FtpdMaxConnections.CounterHelpTitleIndex
  230. += dwFirstHelp;
  231. FtpdDataDefinition.FtpdMaxConnections.CounterOffset =
  232. (DWORD)((LPBYTE)&ftpc.MaxConnections - (LPBYTE)&ftpc);
  233. FtpdDataDefinition.FtpdConnectionAttempts.CounterNameTitleIndex
  234. += dwFirstCounter;
  235. FtpdDataDefinition.FtpdConnectionAttempts.CounterHelpTitleIndex
  236. += dwFirstHelp;
  237. FtpdDataDefinition.FtpdConnectionAttempts.CounterOffset =
  238. (DWORD)((LPBYTE)&ftpc.ConnectionAttempts - (LPBYTE)&ftpc);
  239. FtpdDataDefinition.FtpdLogonAttempts.CounterNameTitleIndex
  240. += dwFirstCounter;
  241. FtpdDataDefinition.FtpdLogonAttempts.CounterHelpTitleIndex
  242. += dwFirstHelp;
  243. FtpdDataDefinition.FtpdLogonAttempts.CounterOffset =
  244. (DWORD)((LPBYTE)&ftpc.LogonAttempts - (LPBYTE)&ftpc);
  245. FtpdDataDefinition.FtpdServiceUptime.CounterNameTitleIndex
  246. += dwFirstCounter;
  247. FtpdDataDefinition.FtpdServiceUptime.CounterHelpTitleIndex
  248. += dwFirstHelp;
  249. FtpdDataDefinition.FtpdServiceUptime.CounterOffset =
  250. (DWORD)((LPBYTE)&ftpc.ServiceUptime - (LPBYTE)&ftpc);
  251. // These counters are currently meaningless, but should be restored if we
  252. // ever enable per-FTP-instance bandwidth throttling.
  253. /*
  254. FtpdDataDefinition.FtpdBlockedRequests.CounterNameTitleIndex
  255. += dwFirstCounter;
  256. FtpdDataDefinition.FtpdBlockedRequests.CounterHelpTitleIndex
  257. += dwFirstHelp;
  258. FtpdDataDefinition.FtpdBlockedRequests.CounterOffset =
  259. (DWORD)((LPBYTE)&ftpc.BlockedRequests - (LPBYTE)&ftpc);
  260. FtpdDataDefinition.FtpdAllowedRequests.CounterNameTitleIndex
  261. += dwFirstCounter;
  262. FtpdDataDefinition.FtpdAllowedRequests.CounterHelpTitleIndex
  263. += dwFirstHelp;
  264. FtpdDataDefinition.FtpdAllowedRequests.CounterOffset =
  265. (DWORD)((LPBYTE)&ftpc.AllowedRequests - (LPBYTE)&ftpc);
  266. FtpdDataDefinition.FtpdRejectedRequests.CounterNameTitleIndex
  267. += dwFirstCounter;
  268. FtpdDataDefinition.FtpdRejectedRequests.CounterHelpTitleIndex
  269. += dwFirstHelp;
  270. FtpdDataDefinition.FtpdRejectedRequests.CounterOffset =
  271. (DWORD)((LPBYTE)&ftpc.RejectedRequests - (LPBYTE)&ftpc);
  272. FtpdDataDefinition.FtpdCurrentBlockedRequests.CounterNameTitleIndex
  273. += dwFirstCounter;
  274. FtpdDataDefinition.FtpdCurrentBlockedRequests.CounterHelpTitleIndex
  275. += dwFirstHelp;
  276. FtpdDataDefinition.FtpdCurrentBlockedRequests.CounterOffset =
  277. (DWORD)((LPBYTE)&ftpc.CurrentBlockedRequests - (LPBYTE)&ftpc);
  278. FtpdDataDefinition.FtpdMeasuredBandwidth.CounterNameTitleIndex
  279. += dwFirstCounter;
  280. FtpdDataDefinition.FtpdMeasuredBandwidth.CounterHelpTitleIndex
  281. += dwFirstHelp;
  282. FtpdDataDefinition.FtpdMeasuredBandwidth.CounterOffset =
  283. (DWORD)((LPBYTE)&ftpc.MeasuredBandwidth - (LPBYTE)&ftpc);
  284. */
  285. //
  286. // Remember that we initialized OK.
  287. //
  288. fInitOK = TRUE;
  289. } else {
  290. ReportEvent (hEventLog, EVENTLOG_ERROR_TYPE,
  291. 0, FTP_UNABLE_QUERY_DATA,
  292. (PSID)NULL, 0,
  293. sizeof(err), NULL,
  294. (PVOID)(&err));
  295. }
  296. } else {
  297. ReportEvent (hEventLog, EVENTLOG_ERROR_TYPE,
  298. 0, FTP_UNABLE_QUERY_DATA,
  299. (PSID)NULL, 0,
  300. sizeof(err), NULL,
  301. (PVOID)(&err));
  302. }
  303. //
  304. // Close the registry if we managed to actually open it.
  305. //
  306. if( hkey != NULL )
  307. {
  308. RegCloseKey( hkey );
  309. hkey = NULL;
  310. }
  311. } else {
  312. ReportEvent (hEventLog, EVENTLOG_ERROR_TYPE,
  313. 0, FTP_UNABLE_QUERY_DATA,
  314. (PSID)NULL, 0,
  315. sizeof(err), NULL,
  316. (PVOID)(&err));
  317. }
  318. }
  319. //
  320. // Bump open counter.
  321. //
  322. InterlockedIncrement((LPLONG )&cOpens);
  323. return err;
  324. } // OpenFTPPerformanceData
  325. /*******************************************************************
  326. NAME: CollectFtpPerformanceData
  327. SYNOPSIS: Initializes the data structures used to communicate
  328. ENTRY: lpValueName - The name of the value to retrieve.
  329. lppData - On entry contains a pointer to the buffer to
  330. receive the completed PerfDataBlock & subordinate
  331. structures. On exit, points to the first bytes
  332. *after* the data structures added by this routine.
  333. lpcbTotalBytes - On entry contains a pointer to the
  334. size (in BYTEs) of the buffer referenced by lppData.
  335. On exit, contains the number of BYTEs added by this
  336. routine.
  337. lpNumObjectTypes - Receives the number of objects added
  338. by this routine.
  339. RETURNS: DWORD - Win32 status code. MUST be either NO_ERROR
  340. or ERROR_MORE_DATA.
  341. HISTORY:
  342. KeithMo 07-Jun-1993 Created.
  343. ********************************************************************/
  344. DWORD CollectFtpPerformanceData( LPWSTR lpValueName,
  345. LPVOID * lppData,
  346. LPDWORD lpcbTotalBytes,
  347. LPDWORD lpNumObjectTypes )
  348. {
  349. PERF_INSTANCE_DEFINITION * pPerfInstanceDefinition;
  350. DWORD dwInstanceIndex = 0;
  351. DWORD dwInstanceCount = 0;
  352. DWORD i = 0;
  353. DWORD dwQueryType;
  354. ULONG cbRequired;
  355. DWORD * pdwCounter;
  356. LARGE_INTEGER * pliCounter;
  357. FTPD_COUNTER_BLOCK * pCounterBlock;
  358. FTPD_COUNTER_BLOCK * pTotal;
  359. FTPD_DATA_DEFINITION * pFtpdDataDefinition;
  360. FTP_STATISTICS_0 * pFTPStats;
  361. NET_API_STATUS neterr;
  362. HRESULT hresErr;
  363. DWORD dwBufferSize = 0;
  364. LPINET_INFO_SITE_LIST pSites;
  365. //
  366. // No need to even try if we failed to open...
  367. //
  368. if( !fInitOK )
  369. {
  370. *lpcbTotalBytes = 0;
  371. *lpNumObjectTypes = 0;
  372. //
  373. // According to the Performance Counter design, this
  374. // is a successful exit. Go figure.
  375. //
  376. return NO_ERROR;
  377. }
  378. //
  379. // Determine the query type.
  380. //
  381. dwQueryType = GetQueryType( lpValueName );
  382. if (( dwQueryType == QUERY_FOREIGN ) || (dwQueryType == QUERY_COSTLY))
  383. {
  384. //
  385. // We don't do foreign queries.
  386. //
  387. *lpcbTotalBytes = 0;
  388. *lpNumObjectTypes = 0;
  389. return NO_ERROR;
  390. }
  391. if( dwQueryType == QUERY_ITEMS )
  392. {
  393. //
  394. // The registry is asking for a specific object. Let's
  395. // see if we're one of the chosen.
  396. //
  397. if( !IsNumberInUnicodeList(
  398. FtpdDataDefinition.FtpdObjectType.ObjectNameTitleIndex,
  399. lpValueName ) )
  400. {
  401. *lpcbTotalBytes = 0;
  402. *lpNumObjectTypes = 0;
  403. return NO_ERROR;
  404. }
  405. }
  406. //
  407. // Enumerate and get total number of instances count.
  408. //
  409. pFtpdDataDefinition = (FTPD_DATA_DEFINITION *)*lppData;
  410. neterr = InetInfoGetSites(
  411. NULL,
  412. INET_FTP_SVC_ID,
  413. &pSites
  414. );
  415. if( neterr != NERR_Success )
  416. {
  417. //
  418. // Only event log once for each server down
  419. //
  420. // if the server is down, we don't log an error.
  421. if ( !( neterr == RPC_S_SERVER_UNAVAILABLE ||
  422. neterr == RPC_S_UNKNOWN_IF ||
  423. neterr == ERROR_SERVICE_NOT_ACTIVE ||
  424. neterr == RPC_S_CALL_FAILED_DNE ))
  425. {
  426. //
  427. // Error retrieving statistics.
  428. //
  429. ReportEvent(hEventLog, EVENTLOG_ERROR_TYPE,
  430. 0, FTP_UNABLE_QUERY_DATA,
  431. (PSID)NULL, 0,
  432. sizeof(neterr), NULL,
  433. (PVOID)(&neterr));
  434. }
  435. cbRequired = sizeof(FTPD_DATA_DEFINITION) +
  436. sizeof(PERF_INSTANCE_DEFINITION) +
  437. (sizeof(WCHAR) * MAX_SIZEOF_INSTANCE_NAME ) +
  438. sizeof(FTPD_COUNTER_BLOCK);
  439. if( *lpcbTotalBytes < cbRequired )
  440. {
  441. //
  442. // Nope.
  443. //
  444. *lpcbTotalBytes = 0;
  445. *lpNumObjectTypes = 0;
  446. return ERROR_MORE_DATA;
  447. }
  448. // We always return the _Total instance even if the counters
  449. // are not accessible. This is so we stay in sync with how the
  450. // other IIS counters work. For the PERF_NO_INSTANCE counters we need
  451. // to do this so that WMI get's it's definition data even if the service
  452. // is not running, and logging doesn't get confused by a PERF_NO_INSTANCE
  453. // counter returning zero instances.
  454. // First we move in the definition.
  455. memmove( pFtpdDataDefinition,
  456. &FtpdDataDefinition,
  457. sizeof(FTPD_DATA_DEFINITION) );
  458. ((PERF_OBJECT_TYPE*) pFtpdDataDefinition)->NumInstances = 1;
  459. ((PERF_OBJECT_TYPE*) pFtpdDataDefinition)->TotalByteLength = cbRequired;
  460. LPVOID pData = (LPVOID) ( pFtpdDataDefinition + 1 );
  461. // Now we setup the PERF_INSTANCE_DEFINITION
  462. ((PERF_INSTANCE_DEFINITION*) pData)->ByteLength = sizeof(PERF_INSTANCE_DEFINITION) +
  463. MAX_SIZEOF_INSTANCE_NAME * sizeof(WCHAR);
  464. ((PERF_INSTANCE_DEFINITION*) pData)->ParentObjectTitleIndex = 0;
  465. ((PERF_INSTANCE_DEFINITION*) pData)->ParentObjectInstance = 0;
  466. ((PERF_INSTANCE_DEFINITION*) pData)->UniqueID = PERF_NO_UNIQUE_ID;
  467. ((PERF_INSTANCE_DEFINITION*) pData)->NameOffset = sizeof(PERF_INSTANCE_DEFINITION);
  468. ((PERF_INSTANCE_DEFINITION*) pData)->NameLength = (wcslen(TOTAL_INSTANCE_NAME) + 1) * sizeof(WCHAR);
  469. pData = (LPBYTE) pData + sizeof(PERF_INSTANCE_DEFINITION);
  470. // Next copy in the Instance Name including the
  471. // NULL, we know we have enough room because of
  472. // the check above for size.
  473. wcsncpy ( (LPWSTR) pData, TOTAL_INSTANCE_NAME, wcslen(TOTAL_INSTANCE_NAME) + 1 );
  474. // To avoid suttle differences we use the same MAX_INSTANCE_NAME
  475. // amount of space even for this faked up _Total Site.
  476. pData = (LPBYTE) pData + ( MAX_SIZEOF_INSTANCE_NAME * sizeof(WCHAR));
  477. // Lastly copy in a block of zero's for the _Total site data.
  478. memset ( pData, 0, sizeof(FTPD_COUNTER_BLOCK) );
  479. // This is setting the size in the structure, it is the first
  480. // DWORD in the W3_CONTER_BLOCK.
  481. *((DWORD*) (pData)) = sizeof(FTPD_COUNTER_BLOCK);
  482. *lpcbTotalBytes = cbRequired;
  483. *lpNumObjectTypes = 1;
  484. *lppData = (LPBYTE) pFtpdDataDefinition + cbRequired;
  485. return NO_ERROR;
  486. }
  487. //
  488. // add 1 to dwInstanceCount for _Total instance
  489. //
  490. dwInstanceCount = pSites->cEntries + 1;
  491. //
  492. // always return an "instance sized" buffer after the definition
  493. // blocks to prevent perfmon from reading bogus data. This is strictly
  494. // a hack to accomodate how PERFMON handles the "0" instance case.
  495. // By doing this, perfmon won't choke when there are no instances
  496. // and the counter object & counters will be displayed in the list
  497. // boxes, even though no instances will be listed.
  498. //
  499. cbRequired = sizeof(FTPD_DATA_DEFINITION) +
  500. (dwInstanceCount * (sizeof(PERF_INSTANCE_DEFINITION) +
  501. MAX_SIZEOF_INSTANCE_NAME +
  502. sizeof (FTPD_COUNTER_BLOCK)));
  503. //
  504. // See if there's enough space.
  505. //
  506. if( *lpcbTotalBytes < cbRequired )
  507. {
  508. //
  509. // Nope.
  510. //
  511. *lpcbTotalBytes = 0;
  512. *lpNumObjectTypes = 0;
  513. MIDL_user_free(pSites);
  514. return ERROR_MORE_DATA;
  515. }
  516. //
  517. // Copy the (constant, initialized) Object Type and counter definitions
  518. // to the caller's data buffer
  519. //
  520. memmove( pFtpdDataDefinition,
  521. &FtpdDataDefinition,
  522. sizeof(FTPD_DATA_DEFINITION) );
  523. //
  524. // Create data for return for each instance
  525. //
  526. pPerfInstanceDefinition = (PERF_INSTANCE_DEFINITION *)
  527. &pFtpdDataDefinition[1];
  528. //
  529. // Set first block of Buffer for _Total
  530. //
  531. MonBuildInstanceDefinition(
  532. pPerfInstanceDefinition,
  533. (PVOID *)&pCounterBlock,
  534. 0,
  535. 0,
  536. (DWORD)-1, // use name
  537. TOTAL_INSTANCE_NAME ); // pass in instance name
  538. pTotal = pCounterBlock;
  539. memset( pTotal, 0, sizeof(FTPD_COUNTER_BLOCK ));
  540. pTotal->PerfCounterBlock.ByteLength = sizeof (FTPD_COUNTER_BLOCK);
  541. pPerfInstanceDefinition =
  542. (PERF_INSTANCE_DEFINITION *)((LPBYTE)pCounterBlock +
  543. sizeof(FTPD_COUNTER_BLOCK));
  544. neterr = FtpQueryStatistics2(
  545. NULL,
  546. 0,
  547. 0, // instance id, 0 for global stats
  548. 0,
  549. (LPBYTE *)&pFTPStats );
  550. if( neterr == NERR_Success )
  551. {
  552. pTotal->ServiceUptime = pFTPStats->ServiceUptime;
  553. }
  554. MIDL_user_free( pFTPStats );
  555. for ( i = 0; i < pSites->cEntries; i++)
  556. {
  557. MonBuildInstanceDefinition(
  558. pPerfInstanceDefinition,
  559. (PVOID *)&pCounterBlock,
  560. 0,
  561. 0,
  562. (DWORD)-1, // use name
  563. pSites->aSiteEntry[i].pszComment // pass in instance name
  564. );
  565. //
  566. // query for statistics info
  567. //
  568. neterr = FtpQueryStatistics2(
  569. NULL,
  570. 0,
  571. pSites->aSiteEntry[i].dwInstance, // instance id
  572. 0,
  573. (LPBYTE *)&pFTPStats );
  574. if( neterr != NERR_Success )
  575. {
  576. //
  577. // Only event log once for each server down
  578. //
  579. // if the server is down, we don't log an error.
  580. if ( !( neterr == RPC_S_SERVER_UNAVAILABLE ||
  581. neterr == RPC_S_UNKNOWN_IF ||
  582. neterr == ERROR_SERVICE_NOT_ACTIVE ||
  583. neterr == RPC_S_CALL_FAILED_DNE ))
  584. {
  585. //
  586. // Error retrieving statistics.
  587. //
  588. ReportEvent (hEventLog, EVENTLOG_ERROR_TYPE,
  589. 0, FTP_UNABLE_QUERY_DATA,
  590. (PSID)NULL, 0,
  591. sizeof(neterr), NULL,
  592. (PVOID)(&neterr));
  593. }
  594. *lpcbTotalBytes = 0;
  595. *lpNumObjectTypes = 0;
  596. MIDL_user_free(pSites);
  597. return NO_ERROR;
  598. }
  599. //
  600. // Format the FTP Server data.
  601. //
  602. CopyStatisticsData( pFTPStats,
  603. pCounterBlock );
  604. //
  605. // update _total instance counters
  606. //
  607. Update_TotalStatisticsData( pCounterBlock,
  608. pTotal );
  609. pPerfInstanceDefinition =
  610. (PERF_INSTANCE_DEFINITION *)((LPBYTE)pCounterBlock +
  611. sizeof(FTPD_COUNTER_BLOCK));
  612. //
  613. // Free the API buffer.
  614. //
  615. MIDL_user_free( pFTPStats );
  616. }
  617. if (dwInstanceCount == 1) {
  618. DWORD cbSize = sizeof(PERF_INSTANCE_DEFINITION) +
  619. MAX_SIZEOF_INSTANCE_NAME +
  620. sizeof(FTPD_COUNTER_BLOCK);
  621. //
  622. // zero fill one instance sized block of data if there's no data
  623. // instances
  624. //
  625. memset (pPerfInstanceDefinition, 0, cbSize);
  626. // adjust pointer to point to end of zeroed block
  627. pPerfInstanceDefinition += cbSize;
  628. }
  629. //
  630. // Update arguments for return.
  631. //
  632. *lppData = (PVOID)(pPerfInstanceDefinition);
  633. *lpNumObjectTypes = 1;
  634. pFtpdDataDefinition->FtpdObjectType.NumInstances = dwInstanceCount;
  635. pFtpdDataDefinition->FtpdObjectType.TotalByteLength =
  636. *lpcbTotalBytes = DIFF((PBYTE)pPerfInstanceDefinition -
  637. (PBYTE)pFtpdDataDefinition);
  638. //
  639. // Success! Honest!!
  640. //
  641. MIDL_user_free(pSites);
  642. return NO_ERROR;
  643. } // CollectFTPPerformanceData
  644. /*******************************************************************
  645. NAME: CloseFtpPerformanceData
  646. SYNOPSIS: Terminates the performance counters.
  647. RETURNS: DWORD - Win32 status code.
  648. HISTORY:
  649. KeithMo 07-Jun-1993 Created.
  650. ********************************************************************/
  651. DWORD CloseFtpPerformanceData( VOID )
  652. {
  653. DWORD dwCount = InterlockedDecrement((LPLONG )&cOpens);
  654. if ((dwCount) == 0) {
  655. if (hEventLog != NULL)
  656. {
  657. DeregisterEventSource (hEventLog);
  658. hEventLog = NULL;
  659. };
  660. }
  661. return NO_ERROR;
  662. } // CloseFTPPerformanceData
  663. VOID
  664. CopyStatisticsData(
  665. IN FTP_STATISTICS_0 * pFTPStats,
  666. OUT FTPD_COUNTER_BLOCK * pCounterBlock
  667. )
  668. {
  669. //
  670. // Format the FTP Server data.
  671. //
  672. pCounterBlock->PerfCounterBlock.ByteLength = sizeof (FTPD_COUNTER_BLOCK);
  673. pCounterBlock->BytesSent = pFTPStats->TotalBytesSent.QuadPart;
  674. pCounterBlock->BytesReceived = pFTPStats->TotalBytesReceived.QuadPart;
  675. pCounterBlock->BytesTotal = pFTPStats->TotalBytesSent.QuadPart +
  676. pFTPStats->TotalBytesReceived.QuadPart;
  677. pCounterBlock->FilesSent = pFTPStats->TotalFilesSent;
  678. pCounterBlock->FilesReceived = pFTPStats->TotalFilesReceived;
  679. pCounterBlock->FilesTotal = pFTPStats->TotalFilesSent +
  680. pFTPStats->TotalFilesReceived;
  681. pCounterBlock->CurrentAnonymous = pFTPStats->CurrentAnonymousUsers;
  682. pCounterBlock->CurrentNonAnonymous = pFTPStats->CurrentNonAnonymousUsers;
  683. pCounterBlock->TotalAnonymous = pFTPStats->TotalAnonymousUsers;
  684. pCounterBlock->TotalNonAnonymous = pFTPStats->TotalNonAnonymousUsers;
  685. pCounterBlock->MaxAnonymous = pFTPStats->MaxAnonymousUsers;
  686. pCounterBlock->MaxNonAnonymous = pFTPStats->MaxNonAnonymousUsers;
  687. pCounterBlock->CurrentConnections = pFTPStats->CurrentConnections;
  688. pCounterBlock->MaxConnections = pFTPStats->MaxConnections;
  689. pCounterBlock->ConnectionAttempts = pFTPStats->ConnectionAttempts;
  690. pCounterBlock->LogonAttempts = pFTPStats->LogonAttempts;
  691. pCounterBlock->ServiceUptime = pFTPStats->ServiceUptime;
  692. // These counters are currently meaningless, but should be restored if we
  693. // ever enable per-FTP-instance bandwidth throttling.
  694. /*
  695. pCounterBlock->BlockedRequests = pFTPStats->TotalBlockedRequests;
  696. pCounterBlock->AllowedRequests = pFTPStats->TotalAllowedRequests;
  697. pCounterBlock->RejectedRequests = pFTPStats->TotalRejectedRequests;
  698. pCounterBlock->MeasuredBandwidth= pFTPStats->MeasuredBandwidth;
  699. pCounterBlock->CurrentBlockedRequests = pFTPStats->CurrentBlockedRequests;
  700. */
  701. } // CopyStatisticsData
  702. VOID
  703. Update_TotalStatisticsData(
  704. IN FTPD_COUNTER_BLOCK * pCounterBlock,
  705. OUT FTPD_COUNTER_BLOCK * pTotal
  706. )
  707. {
  708. //
  709. // update _total instance counters
  710. //
  711. pTotal->BytesSent += pCounterBlock->BytesSent;
  712. pTotal->BytesReceived += pCounterBlock->BytesReceived;
  713. pTotal->BytesTotal += pCounterBlock->BytesTotal;
  714. pTotal->FilesSent += pCounterBlock->FilesSent;
  715. pTotal->FilesReceived += pCounterBlock->FilesReceived;
  716. pTotal->FilesTotal += pCounterBlock->FilesTotal;
  717. pTotal->CurrentAnonymous += pCounterBlock->CurrentAnonymous;
  718. pTotal->CurrentNonAnonymous += pCounterBlock->CurrentNonAnonymous;
  719. pTotal->TotalAnonymous += pCounterBlock->TotalAnonymous;
  720. pTotal->TotalNonAnonymous += pCounterBlock->TotalNonAnonymous;
  721. pTotal->MaxAnonymous += pCounterBlock->MaxAnonymous;
  722. pTotal->MaxNonAnonymous += pCounterBlock->MaxNonAnonymous;
  723. pTotal->CurrentConnections += pCounterBlock->CurrentConnections;
  724. pTotal->MaxConnections += pCounterBlock->MaxConnections;
  725. pTotal->ConnectionAttempts = pCounterBlock->ConnectionAttempts;
  726. pTotal->LogonAttempts += pCounterBlock->LogonAttempts;
  727. // These counters are currently meaningless, but should be restored if we
  728. // ever enable per-FTP-instance bandwidth throttling.
  729. /*
  730. pTotal->BlockedRequests += pCounterBlock->BlockedRequests;
  731. pTotal->RejectedRequests += pCounterBlock->RejectedRequests;
  732. pTotal->AllowedRequests += pCounterBlock->AllowedRequests;
  733. pTotal->MeasuredBandwidth += pCounterBlock->MeasuredBandwidth;
  734. pTotal->CurrentBlockedRequests += pCounterBlock->CurrentBlockedRequests;
  735. */
  736. } // Update_TotalStatisticsData