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.

1101 lines
40 KiB

  1. /////////////////////////////////////////////////////////////
  2. // Copyright(c) 1998, Microsoft Corporation
  3. //
  4. // ipseccmd.cpp
  5. //
  6. // Created on 4/5/98 by Randyram
  7. // Revisions:
  8. // 3/2/00 DKalin
  9. // Update for whistler (dynamic mode only)
  10. // 5/24/00 DKalin
  11. // Static store support (v1.35)
  12. // 7/30/00 DKalin
  13. // Interface-based filtering support (v1.36)
  14. // 8/23/00 DKalin
  15. // Got rid of ipsecposvc service, moved to the usage of persistent APIs
  16. // Default response rule support added
  17. //
  18. // This is the main file for the policy "on the fly" tool
  19. //
  20. /////////////////////////////////////////////////////////////
  21. #include "ipseccmd.h"
  22. int _cdecl ipseccmd_main(int argc, char *argv[])
  23. {
  24. // status holders
  25. ULONG ulRpcErr = 0;
  26. DWORD dwStatus = ERROR_SUCCESS;
  27. BOOL bMMFiltersSpecified = FALSE;
  28. int i;
  29. PHANDLE hMMFilters = NULL, hFilters = NULL;
  30. LPVOID myErrString = NULL;
  31. BOOL bShowSpecified = FALSE;
  32. int show_start_index = 0;
  33. // check to see if they're asking for help
  34. if ( (argc == 1) )
  35. {
  36. usage(stdout);
  37. printf("\nThe command completed successfully.\n");
  38. return 0;
  39. }
  40. else if ( (strchr(POTF_FLAG_TOKENS, argv[1][0]) != NULL) &&
  41. (strchr(POTF_HELP_FLAGS, argv[1][1]) != NULL) )
  42. {
  43. usage(stdout, true);
  44. printf("\nThe command completed successfully.\n");
  45. return 0;
  46. }
  47. // check to see if there are arguments longer than POTF_MAX_STRLEN - we won't accept these
  48. for (i = 1; i < argc; i++)
  49. {
  50. if (strlen(argv[i]) >= POTF_MAX_STRLEN)
  51. {
  52. printf("Error: the argument is too long (>%d symbols)\n", POTF_MAX_STRLEN-1);
  53. return 1;
  54. }
  55. }
  56. // check for MM filters and show option
  57. for (i = 1; i < argc; i++)
  58. {
  59. if (strchr(POTF_FLAG_TOKENS, argv[i][0]) != NULL &&
  60. strncmp(&argv[i][1], POTF_MMFILTER_FLAG,
  61. strlen(POTF_MMFILTER_FLAG)) == 0)
  62. {
  63. bMMFiltersSpecified = TRUE;
  64. break;
  65. }
  66. if (!_stricmp(argv[i], KEYWORD_SHOW))
  67. {
  68. bShowSpecified = TRUE;
  69. show_start_index = i;
  70. break;
  71. }
  72. }
  73. // our RPC info-- process machine name here
  74. RPC_STATUS RpcStat = RPC_S_OK;
  75. handle_t hIpsecpolsvc;
  76. // items for storage and conversion to storage
  77. STORAGE_INFO StoreInfo;
  78. memset(&StoreInfo, 0, sizeof(StoreInfo));
  79. // used to shift machinename out after processing
  80. char **myArgv = argv;
  81. int myArgc = argc;
  82. if ((argc > 1) && (strncmp(argv[1], "\\\\", 2) == 0))
  83. {
  84. TCHAR buf[POTF_MAX_STRLEN];
  85. _stprintf(buf, TEXT("%S"), argv[1]);
  86. _tcscpy((TCHAR *)szServ, buf+2);
  87. myArgv = argv + 1;
  88. myArgc = myArgc - 1;
  89. bLocalMachine = false;
  90. }
  91. // OK, now call show code if they asked us to
  92. if (bShowSpecified)
  93. {
  94. i = ipseccmd_show_main(argc, argv, show_start_index);
  95. if (i == IPSECCMD_USAGE)
  96. {
  97. usage(stdout, true);
  98. printf("\nThe command completed successfully.\n");
  99. return 0;
  100. }
  101. else if (i != 0)
  102. {
  103. return i;
  104. }
  105. else
  106. {
  107. printf("\nThe command completed successfully.\n");
  108. return 0;
  109. }
  110. }
  111. // check if they requested us to remove the persisted policy
  112. if ( myArgc == 2 && (strchr(POTF_FLAG_TOKENS, myArgv[1][0]) != NULL) &&
  113. (POTF_DELETERULE_FLAG == tolower(myArgv[1][1])) )
  114. {
  115. DeletePersistedIPSecPolicy(bLocalMachine ? L"" : szServ, pszIpsecpolPrefix);
  116. printf("\nThe command completed successfully.\n");
  117. return 0;
  118. }
  119. // init the policies
  120. DWORD dwConversionStatus = ERROR_SUCCESS;
  121. bool bConfirm = false; // if true, print policy before plumbing
  122. char cConfirm = 'n';
  123. ///////////////////////////////////////////////////////////////////////
  124. //
  125. // CmdLineToPolicy changed to reduce the number of params.
  126. // To reduce the amount of code changed, I simply copy the
  127. // stuff I used to pass into the main policy struct and then
  128. // copy it back out.
  129. //
  130. ///////////////////////////////////////////////////////////////////////
  131. IPSEC_IKE_POLICY IPSecIkePol;
  132. memset(&IPSecIkePol, 0, sizeof(IPSecIkePol));
  133. // Action!
  134. if (InitWinsock(MAKEWORD(2,0)))
  135. {
  136. // print the respective error from Text2Pol
  137. FormatMessage(FORMAT_MESSAGE_FROM_HMODULE | FORMAT_MESSAGE_ALLOCATE_BUFFER,
  138. GetModuleHandle(TEXT("text2pol.dll")),
  139. T2P_WINSOCK_INITFAIL, 0,
  140. (LPTSTR)&myErrString, 0, NULL);
  141. _ftprintf(stderr,
  142. TEXT("Error 0x%x occurred:\n\n%s\n"), T2P_WINSOCK_INITFAIL, myErrString);
  143. LocalFree(myErrString);
  144. return 1;
  145. }
  146. dwStatus =
  147. CmdLineToPolicy(myArgc, myArgv, IPSecIkePol, bConfirm, &StoreInfo);
  148. WSACleanup();
  149. if (!bLocalMachine)
  150. {
  151. //
  152. // copy the machine name into storage info in case it's needed
  153. //
  154. wcscpy(StoreInfo.szLocationName, szServ);
  155. }
  156. if (T2P_SUCCESS(dwStatus))
  157. {
  158. /// POST PROCESS POLICY////////////////////
  159. // Get defaults, do fix ups
  160. ///////////////////////////////////////////
  161. // use default p1 policy if there isn't one specified
  162. if (IPSecIkePol.IkePol.dwOfferCount == 0)
  163. {
  164. LoadIkeDefaults(IPSecIkePol.IkePol);
  165. }
  166. // new with version 1.21: -f is required
  167. // new with version 1.35: -f is not required if -y or -o specified
  168. // FIX: allow zero filters if -x is specified
  169. if (IPSecIkePol.dwNumFilters == 0 && !StoreInfo.bSetInActive && !StoreInfo.bDeletePolicy && !StoreInfo.bSetActive)
  170. {
  171. //LoadFilterDefaults(myFilters, myNumFilters);
  172. fprintf(stderr, "You must supply filters with -f\n\n");
  173. usage(stderr);
  174. return 1;
  175. }
  176. if (IPSecIkePol.IpsPol.dwOfferCount == 0)
  177. LoadOfferDefaults(IPSecIkePol.IpsPol.pOffers, IPSecIkePol.IpsPol.dwOfferCount);
  178. if (IPSecIkePol.AuthInfos.dwNumAuthInfos == 0)
  179. {
  180. IPSecIkePol.AuthInfos.pAuthenticationInfo = new IPSEC_MM_AUTH_INFO[1];
  181. assert(IPSecIkePol.AuthInfos.pAuthenticationInfo != NULL);
  182. IPSecIkePol.AuthInfos.pAuthenticationInfo[0].AuthMethod = IKE_SSPI;
  183. IPSecIkePol.AuthInfos.pAuthenticationInfo[0].dwAuthInfoSize = 0;
  184. IPSecIkePol.AuthInfos.pAuthenticationInfo[0].pAuthInfo = (LPBYTE) new wchar_t[1];
  185. IPSecIkePol.AuthInfos.pAuthenticationInfo[0].pAuthInfo[0] = UNICODE_NULL;
  186. IPSecIkePol.AuthInfos.dwNumAuthInfos = 1;
  187. }
  188. if (StoreInfo.Type != STORAGE_TYPE_NONE)
  189. {
  190. //
  191. // store the policy
  192. //
  193. // XX put machine name into szLocationName
  194. dwConversionStatus =
  195. StorePolicy(StoreInfo, IPSecIkePol);
  196. if (dwConversionStatus != ERROR_SUCCESS)
  197. {
  198. if ( dwConversionStatus == P2STORE_MISSING_NAME )
  199. {
  200. fprintf(stderr, "You must supply a policy name\n");
  201. }
  202. else
  203. {
  204. fprintf(stderr, "Error converting policy: 0x%x\n",
  205. dwConversionStatus);
  206. }
  207. }
  208. }
  209. else
  210. {
  211. /*
  212. // Check to make sure PA is up //
  213. We do our best to get the PA up and running. If, for some reason,
  214. we can't open SCM or get the service status, we don't abort, but
  215. let the exception handling around the actual RPC calls handle errors
  216. */
  217. if (!(PAIsRunning(dwStatus, (bLocalMachine) ? NULL : (TCHAR*)szServ))
  218. && dwStatus == ERROR_SUCCESS)
  219. {
  220. fprintf(stderr, "Policy Agent Service not running, starting it now...\n");
  221. if (!StartPA(dwStatus, (bLocalMachine) ? NULL : (TCHAR*)szServ))
  222. {
  223. fprintf(stderr,"Couldn't start Policy Agent service, error 0x%x, Exiting.\n",
  224. dwStatus);
  225. return 1;
  226. }
  227. }
  228. else if (dwStatus != ERROR_SUCCESS)
  229. {
  230. fprintf(stderr,"Couldn't check status of Policy Agent service, error 0x%x, Exiting.\n",
  231. dwStatus);
  232. return 1;
  233. }
  234. // get GUIDs
  235. // first delete
  236. DeleteGuidsNames(IPSecIkePol);
  237. GenerateGuidsNames(pszIpsecpolPrefix, IPSecIkePol);
  238. if (bConfirm)
  239. {
  240. PrintPolicies(IPSecIkePol);
  241. _flushall();
  242. cout << "Continue? [y/n]: ";
  243. cin >> cConfirm;
  244. if (cConfirm == 'n' || cConfirm == 'N')
  245. return 1;
  246. }
  247. RpcStat = PlumbIPSecPolicy((bLocalMachine) ? L"" : szServ, &IPSecIkePol, bMMFiltersSpecified, &hMMFilters, &hFilters, TRUE);
  248. if (RpcStat == RPC_S_UNKNOWN_IF)
  249. {
  250. fprintf(stderr, "PA RPC not ready. Sleeping for %d seconds...\n", POTF_PARPC_SLEEPTIME/1000);
  251. Sleep(POTF_PARPC_SLEEPTIME);
  252. RpcStat = PlumbIPSecPolicy((bLocalMachine) ? L"" : szServ, &IPSecIkePol, bMMFiltersSpecified, &hMMFilters, &hFilters, TRUE);
  253. }
  254. if (RpcStat != RPC_S_OK)
  255. {
  256. // PAAddPolicy can return alot of different codes, not well defined.
  257. // so, process the ones we care about or can diagnose
  258. if (RpcStat == RPC_E_ACCESS_DENIED)
  259. {
  260. fprintf(stderr,"Access denied. You do not have admin privileges on the target machine\n");
  261. }
  262. else
  263. {
  264. fprintf(stderr, "Failed to add policy, error 0x%x\n", RpcStat);
  265. }
  266. }
  267. else
  268. {
  269. // everything is fine, must close handles
  270. if (hMMFilters)
  271. {
  272. for (i = 0; i < (int) IPSecIkePol.dwNumMMFilters; i++)
  273. {
  274. CloseMMFilterHandle(hMMFilters[i]);
  275. }
  276. }
  277. if (hFilters)
  278. {
  279. for (i = 0; i < (int) IPSecIkePol.dwNumFilters; i++)
  280. {
  281. if (IPSecIkePol.QMFilterType == QM_TRANSPORT_FILTER)
  282. {
  283. CloseTransportFilterHandle(hFilters[i]);
  284. }
  285. else
  286. {
  287. CloseTunnelFilterHandle(hFilters[i]);
  288. }
  289. }
  290. }
  291. }
  292. }
  293. }
  294. else
  295. {
  296. if (dwStatus != POTF_FAILED)
  297. {
  298. // print the respective error from Text2Pol
  299. FormatMessage(FORMAT_MESSAGE_FROM_HMODULE | FORMAT_MESSAGE_ALLOCATE_BUFFER,
  300. GetModuleHandle(TEXT("text2pol.dll")),
  301. dwStatus, 0,
  302. (LPTSTR)&myErrString, 0, NULL);
  303. _ftprintf(stderr,
  304. TEXT("Error 0x%x occurred:\n\n%s\n"), dwStatus, myErrString);
  305. LocalFree(myErrString);
  306. }
  307. usage(stderr);
  308. return 1;
  309. }
  310. if (StoreInfo.Type != STORAGE_TYPE_NONE)
  311. {
  312. if ( StoreInfo.FilterList )
  313. delete [] StoreInfo.FilterList;
  314. }
  315. DeleteGuidsNames(IPSecIkePol);
  316. if (IPSecIkePol.IkePol.pOffers != NULL)
  317. {
  318. delete [] IPSecIkePol.IkePol.pOffers;
  319. IPSecIkePol.IkePol.pOffers = NULL;
  320. }
  321. if (IPSecIkePol.IpsPol.pOffers != NULL)
  322. {
  323. delete [] IPSecIkePol.IpsPol.pOffers;
  324. IPSecIkePol.IpsPol.pOffers = NULL;
  325. }
  326. if (IPSecIkePol.pMMFilters != NULL)
  327. {
  328. delete [] IPSecIkePol.pMMFilters;
  329. IPSecIkePol.pMMFilters = NULL;
  330. }
  331. if (IPSecIkePol.pTransportFilters != NULL)
  332. {
  333. delete [] IPSecIkePol.pTransportFilters;
  334. IPSecIkePol.pTransportFilters = NULL;
  335. }
  336. if (IPSecIkePol.pTunnelFilters != NULL)
  337. {
  338. delete [] IPSecIkePol.pTunnelFilters;
  339. IPSecIkePol.pTunnelFilters = NULL;
  340. }
  341. if (IPSecIkePol.AuthInfos.pAuthenticationInfo != NULL)
  342. {
  343. delete [] IPSecIkePol.AuthInfos.pAuthenticationInfo;
  344. IPSecIkePol.AuthInfos.pAuthenticationInfo = NULL;
  345. }
  346. memset(&IPSecIkePol, 0, sizeof(IPSecIkePol));
  347. printf("\nThe command completed successfully.\n");
  348. return 0;
  349. }
  350. /////////////////////// UTILITY FUNCTIONS //////////////////////////
  351. /*
  352. // Loads Me to Any
  353. void LoadFilterDefaults(OUT FILTER * & Filters, OUT UINT & NumFilters)
  354. {
  355. NumFilters = 2;
  356. Filters = new FILTER[NumFilters];
  357. assert(Filters != NULL);
  358. memset(Filters, 0, NumFilters * sizeof(FILTER));
  359. Filters[0].FilterSpec.DestAddr = 0;
  360. Filters[0].FilterSpec.DestMask = 0;
  361. Filters[0].FilterSpec.SrcAddr = 0;
  362. Filters[0].FilterSpec.SrcMask = -1;
  363. Filters[1].FilterSpec.DestAddr = 0;
  364. Filters[1].FilterSpec.DestMask = -1;
  365. Filters[1].FilterSpec.SrcAddr = 0;
  366. Filters[1].FilterSpec.SrcMask = 0;
  367. //Filters[0].InterfaceType = Filters[1].InterfaceType = PA_INTERFACE_TYPE_ALL;
  368. // set the GUID
  369. RPC_STATUS RpcStat;
  370. for (UINT i = 0; i < NumFilters; ++i)
  371. {
  372. RpcStat = UuidCreate(&Filters[i].FilterID);
  373. if (RpcStat != RPC_S_OK && RpcStat != RPC_S_UUID_LOCAL_ONLY)
  374. {
  375. sprintf(STRLASTERR, "Couldn't get GUID failed with status: %ul\n",
  376. RpcStat);
  377. WARN;
  378. }
  379. }
  380. }
  381. */
  382. HRESULT
  383. StorePolicy(
  384. IN STORAGE_INFO StoreInfo,
  385. IN IPSEC_IKE_POLICY PolicyToStore
  386. )
  387. {
  388. HRESULT hr = S_OK; // holds polstore info
  389. IPSECPolicyToStorage Polstore;
  390. WCHAR wszPolicyName[POTF_MAX_STRLEN],
  391. wszRuleName[POTF_MAX_STRLEN],
  392. wszLocationName[POTF_MAX_STRLEN];
  393. bool bLocationUsed = false;
  394. DWORD dwLocation;
  395. HANDLE hStore = NULL;
  396. int i;
  397. // do some initialization and checks before we write it
  398. if (StoreInfo.szLocationName[0] != '\0')
  399. {
  400. wcscpy(wszLocationName, StoreInfo.szLocationName);
  401. // MultiByteToWideChar(CP_THREAD_ACP, 0, StoreInfo.szLocationName, -1,
  402. // wszLocationName, POTF_MAX_STRLEN);
  403. bLocationUsed = true;
  404. }
  405. wcscpy(wszPolicyName, StoreInfo.szPolicyName);
  406. wcscpy(wszRuleName, StoreInfo.szRuleName);
  407. // MultiByteToWideChar(CP_THREAD_ACP, 0, StoreInfo.szRuleName, -1,
  408. // wszRuleName, POTF_MAX_STRLEN);
  409. // MultiByteToWideChar(CP_THREAD_ACP, 0, StoreInfo.szPolicyName, -1,
  410. // wszPolicyName, POTF_MAX_STRLEN);
  411. dwLocation = IPSEC_REGISTRY_PROVIDER;
  412. if (StoreInfo.Type == STORAGE_TYPE_DS)
  413. dwLocation = IPSEC_DIRECTORY_PROVIDER;
  414. //
  415. // Special handling for setting active policy to NULL
  416. //
  417. if ( StoreInfo.bSetInActive )
  418. {
  419. // deactivate policy in specified storage
  420. PIPSEC_POLICY_DATA pipspd = NULL;
  421. hr = IPSecOpenPolicyStore((bLocationUsed) ? wszLocationName : NULL, dwLocation, NULL, &hStore);
  422. if (hr == ERROR_SUCCESS && hStore != NULL)
  423. {
  424. hr = IPSecGetAssignedPolicyData(hStore, &pipspd);
  425. if (hr == ERROR_SUCCESS && pipspd != NULL)
  426. {
  427. hr = IPSecUnassignPolicy(hStore, pipspd[0].PolicyIdentifier);
  428. IPSecFreePolicyData(pipspd);
  429. }
  430. IPSecClosePolicyStore(hStore);
  431. }
  432. // now if there are no filters specified, bail out
  433. if (PolicyToStore.dwNumFilters == 0)
  434. {
  435. return hr;
  436. }
  437. }
  438. hr = Polstore.Open(dwLocation,
  439. (bLocationUsed) ? wszLocationName : NULL,
  440. wszPolicyName,
  441. NULL,
  442. (StoreInfo.tPollingInterval)
  443. ? StoreInfo.tPollingInterval : P2STORE_DEFAULT_POLLINT,
  444. true);
  445. if ( hr != ERROR_SUCCESS || hr == P2STORE_MISSING_NAME )
  446. return hr;
  447. //
  448. // Look through the rules to see if we should add or update
  449. //
  450. PIPSEC_POLICY_DATA pPolicy = Polstore.GetPolicy();
  451. assert(pPolicy);
  452. hStore = Polstore.GetStorageHandle();
  453. if (PolicyToStore.dwNumFilters == 0 && StoreInfo.bSetActive)
  454. {
  455. // just set active
  456. hr = Polstore.SetAssignedPolicy(pPolicy);
  457. return hr;
  458. }
  459. if ( StoreInfo.bDeletePolicy )
  460. {
  461. // remove policy from storage
  462. if (!Polstore.IsPolicyInStorage())
  463. {
  464. return ERROR_FILE_NOT_FOUND;
  465. }
  466. // proceed and remove rule internals first (incl. filter and negpol)
  467. for (i = 0; i < (int) pPolicy->dwNumNFACount; i++)
  468. {
  469. if (hr == ERROR_SUCCESS)
  470. {
  471. hr = IPSecDeleteNFAData(hStore, pPolicy->PolicyIdentifier, pPolicy->ppIpsecNFAData[i]);
  472. }
  473. }
  474. for (i = 0; i < (int) pPolicy->dwNumNFACount; i++)
  475. {
  476. if (hr == ERROR_SUCCESS)
  477. {
  478. RPC_STATUS RpcStat;
  479. if (!UuidIsNil(&(pPolicy->ppIpsecNFAData[i]->FilterIdentifier), &RpcStat))
  480. {
  481. hr = IPSecDeleteFilterData(hStore, pPolicy->ppIpsecNFAData[i]->FilterIdentifier);
  482. }
  483. }
  484. if (hr == ERROR_SUCCESS)
  485. {
  486. hr = IPSecDeleteNegPolData(hStore, pPolicy->ppIpsecNFAData[i]->NegPolIdentifier);
  487. }
  488. }
  489. // now proceed and remove ISAKMP and policy
  490. if (hr == ERROR_SUCCESS)
  491. {
  492. hr = IPSecDeletePolicyData(hStore, pPolicy);
  493. }
  494. if (hr == ERROR_SUCCESS)
  495. {
  496. hr = IPSecDeleteISAKMPData(hStore, pPolicy->ISAKMPIdentifier);
  497. }
  498. }
  499. else
  500. {
  501. PIPSEC_NFA_DATA pNFA = NULL;
  502. hr = Polstore.SetISAKMPPolicy(PolicyToStore.IkePol);
  503. if (!pPolicy->dwNumNFACount)
  504. {
  505. // add default response
  506. hr = Polstore.AddDefaultResponseRule();
  507. pPolicy = Polstore.GetPolicy();
  508. }
  509. // in the case of default response, always update the first rule (if it exists)
  510. if (PolicyToStore.QMFilterType == QM_TRANSPORT_FILTER
  511. && PolicyToStore.pTransportFilters[0].SrcAddr.AddrType == IP_ADDR_UNIQUE
  512. && PolicyToStore.pTransportFilters[0].SrcAddr.uIpAddr == IP_ADDRESS_ME
  513. && PolicyToStore.pTransportFilters[0].DesAddr.AddrType == IP_ADDR_UNIQUE
  514. && PolicyToStore.pTransportFilters[0].DesAddr.uIpAddr == IP_ADDRESS_ME
  515. && PolicyToStore.pTransportFilters[0].InboundFilterFlag == (FILTER_FLAG) POTF_DEFAULT_RESPONSE_FLAG
  516. && PolicyToStore.pTransportFilters[0].OutboundFilterFlag == (FILTER_FLAG) POTF_DEFAULT_RESPONSE_FLAG)
  517. {
  518. if (pPolicy->dwNumNFACount)
  519. {
  520. pNFA = pPolicy->ppIpsecNFAData[pPolicy->dwNumNFACount-1];
  521. }
  522. }
  523. else
  524. {
  525. for (i = 0; i < (int) pPolicy->dwNumNFACount; i++)
  526. {
  527. if (pPolicy->ppIpsecNFAData[i]->pszIpsecName && wcscmp(pPolicy->ppIpsecNFAData[i]->pszIpsecName, wszRuleName) == 0)
  528. {
  529. // found a rule
  530. pNFA = pPolicy->ppIpsecNFAData[i];
  531. break;
  532. }
  533. }
  534. }
  535. if ( !pNFA )
  536. {
  537. hr = Polstore.AddRule(PolicyToStore, &StoreInfo);
  538. }
  539. else
  540. {
  541. hr = Polstore.UpdateRule(pNFA, PolicyToStore, &StoreInfo);
  542. }
  543. if (hr == ERROR_SUCCESS)
  544. {
  545. if (!Polstore.IsPolicyInStorage())
  546. {
  547. return ERROR_ACCESS_DENIED;
  548. }
  549. if (StoreInfo.bSetActive)
  550. {
  551. hr = Polstore.SetAssignedPolicy(pPolicy);
  552. }
  553. }
  554. }
  555. return hr;
  556. }
  557. void usage(FILE *fh_usage, bool bExtendedUsage)
  558. {
  559. fprintf(fh_usage, "\n");
  560. fprintf(fh_usage, "%s\n", POTF_VERSION);
  561. fprintf(fh_usage, "USAGE: \n");
  562. fprintf(fh_usage, "ipseccmd \\\\machinename -f FilterList -n NegotiationPolicyList -t TunnelAddr\n");
  563. fprintf(fh_usage, " -a AuthMethodList -1s SecurityMethodList -1k Phase1RekeyAfter -1p\n");
  564. fprintf(fh_usage, " -1f MMFilterList -1e SoftSAExpirationTime -soft -confirm\n");
  565. fprintf(fh_usage, " [-dialup OR -lan]\n");
  566. fprintf(fh_usage, " {-w TYPE:DOMAIN -p PolicyName:PollInterval -r RuleName -x -y -o}\n");
  567. fprintf(fh_usage, "ipseccmd \\\\machinename show filters policies auth stats sas all\n");
  568. fprintf(fh_usage, " \n");
  569. fprintf(fh_usage, "\nBATCH MODE:\n");
  570. fprintf(fh_usage, "ipseccmd -file filename\n");
  571. fprintf(fh_usage, " File must contain regular ipseccmd commands,\n");
  572. fprintf(fh_usage, " all these commands will be executed in one shot.\n");
  573. fprintf(fh_usage, " \n");
  574. if ( !bExtendedUsage )
  575. {
  576. fprintf(fh_usage, "For extended usage, run: ipseccmd -?");
  577. fprintf(fh_usage, "\n");
  578. return;
  579. }
  580. //
  581. // new usage w/ storage added
  582. //
  583. fprintf(fh_usage, "\n");
  584. fprintf(fh_usage, "ipseccmd has three mutually exclusive modes: static, dynamic, and query. \n");
  585. fprintf(fh_usage, "\n");
  586. fprintf(fh_usage, "The default mode is dynamic. \n");
  587. fprintf(fh_usage, "\n");
  588. fprintf(fh_usage, "Dynamic mode will plumb policy directly into the IPSec Services\n");
  589. fprintf(fh_usage, "Security Policies Database. The policy will be persisted, i.e. it will stay\n");
  590. fprintf(fh_usage, "after a reboot. The benefit of dynamic policy is that it can co-exist with\n");
  591. fprintf(fh_usage, "DS based policy.\n");
  592. fprintf(fh_usage, "\nTo delete all dynamic policies, execute \"ipseccmd -u\" command\n");
  593. fprintf(fh_usage, "\nWhen the tool is used in static mode,\n");
  594. fprintf(fh_usage, "it creates or modifies stored policy. This policy can be used again and \n");
  595. fprintf(fh_usage, "will last the lifetime of the store. Static mode is indicated by the -w\n");
  596. fprintf(fh_usage, "flag. The flags in the {} braces are only valid for static mode. The usage \n");
  597. fprintf(fh_usage, "for static mode is an extension of dynamic mode, so please read through\n");
  598. fprintf(fh_usage, "the dynamic mode section.\n");
  599. fprintf(fh_usage, "\nIn query mode, the tool queries IPSec Security Policies Database.\n");
  600. fprintf(fh_usage, "\n");
  601. fprintf(fh_usage, "NOTE: references to SHA in ipseccmd are referring to the SHA1 algorithm.\n");
  602. fprintf(fh_usage, "\n");
  603. fprintf(fh_usage, "------------\n");
  604. fprintf(fh_usage, " QUERY MODE \n");
  605. fprintf(fh_usage, "------------\n");
  606. fprintf(fh_usage, "\n");
  607. fprintf(fh_usage, "The tool displays requested type of data from IPSec Security Policies Database\n");
  608. fprintf(fh_usage, "\n");
  609. fprintf(fh_usage, " filters - shows main mode and quick mode filters\n");
  610. fprintf(fh_usage, " policies - shows main mode and quick mode policies\n");
  611. fprintf(fh_usage, " auth - shows main mode authentication methods\n");
  612. fprintf(fh_usage, " stats - shows Internet Key Exchange (IKE) and IPSec statistics\n");
  613. fprintf(fh_usage, " sas - shows main mode and quick mode Security Associations\n");
  614. fprintf(fh_usage, " all - shows all of the above data\n");
  615. fprintf(fh_usage, "It is possible to combine several flags\n");
  616. fprintf(fh_usage, "EXAMPLE: ipseccmd show filters policies\n");
  617. fprintf(fh_usage, "\n");
  618. fprintf(fh_usage, "------------\n");
  619. fprintf(fh_usage, "DYNAMIC MODE\n");
  620. fprintf(fh_usage, "------------\n");
  621. fprintf(fh_usage, "\n");
  622. fprintf(fh_usage, "Each execution of the tool sets an IPSec rule, an IKE policy,\n");
  623. fprintf(fh_usage, "or both. When setting the IPSec policy, think of it as setting an \"IP Security Rule\" \nin the UI. So, if you need to set up a tunnel policy, you will need\n");
  624. fprintf(fh_usage, "to execute the tool twice, once for the outbound filters and outgoing tunnel\n");
  625. fprintf(fh_usage, "endpoint, and once for the inbound filters and incoming tunnel endpoint.\n");
  626. fprintf(fh_usage, "\n");
  627. fprintf(fh_usage, "\n");
  628. fprintf(fh_usage, "OPTIONS:\n");
  629. fprintf(fh_usage, " \n");
  630. fprintf(fh_usage, " \\\\machinename sets policies on that machine. If not included, the \n");
  631. fprintf(fh_usage, " local machine is assumed.\n");
  632. fprintf(fh_usage, " NOTE: that if you use this it must be the first argument AND\n");
  633. fprintf(fh_usage, " you MUST have administrative privileges on that machine.\n");
  634. fprintf(fh_usage, "\n");
  635. fprintf(fh_usage, " -confirm will ask you to confirm before setting policy\n");
  636. fprintf(fh_usage, " can be abbreviated to -c\n");
  637. fprintf(fh_usage, " *OPTIONAL, DYNAMIC MODE ONLY*\n");
  638. fprintf(fh_usage, "\n");
  639. fprintf(fh_usage, " The following flags deal with IPSec policy. If omitted, a default value \n");
  640. fprintf(fh_usage, " is used where specified.\n");
  641. fprintf(fh_usage, "\n");
  642. fprintf(fh_usage, " -f FilterList \n");
  643. fprintf(fh_usage, " where FilterList is one or more space separated filterspecs\n");
  644. fprintf(fh_usage, " a filterspec is of the format:\n");
  645. fprintf(fh_usage, " A.B.C.D/mask:port=A.B.C.D/mask:port:protocol\n");
  646. fprintf(fh_usage, " you can also specify DEFAULT to create default response rule\n");
  647. fprintf(fh_usage, "\n");
  648. fprintf(fh_usage, " The Source address is always on the left of the '=' and the Destination \n");
  649. fprintf(fh_usage, " address is always on the right. \n");
  650. fprintf(fh_usage, "\n");
  651. fprintf(fh_usage, " MIRRORING: If you replace the '=' with a '+' two filters will be created, \n");
  652. fprintf(fh_usage, " one in each direction.\n");
  653. fprintf(fh_usage, "\n");
  654. fprintf(fh_usage, " mask and port are optional. If omitted, Any port and\n");
  655. fprintf(fh_usage, " mask 255.255.255.255 will be used for the filter. \n");
  656. fprintf(fh_usage, "\n");
  657. fprintf(fh_usage, " You can replace A.B.C.D/mask with the following for \n");
  658. fprintf(fh_usage, " special meaning:\n");
  659. fprintf(fh_usage, " 0 means My address(es)\n");
  660. fprintf(fh_usage, " * means Any address\n");
  661. fprintf(fh_usage, " a DNS name (NOTE: multiple resolutions are ignored)\n");
  662. fprintf(fh_usage, " a GUID of the local network interface in the form {12345678-1234-1234-1234-123456789ABC}\n");
  663. fprintf(fh_usage, " GUIDs are NOT supported for static mode\n");
  664. fprintf(fh_usage, "\n");
  665. fprintf(fh_usage, " protocol is optional, if omitted, Any protocol is assumed. If you \n");
  666. fprintf(fh_usage, " indicate a protocol, a port must precede it or :: must preceded it.\n");
  667. fprintf(fh_usage, " NOTE BENE: if protocol is specified, it must be the last item in \n");
  668. fprintf(fh_usage, " the filter spec. \n");
  669. fprintf(fh_usage, "\n");
  670. fprintf(fh_usage, " Examples:\n");
  671. fprintf(fh_usage, " Machine1+Machine2::6 will filter TCP traffic between Machine1 and Machine2\n");
  672. fprintf(fh_usage, " 172.31.0.0/255.255.0.0:80=157.0.0.0/255.0.0.0:80:TCP will filter\n");
  673. fprintf(fh_usage, " all TCP traffic from the first subnet, port 80 to the second subnet, \n");
  674. fprintf(fh_usage, " port 80\n");
  675. fprintf(fh_usage, "\n");
  676. fprintf(fh_usage, " PASSTHRU and DROP filters: By surrounding a filter specification with (), \n");
  677. fprintf(fh_usage, " the filter will be a passthru filter. If you surround it with [], the \n");
  678. fprintf(fh_usage, " filter will be a blocking, or drop, filter. \n");
  679. fprintf(fh_usage, " Example: (0+128.2.1.1) will create 2 filters (it's mirrored) that will \n");
  680. fprintf(fh_usage, " be exempted from policy.\n");
  681. fprintf(fh_usage, "\n");
  682. fprintf(fh_usage, " You can use the following protocol symbols: ICMP UDP RAW TCP \n");
  683. fprintf(fh_usage, "\n");
  684. fprintf(fh_usage, " Star notation:\n");
  685. fprintf(fh_usage, " If you're subnet masks are along octet boundaries, then you\n");
  686. fprintf(fh_usage, " can use the star notation to wildcard subnets.\n");
  687. fprintf(fh_usage, " Examples:\n");
  688. fprintf(fh_usage, " 128.*.*.* is same as 128.0.0.0/255.0.0.0\n");
  689. fprintf(fh_usage, " 128.*.* is the same as above\n");
  690. fprintf(fh_usage, " 128.* is the same as above\n");
  691. fprintf(fh_usage, " 144.92.*.* is same as 144.92.0.0/255.255.0.0\n");
  692. fprintf(fh_usage, "\n");
  693. fprintf(fh_usage, " There is no DEFAULT, -f is required\n");
  694. fprintf(fh_usage, " \n");
  695. fprintf(fh_usage, " -n NegotiationPolicyList \n");
  696. fprintf(fh_usage, " where NegotiationPolicyList is one or more space separated \n");
  697. fprintf(fh_usage, " IPSec policies in the one of the following forms:\n");
  698. fprintf(fh_usage, "\n");
  699. fprintf(fh_usage, " ESP[ConfAlg,AuthAlg]RekeyPFS[Group] \n");
  700. fprintf(fh_usage, " AH[HashAlg] \n");
  701. fprintf(fh_usage, " AH[HashAlg]+ESP[ConfAlg,AuthAlg]\n");
  702. fprintf(fh_usage, "\n");
  703. fprintf(fh_usage, " where ConfAlg can be NONE, DES, or 3DES\n");
  704. fprintf(fh_usage, " and AuthAlg can be NONE, MD5, or SHA\n");
  705. fprintf(fh_usage, " and HashAlg is MD5 or SHA\n");
  706. fprintf(fh_usage, "\n");
  707. fprintf(fh_usage, " NOTE: ESP[NONE,NONE] is not a supported config\n");
  708. fprintf(fh_usage, " NOTE: SHA refers the SHA1 hash algorithm\n");
  709. fprintf(fh_usage, "\n");
  710. fprintf(fh_usage, " Rekey is number of KBytes or number of seconds to rekey \n");
  711. fprintf(fh_usage, " put K or S after the number to indicate KBytes or seconds, respectively\n");
  712. fprintf(fh_usage, " Example: 3600S will rekey after 1 hour\n");
  713. fprintf(fh_usage, " To use both, separate with a slash.\n");
  714. fprintf(fh_usage, " Example: 3600S/5000K will rekey every hour and 5 MB.\n");
  715. fprintf(fh_usage, "\n");
  716. fprintf(fh_usage, " REKEY PARAMETERS ARE OPTIONAL\n");
  717. fprintf(fh_usage, "\n");
  718. fprintf(fh_usage, " PFS this is OPTIONAL, if it is present it will enable phase 2 perfect\n");
  719. fprintf(fh_usage, " forward secrecy. You may use just P for short.\n");
  720. fprintf(fh_usage, " It is also possible to specify which PFS Group to use: \n");
  721. fprintf(fh_usage, " PFS1 or P1, PFS2 or P2\n");
  722. fprintf(fh_usage, " By Default, PFS Group value will be taken from current Main Mode settings");
  723. fprintf(fh_usage, "\n");
  724. fprintf(fh_usage, " DEFAULT: ESP[3DES,SHA] ESP[3DES,MD5] ESP[DES,SHA]\n");
  725. fprintf(fh_usage, " ESP[DES,MD5]\n");
  726. fprintf(fh_usage, "\n");
  727. fprintf(fh_usage, " -t tunnel address in one of the following forms:\n");
  728. fprintf(fh_usage, " A.B.C.D\n");
  729. fprintf(fh_usage, " DNS name\n");
  730. fprintf(fh_usage, "\n");
  731. fprintf(fh_usage, " DEFAULT: omission of tunnel address assumes transport mode\n");
  732. fprintf(fh_usage, "\n");
  733. fprintf(fh_usage, " -a AuthMethodList \n");
  734. fprintf(fh_usage, " A list of space separated auth methods of the form:\n");
  735. fprintf(fh_usage, " PRESHARE:\"preshared key string\"\n");
  736. fprintf(fh_usage, " KERBEROS\n");
  737. fprintf(fh_usage, " CERT:\"CA Info\"\n");
  738. fprintf(fh_usage, "\n");
  739. fprintf(fh_usage, " The strings provided to preshared key and CA info ARE case sensitive.\n");
  740. fprintf(fh_usage, " You can abbreviate the method with the first letter, ie. P, K, or C.\n");
  741. fprintf(fh_usage, "\n");
  742. fprintf(fh_usage, " DEFAULT: KERBEROS\n");
  743. fprintf(fh_usage, "\n");
  744. fprintf(fh_usage, " -soft will allow soft associations\n");
  745. fprintf(fh_usage, " DEFAULT: don't allow soft SAs\n");
  746. fprintf(fh_usage, "\n");
  747. fprintf(fh_usage, " -lan will set policy only for lan adapters\n");
  748. fprintf(fh_usage, " -dialup will set policy only for dialup adapters\n");
  749. fprintf(fh_usage, " *BOTH ARE OPTIONAL, if not specified, All adapters are used*\n");
  750. fprintf(fh_usage, " DEFAULT: All adapters\n");
  751. fprintf(fh_usage, "\n");
  752. fprintf(fh_usage, " The following deal with IKE phase 1 policy. An easy way to remember\n");
  753. fprintf(fh_usage, " is that all IKE phase 1 parameters are passed with a 1 in the flag.\n");
  754. fprintf(fh_usage, "\n");
  755. fprintf(fh_usage, " If no IKE flags are specified, the current IKE policy\n");
  756. fprintf(fh_usage, " will be used. If there is no current IKE policy, the defaults \n");
  757. fprintf(fh_usage, " specified below will be used.\n");
  758. fprintf(fh_usage, "\n");
  759. fprintf(fh_usage, " -1s SecurityMethodList\n");
  760. fprintf(fh_usage, " where SecurityMethodList is one or more space separated SecurityMethods\n");
  761. fprintf(fh_usage, " in the form:\n");
  762. fprintf(fh_usage, " ConfAlg-HashAlg-GroupNum\n");
  763. fprintf(fh_usage, " where ConfAlg can be DES or 3DES\n");
  764. fprintf(fh_usage, " and HashAlg is MD5 or SHA\n");
  765. fprintf(fh_usage, " and GroupNum is:\n");
  766. fprintf(fh_usage, " 1 (Low)\n");
  767. fprintf(fh_usage, " 2 (Med)\n");
  768. fprintf(fh_usage, "\n");
  769. fprintf(fh_usage, " Example: DES-SHA-1\n");
  770. fprintf(fh_usage, " DEFAULT: 3DES-SHA-2 3DES-MD5-2 DES-SHA-1 DES-MD5-1\n");
  771. fprintf(fh_usage, "\n");
  772. fprintf(fh_usage, " -1p enable PFS for phase 1\n");
  773. fprintf(fh_usage, " DEFAULT: not enabled\n");
  774. fprintf(fh_usage, "\n");
  775. fprintf(fh_usage, " -1k number of Quick Modes or number of seconds to rekey for phase 1\n");
  776. fprintf(fh_usage, " put Q or S after the number to indicate Quick Modes or seconds,\n");
  777. fprintf(fh_usage, " respectively\n");
  778. fprintf(fh_usage, " Example: 10Q will rekey after 10 quick modes\n");
  779. fprintf(fh_usage, " To use both, separate with a slash.\n");
  780. fprintf(fh_usage, " Example: 10Q/3600S will rekey every hour and 10 quick modes\n");
  781. fprintf(fh_usage, " *OPTIONAL*\n");
  782. fprintf(fh_usage, " DEFAULT: no QM limit, 480 min lifetime\n");
  783. fprintf(fh_usage, " \n");
  784. fprintf(fh_usage, " -1e SoftSAExpirationTime\n");
  785. fprintf(fh_usage, " set Soft SA expiration time attribute of the main mode policy\n");
  786. fprintf(fh_usage, " value is specified in seconds\n");
  787. fprintf(fh_usage, " DEFAULT: not set if Soft SA is not allowed\n");
  788. fprintf(fh_usage, " set to 300 seconds if Soft SA is allowed\n");
  789. fprintf(fh_usage, "\n");
  790. fprintf(fh_usage, " -1f MMFilterList\n");
  791. fprintf(fh_usage, " set specific main mode filters. Syntax is the same as for -f option\n");
  792. fprintf(fh_usage, " except that you cannot specify passthru, block filters, ports and protocols\n");
  793. fprintf(fh_usage, " DEFAULT: filters are generated automatically based on quick mode filters\n");
  794. fprintf(fh_usage, " \n");
  795. fprintf(fh_usage, "-----------\n");
  796. fprintf(fh_usage, "STATIC MODE\n");
  797. fprintf(fh_usage, "-----------\n");
  798. fprintf(fh_usage, "\n");
  799. fprintf(fh_usage, "Static mode uses most of the dynamic mode syntax, but adds a few flags\n");
  800. fprintf(fh_usage, "that enable it work at a policy level as well. Remember, dynamic mode\n");
  801. fprintf(fh_usage, "just lets you add anonymous rules to the policy agent. Static mode\n");
  802. fprintf(fh_usage, "allows you to create named policies and named rules. It also has some\n");
  803. fprintf(fh_usage, "functionality to modify existing policies and rules, provided they were\n");
  804. fprintf(fh_usage, "originally created with ipseccmd.\n");
  805. fprintf(fh_usage, "\n");
  806. fprintf(fh_usage, "Static mode is supposed to provide most of the functionality of the IPSec UI\n");
  807. fprintf(fh_usage, "in a command line tool, so there are references here to the UI.\n");
  808. fprintf(fh_usage, "\n");
  809. fprintf(fh_usage, "First, there is one change to the dynamic mode usage that static mode\n");
  810. fprintf(fh_usage, "requires. In static mode, pass through and block filters are indicated\n");
  811. fprintf(fh_usage, "in the NegotiationPolicyList that is specified by -n. There are three\n");
  812. fprintf(fh_usage, "items you can pass in the NegotiationPolicyList that have special meaning:\n");
  813. fprintf(fh_usage, "\n");
  814. fprintf(fh_usage, "BLOCK will ignore the rest of the policies in NegotiationPolicyList and \n");
  815. fprintf(fh_usage, " will make all of the filters blocking or drop filters.\n");
  816. fprintf(fh_usage, " This is the same as checking the \"Block\" radio button\n");
  817. fprintf(fh_usage, " in the UI\n");
  818. fprintf(fh_usage, "\n");
  819. fprintf(fh_usage, "PASS will ignore the rest of the policies in NegotiationPolicyList and \n");
  820. fprintf(fh_usage, " will make all of the filters pass through filters.\n");
  821. fprintf(fh_usage, " This is the same as checking the \"Permit\"\n");
  822. fprintf(fh_usage, " radio button in the UI\n");
  823. fprintf(fh_usage, "\n");
  824. fprintf(fh_usage, "INPASS will plumb any inbound filters as pass through.\n");
  825. fprintf(fh_usage, " This is the same as checking the \"Allow unsecured communication,\n");
  826. fprintf(fh_usage, " but always respond using IPSEC\" check box in the UI\n");
  827. fprintf(fh_usage, " \n");
  828. fprintf(fh_usage, "\n");
  829. fprintf(fh_usage, "Static Mode flags:\n");
  830. fprintf(fh_usage, "All flags are REQUIRED unless otherwise indicated.\n");
  831. fprintf(fh_usage, "\n");
  832. fprintf(fh_usage, "-w Write the policy to storage indicated by TYPE:LOCATION\n");
  833. fprintf(fh_usage, " TYPE can be either REG for registry or DS for Directory Storage\n");
  834. fprintf(fh_usage, " if \\\\machinename was specified and TYPE is REG, will be written\n");
  835. fprintf(fh_usage, " to the remote machine's registry\n");
  836. fprintf(fh_usage, " DOMAIN for the DS case only. Indicates the domain name of the\n");
  837. fprintf(fh_usage, " DS to write to. If omitted, use the domain the local machine is in.\n");
  838. fprintf(fh_usage, " OPTIONAL\n");
  839. fprintf(fh_usage, "\n");
  840. fprintf(fh_usage, "-p PolicyName:PollInterval\n");
  841. fprintf(fh_usage, " Name the policy with this string. If a policy with this name is\n");
  842. fprintf(fh_usage, " already in storage, this rule will be added to the policy. \n");
  843. fprintf(fh_usage, " Otherwise a new policy will be created. If PollInterval is specified,\n");
  844. fprintf(fh_usage, " the polling interval for the policy will be set.\n");
  845. fprintf(fh_usage, "\n");
  846. fprintf(fh_usage, "-r RuleName\n");
  847. fprintf(fh_usage, " Name the rule with this string. If a rule with that name already exists,\n");
  848. fprintf(fh_usage, " that rule is modified to reflect the information supplied to ipseccmd.\n");
  849. fprintf(fh_usage, " For example, if only -f is specified and the rule exists,\n");
  850. fprintf(fh_usage, " only the filters of that rule will be replaced. \n");
  851. fprintf(fh_usage, "\n");
  852. fprintf(fh_usage, "-x will set the policy active in the LOCAL registry case OPTIONAL\n");
  853. fprintf(fh_usage, "\n");
  854. fprintf(fh_usage, "-y will set the policy inactive in the LOCAL registry case OPTIONAL\n");
  855. fprintf(fh_usage, "\n");
  856. fprintf(fh_usage, "-o will delete the policy specified by -p OPTIONAL\n");
  857. fprintf(fh_usage, " (NOTE: this will delete all aspects of the specified policy\n");
  858. fprintf(fh_usage, " don't use if you have other policies pointing to the objects in that policy)\n");
  859. return;
  860. }
  861. // need to init/cleanup Winsock
  862. int
  863. InitWinsock(
  864. WORD wVersionRequested
  865. )
  866. {
  867. WSADATA wsaData;
  868. int err;
  869. err = WSAStartup( wVersionRequested, &wsaData );
  870. if ( !err )
  871. {
  872. //
  873. // CHECK VERSION
  874. //
  875. if ( LOBYTE(wsaData.wVersion) != LOBYTE(wVersionRequested) ||
  876. HIBYTE(wsaData.wVersion) != HIBYTE(wVersionRequested) )
  877. {
  878. /* Tell the user that we could not find a usable */
  879. /* WinSock DLL. */
  880. WSACleanup();
  881. err = (-1);
  882. }
  883. }
  884. return err;
  885. }
  886. // fgets function that reads a string from the file. Buffer is reallocated to make sure we read the whole string
  887. char* my_fgets (char** ppbuf, int* pbufsize, FILE* f)
  888. {
  889. char* tmp;
  890. int itmp;
  891. tmp = fgets(*ppbuf, *pbufsize, f);
  892. if (!tmp)
  893. return NULL;
  894. itmp = strlen(*ppbuf);
  895. while (itmp == *pbufsize-1 && (*ppbuf)[*pbufsize-2] != '\r' && (*ppbuf)[*pbufsize-2] != '\n')
  896. {
  897. // need to read longer string
  898. *pbufsize *= 2;
  899. *ppbuf = (char*) realloc(*ppbuf, *pbufsize);
  900. assert(*ppbuf);
  901. // read again
  902. tmp = fgets(*ppbuf+itmp, *pbufsize/2+1, f);
  903. if (!tmp)
  904. return NULL;
  905. itmp = strlen(*ppbuf);
  906. }
  907. return tmp;
  908. }
  909. // "real" main that reads from file if needed (v 1.51)
  910. int _cdecl main(int argc, char *argv[])
  911. {
  912. FILE *f;
  913. char *commandLine = NULL; // temp storage for cmd line
  914. int clSize = POTF_MAX_STRLEN;
  915. char * tArgv[8192];
  916. if (argc < 2 || _stricmp(argv[1], "-file"))
  917. {
  918. // regular ipseccmd
  919. return ipseccmd_main(argc, argv);
  920. }
  921. tArgv[0] = _strdup("ipseccmd");
  922. commandLine = new char[clSize];
  923. assert(commandLine);
  924. int tArgc = 1;
  925. int iLine = 1;
  926. if ((f = fopen(argv[2], "r")) == NULL)
  927. {
  928. printf("%s could not be opened for read! GetLastError = 0x%x\n", argv[2], GetLastError());
  929. return 1;
  930. }
  931. // now read the contents of the file
  932. while (!feof(f))
  933. {
  934. if (my_fgets(&commandLine, &clSize, f) != NULL)
  935. {
  936. if (strlen(commandLine) > 0)
  937. {
  938. if (commandLine[strlen(commandLine)-1] == '\r' || commandLine[strlen(commandLine)-1] == '\n')
  939. {
  940. commandLine[strlen(commandLine)-1] = 0;
  941. }
  942. }
  943. if (strlen(commandLine) > 0)
  944. {
  945. if (commandLine[strlen(commandLine)-1] == '\r' || commandLine[strlen(commandLine)-1] == '\n')
  946. {
  947. commandLine[strlen(commandLine)-1] = 0;
  948. }
  949. }
  950. // parse the command line
  951. tArgv[1] = strtok(commandLine, " \t");
  952. if (!tArgv[1])
  953. {
  954. // empty string, continue;
  955. continue;
  956. }
  957. tArgc++;
  958. if (!_stricmp(tArgv[1], tArgv[0]))
  959. {
  960. // line in file starts with "ipseccmd", skip the word;
  961. tArgc--;
  962. }
  963. while (tArgv[tArgc] = strtok(NULL, " \t"))
  964. {
  965. tArgc++;
  966. }
  967. // end parse
  968. ipseccmd_main (tArgc, tArgv);
  969. tArgc = 1;
  970. }
  971. }
  972. }