Source code of Windows XP (NT5)
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.

1320 lines
35 KiB

  1. /********************************************************************/
  2. /** Copyright(c) 1989 Microsoft Corporation. **/
  3. /********************************************************************/
  4. //***
  5. //
  6. // Filename: init.c
  7. //
  8. // Description: This module contains all the code to initialize the PPP
  9. // engine.
  10. //
  11. // History:
  12. // Nov 11,1993. NarenG Created original version.
  13. //
  14. #include <nt.h>
  15. #include <ntrtl.h>
  16. #include <nturtl.h> // needed for winbase.h
  17. #include <windows.h> // Win32 base API's
  18. #include <stdlib.h>
  19. #include <string.h>
  20. #include <time.h>
  21. #include <wchar.h>
  22. #include <raserror.h>
  23. #include <rasman.h>
  24. #include <rtutils.h>
  25. #include <mprlog.h>
  26. #include <lmcons.h>
  27. #include <rasppp.h>
  28. #include <pppcp.h>
  29. #include <lcp.h>
  30. #define _ALLOCATE_GLOBALS_
  31. #include <ppp.h>
  32. #include <timer.h>
  33. #include <util.h>
  34. #include <worker.h>
  35. #include <init.h>
  36. #include <rasauth.h>
  37. #include <bap.h>
  38. #include <raseapif.h>
  39. #define __NOT_INCLUDE_OpenRAS_IASProfileDlg__
  40. #include <dialinusr.h>
  41. #define ALLOC_BLTINCPS_GLOBALS
  42. #include <bltincps.h>
  43. // AFP Server Service registry parameter structure
  44. //
  45. typedef struct _PPP_REGISTRY_PARAMS {
  46. LPSTR pszValueName;
  47. DWORD * pValue;
  48. DWORD Max;
  49. DWORD dwDefValue;
  50. } PPP_REGISTRY_PARAMS, *PPPP_REGISTRY_PARAMS;
  51. PPP_REGISTRY_PARAMS PppRegParams[] =
  52. {
  53. RAS_VALUENAME_MAXTERMINATE,
  54. &(PppConfigInfo.MaxTerminate),
  55. 255,
  56. PPP_DEF_MAXTERMINATE,
  57. RAS_VALUENAME_MAXCONFIGURE,
  58. &(PppConfigInfo.MaxConfigure),
  59. 255,
  60. PPP_DEF_MAXCONFIGURE,
  61. RAS_VALUENAME_MAXFAILURE,
  62. &(PppConfigInfo.MaxFailure),
  63. 255,
  64. PPP_DEF_MAXFAILURE,
  65. RAS_VALUENAME_MAXREJECT,
  66. &(PppConfigInfo.MaxReject),
  67. 255,
  68. PPP_DEF_MAXREJECT,
  69. RAS_VALUENAME_RESTARTTIMER,
  70. &(PppConfigInfo.DefRestartTimer),
  71. 0xFFFFFFFF,
  72. PPP_DEF_RESTARTTIMER,
  73. RAS_VALUENAME_NEGOTIATETIME,
  74. &(PppConfigInfo.NegotiateTime),
  75. 0xFFFFFFFF,
  76. PPP_DEF_NEGOTIATETIME,
  77. RAS_VALUENAME_CALLBACKDELAY,
  78. &(PppConfigInfo.dwCallbackDelay),
  79. 255,
  80. PPP_DEF_CALLBACKDELAY,
  81. RAS_VALUENAME_PORTLIMIT,
  82. &(PppConfigInfo.dwDefaultPortLimit),
  83. 0xFFFFFFFF,
  84. PPP_DEF_PORTLIMIT,
  85. RAS_VALUENAME_SESSIONTIMEOUT,
  86. &(PppConfigInfo.dwDefaultSessionTimeout),
  87. 0xFFFFFFFF,
  88. PPP_DEF_SESSIONTIMEOUT,
  89. RAS_VALUENAME_IDLETIMEOUT,
  90. &(PppConfigInfo.dwDefaulIdleTimeout),
  91. 0xFFFFFFFF,
  92. PPP_DEF_IDLETIMEOUT,
  93. RAS_VALUENAME_BAPTHRESHOLD,
  94. &(PppConfigInfo.dwHangupExtraPercent),
  95. 100,
  96. RAS_DEF_BAPLINEDNLIMIT,
  97. RAS_VALUENAME_BAPTIME,
  98. &(PppConfigInfo.dwHangUpExtraSampleSeconds),
  99. 0xFFFFFFFF,
  100. RAS_DEF_BAPLINEDNTIME,
  101. RAS_VALUENAME_BAPLISTENTIME,
  102. &(PppConfigInfo.dwBapListenTimeoutSeconds),
  103. 0xFFFFFFFF,
  104. PPP_DEF_BAPLISTENTIME,
  105. RAS_VALUENAME_UNKNOWNPACKETTRACESIZE,
  106. &(PppConfigInfo.dwUnknownPacketTraceSize),
  107. 0xFFFFFFFF,
  108. PPP_DEF_UNKNOWNPACKETTRACESIZE,
  109. RAS_ECHO_REQUEST_INTERVAL,
  110. &(PppConfigInfo.dwLCPEchoTimeInterval),
  111. 0xFFFFFFFF,
  112. PPP_DEF_ECHO_REQUEST_INTERVAL, //Default of 60 seconds
  113. RAS_ECHO_REQUEST_IDLE,
  114. &(PppConfigInfo.dwIdleBeforeEcho),
  115. 0xFFFFFFFF,
  116. PPP_DEF_ECHO_REQUEST_IDLE, //Default of 300 seconds
  117. RAS_ECHO_NUM_MISSED_ECHOS,
  118. &(PppConfigInfo.dwNumMissedEchosBeforeDisconnect),
  119. 0xFFFFFFFF,
  120. PPP_DEF_ECHO_NUM_MISSED_ECHOS, //Default of 3 tries
  121. RAS_DONTNEGOTIATE_MULTILINKONSINGLELINK,
  122. &(PppConfigInfo.dwDontNegotiateMultiLinkOnSingleLink),
  123. 0xFFFFFFFF,
  124. 0,
  125. NULL, NULL, 0, 0
  126. };
  127. static DLL_ENTRY_POINTS * pCpDlls = (DLL_ENTRY_POINTS*)NULL;
  128. HANDLE HInstDLL;
  129. //**
  130. //
  131. // Call: LoadProtocolDlls
  132. //
  133. // Returns: NO_ERROR - Success
  134. // non-zero code - Failure
  135. //
  136. // Description: This procedure enumerates all the Subkeys under the PPP key
  137. // and loads each AP or CP and fills up the DLL_ENTRY_POINTS
  138. // structure with the required entry points. It also will return
  139. // the total number of protocols in all the Dlls. Note that each
  140. // DLL could have up to PPPCP_MAXCPSPERDLL protocols.
  141. //
  142. DWORD
  143. LoadProtocolDlls(
  144. IN DLL_ENTRY_POINTS * pCpDlls,
  145. IN DWORD cCpDlls,
  146. IN HKEY hKeyProtocols,
  147. OUT DWORD * pcTotalNumProtocols
  148. )
  149. {
  150. HKEY hKeyCp = (HKEY)NULL;
  151. LPSTR pCpDllPath = (LPSTR)NULL;
  152. LPSTR pCpDllExpandedPath = (LPSTR)NULL;
  153. DWORD dwKeyIndex;
  154. DWORD dwRetCode;
  155. CHAR chSubKeyName[100];
  156. DWORD cbSubKeyName;
  157. DWORD dwNumSubKeys;
  158. DWORD dwMaxSubKeySize;
  159. DWORD dwNumValues;
  160. DWORD cbMaxValNameLen;
  161. DWORD cbMaxValueDataSize;
  162. DWORD dwSecDescLen;
  163. DWORD ProtocolIds[PPPCP_MAXCPSPERDLL];
  164. DWORD dwNumProtocolIds;
  165. FARPROC pRasCpEnumProtocolIds;
  166. FARPROC pRasCpGetInfo;
  167. DWORD cbSize;
  168. DWORD dwType;
  169. HINSTANCE hInstance;
  170. //
  171. // Read the registry to find out the various control protocols to load.
  172. //
  173. for ( dwKeyIndex = 0; dwKeyIndex < cCpDlls; dwKeyIndex++ )
  174. {
  175. cbSubKeyName = sizeof( chSubKeyName );
  176. dwRetCode = RegEnumKeyEx(
  177. hKeyProtocols,
  178. dwKeyIndex,
  179. chSubKeyName,
  180. &cbSubKeyName,
  181. NULL,
  182. NULL,
  183. NULL,
  184. NULL
  185. );
  186. if ( ( dwRetCode != NO_ERROR ) &&
  187. ( dwRetCode != ERROR_MORE_DATA ) &&
  188. ( dwRetCode != ERROR_NO_MORE_ITEMS ) )
  189. {
  190. PppLogErrorString(ROUTERLOG_CANT_ENUM_REGKEYVALUES,0,
  191. NULL,dwRetCode,0);
  192. break;
  193. }
  194. dwRetCode = RegOpenKeyEx(
  195. hKeyProtocols,
  196. chSubKeyName,
  197. 0,
  198. KEY_QUERY_VALUE,
  199. &hKeyCp );
  200. if ( dwRetCode != NO_ERROR )
  201. {
  202. PppLogErrorString(ROUTERLOG_CANT_OPEN_PPP_REGKEY,0,NULL,
  203. dwRetCode,0);
  204. break;
  205. }
  206. //
  207. // Find out the size of the path value.
  208. //
  209. dwRetCode = RegQueryInfoKey(
  210. hKeyCp,
  211. NULL,
  212. NULL,
  213. NULL,
  214. &dwNumSubKeys,
  215. &dwMaxSubKeySize,
  216. NULL,
  217. &dwNumValues,
  218. &cbMaxValNameLen,
  219. &cbMaxValueDataSize,
  220. NULL,
  221. NULL
  222. );
  223. if ( dwRetCode != NO_ERROR )
  224. {
  225. PppLogErrorString(ROUTERLOG_CANT_OPEN_PPP_REGKEY,0,NULL,
  226. dwRetCode,0);
  227. break;
  228. }
  229. //
  230. // Allocate space for path and add one for NULL terminator
  231. //
  232. pCpDllPath = (LPBYTE)LOCAL_ALLOC( LPTR, ++cbMaxValueDataSize );
  233. if ( pCpDllPath == (LPBYTE)NULL )
  234. {
  235. dwRetCode = GetLastError();
  236. PppLogError( ROUTERLOG_NOT_ENOUGH_MEMORY, 0, NULL, dwRetCode);
  237. break;
  238. }
  239. //
  240. // Read in the path
  241. //
  242. dwRetCode = RegQueryValueEx(
  243. hKeyCp,
  244. RAS_VALUENAME_PATH,
  245. NULL,
  246. &dwType,
  247. pCpDllPath,
  248. &cbMaxValueDataSize
  249. );
  250. if ( dwRetCode != NO_ERROR )
  251. {
  252. PppLogError(ROUTERLOG_CANT_GET_REGKEYVALUES, 0, NULL, dwRetCode );
  253. break;
  254. }
  255. if ( ( dwType != REG_EXPAND_SZ ) && ( dwType != REG_SZ ) )
  256. {
  257. dwRetCode = ERROR_REGISTRY_CORRUPT;
  258. PppLogError( ROUTERLOG_CANT_GET_REGKEYVALUES, 0, NULL, dwRetCode );
  259. break;
  260. }
  261. //
  262. // Replace the %SystemRoot% with the actual path.
  263. //
  264. cbSize = ExpandEnvironmentStrings( pCpDllPath, NULL, 0 );
  265. if ( cbSize == 0 )
  266. {
  267. dwRetCode = GetLastError();
  268. PppLogError( ROUTERLOG_CANT_GET_REGKEYVALUES, 0, NULL, dwRetCode );
  269. break;
  270. }
  271. pCpDllExpandedPath = (LPSTR)LOCAL_ALLOC( LPTR, ++cbSize );
  272. if ( pCpDllExpandedPath == (LPSTR)NULL )
  273. {
  274. dwRetCode = GetLastError();
  275. PppLogError( ROUTERLOG_NOT_ENOUGH_MEMORY, 0, NULL, dwRetCode);
  276. break;
  277. }
  278. cbSize = ExpandEnvironmentStrings(
  279. pCpDllPath,
  280. pCpDllExpandedPath,
  281. cbSize );
  282. if ( cbSize == 0 )
  283. {
  284. dwRetCode = GetLastError();
  285. PppLogError(ROUTERLOG_CANT_GET_REGKEYVALUES,0,NULL,dwRetCode);
  286. break;
  287. }
  288. hInstance = LoadLibrary( pCpDllExpandedPath );
  289. if ( hInstance == (HINSTANCE)NULL )
  290. {
  291. dwRetCode = GetLastError();
  292. PppLogErrorString( ROUTERLOG_PPP_CANT_LOAD_DLL,1,
  293. &pCpDllExpandedPath,dwRetCode, 1);
  294. break;
  295. }
  296. pRasCpEnumProtocolIds = GetProcAddress( hInstance,
  297. "RasCpEnumProtocolIds" );
  298. if ( pRasCpEnumProtocolIds == (FARPROC)NULL )
  299. {
  300. dwRetCode = GetLastError();
  301. PppLogErrorString( ROUTERLOG_PPPCP_DLL_ERROR, 1,
  302. &pCpDllExpandedPath, dwRetCode, 1);
  303. break;
  304. }
  305. pCpDlls[dwKeyIndex].pRasCpEnumProtocolIds = pRasCpEnumProtocolIds;
  306. dwRetCode = (DWORD) (*pRasCpEnumProtocolIds)( ProtocolIds, &dwNumProtocolIds );
  307. if ( dwRetCode != NO_ERROR )
  308. {
  309. PppLogErrorString(ROUTERLOG_PPPCP_DLL_ERROR, 1,
  310. &pCpDllExpandedPath, dwRetCode, 1);
  311. break;
  312. }
  313. (*pcTotalNumProtocols) += dwNumProtocolIds;
  314. pRasCpGetInfo = GetProcAddress( hInstance, "RasCpGetInfo" );
  315. if ( pRasCpGetInfo == (FARPROC)NULL )
  316. {
  317. dwRetCode = GetLastError();
  318. PppLogErrorString(ROUTERLOG_PPPCP_DLL_ERROR, 1,
  319. &pCpDllExpandedPath, dwRetCode, 1);
  320. break;
  321. }
  322. pCpDlls[dwKeyIndex].pRasCpGetInfo = pRasCpGetInfo;
  323. RegCloseKey( hKeyCp );
  324. hKeyCp = (HKEY)NULL;
  325. pCpDlls[dwKeyIndex].pszModuleName = pCpDllExpandedPath;
  326. pCpDlls[dwKeyIndex].hInstance = hInstance;
  327. if ( NULL != pCpDllPath )
  328. LOCAL_FREE( pCpDllPath );
  329. pCpDllPath = (LPSTR)NULL;
  330. }
  331. if ( hKeyCp != (HKEY)NULL )
  332. RegCloseKey( hKeyCp );
  333. if ( pCpDllPath != (LPSTR)NULL )
  334. LOCAL_FREE( pCpDllPath );
  335. return( dwRetCode );
  336. }
  337. //**
  338. //
  339. // Call: ReadPPPKeyValues
  340. //
  341. // Returns: NO_ERROR - Success
  342. // Non-zero - Failure
  343. //
  344. // Description: Will read in all the values in the PPP key.
  345. //
  346. DWORD
  347. ReadPPPKeyValues(
  348. IN HKEY hKeyPpp
  349. )
  350. {
  351. DWORD dwIndex;
  352. DWORD dwRetCode;
  353. DWORD cbValueBuf;
  354. DWORD dwType;
  355. //
  356. // Run through and get all the PPP values
  357. //
  358. for ( dwIndex = 0; PppRegParams[dwIndex].pszValueName != NULL; dwIndex++ )
  359. {
  360. cbValueBuf = sizeof( DWORD );
  361. dwRetCode = RegQueryValueEx(
  362. hKeyPpp,
  363. PppRegParams[dwIndex].pszValueName,
  364. NULL,
  365. &dwType,
  366. (LPBYTE)(PppRegParams[dwIndex].pValue),
  367. &cbValueBuf
  368. );
  369. if ((dwRetCode != NO_ERROR) && (dwRetCode != ERROR_FILE_NOT_FOUND))
  370. {
  371. PppLogError(ROUTERLOG_CANT_GET_REGKEYVALUES,0,NULL,dwRetCode);
  372. break;
  373. }
  374. if ( dwRetCode == ERROR_FILE_NOT_FOUND )
  375. {
  376. *(PppRegParams[dwIndex].pValue) = PppRegParams[dwIndex].dwDefValue;
  377. dwRetCode = NO_ERROR;
  378. }
  379. else
  380. {
  381. if ( ( dwType != REG_DWORD ) ||
  382. ( *(PppRegParams[dwIndex].pValue) > PppRegParams[dwIndex].Max))
  383. {
  384. CHAR * pChar = PppRegParams[dwIndex].pszValueName;
  385. PppLogWarning(ROUTERLOG_REGVALUE_OVERIDDEN, 1,&pChar);
  386. *(PppRegParams[dwIndex].pValue)
  387. = PppRegParams[dwIndex].dwDefValue;
  388. }
  389. }
  390. }
  391. if ( dwRetCode != NO_ERROR )
  392. {
  393. return( ERROR_REGISTRY_CORRUPT );
  394. }
  395. //
  396. // If value is zero use defaults.
  397. //
  398. if ( PppConfigInfo.MaxTerminate == 0 )
  399. {
  400. PppConfigInfo.MaxTerminate = PPP_DEF_MAXTERMINATE;
  401. }
  402. if ( PppConfigInfo.MaxFailure == 0 )
  403. {
  404. PppConfigInfo.MaxFailure = PPP_DEF_MAXFAILURE;
  405. }
  406. if ( PppConfigInfo.MaxConfigure == 0 )
  407. {
  408. PppConfigInfo.MaxConfigure = PPP_DEF_MAXCONFIGURE;
  409. }
  410. if ( PppConfigInfo.MaxReject == 0 )
  411. {
  412. PppConfigInfo.MaxReject = PPP_DEF_MAXREJECT;
  413. }
  414. //
  415. // Really the number for request retries so subtract one.
  416. //
  417. PppConfigInfo.MaxTerminate--;
  418. PppConfigInfo.MaxConfigure--;
  419. return( NO_ERROR );
  420. }
  421. //**
  422. //
  423. // Call: ReadRegistryInfo
  424. //
  425. // Returns: NO_ERROR - Success
  426. // non-zero WIN32 error - failure
  427. //
  428. // Description: Will read all PPP information in the registry. Will load the
  429. // control and authentication protocol dlls and
  430. // initialze the CpTable with information about the protocols.
  431. //
  432. DWORD
  433. ReadRegistryInfo(
  434. OUT HKEY * phKeyPpp
  435. )
  436. {
  437. HKEY hKeyProtocols = (HKEY)NULL;
  438. DWORD dwNumSubKeys = 0;
  439. DWORD dwMaxSubKeySize;
  440. DWORD dwNumValues;
  441. DWORD cbMaxValNameLen;
  442. DWORD cbMaxValueDataSize;
  443. DWORD dwSecDescLen;
  444. FILETIME LastWrite;
  445. DWORD dwRetCode;
  446. DWORD ProtocolIds[PPPCP_MAXCPSPERDLL];
  447. DWORD dwNumProtocolIds;
  448. DWORD cTotalNumProtocols = 0;
  449. PPPCP_ENTRY CpEntry;
  450. DWORD dwIndex;
  451. DWORD cbValueBuf;
  452. DWORD dwType;
  453. DWORD dwValue;
  454. do
  455. {
  456. dwRetCode = RegOpenKeyEx(
  457. HKEY_LOCAL_MACHINE,
  458. RAS_KEYPATH_PPP,
  459. 0,
  460. KEY_READ,
  461. phKeyPpp );
  462. if ( dwRetCode != NO_ERROR)
  463. {
  464. PppLogErrorString(ROUTERLOG_CANT_OPEN_PPP_REGKEY,0,NULL,
  465. dwRetCode,0);
  466. break;
  467. }
  468. dwRetCode = RegOpenKeyEx(
  469. HKEY_LOCAL_MACHINE,
  470. RAS_KEYPATH_PROTOCOLS,
  471. 0,
  472. KEY_READ,
  473. &hKeyProtocols );
  474. if ( dwRetCode != NO_ERROR)
  475. {
  476. PppLogErrorString(ROUTERLOG_CANT_OPEN_PPP_REGKEY,0,NULL,
  477. dwRetCode,0);
  478. break;
  479. }
  480. //
  481. // Find out how many sub-keys or dlls there are
  482. //
  483. dwRetCode = RegQueryInfoKey(
  484. hKeyProtocols,
  485. NULL,
  486. NULL,
  487. NULL,
  488. &dwNumSubKeys,
  489. &dwMaxSubKeySize,
  490. NULL,
  491. &dwNumValues,
  492. &cbMaxValNameLen,
  493. &cbMaxValueDataSize,
  494. NULL,
  495. NULL
  496. );
  497. if ( dwRetCode != NO_ERROR )
  498. {
  499. PppLogErrorString(ROUTERLOG_CANT_OPEN_PPP_REGKEY,0,
  500. NULL,dwRetCode,0);
  501. break;
  502. }
  503. //
  504. // Cannot have no APs or NCPs
  505. //
  506. if ( dwNumSubKeys == 0 )
  507. {
  508. PppLogError( ROUTERLOG_NO_AUTHENTICATION_CPS, 0, NULL, 0 );
  509. dwRetCode = ERROR_REGISTRY_CORRUPT;
  510. break;
  511. }
  512. dwRetCode = ReadPPPKeyValues( *phKeyPpp );
  513. if ( dwRetCode != NO_ERROR )
  514. {
  515. break;
  516. }
  517. LoadParserDll( PppConfigInfo.hKeyPpp );
  518. //
  519. // Allocate space to hold entry points for all the CP dlls
  520. //
  521. pCpDlls = (DLL_ENTRY_POINTS*)LOCAL_ALLOC( LPTR,
  522. sizeof( DLL_ENTRY_POINTS )
  523. * (dwNumSubKeys + 1) );
  524. if ( pCpDlls == (DLL_ENTRY_POINTS*)NULL )
  525. {
  526. dwRetCode = GetLastError();
  527. PppLogError( ROUTERLOG_NOT_ENOUGH_MEMORY, 0, NULL, dwRetCode );
  528. break;
  529. }
  530. pCpDlls[dwNumSubKeys].hInstance = INVALID_HANDLE_VALUE;
  531. //
  532. // Load all the AP and CP dlls and get their entry points
  533. //
  534. dwRetCode = LoadProtocolDlls(
  535. pCpDlls,
  536. dwNumSubKeys,
  537. hKeyProtocols,
  538. &cTotalNumProtocols );
  539. if ( dwRetCode != NO_ERROR )
  540. break;
  541. //
  542. // We now know how big the CpTable structure has to be so allocate space
  543. // for it. Add one for LCP.
  544. //
  545. CpTable = (PPPCP_ENTRY *)LOCAL_ALLOC( LPTR, sizeof( PPPCP_ENTRY ) *
  546. ( cTotalNumProtocols + 1 ) );
  547. if ( CpTable == (PPPCP_ENTRY *)NULL)
  548. {
  549. dwRetCode = GetLastError();
  550. PppLogError( ROUTERLOG_NOT_ENOUGH_MEMORY, 0, NULL, dwRetCode );
  551. break;
  552. }
  553. //
  554. // Now fill up the table. First fill up information for LCP
  555. //
  556. dwRetCode = LcpGetInfo( PPP_LCP_PROTOCOL,
  557. &(CpTable[LCP_INDEX].CpInfo) );
  558. if ( dwRetCode != NO_ERROR )
  559. {
  560. CHAR * pChar = "LCP";
  561. PppLogErrorString(ROUTERLOG_PPPCP_DLL_ERROR, 1,
  562. &pChar, dwRetCode, 1);
  563. break;
  564. }
  565. PppConfigInfo.NumberOfCPs = 1;
  566. PppConfigInfo.NumberOfAPs = 0;
  567. //
  568. // Fill up the table with the loaded APs and CPs. The CPs start from
  569. // 1 and increase the APs start from cTotolNumProtocols and go down.
  570. //
  571. for ( dwIndex = 0; dwIndex < dwNumSubKeys; dwIndex++ )
  572. {
  573. dwRetCode = (DWORD)(pCpDlls[dwIndex].pRasCpEnumProtocolIds)(
  574. ProtocolIds,
  575. &dwNumProtocolIds );
  576. if ( dwRetCode != NO_ERROR )
  577. {
  578. PppLogErrorString(
  579. ROUTERLOG_PPPCP_DLL_ERROR,
  580. 1,
  581. &(pCpDlls[dwIndex].pszModuleName),
  582. dwRetCode,
  583. 1 );
  584. break;
  585. }
  586. if ( ( dwNumProtocolIds == 0 ) ||
  587. ( dwNumProtocolIds > PPPCP_MAXCPSPERDLL ) )
  588. {
  589. dwRetCode = ERROR_INVALID_PARAMETER;
  590. PppLogErrorString(
  591. ROUTERLOG_PPPCP_DLL_ERROR,
  592. 1,
  593. &(pCpDlls[dwIndex].pszModuleName),
  594. dwRetCode,
  595. 1 );
  596. break;
  597. }
  598. while( dwNumProtocolIds-- > 0 )
  599. {
  600. ZeroMemory( &CpEntry, sizeof( CpEntry ) );
  601. dwRetCode = (DWORD)(pCpDlls[dwIndex].pRasCpGetInfo)(
  602. ProtocolIds[dwNumProtocolIds],
  603. &CpEntry.CpInfo );
  604. if ( dwRetCode != NO_ERROR )
  605. {
  606. PppLogErrorString(
  607. ROUTERLOG_PPPCP_DLL_ERROR,
  608. 1,
  609. &(pCpDlls[dwIndex].pszModuleName),
  610. dwRetCode,
  611. 1 );
  612. break;
  613. }
  614. if ( CpEntry.CpInfo.Protocol == PPP_IPCP_PROTOCOL )
  615. {
  616. PppConfigInfo.RasIpcpDhcpInform =
  617. (DWORD(*)(VOID*, PPP_DHCP_INFORM*))
  618. IpcpDhcpInform;
  619. PppConfigInfo.RasIphlpDhcpCallback =
  620. (VOID(*)(ULONG))
  621. RasSrvrDhcpCallback;
  622. }
  623. if ( CpEntry.CpInfo.RasCpInit != NULL )
  624. {
  625. if ( (PPP_IPCP_PROTOCOL == CpEntry.CpInfo.Protocol)
  626. || (PPP_IPXCP_PROTOCOL == CpEntry.CpInfo.Protocol)
  627. || (PPP_NBFCP_PROTOCOL == CpEntry.CpInfo.Protocol)
  628. || (PPP_ATCP_PROTOCOL == CpEntry.CpInfo.Protocol)
  629. )
  630. {
  631. // Do not init the CP.
  632. }
  633. else
  634. {
  635. PppLog(1, "RasCpInit(%x, TRUE)", CpEntry.CpInfo.Protocol);
  636. dwRetCode = CpEntry.CpInfo.RasCpInit(
  637. TRUE/* fInitialize */);
  638. CpEntry.fFlags |= PPPCP_FLAG_INIT_CALLED;
  639. if ( dwRetCode != NO_ERROR )
  640. {
  641. CHAR* SubStringArray[2];
  642. SubStringArray[0] = CpEntry.CpInfo.SzProtocolName;
  643. SubStringArray[1] = pCpDlls[dwIndex].pszModuleName;
  644. PppLogErrorString(
  645. ROUTERLOG_PPPCP_INIT_ERROR,
  646. 2,
  647. SubStringArray,
  648. dwRetCode,
  649. 2 );
  650. break;
  651. }
  652. else
  653. {
  654. CpEntry.fFlags |= PPPCP_FLAG_AVAILABLE;
  655. }
  656. }
  657. }
  658. //
  659. // If this entry point is NULL we assume that this is a CP.
  660. //
  661. if ( CpEntry.CpInfo.RasApMakeMessage == NULL )
  662. {
  663. if ( ( CpEntry.CpInfo.RasCpBegin == NULL ) ||
  664. ( CpEntry.CpInfo.RasCpEnd == NULL ) ||
  665. ( CpEntry.CpInfo.RasCpReset == NULL ) ||
  666. ( CpEntry.CpInfo.RasCpMakeConfigRequest == NULL ) ||
  667. ( CpEntry.CpInfo.RasCpMakeConfigResult == NULL ) ||
  668. ( CpEntry.CpInfo.RasCpConfigAckReceived == NULL ) ||
  669. ( CpEntry.CpInfo.RasCpConfigNakReceived == NULL ) ||
  670. ( CpEntry.CpInfo.RasCpConfigRejReceived == NULL ) ||
  671. ( CpEntry.CpInfo.Recognize > ( DISCARD_REQ + 1) ) )
  672. {
  673. dwRetCode = ERROR_INVALID_PARAMETER;
  674. PppLogErrorString(
  675. ROUTERLOG_PPPCP_DLL_ERROR,
  676. 1,
  677. &(pCpDlls[dwIndex].pszModuleName),
  678. dwRetCode,
  679. 1 );
  680. break;
  681. }
  682. CpTable[PppConfigInfo.NumberOfCPs++] = CpEntry;
  683. }
  684. else
  685. {
  686. CpTable[cTotalNumProtocols-PppConfigInfo.NumberOfAPs]
  687. = CpEntry;
  688. PppConfigInfo.NumberOfAPs++;
  689. }
  690. }
  691. if ( dwRetCode != NO_ERROR )
  692. break;
  693. }
  694. if ( GetCpIndexFromProtocol( PPP_BACP_PROTOCOL ) == (DWORD)-1 )
  695. {
  696. PppConfigInfo.ServerConfigInfo.dwConfigMask &= ~PPPCFG_NegotiateBacp;
  697. }
  698. else if ( PppConfigInfo.ServerConfigInfo.dwConfigMask &
  699. PPPCFG_NegotiateBacp )
  700. {
  701. cbValueBuf = sizeof(DWORD);
  702. if (RegQueryValueEx(
  703. *phKeyPpp, RAS_VALUENAME_DOBAPONVPN,
  704. NULL, &dwType,
  705. (LPBYTE )&dwValue, &cbValueBuf ) == 0
  706. && dwType == REG_DWORD
  707. && cbValueBuf == sizeof(DWORD)
  708. && dwValue)
  709. {
  710. FDoBapOnVpn = TRUE;
  711. BapTrace( "Allowing BAP over VPN's" );
  712. }
  713. }
  714. } while( FALSE );
  715. if ( dwRetCode != NO_ERROR )
  716. {
  717. if ( CpTable != (PPPCP_ENTRY *)NULL )
  718. {
  719. LOCAL_FREE( CpTable );
  720. }
  721. }
  722. if ( hKeyProtocols != (HKEY)NULL )
  723. {
  724. RegCloseKey( hKeyProtocols );
  725. }
  726. if ( pCpDlls != (DLL_ENTRY_POINTS*)NULL )
  727. {
  728. for ( dwIndex = 0; dwIndex < dwNumSubKeys; dwIndex++ )
  729. {
  730. if ( pCpDlls[dwIndex].pszModuleName != (LPSTR)NULL )
  731. {
  732. LOCAL_FREE( pCpDlls[dwIndex].pszModuleName );
  733. }
  734. }
  735. if ( dwRetCode != NO_ERROR )
  736. {
  737. LOCAL_FREE( pCpDlls );
  738. pCpDlls = NULL;
  739. }
  740. }
  741. return( dwRetCode );
  742. }
  743. //**
  744. //
  745. // Call: InitializePPP
  746. //
  747. // Returns: NO_ERROR - Success
  748. // non-zero code - Failure
  749. //
  750. // Description: Will initialize all global data and load and initialize the
  751. // Control and Authentication protocol dll.s
  752. //
  753. DWORD
  754. InitializePPP(
  755. VOID
  756. )
  757. {
  758. DWORD dwIndex;
  759. DWORD dwTId;
  760. DWORD dwRetCode;
  761. HANDLE hThread;
  762. NT_PRODUCT_TYPE NtProductType;
  763. srand ( (unsigned int)time ( NULL ) );
  764. PppConfigInfo.dwTraceId = TraceRegisterA( "PPP" );
  765. DwBapTraceId = TraceRegisterA( "BAP" );
  766. // PrivateTraceId = TraceRegisterA( "Private" );
  767. PppConfigInfo.hLogEvents = RouterLogRegister( TEXT("RemoteAccess") );
  768. PppConfigInfo.RasIpcpDhcpInform = NULL;
  769. PppConfigInfo.RasIphlpDhcpCallback = NULL;
  770. PppConfigInfo.dwLoggingLevel = 3;
  771. //
  772. // Create DDM private heap
  773. //
  774. PppConfigInfo.hHeap = HeapCreate( 0, PPP_HEAP_INITIAL_SIZE,
  775. PPP_HEAP_MAX_SIZE );
  776. if ( PppConfigInfo.hHeap == NULL )
  777. {
  778. return( GetLastError() );
  779. }
  780. if ( (dwRetCode = ReadRegistryInfo(&(PppConfigInfo.hKeyPpp))) != NO_ERROR )
  781. {
  782. return( dwRetCode );
  783. }
  784. dwRetCode = InitEndpointDiscriminator(PppConfigInfo.EndPointDiscriminator);
  785. if ( dwRetCode != NO_ERROR )
  786. {
  787. return( dwRetCode );
  788. }
  789. PppConfigInfo.PortUIDGenerator = 0;
  790. //
  791. // Initialize global data-structures
  792. //
  793. //
  794. // Allocate hash table for PCBs
  795. //
  796. PcbTable.PcbBuckets = LOCAL_ALLOC( LPTR,
  797. sizeof( PCB_BUCKET ) *
  798. PcbTable.NumPcbBuckets );
  799. if ( PcbTable.PcbBuckets == NULL )
  800. {
  801. return( GetLastError() );
  802. }
  803. //
  804. // Allocate hash table for BCBs
  805. //
  806. PcbTable.BcbBuckets = LOCAL_ALLOC( LPTR,
  807. sizeof( BCB_BUCKET ) *
  808. PcbTable.NumPcbBuckets );
  809. if ( PcbTable.BcbBuckets == NULL )
  810. {
  811. LOCAL_FREE( PcbTable.PcbBuckets );
  812. return( GetLastError() );
  813. }
  814. for( dwIndex = 0; dwIndex < PcbTable.NumPcbBuckets; dwIndex++ )
  815. {
  816. PcbTable.PcbBuckets[dwIndex].pPorts = (PCB *)NULL;
  817. PcbTable.BcbBuckets[dwIndex].pBundles = (BCB *)NULL;
  818. }
  819. WorkItemQ.pQHead = (PCB_WORK_ITEM*)NULL;
  820. WorkItemQ.pQTail = (PCB_WORK_ITEM*)NULL;
  821. InitializeCriticalSection( &(WorkItemQ.CriticalSection) );
  822. WorkItemQ.hEventNonEmpty = CreateEvent( NULL, TRUE, FALSE, NULL );
  823. if ( WorkItemQ.hEventNonEmpty == (HANDLE)NULL )
  824. {
  825. return( GetLastError() );
  826. }
  827. TimerQ.hEventNonEmpty = CreateEvent( NULL, FALSE, FALSE, NULL );
  828. if ( TimerQ.hEventNonEmpty == (HANDLE)NULL )
  829. {
  830. return( GetLastError() );
  831. }
  832. PppConfigInfo.hEventChangeNotification = CreateEvent(NULL,FALSE,FALSE,NULL);
  833. if ( PppConfigInfo.hEventChangeNotification == (HANDLE)NULL )
  834. {
  835. return( GetLastError() );
  836. }
  837. RtlGetNtProductType( &NtProductType );
  838. if ( NtProductWinNt == NtProductType )
  839. {
  840. PppConfigInfo.fFlags |= PPPCONFIG_FLAG_WKSTA;
  841. }
  842. //
  843. // Create worker thread.
  844. //
  845. hThread = CreateThread( NULL, 0, WorkerThread, NULL, 0, &dwTId );
  846. if ( hThread == (HANDLE)NULL )
  847. {
  848. return( GetLastError() );
  849. }
  850. CloseHandle(hThread);
  851. return( NO_ERROR );
  852. }
  853. //**
  854. //
  855. // Call: PPPCleanUp
  856. //
  857. // Returns: NO_ERROR - Success
  858. // Non-zero returns - Failure
  859. //
  860. // Description: Will de-allocate all allocated memory, close all handles and
  861. // reset all the global structures to 0.
  862. //
  863. VOID
  864. PPPCleanUp(
  865. VOID
  866. )
  867. {
  868. DWORD dwIndex;
  869. DWORD dwError;
  870. DWORD cTotalNumProtocols;
  871. //
  872. // Unload DLLs.
  873. //
  874. cTotalNumProtocols = PppConfigInfo.NumberOfCPs + PppConfigInfo.NumberOfAPs;
  875. if ( pCpDlls != NULL )
  876. {
  877. for ( dwIndex = 0; dwIndex < cTotalNumProtocols; dwIndex++ )
  878. {
  879. if ( CpTable[dwIndex].fFlags & PPPCP_FLAG_INIT_CALLED )
  880. {
  881. PppLog( 1, "RasCpInit(%x, FALSE)",
  882. CpTable[dwIndex].CpInfo.Protocol );
  883. dwError = CpTable[dwIndex].CpInfo.RasCpInit(
  884. FALSE /* fInitialize */ );
  885. if ( NO_ERROR != dwError )
  886. {
  887. PppLog(
  888. 1,
  889. "RasCpInit(FALSE) for protocol 0x%x returned error %d",
  890. CpTable[dwIndex].CpInfo.Protocol,
  891. dwError );
  892. }
  893. CpTable[dwIndex].fFlags &= ~PPPCP_FLAG_INIT_CALLED;
  894. CpTable[dwIndex].fFlags &= ~PPPCP_FLAG_AVAILABLE;
  895. }
  896. }
  897. for ( dwIndex = 0;
  898. pCpDlls[dwIndex].hInstance != INVALID_HANDLE_VALUE;
  899. dwIndex++ )
  900. {
  901. if ( pCpDlls[dwIndex].hInstance != NULL )
  902. {
  903. FreeLibrary( pCpDlls[dwIndex].hInstance );
  904. }
  905. }
  906. if ( pCpDlls )
  907. LOCAL_FREE( pCpDlls );
  908. pCpDlls = NULL;
  909. }
  910. RouterLogDeregister( PppConfigInfo.hLogEvents );
  911. DeleteCriticalSection( &(WorkItemQ.CriticalSection) );
  912. if ( TimerQ.hEventNonEmpty != NULL )
  913. {
  914. CloseHandle( TimerQ.hEventNonEmpty );
  915. }
  916. if ( WorkItemQ.hEventNonEmpty != NULL )
  917. {
  918. CloseHandle( WorkItemQ.hEventNonEmpty );
  919. }
  920. if ( PppConfigInfo.hEventChangeNotification != NULL )
  921. {
  922. CloseHandle( PppConfigInfo.hEventChangeNotification );
  923. }
  924. //
  925. // Destroy private heap
  926. //
  927. if ( PppConfigInfo.hHeap != NULL )
  928. {
  929. HeapDestroy( PppConfigInfo.hHeap );
  930. }
  931. if ( PppConfigInfo.dwTraceId != INVALID_TRACEID )
  932. {
  933. TraceDeregisterA( PppConfigInfo.dwTraceId );
  934. }
  935. if ( PppConfigInfo.hKeyPpp != (HKEY)NULL )
  936. {
  937. RegCloseKey( PppConfigInfo.hKeyPpp );
  938. }
  939. if ( NULL != PppConfigInfo.hInstanceParserDll )
  940. {
  941. FreeLibrary( PppConfigInfo.hInstanceParserDll );
  942. }
  943. if (NULL != PppConfigInfo.pszParserDllPath)
  944. {
  945. LOCAL_FREE(PppConfigInfo.pszParserDllPath);
  946. }
  947. PppConfigInfo.pszParserDllPath = NULL;
  948. PppConfigInfo.PacketFromPeer = NULL;
  949. PppConfigInfo.PacketToPeer = NULL;
  950. PppConfigInfo.PacketFree = NULL;
  951. //
  952. // TraceDeregisterA can handle INVALID_TRACEID gracefully
  953. //
  954. TraceDeregisterA( DwBapTraceId );
  955. ZeroMemory( &PcbTable, sizeof( PcbTable ) );
  956. ZeroMemory( &WorkItemQ, sizeof( WorkItemQ ) );
  957. ZeroMemory( &PppConfigInfo, sizeof( PppConfigInfo ) );
  958. ZeroMemory( &TimerQ, sizeof( TimerQ ) );
  959. CpTable = NULL;
  960. }
  961. //**
  962. //
  963. // Call: DllEntryPoint
  964. //
  965. // Returns: TRUE - Success
  966. // FALSE - Failure
  967. //
  968. // Description:
  969. //
  970. BOOL
  971. DllEntryPoint(
  972. IN HANDLE hInstDLL,
  973. IN DWORD fdwReason,
  974. IN LPVOID lpvReserved
  975. )
  976. {
  977. switch (fdwReason)
  978. {
  979. case DLL_PROCESS_ATTACH:
  980. HInstDLL = hInstDLL;
  981. DisableThreadLibraryCalls(hInstDLL);
  982. break;
  983. }
  984. return(TRUE);
  985. }
  986. //**
  987. //
  988. // Call: RasCpEnumProtocolIds
  989. //
  990. // Returns: NO_ERROR - Success
  991. //
  992. // Description: This entry point is called to enumerate the number and the
  993. // control protocol Ids for the protocols contained in the module.
  994. //
  995. DWORD
  996. RasCpEnumProtocolIds(
  997. OUT DWORD * pdwProtocolIds,
  998. IN OUT DWORD * pcProtocolIds
  999. )
  1000. {
  1001. DWORD dwIndex;
  1002. HKEY hKey;
  1003. DWORD dwErr;
  1004. DWORD dwType;
  1005. DWORD dwValue;
  1006. DWORD dwSize;
  1007. PppLog(1, "RasCpEnumProtocolIds");
  1008. RTASSERT(NUM_BUILT_IN_CPS <= PPPCP_MAXCPSPERDLL);
  1009. *pcProtocolIds = 0;
  1010. dwErr = RegOpenKeyEx(HKEY_LOCAL_MACHINE, RAS_KEYPATH_BUILTIN, 0, KEY_READ,
  1011. &hKey);
  1012. if (ERROR_SUCCESS == dwErr)
  1013. {
  1014. for (dwIndex = 0; dwIndex < NUM_BUILT_IN_CPS; dwIndex++)
  1015. {
  1016. dwSize = sizeof(dwValue);
  1017. dwErr = RegQueryValueEx(hKey, BuiltInCps[dwIndex].szNegotiateCp,
  1018. NULL, &dwType, (BYTE*)&dwValue, &dwSize);
  1019. if ( ERROR_SUCCESS == dwErr
  1020. && REG_DWORD == dwType
  1021. && sizeof(DWORD) == dwSize
  1022. && !dwValue)
  1023. {
  1024. BuiltInCps[dwIndex].fLoad = FALSE;
  1025. PppLog(1, "%s is FALSE", BuiltInCps[dwIndex].szNegotiateCp);
  1026. }
  1027. }
  1028. RegCloseKey(hKey);
  1029. }
  1030. for (dwIndex = 0; dwIndex < NUM_BUILT_IN_CPS; dwIndex++)
  1031. {
  1032. if (BuiltInCps[dwIndex].fLoad)
  1033. {
  1034. pdwProtocolIds[*pcProtocolIds] = BuiltInCps[dwIndex].dwProtocolId;
  1035. PppLog(1, "Protocol %x", BuiltInCps[dwIndex].dwProtocolId);
  1036. *pcProtocolIds += 1;
  1037. }
  1038. }
  1039. return(NO_ERROR);
  1040. }
  1041. //**
  1042. //
  1043. // Call: RasCpGetInfo
  1044. //
  1045. // Returns: NO_ERROR - Success
  1046. // ERROR_INVALID_PARAMETER - Protocol id is unrecogized
  1047. //
  1048. // Description: This entry point is called for get all information for the
  1049. // control protocol in this module.
  1050. //
  1051. DWORD
  1052. RasCpGetInfo(
  1053. IN DWORD dwProtocolId,
  1054. OUT PPPCP_INFO* pCpInfo
  1055. )
  1056. {
  1057. DWORD dwIndex;
  1058. PppLog(1, "RasCpGetInfo %x", dwProtocolId);
  1059. for (dwIndex = 0; dwIndex < NUM_BUILT_IN_CPS; dwIndex++)
  1060. {
  1061. if ( (BuiltInCps[dwIndex].dwProtocolId == dwProtocolId)
  1062. && BuiltInCps[dwIndex].fLoad)
  1063. {
  1064. return((DWORD)BuiltInCps[dwIndex].pRasCpGetInfo(
  1065. dwProtocolId, pCpInfo));
  1066. }
  1067. }
  1068. return(ERROR_INVALID_PARAMETER);
  1069. }
  1070. VOID
  1071. PrivateTrace(
  1072. IN CHAR* Format,
  1073. ...
  1074. )
  1075. {
  1076. va_list arglist;
  1077. va_start(arglist, Format);
  1078. TraceVprintfEx(PrivateTraceId,
  1079. 0x00010000 | TRACE_USE_MASK | TRACE_USE_MSEC,
  1080. Format,
  1081. arglist);
  1082. va_end(arglist);
  1083. }