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.

881 lines
19 KiB

  1. /*++
  2. Copyright (c) 1990 - 1995 Microsoft Corporation
  3. Module Name:
  4. util.c
  5. Abstract:
  6. Client Side Utility Routines
  7. Author:
  8. Dave Snipp (DaveSn) 15-Mar-1991
  9. Revision History:
  10. --*/
  11. #include "precomp.h"
  12. #pragma hdrstop
  13. #include "client.h"
  14. HANDLE hCSR = INVALID_HANDLE_VALUE;
  15. BOOL
  16. InCSRProcess(
  17. VOID
  18. )
  19. {
  20. //
  21. // hCSR == INVALID_HANDLE_VALUE Not initialized, must check.
  22. // NULL Not running in CSR
  23. // hModule value Running in CSR
  24. //
  25. if (hCSR != NULL) {
  26. //
  27. // Check if we are running in CSR. If so, then don't try
  28. // any notifications.
  29. //
  30. if (hCSR == INVALID_HANDLE_VALUE) {
  31. hCSR = GetModuleHandle( gszCSRDll );
  32. }
  33. }
  34. return hCSR != NULL;
  35. }
  36. LPVOID
  37. DllAllocSplMem(
  38. DWORD cb
  39. )
  40. /*++
  41. Routine Description:
  42. This function will allocate local memory. It will possibly allocate extra
  43. memory and fill this with debugging information for the debugging version.
  44. Arguments:
  45. cb - The amount of memory to allocate
  46. Return Value:
  47. NON-NULL - A pointer to the allocated memory
  48. FALSE/NULL - The operation failed. Extended error status is available
  49. using GetLastError.
  50. --*/
  51. {
  52. PDWORD_PTR pMem;
  53. DWORD cbNew;
  54. cb = DWORD_ALIGN_UP(cb);
  55. cbNew = cb+sizeof(DWORD_PTR)+sizeof(DWORD);
  56. pMem= LocalAlloc(LPTR, cbNew);
  57. if (!pMem) {
  58. DBGMSG( DBG_WARNING, ("Memory Allocation failed for %d bytes\n", cbNew ));
  59. return 0;
  60. }
  61. *pMem=cb;
  62. *(LPDWORD)((LPBYTE)pMem+cbNew-sizeof(DWORD))=0xdeadbeef;
  63. return (LPVOID)(pMem+1);
  64. }
  65. BOOL
  66. DllFreeSplMem(
  67. LPVOID pMem
  68. )
  69. {
  70. DWORD_PTR cbNew;
  71. PDWORD_PTR pNewMem;
  72. if( !pMem ){
  73. return TRUE;
  74. }
  75. pNewMem = pMem;
  76. pNewMem--;
  77. cbNew = *pNewMem;
  78. if (*(LPDWORD)((LPBYTE)pMem + cbNew) != 0xdeadbeef) {
  79. DBGMSG(DBG_ERROR, ("DllFreeSplMem Corrupt Memory in winspool : %0p\n", pNewMem));
  80. return FALSE;
  81. }
  82. memset(pNewMem, 0x65, (size_t) cbNew);
  83. LocalFree((LPVOID)pNewMem);
  84. return TRUE;
  85. }
  86. LPVOID
  87. ReallocSplMem(
  88. LPVOID pOldMem,
  89. DWORD cbOld,
  90. DWORD cbNew
  91. )
  92. {
  93. LPVOID pNewMem;
  94. pNewMem=AllocSplMem(cbNew);
  95. if (pOldMem && pNewMem) {
  96. if (cbOld) {
  97. CopyMemory( pNewMem, pOldMem, min(cbNew, cbOld));
  98. }
  99. FreeSplMem(pOldMem);
  100. }
  101. return pNewMem;
  102. }
  103. LPTSTR
  104. AllocSplStr(
  105. LPCTSTR pStr
  106. )
  107. /*++
  108. Routine Description:
  109. This function will allocate enough local memory to store the specified
  110. string, and copy that string to the allocated memory
  111. Arguments:
  112. pStr - Pointer to the string that needs to be allocated and stored
  113. Return Value:
  114. NON-NULL - A pointer to the allocated memory containing the string
  115. FALSE/NULL - The operation failed. Extended error status is available
  116. using GetLastError.
  117. --*/
  118. {
  119. LPTSTR pMem;
  120. if (!pStr)
  121. return 0;
  122. if (pMem = AllocSplMem( _tcslen(pStr)*sizeof(TCHAR) + sizeof(TCHAR) ))
  123. _tcscpy(pMem, pStr);
  124. return pMem;
  125. }
  126. BOOL
  127. DllFreeSplStr(
  128. LPTSTR pStr
  129. )
  130. {
  131. return pStr ?
  132. DllFreeSplMem(pStr) :
  133. FALSE;
  134. }
  135. BOOL
  136. ReallocSplStr(
  137. LPTSTR *ppStr,
  138. LPCTSTR pStr
  139. )
  140. {
  141. LPWSTR pOldStr = *ppStr;
  142. *ppStr=AllocSplStr(pStr);
  143. FreeSplStr(pOldStr);
  144. return TRUE;
  145. }
  146. /* Message
  147. *
  148. * Displays a message by loading the strings whose IDs are passed into
  149. * the function, and substituting the supplied variable argument list
  150. * using the varargs macros.
  151. *
  152. */
  153. INT
  154. Message(
  155. HWND hwnd,
  156. DWORD Type,
  157. INT CaptionID,
  158. INT TextID,
  159. ...
  160. )
  161. {
  162. TCHAR MsgText[256];
  163. TCHAR MsgFormat[256];
  164. TCHAR MsgCaption[40];
  165. va_list vargs;
  166. if( ( LoadString( hInst, TextID, MsgFormat,
  167. COUNTOF(MsgFormat)) > 0 )
  168. && ( LoadString( hInst, CaptionID, MsgCaption, COUNTOF(MsgCaption) ) > 0 ) )
  169. {
  170. va_start( vargs, TextID );
  171. wvsprintf( MsgText, MsgFormat, vargs );
  172. va_end( vargs );
  173. return MessageBox( hwnd, MsgText, MsgCaption, Type );
  174. }
  175. else
  176. return 0;
  177. }
  178. /*
  179. *
  180. */
  181. LPTSTR
  182. GetErrorString(
  183. DWORD Error
  184. )
  185. {
  186. TCHAR Buffer[1024];
  187. LPTSTR pErrorString = NULL;
  188. DWORD dwFlags;
  189. HANDLE hModule;
  190. if ((Error >= NERR_BASE) && (Error <= MAX_NERR)){
  191. hModule = LoadLibrary(szNetMsgDll);
  192. dwFlags = FORMAT_MESSAGE_FROM_HMODULE;
  193. }
  194. else {
  195. hModule = NULL;
  196. dwFlags = FORMAT_MESSAGE_FROM_SYSTEM;
  197. }
  198. if( FormatMessage( dwFlags, hModule,
  199. Error, 0, Buffer,
  200. COUNTOF(Buffer), NULL )
  201. == 0 )
  202. LoadString( hInst, IDS_UNKNOWN_ERROR, Buffer,
  203. COUNTOF(Buffer));
  204. pErrorString = AllocSplStr(Buffer);
  205. if (hModule) {
  206. FreeLibrary(hModule);
  207. }
  208. return pErrorString;
  209. }
  210. DWORD ReportFailure( HWND hwndParent,
  211. DWORD idTitle,
  212. DWORD idDefaultError )
  213. {
  214. DWORD ErrorID;
  215. DWORD MsgType;
  216. LPTSTR pErrorString;
  217. ErrorID = GetLastError( );
  218. MsgType = MB_OK | MB_ICONSTOP;
  219. pErrorString = GetErrorString( ErrorID );
  220. Message( hwndParent, MsgType, idTitle,
  221. idDefaultError, pErrorString );
  222. FreeSplStr( pErrorString );
  223. return ErrorID;
  224. }
  225. /*
  226. *
  227. */
  228. #define ENTRYFIELD_LENGTH 256
  229. LPTSTR AllocDlgItemText(HWND hwnd, int id)
  230. {
  231. TCHAR string[ENTRYFIELD_LENGTH];
  232. GetDlgItemText (hwnd, id, string, COUNTOF(string));
  233. return ( *string ? AllocSplStr(string) : NULL );
  234. }
  235. PSECURITY_DESCRIPTOR
  236. BuildInputSD(
  237. PSECURITY_DESCRIPTOR pPrinterSD,
  238. PDWORD pSizeSD
  239. )
  240. /*++
  241. --*/
  242. {
  243. SECURITY_DESCRIPTOR AbsoluteSD;
  244. PSECURITY_DESCRIPTOR pRelative;
  245. BOOL Defaulted = FALSE;
  246. BOOL DaclPresent = FALSE;
  247. BOOL SaclPresent = FALSE;
  248. PSID pOwnerSid = NULL;
  249. PSID pGroupSid = NULL;
  250. PACL pDacl = NULL;
  251. PACL pSacl = NULL;
  252. DWORD SDLength = 0;
  253. //
  254. // Initialize *pSizeSD = 0;
  255. //
  256. *pSizeSD = 0;
  257. if (!IsValidSecurityDescriptor(pPrinterSD)) {
  258. return(NULL);
  259. }
  260. if (!InitializeSecurityDescriptor (&AbsoluteSD, SECURITY_DESCRIPTOR_REVISION1)) {
  261. return(NULL);
  262. }
  263. if(!GetSecurityDescriptorOwner(pPrinterSD,
  264. &pOwnerSid, &Defaulted)){
  265. return(NULL);
  266. }
  267. SetSecurityDescriptorOwner(&AbsoluteSD,
  268. pOwnerSid, Defaulted );
  269. if(! GetSecurityDescriptorGroup( pPrinterSD,
  270. &pGroupSid, &Defaulted )){
  271. return(NULL);
  272. }
  273. SetSecurityDescriptorGroup( &AbsoluteSD,
  274. pGroupSid, Defaulted );
  275. if(!GetSecurityDescriptorDacl( pPrinterSD,
  276. &DaclPresent, &pDacl, &Defaulted )){
  277. return(NULL);
  278. }
  279. SetSecurityDescriptorDacl( &AbsoluteSD,
  280. DaclPresent, pDacl, Defaulted );
  281. if(!GetSecurityDescriptorSacl( pPrinterSD,
  282. &SaclPresent, &pSacl, &Defaulted)){
  283. return(NULL);
  284. }
  285. SetSecurityDescriptorSacl( &AbsoluteSD,
  286. SaclPresent, pSacl, Defaulted );
  287. SDLength = GetSecurityDescriptorLength( &AbsoluteSD);
  288. pRelative = LocalAlloc(LPTR, SDLength);
  289. if (!pRelative) {
  290. return(NULL);
  291. }
  292. if (!MakeSelfRelativeSD (&AbsoluteSD, pRelative, &SDLength)) {
  293. LocalFree(pRelative);
  294. return(NULL);
  295. }
  296. *pSizeSD = SDLength;
  297. return(pRelative);
  298. }
  299. PKEYDATA
  300. CreateTokenList(
  301. LPWSTR pKeyData
  302. )
  303. {
  304. DWORD cTokens;
  305. DWORD cb;
  306. PKEYDATA pResult;
  307. LPWSTR pDest;
  308. LPWSTR psz = pKeyData;
  309. LPWSTR *ppToken;
  310. if (!psz || !*psz)
  311. return NULL;
  312. cTokens=1;
  313. /* Scan through the string looking for commas,
  314. * ensuring that each is followed by a non-NULL character:
  315. */
  316. while ((psz = wcschr(psz, L',')) && psz[1]) {
  317. cTokens++;
  318. psz++;
  319. }
  320. cb = sizeof(KEYDATA) + (cTokens-1) * sizeof(LPWSTR) +
  321. wcslen(pKeyData)*sizeof(WCHAR) + sizeof(WCHAR);
  322. if (!(pResult = (PKEYDATA)AllocSplMem(cb)))
  323. return NULL;
  324. pResult->cb = cb;
  325. /* Initialise pDest to point beyond the token pointers:
  326. */
  327. pDest = (LPWSTR)((LPBYTE)pResult + sizeof(KEYDATA) +
  328. (cTokens-1) * sizeof(LPWSTR));
  329. /* Then copy the key data buffer there:
  330. */
  331. wcscpy(pDest, pKeyData);
  332. ppToken = pResult->pTokens;
  333. psz = pDest;
  334. do {
  335. *ppToken++ = psz;
  336. if ( psz = wcschr(psz, L',') )
  337. *psz++ = L'\0';
  338. } while (psz);
  339. pResult->cTokens = cTokens;
  340. return( pResult );
  341. }
  342. LPWSTR
  343. GetPrinterPortList(
  344. HANDLE hPrinter
  345. )
  346. {
  347. LPBYTE pMem;
  348. LPTSTR pPort;
  349. DWORD dwPassed = 1024; //Try 1K to start with
  350. LPPRINTER_INFO_2 pPrinter;
  351. DWORD dwLevel = 2;
  352. DWORD dwNeeded;
  353. PKEYDATA pKeyData;
  354. DWORD i = 0;
  355. LPWSTR pPortNames = NULL;
  356. pMem = AllocSplMem(dwPassed);
  357. if (pMem == NULL) {
  358. return FALSE;
  359. }
  360. if (!GetPrinter(hPrinter, dwLevel, pMem, dwPassed, &dwNeeded)) {
  361. DBGMSG(DBG_TRACE, ("Last error is %d\n", GetLastError()));
  362. if (GetLastError() != ERROR_INSUFFICIENT_BUFFER) {
  363. return NULL;
  364. }
  365. pMem = ReallocSplMem(pMem, dwPassed, dwNeeded);
  366. dwPassed = dwNeeded;
  367. if (!GetPrinter(hPrinter, dwLevel, pMem, dwPassed, &dwNeeded)) {
  368. FreeSplMem(pMem);
  369. return (NULL);
  370. }
  371. }
  372. pPrinter = (LPPRINTER_INFO_2)pMem;
  373. //
  374. // Fixes the null pPrinter->pPortName problem where
  375. // downlevel may return null
  376. //
  377. if (!pPrinter->pPortName) {
  378. FreeSplMem(pMem);
  379. return(NULL);
  380. }
  381. pPortNames = AllocSplStr(pPrinter->pPortName);
  382. FreeSplMem(pMem);
  383. return(pPortNames);
  384. }
  385. BOOL
  386. UpdateString(
  387. IN LPCTSTR pszString, OPTIONAL
  388. OUT LPTSTR* ppszOut
  389. )
  390. /*++
  391. Routine Description:
  392. Updates an output string if the input is non-NULL.
  393. Arguments:
  394. pszString - String to update to. If NULL or -1, function does nothing.
  395. Return Value:
  396. TRUE - Success
  397. FALSE - FAILED. *ppszOut = NULL
  398. --*/
  399. {
  400. if( pszString && pszString != (LPCTSTR)-1 ){
  401. FreeSplStr( *ppszOut );
  402. *ppszOut = AllocSplStr( pszString );
  403. if( !*ppszOut ){
  404. return FALSE;
  405. }
  406. }
  407. return TRUE;
  408. }
  409. DWORD
  410. RouterFreeBidiResponseContainer(
  411. PBIDI_RESPONSE_CONTAINER pData
  412. )
  413. {
  414. BIDI_RESPONSE_DATA *p;
  415. DWORD Count = 0;
  416. DWORD NumOfRspns;
  417. DWORD dwRet = ERROR_SUCCESS;
  418. try
  419. {
  420. if(pData)
  421. {
  422. Count = pData->Count;
  423. for(NumOfRspns= 0,
  424. p = &pData->aData[0];
  425. NumOfRspns < Count;
  426. NumOfRspns++,
  427. p++
  428. )
  429. {
  430. if(p)
  431. {
  432. if(p->pSchema)
  433. {
  434. MIDL_user_free(p->pSchema);
  435. }
  436. switch(p->data.dwBidiType)
  437. {
  438. //
  439. // Text Data (ANSI String)
  440. //
  441. case BIDI_TEXT:
  442. //
  443. // String (UNICODE string)
  444. //
  445. case BIDI_ENUM:
  446. //
  447. // Enumeration Data (ANSI String)
  448. //
  449. case BIDI_STRING:
  450. {
  451. if(p->data.u.sData)
  452. {
  453. MIDL_user_free(p->data.u.sData);
  454. }
  455. }
  456. break;
  457. //
  458. // Binary Data (BLOB)
  459. //
  460. case BIDI_BLOB:
  461. {
  462. if(p->data.u.biData.pData)
  463. {
  464. MIDL_user_free(p->data.u.biData.pData);
  465. }
  466. }
  467. break;
  468. //
  469. // Undefined Type
  470. //
  471. default:
  472. {
  473. //
  474. // Nothing really , just return
  475. //
  476. }
  477. break;
  478. }
  479. }
  480. }
  481. MIDL_user_free(pData);
  482. }
  483. }
  484. except(1)
  485. {
  486. dwRet = TranslateExceptionCode(GetExceptionCode());
  487. DBGMSG(DBG_ERROR, ("RouterFreeBidiResponseContainer raised an exception : %u \n", dwRet));
  488. }
  489. return(dwRet);
  490. }
  491. ClientVersion
  492. GetClientVer()
  493. {
  494. ClientVersion ClientVer;
  495. return(ClientVer = sizeof(ULONG_PTR));
  496. }
  497. ServerVersion
  498. GetServerVer()
  499. {
  500. ULONG_PTR ul;
  501. NTSTATUS st;
  502. ServerVersion CurrVer;
  503. st = NtQueryInformationProcess(NtCurrentProcess(),
  504. ProcessWow64Information,
  505. &ul,
  506. sizeof(ul),
  507. NULL);
  508. if (NT_SUCCESS(st))
  509. {
  510. // If this call succeeds, we're on Win2000 or newer machines.
  511. if (0 != ul)
  512. {
  513. // 32-bit code running on Win64
  514. CurrVer = THUNKVERSION;
  515. } else
  516. {
  517. // 32-bit code running on Win2000 or later 32-bit OS
  518. CurrVer = NATIVEVERSION;
  519. }
  520. } else
  521. {
  522. CurrVer = NATIVEVERSION;
  523. }
  524. return(CurrVer);
  525. }
  526. BOOL
  527. RunInWOW64()
  528. {
  529. return((GetClientVer() == RUN32BINVER) &&
  530. (GetServerVer() == THUNKVERSION) &&
  531. !bLoadedBySpooler);
  532. /*return(bLoadedBySpooler ? FALSE : TRUE);*/
  533. }
  534. HWND
  535. GetForeGroundWindow(
  536. VOID
  537. )
  538. {
  539. //
  540. // get the foreground window first
  541. //
  542. HWND hWndOwner;
  543. HWND hWndLastPopup;
  544. HWND hWndForeground = GetForegroundWindow();
  545. //
  546. // climb up to the top parent in case it's a child window...
  547. //
  548. HWND hWndParent = hWndForeground;
  549. while ( hWndParent = GetParent(hWndParent) )
  550. {
  551. hWndForeground = hWndParent;
  552. }
  553. //
  554. // get the owner in case the top parent is owned
  555. //
  556. hWndOwner= GetWindow(hWndForeground, GW_OWNER);
  557. if ( hWndOwner )
  558. {
  559. hWndForeground = hWndOwner;
  560. }
  561. hWndLastPopup = GetLastActivePopup(hWndForeground);
  562. return(hWndLastPopup);
  563. }
  564. LPCWSTR
  565. FindFileName(
  566. IN LPCWSTR pPathName
  567. )
  568. /*++
  569. Routine Description:
  570. Retrieve the filename portion of a path.
  571. This will can the input string until it finds the last backslash,
  572. then return the portion of the string immediately following it.
  573. If the string terminates with a backslash, then NULL is returned.
  574. Note: this can return illegal file names; no validation is done.
  575. Arguments:
  576. pPathName - Path name to parse.
  577. Return Value:
  578. Last portion of file name or NULL if none available.
  579. --*/
  580. {
  581. LPCWSTR pSlash;
  582. LPCWSTR pTemp;
  583. if( !pPathName ){
  584. return NULL;
  585. }
  586. pTemp = pPathName;
  587. while( pSlash = wcschr( pTemp, L'\\' )) {
  588. pTemp = pSlash+1;
  589. }
  590. if( !*pTemp ){
  591. return NULL;
  592. }
  593. return pTemp;
  594. }
  595. /*++
  596. Name:
  597. BuildSpoolerObjectPath
  598. Description:
  599. This function addresses bug 461462. When the spooler is not running,
  600. this function can build the path to the drivers directory or the print
  601. processors directory. The result of this function is a path like:
  602. C:\Windows\System32\spool\Drivers\w32x86
  603. or
  604. C:\Windows\System32\spool\Prtprocs\w32x86
  605. depending on the pszPath argument.
  606. Arguments:
  607. pszPath - can be "drivers" or "prtprocs"
  608. pszName - must be NULL or ""
  609. pszEnvironment - can be "windows nt x86" etc. This argument must be validated by the caller
  610. (GetPrinterDriverDirectoryW and GetPrintProcessorDirectoryW)
  611. Level - must be 1
  612. pDriverDirectory - buffer where to store the path
  613. cbBuf - count of bytes in the buffer
  614. pcbNeeded - count of bytes needed to store the path
  615. Return Value:
  616. TRUE - the function succeeded, pDriverDirectory can be used
  617. FALSE - the function failed, pDriverDirectory cannot be used.
  618. Last error:
  619. This function sets the last error in both failure and success cases.
  620. --*/
  621. BOOL
  622. BuildSpoolerObjectPath(
  623. IN PCWSTR pszPath,
  624. IN PCWSTR pszName,
  625. IN PCWSTR pszEnvironment,
  626. IN DWORD Level,
  627. IN PBYTE pDriverDirectory,
  628. IN DWORD cbBuf,
  629. IN PDWORD pcbNeeded
  630. )
  631. {
  632. DWORD Error = ERROR_INVALID_PARAMETER;
  633. if (pcbNeeded && (!pszName || !*pszName))
  634. {
  635. HKEY hkRoot = NULL;
  636. HKEY hkEnv = NULL;
  637. DWORD cchDir = MAX_PATH;
  638. WCHAR szDir[MAX_PATH];
  639. Error = GetSystemWindowsDirectory(szDir, cchDir) ? ERROR_SUCCESS : GetLastError();
  640. if (Error == ERROR_SUCCESS &&
  641. (Error = RegOpenKeyEx(HKEY_LOCAL_MACHINE,
  642. gszRegEnvironments,
  643. 0,
  644. KEY_READ,
  645. &hkRoot)) == ERROR_SUCCESS &&
  646. (Error = RegOpenKeyEx(hkRoot,
  647. pszEnvironment,
  648. 0,
  649. KEY_READ,
  650. &hkEnv)) == ERROR_SUCCESS &&
  651. (Error = StrNCatBuff(szDir,
  652. cchDir,
  653. szDir,
  654. szSlash,
  655. gszSystem32Spool,
  656. szSlash,
  657. pszPath,
  658. szSlash,
  659. NULL)) == ERROR_SUCCESS)
  660. {
  661. DWORD Length = wcslen(szDir);
  662. DWORD cbAvailable = (cchDir - Length) * sizeof(WCHAR);
  663. if ((Error = RegQueryValueEx(hkEnv,
  664. gszEnivronmentDirectory,
  665. NULL,
  666. NULL,
  667. (PBYTE)&szDir[Length],
  668. &cbAvailable)) == ERROR_SUCCESS)
  669. {
  670. *pcbNeeded = (wcslen(szDir) + 1) * sizeof(WCHAR);
  671. if (cbBuf >= *pcbNeeded)
  672. {
  673. wcscpy((PWSTR)pDriverDirectory, szDir);
  674. }
  675. else
  676. {
  677. Error = ERROR_INSUFFICIENT_BUFFER;
  678. }
  679. }
  680. }
  681. else if (Error == ERROR_FILE_NOT_FOUND)
  682. {
  683. //
  684. // If we cannot open the "pszEnvironment" key then the environment is invalid.
  685. //
  686. Error = ERROR_INVALID_ENVIRONMENT;
  687. }
  688. if (hkRoot) RegCloseKey(hkRoot);
  689. if (hkEnv) RegCloseKey(hkEnv);
  690. }
  691. SetLastError(Error);
  692. return Error == ERROR_SUCCESS;
  693. }