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.

696 lines
15 KiB

  1. /*++
  2. Copyright (c) 1990-1994 Microsoft Corporation
  3. All rights reserved
  4. Module Name:
  5. threads.c (thread manager)
  6. Abstract:
  7. Handles the threads used to for notifications (WPC, FFPCN)
  8. Author:
  9. Albert Ting (AlbertT) 25-Jan-94
  10. Environment:
  11. User Mode -Win32
  12. Revision History:
  13. --*/
  14. #include "precomp.h"
  15. #pragma hdrstop
  16. #include "threadm.h"
  17. #include "ntfytab.h"
  18. #define ENTER_THREAD_LIST() EnterCriticalSection(tmStateStatic.pCritSec)
  19. #define EXIT_THREAD_LIST() LeaveCriticalSection(tmStateStatic.pCritSec)
  20. extern CRITICAL_SECTION RouterNotifySection;
  21. DWORD
  22. ThreadNotifyProcessJob(
  23. PTMSTATEVAR pTMStateVar,
  24. PJOB pJob);
  25. PJOB
  26. ThreadNotifyNextJob(
  27. PTMSTATEVAR ptmStateVar);
  28. TMSTATESTATIC tmStateStatic = {
  29. 10,
  30. 2500,
  31. (PFNPROCESSJOB)ThreadNotifyProcessJob,
  32. (PFNNEXTJOB)ThreadNotifyNextJob,
  33. NULL,
  34. NULL,
  35. &RouterNotifySection
  36. };
  37. TMSTATEVAR tmStateVar;
  38. PCHANGE pChangeList;
  39. WCHAR szThreadMax[] = L"ThreadNotifyMax";
  40. WCHAR szThreadIdleLife[] = L"ThreadNotifyIdleLife";
  41. WCHAR szThreadNotifySleep[] = L"ThreadNotifySleep";
  42. DWORD dwThreadNotifySleep = 1000;
  43. BOOL
  44. ThreadInit()
  45. {
  46. HKEY hKey;
  47. DWORD dwType = REG_DWORD;
  48. DWORD cbData;
  49. if (!TMCreateStatic(&tmStateStatic))
  50. return FALSE;
  51. if (!TMCreate(&tmStateStatic, &tmStateVar))
  52. return FALSE;
  53. if (!RegOpenKeyEx(HKEY_LOCAL_MACHINE,
  54. szPrintKey,
  55. 0,
  56. KEY_READ,
  57. &hKey)) {
  58. cbData = sizeof(tmStateStatic.uMaxThreads);
  59. //
  60. // Ignore failure case since we default to 10.
  61. //
  62. RegQueryValueEx(hKey,
  63. szThreadMax,
  64. NULL,
  65. &dwType,
  66. (LPBYTE)&tmStateStatic.uMaxThreads,
  67. &cbData);
  68. cbData = sizeof(tmStateStatic.uIdleLife);
  69. //
  70. // Ignore failure case since we default to 1000 (1 sec).
  71. //
  72. RegQueryValueEx(hKey,
  73. szThreadIdleLife,
  74. NULL,
  75. &dwType,
  76. (LPBYTE)&tmStateStatic.uIdleLife,
  77. &cbData);
  78. cbData = sizeof(dwThreadNotifySleep);
  79. //
  80. // Ignore failure case since we default to 2500 (2.5 sec).
  81. //
  82. RegQueryValueEx(hKey,
  83. szThreadNotifySleep,
  84. NULL,
  85. &dwType,
  86. (LPBYTE)&dwThreadNotifySleep,
  87. &cbData);
  88. RegCloseKey(hKey);
  89. }
  90. return TRUE;
  91. }
  92. VOID
  93. ThreadDestroy()
  94. {
  95. TMDestroy(&tmStateVar);
  96. TMDestroyStatic(&tmStateStatic);
  97. }
  98. BOOL
  99. LinkChange(
  100. PCHANGE pChange)
  101. /*++
  102. Routine Description:
  103. Link up the change to the list of jobs that need to be processed.
  104. If the call succeeded but there was an overflow at the client,
  105. then we won't add this to the list until it gets refreshed.
  106. Arguments:
  107. Return Value:
  108. --*/
  109. {
  110. if (pChange->eStatus & STATUS_CHANGE_DISCARDNOTED) {
  111. return FALSE;
  112. }
  113. pChange->cRef++;
  114. pChange->eStatus |= STATUS_CHANGE_ACTIVE;
  115. LinkAdd(&pChange->Link, (PLINK*)&pChangeList);
  116. return TRUE;
  117. }
  118. BOOL
  119. ThreadNotify(
  120. LPPRINTHANDLE pPrintHandle)
  121. /*++
  122. Routine Description:
  123. Handles notifying the remote clients of changes.
  124. Arguments:
  125. pPrintHandle - printer that requires notification
  126. Return Value:
  127. TRUE = success, GetLastError() valid on FALSE.
  128. NOTE: Currenly only supports grouping
  129. --*/
  130. {
  131. PCHANGE pChange = pPrintHandle->pChange;
  132. ENTER_THREAD_LIST();
  133. //
  134. // Only add if we're not on the linked list already.
  135. //
  136. if (!(pChange->eStatus & STATUS_CHANGE_ACTIVE)) {
  137. DBGMSG(DBG_NOTIFY, ("TMN: link added 0x%x cRef++ %d\n",
  138. pChange,
  139. pChange->cRef));
  140. //
  141. // Only add ourseleves to the linked list and
  142. // Notify via TMAddJob if we are not on the list.
  143. //
  144. if (LinkChange(pChange))
  145. TMAddJob(&tmStateVar);
  146. } else {
  147. pChange->eStatus |= STATUS_CHANGE_ACTIVE_REQ;
  148. DBGMSG(DBG_NOTIFY, ("TMN: In LL already 0x%x cRef %d\n",
  149. pChange,
  150. pChange->cRef));
  151. }
  152. EXIT_THREAD_LIST();
  153. return TRUE;
  154. }
  155. PJOB
  156. ThreadNotifyNextJob(
  157. PTMSTATEVAR ptmStateVar)
  158. /*++
  159. Routine Description:
  160. Callback to get the next job.
  161. Arguments:
  162. ptmStateVar - ignored.
  163. Return Value:
  164. pJob (pChange)
  165. --*/
  166. {
  167. PCHANGE pChange;
  168. ENTER_THREAD_LIST();
  169. //
  170. // If there are no jobs left, quit.
  171. //
  172. pChange = (PCHANGE)pChangeList;
  173. DBGMSG(DBG_NOTIFY, ("ThreadNotifyNextJob: Removing pChange 0x%x\n",
  174. pChange));
  175. if (pChange) {
  176. LinkDelete(&pChange->Link, (PLINK*)&pChangeList);
  177. }
  178. EXIT_THREAD_LIST();
  179. return (PJOB)pChange;
  180. }
  181. DWORD
  182. ThreadNotifyProcessJob(
  183. PTMSTATEVAR pTMStateVar,
  184. PJOB pJob)
  185. /*++
  186. Routine Description:
  187. Does the actual RPC call to notify the client.
  188. Arguments:
  189. pJob = pChange structure
  190. Return Value:
  191. --*/
  192. {
  193. PCHANGE pChange = (PCHANGE)pJob;
  194. DWORD fdwChangeFlags;
  195. HANDLE hNotifyRemote;
  196. PPRINTER_NOTIFY_INFO pPrinterNotifyInfo;
  197. PPRINTER_NOTIFY_INFO pPrinterNotifyInfoNew;
  198. RPC_V2_UREPLY_PRINTER Reply;
  199. DWORD dwReturn;
  200. DWORD dwResult = 0;
  201. DWORD dwColor;
  202. ENTER_THREAD_LIST();
  203. if (pChange->eStatus & STATUS_CHANGE_CLOSING) {
  204. //
  205. // Abort this job
  206. //
  207. dwReturn = ERROR_INVALID_PARAMETER;
  208. goto Done;
  209. }
  210. fdwChangeFlags = pChange->fdwChangeFlags;
  211. pChange->fdwChangeFlags = 0;
  212. //
  213. // We must save out this copy in case we have some more info
  214. // while we are RPCing. Turn off LL_REQ because we are about
  215. // to process this batch.
  216. //
  217. pPrinterNotifyInfo = pChange->ChangeInfo.pPrinterNotifyInfo;
  218. #if DBG
  219. if( !pPrinterNotifyInfo && !fdwChangeFlags ){
  220. DBGMSG( DBG_WARN, ( "ThreadNotifyProcessJob: No change information\n" ));
  221. }
  222. #endif
  223. pChange->ChangeInfo.pPrinterNotifyInfo = NULL;
  224. pChange->eStatus &= ~STATUS_CHANGE_ACTIVE_REQ;
  225. dwColor = pChange->dwColor;
  226. //
  227. // We were already marked in use when the job was added.
  228. // If another thread wants to delete it, they should OR in
  229. // STATUS_CHANGE_CLOSING, which we will pickup.
  230. //
  231. EXIT_THREAD_LIST();
  232. if (pChange->hNotifyRemote) {
  233. DBGMSG(DBG_NOTIFY, (">> Remoting pChange 0x%x hNotifyRemote 0x%x\n",
  234. pChange,
  235. pChange->hNotifyRemote));
  236. RpcTryExcept {
  237. //
  238. // Note:
  239. //
  240. // We should not be impersonating at this stage since
  241. // we will get a separate session id.
  242. //
  243. if (pPrinterNotifyInfo) {
  244. Reply.pInfo = (PRPC_V2_NOTIFY_INFO)pPrinterNotifyInfo;
  245. //
  246. // Remote case; bind and call the remote router.
  247. //
  248. dwReturn = RpcRouterReplyPrinterEx(
  249. pChange->hNotifyRemote,
  250. dwColor,
  251. fdwChangeFlags,
  252. &dwResult,
  253. REPLY_PRINTER_CHANGE,
  254. Reply);
  255. } else {
  256. dwReturn = RpcRouterReplyPrinter(
  257. pChange->hNotifyRemote,
  258. fdwChangeFlags,
  259. 1,
  260. (PBYTE)&pPrinterNotifyInfo);
  261. }
  262. } RpcExcept(I_RpcExceptionFilter(RpcExceptionCode())) {
  263. dwReturn = RpcExceptionCode();
  264. } RpcEndExcept
  265. } else {
  266. dwReturn = ERROR_INVALID_HANDLE;
  267. DBGMSG(DBG_WARNING, ("ThreadNotifyProcessJob: no hNotifyRemote\n"));
  268. }
  269. if (dwReturn) {
  270. DBGMSG(DBG_WARNING, ("ThreadNotifyProcessJob: RPC error %d\npChange 0x%x, hNotifyRemote 0x%x, hPrinterRemote 0x%x\n",
  271. dwReturn,
  272. pChange,
  273. pChange->hNotifyRemote,
  274. pChange->dwPrinterRemote));
  275. //
  276. // On error, close and retry
  277. //
  278. CloseReplyRemote(pChange->hNotifyRemote);
  279. if (OpenReplyRemote(pChange->pszLocalMachine,
  280. &pChange->hNotifyRemote,
  281. pChange->dwPrinterRemote,
  282. REPLY_TYPE_NOTIFICATION,
  283. 0,
  284. NULL)) {
  285. pChange->hNotifyRemote = NULL;
  286. }
  287. }
  288. Sleep(dwThreadNotifySleep);
  289. ENTER_THREAD_LIST();
  290. pPrinterNotifyInfoNew = pChange->ChangeInfo.pPrinterNotifyInfo;
  291. //
  292. // Keep dwResult only if the color is current.
  293. //
  294. // The avoids the problem when the server sends a discard (RPC1),
  295. // then the client refreshes (RPC2). RPC2 returns first, clearing
  296. // the client's discard bit. The server overflows the new
  297. // buffer. RPC1 completes, and returns discardnoted, which is
  298. // incorrect since it is stale. If the color isn't checked, then
  299. // the server thinks that the discard has been noted when it really
  300. // has not.
  301. //
  302. if (dwColor != pChange->dwColor) {
  303. dwResult = 0;
  304. }
  305. if (pPrinterNotifyInfo) {
  306. //
  307. // Handle different error states from RPC. Each case/default
  308. // block must either update or pChange->ChangeInfo.pPrinterNotifyInfo
  309. // and free the old or the new if both exist.
  310. //
  311. switch (dwReturn) {
  312. case ERROR_SUCCESS:
  313. //
  314. // On success, see if the client saw the info but couldn't
  315. // store it since they overflowed. In this case, we note
  316. // this and never RPC to them again until they refresh.
  317. //
  318. if (dwResult & PRINTER_NOTIFY_INFO_DISCARDNOTED) {
  319. pChange->eStatus |= (STATUS_CHANGE_DISCARDED |
  320. STATUS_CHANGE_DISCARDNOTED);
  321. }
  322. //
  323. // If new buffer allocated, free our old one, else reuse buffer.
  324. //
  325. if (!pPrinterNotifyInfoNew) {
  326. //
  327. // Clear it since we are reusing it.
  328. //
  329. ClearPrinterNotifyInfo(pPrinterNotifyInfo, pChange);
  330. pChange->ChangeInfo.pPrinterNotifyInfo = pPrinterNotifyInfo;
  331. } else {
  332. //
  333. // Free the old one since we are using the new one now.
  334. //
  335. RouterFreePrinterNotifyInfo(pPrinterNotifyInfo);
  336. }
  337. pChange->ChangeInfo.pPrinterNotifyInfo->Flags |= dwResult;
  338. break;
  339. case RPC_S_CALL_FAILED_DNE:
  340. //
  341. // On DNE, keep the notification info. We are guarenteed by
  342. // rpc that this means no part of the call executed.
  343. //
  344. if (pPrinterNotifyInfoNew) {
  345. //
  346. // We already have some info. Merge it in
  347. // with the exiting data.
  348. //
  349. pChange->ChangeInfo.pPrinterNotifyInfo = pPrinterNotifyInfo;
  350. if (pChange->ChangeInfo.pPrintHandle) {
  351. AppendPrinterNotifyInfo(pChange->ChangeInfo.pPrintHandle,
  352. dwColor,
  353. pPrinterNotifyInfoNew);
  354. }
  355. RouterFreePrinterNotifyInfo(pPrinterNotifyInfoNew);
  356. }
  357. break;
  358. default:
  359. //
  360. // Did it succeed? Maybe, maybe not. Fail it by freeing
  361. // the current one if it exists, then clear the current one
  362. // and set the discard bit.
  363. //
  364. pChange->ChangeInfo.pPrinterNotifyInfo = pPrinterNotifyInfo;
  365. ClearPrinterNotifyInfo(pPrinterNotifyInfo, pChange);
  366. SetDiscardPrinterNotifyInfo(pPrinterNotifyInfo, pChange);
  367. //
  368. // Free the new buffer since we are reusing the old one.
  369. //
  370. if (pPrinterNotifyInfoNew) {
  371. RouterFreePrinterNotifyInfo(pPrinterNotifyInfoNew);
  372. }
  373. }
  374. }
  375. pChange->eStatus &= ~STATUS_CHANGE_ACTIVE;
  376. //
  377. // STATUS_CHANGE_ACTIVE_REQ set, then some notifications came in
  378. // while we were out. Check that we actually do have information
  379. // (it might have been sent with the last RPC) and we aren't closing.
  380. //
  381. if ((pChange->eStatus & STATUS_CHANGE_ACTIVE_REQ) &&
  382. NotifyNeeded(pChange) &&
  383. !(pChange->eStatus & STATUS_CHANGE_CLOSING)) {
  384. DBGMSG(DBG_NOTIFY, ("ThreadNotifyProcessJob: delayed link added 0x%x cRef++ %d\n",
  385. pChange,
  386. pChange->cRef));
  387. pChange->eStatus &= ~STATUS_CHANGE_ACTIVE_REQ;
  388. LinkChange(pChange);
  389. //
  390. // No need to call TMAddJob(&tmStateVar) since this
  391. // thread will pickup the job. If there is a job already waiting,
  392. // then that means it already has a thread spawning to pick it up.
  393. //
  394. }
  395. Done:
  396. //
  397. // Mark ourselves no longer in use. If we were in use when someone
  398. // tried to delete the notify, we need to delete it once we're done.
  399. //
  400. pChange->cRef--;
  401. DBGMSG(DBG_NOTIFY, ("ThreadNotifyProcessJob: Done 0x%x cRef-- %d\n",
  402. pChange,
  403. pChange->cRef));
  404. if (pChange->eStatus & STATUS_CHANGE_CLOSING) {
  405. hNotifyRemote = pChange->hNotifyRemote;
  406. pChange->hNotifyRemote = NULL;
  407. //
  408. // Free the Change struct and close the hNotifyRemote
  409. //
  410. FreeChange(pChange);
  411. EXIT_THREAD_LIST();
  412. CloseReplyRemote(hNotifyRemote);
  413. } else {
  414. EXIT_THREAD_LIST();
  415. }
  416. return 0;
  417. }
  418. //
  419. // Usually a macro
  420. //
  421. #ifndef LINKADDFAST
  422. VOID
  423. LinkAdd(
  424. PLINK pLink,
  425. PLINK* ppLinkHead)
  426. /*++
  427. Routine Description:
  428. Adds the item to linked list.
  429. Arguments:
  430. pLink - item to add
  431. ppLinkHead - linked list head pointer
  432. Return Value:
  433. VOID
  434. NOTE: This appends to the tail of the list; the macro must be changed also.
  435. --*/
  436. {
  437. //
  438. // First check if its in the list
  439. //
  440. PLINK pLinkT;
  441. PLINK pLinkLast = NULL;
  442. for(pLinkT=*ppLinkHead; pLinkT; pLinkT=pLinkT->pNext) {
  443. if (pLinkT == pLink) {
  444. DBGMSG(DBG_ERROR, ("LinkAdd: Duplicate link adding!\n"));
  445. }
  446. pLinkLast = pLinkT;
  447. }
  448. if (pLinkLast) {
  449. pLinkLast->pNext = pLink;
  450. } else {
  451. pLink->pNext = *ppLinkHead;
  452. *ppLinkHead = pLink;
  453. }
  454. }
  455. #endif
  456. VOID
  457. LinkDelete(
  458. PLINK pLink,
  459. PLINK* ppLinkHead)
  460. /*++
  461. Routine Description:
  462. Removes item from list
  463. Arguments:
  464. pLink - Item to delete
  465. ppLinkHead - pointer to link head
  466. Return Value:
  467. VOID
  468. --*/
  469. {
  470. PLINK pLink2 = *ppLinkHead;
  471. if (!pLink)
  472. return;
  473. //
  474. // Check head case first
  475. //
  476. if (pLink2 == pLink) {
  477. *ppLinkHead = pLink->pNext;
  478. } else {
  479. //
  480. // Scan list to delete
  481. //
  482. for(;
  483. pLink2;
  484. pLink2=pLink2->pNext) {
  485. if (pLink == pLink2->pNext) {
  486. pLink2->pNext = pLink->pNext;
  487. break;
  488. }
  489. }
  490. }
  491. pLink->pNext = NULL;
  492. return;
  493. }