Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

320 lines
7.8 KiB

  1. /*++
  2. Copyright (c) 1990-2002 Microsoft Corporation
  3. Module Name:
  4. TPmgr.hxx
  5. Abstract:
  6. Declaration of functions used for initializing and using the Thread Pool.
  7. Author:
  8. Ali Naqvi (alinaqvi) 3-May-2002
  9. Revision History:
  10. --*/
  11. #include "precomp.h"
  12. #include "pool.hxx"
  13. #include "TPmgr.hxx"
  14. TThreadPool *g_pThreadPool = NULL;
  15. HRESULT
  16. InitOpnPrnThreadPool(
  17. VOID
  18. )
  19. {
  20. HRESULT hResult = E_FAIL;
  21. g_pThreadPool = new TThreadPool();
  22. hResult = g_pThreadPool ? S_OK : E_OUTOFMEMORY;
  23. return hResult;
  24. }
  25. VOID
  26. DeleteOpnPrnThreadPool(
  27. VOID
  28. )
  29. {
  30. delete g_pThreadPool;
  31. }
  32. /*++
  33. Name:
  34. BindThreadToHandle
  35. Description:
  36. This takes a handle and binds a background thread to it. This thread can
  37. either come from the pool of threads currently wedged against a server, or
  38. if none in the pool satisfy, the tread will be created.
  39. Arguments:
  40. pSpool - The forground handle that we will be binding the thread to.
  41. Return Value:
  42. HRESULT
  43. --*/
  44. HRESULT
  45. BindThreadToHandle(
  46. IN PWSPOOL pSpool
  47. )
  48. {
  49. HRESULT hResult = pSpool ? S_OK : E_INVALIDARG;
  50. HANDLE hThread = NULL;
  51. DWORD IDThread;
  52. //
  53. // This must be called from inside the win32spl CS.
  54. //
  55. SplInSem();
  56. if (SUCCEEDED(hResult))
  57. {
  58. //
  59. // First just try and pick up a thread that is currently wedged against
  60. // a slow server.
  61. //
  62. hResult = g_pThreadPool->UseThread(pSpool->pName, &(pSpool->pThread), pSpool->PrinterDefaults.DesiredAccess);
  63. }
  64. if (hResult == S_FALSE)
  65. {
  66. hResult = g_pThreadPool->CreateThreadEntry(pSpool->pName, &(pSpool->PrinterDefaults), &(pSpool->pThread));
  67. if (SUCCEEDED(hResult))
  68. {
  69. IncThreadCount();
  70. hThread = CreateThread( NULL, 0, (LPTHREAD_START_ROUTINE)RemoteOpenPrinterThread, pSpool->pThread, 0, &IDThread );
  71. hResult = hThread ? S_OK : HRESULT_FROM_WIN32(GetLastError());
  72. }
  73. }
  74. if (SUCCEEDED(hResult))
  75. {
  76. //
  77. // We are an asynchronous thread and we have a foreground handle.
  78. //
  79. pSpool->Status |= WSPOOL_STATUS_ASYNC;
  80. pSpool->pThread->bForegroundClose = TRUE;
  81. }
  82. else
  83. {
  84. if (pSpool && pSpool->pThread)
  85. {
  86. g_pThreadPool->DeleteThreadEntry(pSpool->pThread);
  87. pSpool->pThread = NULL;
  88. DecThreadCount();
  89. }
  90. }
  91. if (hThread)
  92. {
  93. CloseHandle( hThread );
  94. }
  95. return hResult;
  96. }
  97. /*++
  98. Name:
  99. ReturnThreadFromHandle
  100. Description:
  101. This gets a handle and returns the thread to the pool if it is still running.
  102. If it is not running, (The thread status will indicate this), then we close the
  103. RpcHandle ourselves. We clear the foreground handle from the thread. This is a
  104. signal to it that it must close the RPC handle when it terminates.
  105. Arguments:
  106. pSpool - The forground handle that we are detaching the thread from.
  107. Return Value:
  108. HRESULT
  109. --*/
  110. HRESULT
  111. ReturnThreadFromHandle(
  112. IN PWSPOOL pSpool
  113. )
  114. {
  115. HRESULT hResult = pSpool ? S_OK : E_INVALIDARG;
  116. BOOL bCloseRemote = FALSE;
  117. SplOutSem();
  118. if (SUCCEEDED(hResult) && (pSpool->Status & WSPOOL_STATUS_ASYNC))
  119. {
  120. EnterSplSem();
  121. //
  122. // Let the background thread know that it needs to close, if it is running.
  123. //
  124. pSpool->pThread->bForegroundClose = FALSE;
  125. //
  126. // If the thread terminated in the meanwhile, we should close the RPC
  127. // handle in this thread.
  128. //
  129. if (pSpool->pThread->dwStatus == THREAD_STATUS_TERMINATED)
  130. {
  131. bCloseRemote = TRUE;
  132. //
  133. // If the thread actually terminated, either we have a valid RPC handle
  134. // or the handle is signalled that OpenPrinter failed.
  135. //
  136. pSpool->RpcHandle = pSpool->pThread->hRpcHandle;
  137. pSpool->Status &= ~WSPOOL_STATUS_NO_RPC_HANDLE;
  138. //
  139. // We just need to free the thread object since we know we are not returning it.
  140. //
  141. g_pThreadPool->FreeThread(pSpool->pThread);
  142. //
  143. // Don't need to see bForegroundClose because the thread
  144. //
  145. }
  146. else
  147. {
  148. //
  149. // If there was an error in this thread then we don't want to
  150. // return it to the pool.
  151. //
  152. if (pSpool->pThread->dwRpcOpenPrinterError != ERROR_SUCCESS)
  153. {
  154. //
  155. // If there was an OpenPrinter error, there should not be an RPC handle,
  156. // So, neither thread actually needs to close the RPC handle.
  157. //
  158. SPLASSERT(pSpool->pThread->hRpcHandle == NULL);
  159. //
  160. // We know that the background thread is running and we know
  161. // that it doesn't need to close the handle. But it will delete
  162. // the thread entry. We don't put the thread back on the list because
  163. // we know that it is broken.
  164. //
  165. }
  166. else
  167. {
  168. //
  169. // The thread hasn't terminated yet, so we should return it to the thread pool.
  170. //
  171. g_pThreadPool->ReturnThread(pSpool->pThread);
  172. }
  173. //
  174. // We don't need to close, but we also shoudn't use the RPC handle.
  175. // And if the foreground thread has picked it up then we want to
  176. // make sure that it is cleared or we will assert when we close the
  177. // handle.
  178. //
  179. pSpool->RpcHandle = NULL;
  180. pSpool->Status |= WSPOOL_STATUS_NO_RPC_HANDLE;
  181. }
  182. pSpool->pThread = NULL;
  183. LeaveSplSem();
  184. }
  185. SplOutSem();
  186. if (bCloseRemote && pSpool->RpcHandle)
  187. {
  188. hResult = RemoteClosePrinter((HANDLE)pSpool) ? S_OK : GetLastErrorAsHResultAndFail();
  189. }
  190. return hResult;
  191. }
  192. /*++
  193. Name:
  194. BackgroundThreadFinished
  195. Description:
  196. This is called when the background thread is finished. If we are no longer
  197. linked with a foreground handle, we know we have to close the rpc handle. We
  198. mark the thread as terminated to prevent ourselves being put back on the
  199. active list by the foreground thread.
  200. Arguments:
  201. pThread - The thread that is finished
  202. pBackgroundHandle - The background handle that we might have to close.
  203. Return Value:
  204. HRESULT
  205. --*/
  206. HRESULT
  207. BackgroundThreadFinished(
  208. IN OUT PWIN32THREAD *ppThread,
  209. IN OUT PWSPOOL pBackgroundHandle
  210. )
  211. {
  212. HRESULT hr = ppThread && *ppThread ? S_OK : E_INVALIDARG;
  213. SplInSem();
  214. if (SUCCEEDED(hr))
  215. {
  216. PWIN32THREAD pThread = *ppThread;
  217. //
  218. // If we still have a forground handle, then NULL our RPC handle
  219. // so that we won't close the handle ourselves and instead leave it
  220. // up to the foreground thread.
  221. //
  222. pThread->dwStatus = THREAD_STATUS_TERMINATED;
  223. if (pThread->bForegroundClose)
  224. {
  225. if (pBackgroundHandle)
  226. {
  227. pBackgroundHandle->RpcHandle = NULL;
  228. }
  229. }
  230. else
  231. {
  232. //
  233. // We must also delete ourselves from the pool.
  234. //
  235. g_pThreadPool->DeleteThreadEntry(pThread);
  236. }
  237. *ppThread = NULL;
  238. }
  239. return hr;
  240. }