VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/misc/reqpool.cpp@ 99758

Last change on this file since 99758 was 98103, checked in by vboxsync, 2 years ago

Copyright year updates by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 43.6 KB
Line 
1/* $Id: reqpool.cpp 98103 2023-01-17 14:15:46Z vboxsync $ */
2/** @file
3 * IPRT - Request Pool.
4 */
5
6/*
7 * Copyright (C) 2006-2023 Oracle and/or its affiliates.
8 *
9 * This file is part of VirtualBox base platform packages, as
10 * available from https://www.virtualbox.org.
11 *
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation, in version 3 of the
15 * License.
16 *
17 * This program is distributed in the hope that it will be useful, but
18 * WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 * General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, see <https://www.gnu.org/licenses>.
24 *
25 * The contents of this file may alternatively be used under the terms
26 * of the Common Development and Distribution License Version 1.0
27 * (CDDL), a copy of it is provided in the "COPYING.CDDL" file included
28 * in the VirtualBox distribution, in which case the provisions of the
29 * CDDL are applicable instead of those of the GPL.
30 *
31 * You may elect to license modified versions of this file under the
32 * terms and conditions of either the GPL or the CDDL or both.
33 *
34 * SPDX-License-Identifier: GPL-3.0-only OR CDDL-1.0
35 */
36
37
38/*********************************************************************************************************************************
39* Header Files *
40*********************************************************************************************************************************/
41#include <iprt/req.h>
42#include "internal/iprt.h"
43
44#include <iprt/assert.h>
45#include <iprt/asm.h>
46#include <iprt/critsect.h>
47#include <iprt/err.h>
48#include <iprt/list.h>
49#include <iprt/log.h>
50#include <iprt/mem.h>
51#include <iprt/string.h>
52#include <iprt/time.h>
53#include <iprt/semaphore.h>
54#include <iprt/thread.h>
55
56#include "internal/req.h"
57#include "internal/magics.h"
58
59
60/*********************************************************************************************************************************
61* Defined Constants And Macros *
62*********************************************************************************************************************************/
63/** The max number of worker threads. */
64#define RTREQPOOL_MAX_THREADS UINT32_C(16384)
65/** The max number of milliseconds to push back. */
66#define RTREQPOOL_PUSH_BACK_MAX_MS RT_MS_1MIN
67/** The max number of free requests to keep around. */
68#define RTREQPOOL_MAX_FREE_REQUESTS (RTREQPOOL_MAX_THREADS * 2U)
69
70
71/*********************************************************************************************************************************
72* Structures and Typedefs *
73*********************************************************************************************************************************/
74typedef struct RTREQPOOLTHREAD
75{
76 /** Node in the RTREQPOOLINT::IdleThreads list. */
77 RTLISTNODE IdleNode;
78 /** Node in the RTREQPOOLINT::WorkerThreads list. */
79 RTLISTNODE ListNode;
80
81 /** The submit timestamp of the pending request. */
82 uint64_t uPendingNanoTs;
83 /** The submit timestamp of the request processing. */
84 uint64_t uProcessingNanoTs;
85 /** When this CPU went idle the last time. */
86 uint64_t uIdleNanoTs;
87 /** The number of requests processed by this thread. */
88 uint64_t cReqProcessed;
89 /** Total time the requests processed by this thread took to process. */
90 uint64_t cNsTotalReqProcessing;
91 /** Total time the requests processed by this thread had to wait in
92 * the queue before being scheduled. */
93 uint64_t cNsTotalReqQueued;
94 /** The CPU this was scheduled last time we checked. */
95 RTCPUID idLastCpu;
96
97 /** The submitter will put an incoming request here when scheduling an idle
98 * thread. */
99 PRTREQINT volatile pTodoReq;
100 /** The request the thread is currently processing. */
101 PRTREQINT volatile pPendingReq;
102
103 /** The thread handle. */
104 RTTHREAD hThread;
105 /** Nano seconds timestamp representing the birth time of the thread. */
106 uint64_t uBirthNanoTs;
107 /** Pointer to the request thread pool instance the thread is associated
108 * with. */
109 struct RTREQPOOLINT *pPool;
110} RTREQPOOLTHREAD;
111/** Pointer to a worker thread. */
112typedef RTREQPOOLTHREAD *PRTREQPOOLTHREAD;
113
114/**
115 * Request thread pool instance data.
116 */
117typedef struct RTREQPOOLINT
118{
119 /** Magic value (RTREQPOOL_MAGIC). */
120 uint32_t u32Magic;
121 /** The request pool name. */
122 char szName[12];
123
124 /** @name Config
125 * @{ */
126 /** The worker thread type. */
127 RTTHREADTYPE enmThreadType;
128 /** The work thread flags (RTTHREADFLAGS). */
129 uint32_t fThreadFlags;
130 /** The maximum number of worker threads. */
131 uint32_t cMaxThreads;
132 /** The minimum number of worker threads. */
133 uint32_t cMinThreads;
134 /** The number of milliseconds a thread needs to be idle before it is
135 * considered for retirement. */
136 uint32_t cMsMinIdle;
137 /** cMsMinIdle in nano seconds. */
138 uint64_t cNsMinIdle;
139 /** The idle thread sleep interval in milliseconds. */
140 RTMSINTERVAL cMsIdleSleep;
141 /** The number of threads which should be spawned before throttling kicks
142 * in. */
143 uint32_t cThreadsPushBackThreshold;
144 /** The max number of milliseconds to push back a submitter before creating
145 * a new worker thread once the threshold has been reached. */
146 uint32_t cMsMaxPushBack;
147 /** The minimum number of milliseconds to push back a submitter before
148 * creating a new worker thread once the threshold has been reached. */
149 uint32_t cMsMinPushBack;
150 /** The max number of free requests in the recycle LIFO. */
151 uint32_t cMaxFreeRequests;
152 /** @} */
153
154 /** Signaled by terminating worker threads. */
155 RTSEMEVENTMULTI hThreadTermEvt;
156
157 /** Destruction indicator. The worker threads checks in their loop. */
158 bool volatile fDestructing;
159
160 /** The current submitter push back in milliseconds.
161 * This is recalculated when worker threads come and go. */
162 uint32_t cMsCurPushBack;
163 /** The current number of worker threads. */
164 uint32_t cCurThreads;
165 /** Statistics: The total number of threads created. */
166 uint32_t cThreadsCreated;
167 /** Statistics: The timestamp when the last thread was created. */
168 uint64_t uLastThreadCreateNanoTs;
169 /** Linked list of worker threads. */
170 RTLISTANCHOR WorkerThreads;
171
172 /** The number of requests processed and counted in the time totals. */
173 uint64_t cReqProcessed;
174 /** Total time the requests processed by this thread took to process. */
175 uint64_t cNsTotalReqProcessing;
176 /** Total time the requests processed by this thread had to wait in
177 * the queue before being scheduled. */
178 uint64_t cNsTotalReqQueued;
179
180 /** Reference counter. */
181 uint32_t volatile cRefs;
182 /** The number of idle thread or threads in the process of becoming
183 * idle. This is increased before the to-be-idle thread tries to enter
184 * the critical section and add itself to the list. */
185 uint32_t volatile cIdleThreads;
186 /** Linked list of idle threads. */
187 RTLISTANCHOR IdleThreads;
188
189 /** Head of the request FIFO. */
190 PRTREQINT pPendingRequests;
191 /** Where to insert the next request. */
192 PRTREQINT *ppPendingRequests;
193 /** The number of requests currently pending. */
194 uint32_t cCurPendingRequests;
195 /** The number of requests currently being executed. */
196 uint32_t volatile cCurActiveRequests;
197 /** The number of requests submitted. */
198 uint64_t cReqSubmitted;
199 /** The number of cancelled. */
200 uint64_t cReqCancelled;
201
202 /** Head of the request recycling LIFO. */
203 PRTREQINT pFreeRequests;
204 /** The number of requests in the recycling LIFO. This is read without
205 * entering the critical section, thus volatile. */
206 uint32_t volatile cCurFreeRequests;
207
208 /** Critical section serializing access to members of this structure. */
209 RTCRITSECT CritSect;
210
211} RTREQPOOLINT;
212
213
214/**
215 * Used by exiting thread and the pool destruction code to cancel unexpected
216 * requests.
217 *
218 * @param pReq The request.
219 */
220static void rtReqPoolCancelReq(PRTREQINT pReq)
221{
222 pReq->uOwner.hPool = NIL_RTREQPOOL; /* force free */
223 pReq->enmState = RTREQSTATE_COMPLETED;
224 ASMAtomicWriteS32(&pReq->iStatusX, VERR_CANCELLED);
225 if (pReq->hPushBackEvt != NIL_RTSEMEVENTMULTI)
226 RTSemEventMultiSignal(pReq->hPushBackEvt);
227 RTSemEventSignal(pReq->EventSem);
228
229 RTReqRelease(pReq);
230}
231
232
233/**
234 * Recalculate the max pushback interval when adding or removing worker threads.
235 *
236 * @param pPool The pool. cMsCurPushBack will be changed.
237 */
238static void rtReqPoolRecalcPushBack(PRTREQPOOLINT pPool)
239{
240 uint32_t const cMsRange = pPool->cMsMaxPushBack - pPool->cMsMinPushBack;
241 uint32_t const cSteps = pPool->cMaxThreads - pPool->cThreadsPushBackThreshold;
242 uint32_t const iStep = pPool->cCurThreads - pPool->cThreadsPushBackThreshold;
243
244 uint32_t cMsCurPushBack;
245 if (cSteps == 0 /* disabled */)
246 cMsCurPushBack = 0;
247 else if ((cMsRange >> 2) >= cSteps)
248 cMsCurPushBack = cMsRange / cSteps * iStep;
249 else
250 cMsCurPushBack = (uint32_t)( (uint64_t)cMsRange * RT_NS_1MS / cSteps * iStep / RT_NS_1MS );
251 cMsCurPushBack += pPool->cMsMinPushBack;
252
253 pPool->cMsCurPushBack = cMsCurPushBack;
254}
255
256
257
258/**
259 * Performs thread exit.
260 *
261 * @returns Thread termination status code (VINF_SUCCESS).
262 * @param pPool The pool.
263 * @param pThread The thread.
264 * @param fLocked Whether we are inside the critical section
265 * already.
266 */
267static int rtReqPoolThreadExit(PRTREQPOOLINT pPool, PRTREQPOOLTHREAD pThread, bool fLocked)
268{
269 if (!fLocked)
270 RTCritSectEnter(&pPool->CritSect);
271
272 /* Get out of the idle list. */
273 if (!RTListIsEmpty(&pThread->IdleNode))
274 {
275 RTListNodeRemove(&pThread->IdleNode);
276 Assert(pPool->cIdleThreads > 0);
277 ASMAtomicDecU32(&pPool->cIdleThreads);
278 }
279
280 /* Get out of the thread list. */
281 RTListNodeRemove(&pThread->ListNode);
282 Assert(pPool->cCurThreads > 0);
283 pPool->cCurThreads--;
284 rtReqPoolRecalcPushBack(pPool);
285
286 /* This shouldn't happen... */
287 PRTREQINT pReq = pThread->pTodoReq;
288 if (pReq)
289 {
290 AssertFailed();
291 pThread->pTodoReq = NULL;
292 rtReqPoolCancelReq(pReq);
293 }
294
295 /* If we're the last thread terminating, ping the destruction thread before
296 we leave the critical section. */
297 if ( RTListIsEmpty(&pPool->WorkerThreads)
298 && pPool->hThreadTermEvt != NIL_RTSEMEVENT)
299 RTSemEventMultiSignal(pPool->hThreadTermEvt);
300
301 RTCritSectLeave(&pPool->CritSect);
302
303 RTMemFree(pThread);
304 return VINF_SUCCESS;
305}
306
307
308
309/**
310 * Process one request.
311 *
312 * @param pPool The pool.
313 * @param pThread The worker thread.
314 * @param pReq The request to process.
315 */
316static void rtReqPoolThreadProcessRequest(PRTREQPOOLINT pPool, PRTREQPOOLTHREAD pThread, PRTREQINT pReq)
317{
318 /*
319 * Update thread state.
320 */
321 pThread->uProcessingNanoTs = RTTimeNanoTS();
322 pThread->uPendingNanoTs = pReq->uSubmitNanoTs;
323 pThread->pPendingReq = pReq;
324 ASMAtomicIncU32(&pPool->cCurActiveRequests);
325 Assert(pReq->u32Magic == RTREQ_MAGIC);
326
327 /*
328 * Do the actual processing.
329 */
330 rtReqProcessOne(pReq);
331
332 /*
333 * Update thread statistics and state.
334 */
335 ASMAtomicDecU32(&pPool->cCurActiveRequests);
336 pThread->pPendingReq = NULL;
337 uint64_t const uNsTsEnd = RTTimeNanoTS();
338 pThread->cNsTotalReqProcessing += uNsTsEnd - pThread->uProcessingNanoTs;
339 pThread->cNsTotalReqQueued += pThread->uProcessingNanoTs - pThread->uPendingNanoTs;
340 pThread->cReqProcessed++;
341}
342
343
344
345/**
346 * The Worker Thread Procedure.
347 *
348 * @returns VINF_SUCCESS.
349 * @param hThreadSelf The thread handle (unused).
350 * @param pvArg Pointer to the thread data.
351 */
352static DECLCALLBACK(int) rtReqPoolThreadProc(RTTHREAD hThreadSelf, void *pvArg)
353{
354 PRTREQPOOLTHREAD pThread = (PRTREQPOOLTHREAD)pvArg;
355 PRTREQPOOLINT pPool = pThread->pPool;
356
357 /*
358 * The work loop.
359 */
360 uint64_t cReqPrevProcessedIdle = UINT64_MAX;
361 uint64_t cReqPrevProcessedStat = 0;
362 uint64_t cNsPrevTotalReqProcessing = 0;
363 uint64_t cNsPrevTotalReqQueued = 0;
364 while (!pPool->fDestructing)
365 {
366 /*
367 * Process pending work.
368 */
369
370 /* Check if anything is scheduled directly to us. */
371 PRTREQINT pReq = ASMAtomicXchgPtrT(&pThread->pTodoReq, NULL, PRTREQINT);
372 if (pReq)
373 {
374 Assert(RTListIsEmpty(&pThread->IdleNode)); /* Must not be in the idle list. */
375 rtReqPoolThreadProcessRequest(pPool, pThread, pReq);
376 continue;
377 }
378
379 ASMAtomicIncU32(&pPool->cIdleThreads);
380 RTCritSectEnter(&pPool->CritSect);
381
382 /* Update the global statistics. */
383 if (cReqPrevProcessedStat != pThread->cReqProcessed)
384 {
385 pPool->cReqProcessed += pThread->cReqProcessed - cReqPrevProcessedStat;
386 cReqPrevProcessedStat = pThread->cReqProcessed;
387 pPool->cNsTotalReqProcessing += pThread->cNsTotalReqProcessing - cNsPrevTotalReqProcessing;
388 cNsPrevTotalReqProcessing = pThread->cNsTotalReqProcessing;
389 pPool->cNsTotalReqQueued += pThread->cNsTotalReqQueued - cNsPrevTotalReqQueued;
390 cNsPrevTotalReqQueued = pThread->cNsTotalReqQueued;
391 }
392
393 /* Recheck the todo request pointer after entering the critsect. */
394 pReq = ASMAtomicXchgPtrT(&pThread->pTodoReq, NULL, PRTREQINT);
395 if (pReq)
396 {
397 Assert(RTListIsEmpty(&pThread->IdleNode)); /* Must not be in the idle list. */
398 RTCritSectLeave(&pPool->CritSect);
399
400 rtReqPoolThreadProcessRequest(pPool, pThread, pReq);
401 continue;
402 }
403
404 /* Any pending requests in the queue? */
405 pReq = pPool->pPendingRequests;
406 if (pReq)
407 {
408 pPool->pPendingRequests = pReq->pNext;
409 if (pReq->pNext == NULL)
410 pPool->ppPendingRequests = &pPool->pPendingRequests;
411 Assert(pPool->cCurPendingRequests > 0);
412 pPool->cCurPendingRequests--;
413
414 /* Un-idle ourselves and process the request. */
415 if (!RTListIsEmpty(&pThread->IdleNode))
416 {
417 RTListNodeRemove(&pThread->IdleNode);
418 RTListInit(&pThread->IdleNode);
419 ASMAtomicDecU32(&pPool->cIdleThreads);
420 }
421 ASMAtomicDecU32(&pPool->cIdleThreads);
422 RTCritSectLeave(&pPool->CritSect);
423
424 rtReqPoolThreadProcessRequest(pPool, pThread, pReq);
425 continue;
426 }
427
428 /*
429 * Nothing to do, go idle.
430 */
431 if (cReqPrevProcessedIdle != pThread->cReqProcessed)
432 {
433 cReqPrevProcessedIdle = pThread->cReqProcessed;
434 pThread->uIdleNanoTs = RTTimeNanoTS();
435 }
436 else if (pPool->cCurThreads > pPool->cMinThreads)
437 {
438 uint64_t cNsIdle = RTTimeNanoTS() - pThread->uIdleNanoTs;
439 if (cNsIdle >= pPool->cNsMinIdle)
440 return rtReqPoolThreadExit(pPool, pThread, true /*fLocked*/);
441 }
442
443 if (RTListIsEmpty(&pThread->IdleNode))
444 RTListPrepend(&pPool->IdleThreads, &pThread->IdleNode);
445 else
446 ASMAtomicDecU32(&pPool->cIdleThreads);
447 RTThreadUserReset(hThreadSelf);
448 uint32_t const cMsSleep = pPool->cMsIdleSleep;
449
450 RTCritSectLeave(&pPool->CritSect);
451
452 RTThreadUserWait(hThreadSelf, cMsSleep);
453 }
454
455 return rtReqPoolThreadExit(pPool, pThread, false /*fLocked*/);
456}
457
458
459/**
460 * Create a new worker thread.
461 *
462 * @param pPool The pool needing new worker thread.
463 * @remarks Caller owns the critical section
464 */
465static void rtReqPoolCreateNewWorker(RTREQPOOL pPool)
466{
467 PRTREQPOOLTHREAD pThread = (PRTREQPOOLTHREAD)RTMemAllocZ(sizeof(RTREQPOOLTHREAD));
468 if (!pThread)
469 return;
470
471 pThread->uBirthNanoTs = RTTimeNanoTS();
472 pThread->pPool = pPool;
473 pThread->idLastCpu = NIL_RTCPUID;
474 pThread->hThread = NIL_RTTHREAD;
475 RTListInit(&pThread->IdleNode);
476 RTListAppend(&pPool->WorkerThreads, &pThread->ListNode);
477 pPool->cCurThreads++;
478 pPool->cThreadsCreated++;
479
480 int rc = RTThreadCreateF(&pThread->hThread, rtReqPoolThreadProc, pThread, 0 /*default stack size*/,
481 pPool->enmThreadType, pPool->fThreadFlags, "%s%02u", pPool->szName, pPool->cThreadsCreated);
482 if (RT_SUCCESS(rc))
483 pPool->uLastThreadCreateNanoTs = pThread->uBirthNanoTs;
484 else
485 {
486 pPool->cCurThreads--;
487 RTListNodeRemove(&pThread->ListNode);
488 RTMemFree(pThread);
489 }
490}
491
492
493/**
494 * Repel the submitter, giving the worker threads a chance to process the
495 * incoming request.
496 *
497 * @returns Success if a worker picked up the request, failure if not. The
498 * critical section has been left on success, while we'll be inside it
499 * on failure.
500 * @param pPool The pool.
501 * @param pReq The incoming request.
502 */
503static int rtReqPoolPushBack(PRTREQPOOLINT pPool, PRTREQINT pReq)
504{
505 /*
506 * Lazily create the push back semaphore that we'll be blociing on.
507 */
508 int rc;
509 RTSEMEVENTMULTI hEvt = pReq->hPushBackEvt;
510 if (hEvt == NIL_RTSEMEVENTMULTI)
511 {
512 rc = RTSemEventMultiCreate(&hEvt);
513 if (RT_FAILURE(rc))
514 return rc;
515 pReq->hPushBackEvt = hEvt;
516 }
517
518 /*
519 * Prepare the request and semaphore.
520 */
521 uint32_t const cMsTimeout = pPool->cMsCurPushBack;
522 pReq->fSignalPushBack = true;
523 RTReqRetain(pReq);
524 RTSemEventMultiReset(hEvt);
525
526 RTCritSectLeave(&pPool->CritSect);
527
528 /*
529 * Block.
530 */
531 rc = RTSemEventMultiWait(hEvt, cMsTimeout);
532 if (RT_FAILURE(rc))
533 {
534 AssertMsg(rc == VERR_TIMEOUT, ("%Rrc\n", rc));
535 RTCritSectEnter(&pPool->CritSect);
536 }
537 RTReqRelease(pReq);
538 return rc;
539}
540
541
542
543DECLHIDDEN(void) rtReqPoolSubmit(PRTREQPOOLINT pPool, PRTREQINT pReq)
544{
545 RTCritSectEnter(&pPool->CritSect);
546
547 pPool->cReqSubmitted++;
548
549 /*
550 * Try schedule the request to a thread that's currently idle.
551 */
552 PRTREQPOOLTHREAD pThread = RTListGetFirst(&pPool->IdleThreads, RTREQPOOLTHREAD, IdleNode);
553 if (pThread)
554 {
555 /** @todo CPU affinity??? */
556 ASMAtomicWritePtr(&pThread->pTodoReq, pReq);
557
558 RTListNodeRemove(&pThread->IdleNode);
559 RTListInit(&pThread->IdleNode);
560 ASMAtomicDecU32(&pPool->cIdleThreads);
561
562 RTThreadUserSignal(pThread->hThread);
563
564 RTCritSectLeave(&pPool->CritSect);
565 return;
566 }
567 Assert(RTListIsEmpty(&pPool->IdleThreads));
568
569 /*
570 * Put the request in the pending queue.
571 */
572 pReq->pNext = NULL;
573 *pPool->ppPendingRequests = pReq;
574 pPool->ppPendingRequests = (PRTREQINT *)&pReq->pNext;
575 pPool->cCurPendingRequests++;
576
577 /*
578 * If there is an incoming worker thread already or we've reached the
579 * maximum number of worker threads, we're done.
580 */
581 if ( pPool->cIdleThreads > 0
582 || pPool->cCurThreads >= pPool->cMaxThreads)
583 {
584 RTCritSectLeave(&pPool->CritSect);
585 return;
586 }
587
588 /*
589 * Push back before creating a new worker thread.
590 */
591 if ( pPool->cCurThreads > pPool->cThreadsPushBackThreshold
592 && (RTTimeNanoTS() - pReq->uSubmitNanoTs) / RT_NS_1MS >= pPool->cMsCurPushBack )
593 {
594 int rc = rtReqPoolPushBack(pPool, pReq);
595 if (RT_SUCCESS(rc))
596 return;
597 }
598
599 /*
600 * Create a new thread for processing the request.
601 * For simplicity, we don't bother leaving the critical section while doing so.
602 */
603 rtReqPoolCreateNewWorker(pPool);
604
605 RTCritSectLeave(&pPool->CritSect);
606 return;
607}
608
609
610/**
611 * Worker for RTReqCancel that looks for the request in the pending list and
612 * completes it if found there.
613 *
614 * @param pPool The request thread pool.
615 * @param pReq The request.
616 */
617DECLHIDDEN(void) rtReqPoolCancel(PRTREQPOOLINT pPool, PRTREQINT pReq)
618{
619 RTCritSectEnter(&pPool->CritSect);
620
621 pPool->cReqCancelled++;
622
623 /*
624 * Check if the request is in the pending list.
625 */
626 PRTREQINT pPrev = NULL;
627 PRTREQINT pCur = pPool->pPendingRequests;
628 while (pCur)
629 if (pCur != pReq)
630 {
631 pPrev = pCur;
632 pCur = pCur->pNext;
633 }
634 else
635 {
636 /*
637 * Unlink it and process it.
638 */
639 if (!pPrev)
640 {
641 pPool->pPendingRequests = pReq->pNext;
642 if (!pReq->pNext)
643 pPool->ppPendingRequests = &pPool->pPendingRequests;
644 }
645 else
646 {
647 pPrev->pNext = pReq->pNext;
648 if (!pReq->pNext)
649 pPool->ppPendingRequests = (PRTREQINT *)&pPrev->pNext;
650 }
651 Assert(pPool->cCurPendingRequests > 0);
652 pPool->cCurPendingRequests--;
653
654 rtReqProcessOne(pReq);
655 break;
656 }
657
658 RTCritSectLeave(&pPool->CritSect);
659 return;
660}
661
662
663/**
664 * Frees a requst.
665 *
666 * @returns true if recycled, false if not.
667 * @param pPool The request thread pool.
668 * @param pReq The request.
669 */
670DECLHIDDEN(bool) rtReqPoolRecycle(PRTREQPOOLINT pPool, PRTREQINT pReq)
671{
672 if ( pPool
673 && ASMAtomicReadU32(&pPool->cCurFreeRequests) < pPool->cMaxFreeRequests)
674 {
675 RTCritSectEnter(&pPool->CritSect);
676 if (pPool->cCurFreeRequests < pPool->cMaxFreeRequests)
677 {
678 pReq->pNext = pPool->pFreeRequests;
679 pPool->pFreeRequests = pReq;
680 ASMAtomicIncU32(&pPool->cCurFreeRequests);
681
682 RTCritSectLeave(&pPool->CritSect);
683 return true;
684 }
685
686 RTCritSectLeave(&pPool->CritSect);
687 }
688 return false;
689}
690
691
692RTDECL(int) RTReqPoolCreate(uint32_t cMaxThreads, RTMSINTERVAL cMsMinIdle,
693 uint32_t cThreadsPushBackThreshold, uint32_t cMsMaxPushBack,
694 const char *pszName, PRTREQPOOL phPool)
695{
696 /*
697 * Validate and massage the config.
698 */
699 if (cMaxThreads == UINT32_MAX)
700 cMaxThreads = RTREQPOOL_MAX_THREADS;
701 AssertMsgReturn(cMaxThreads > 0 && cMaxThreads <= RTREQPOOL_MAX_THREADS, ("%u\n", cMaxThreads), VERR_OUT_OF_RANGE);
702 uint32_t const cMinThreads = cMaxThreads > 2 ? 2 : cMaxThreads - 1;
703
704 if (cThreadsPushBackThreshold == 0)
705 cThreadsPushBackThreshold = cMinThreads;
706 else if (cThreadsPushBackThreshold == UINT32_MAX)
707 cThreadsPushBackThreshold = cMaxThreads;
708 AssertMsgReturn(cThreadsPushBackThreshold <= cMaxThreads, ("%u/%u\n", cThreadsPushBackThreshold, cMaxThreads), VERR_OUT_OF_RANGE);
709
710 if (cMsMaxPushBack == UINT32_MAX)
711 cMsMaxPushBack = RTREQPOOL_PUSH_BACK_MAX_MS;
712 AssertMsgReturn(cMsMaxPushBack <= RTREQPOOL_PUSH_BACK_MAX_MS, ("%llu\n", cMsMaxPushBack), VERR_OUT_OF_RANGE);
713 uint32_t const cMsMinPushBack = cMsMaxPushBack >= 200 ? 100 : cMsMaxPushBack / 2;
714
715 AssertPtrReturn(pszName, VERR_INVALID_POINTER);
716 size_t cchName = strlen(pszName);
717 AssertReturn(cchName > 0, VERR_INVALID_PARAMETER);
718 Assert(cchName <= 10);
719
720 AssertPtrReturn(phPool, VERR_INVALID_POINTER);
721
722 /*
723 * Create and initialize the pool.
724 */
725 PRTREQPOOLINT pPool = (PRTREQPOOLINT)RTMemAlloc(sizeof(*pPool));
726 if (!pPool)
727 return VERR_NO_MEMORY;
728
729 pPool->u32Magic = RTREQPOOL_MAGIC;
730 RTStrCopy(pPool->szName, sizeof(pPool->szName), pszName);
731
732 pPool->enmThreadType = RTTHREADTYPE_DEFAULT;
733 pPool->fThreadFlags = 0;
734 pPool->cMaxThreads = cMaxThreads;
735 pPool->cMinThreads = cMinThreads;
736 pPool->cMsMinIdle = cMsMinIdle == RT_INDEFINITE_WAIT || cMsMinIdle >= UINT32_MAX ? UINT32_MAX : cMsMinIdle;
737 pPool->cNsMinIdle = pPool->cMsMinIdle == UINT32_MAX ? UINT64_MAX : cMsMinIdle * RT_NS_1MS_64;
738 pPool->cMsIdleSleep = pPool->cMsMinIdle == UINT32_MAX ? RT_INDEFINITE_WAIT : RT_MAX(RT_MS_1SEC, pPool->cMsMinIdle);
739 pPool->cThreadsPushBackThreshold = cThreadsPushBackThreshold;
740 pPool->cMsMaxPushBack = cMsMaxPushBack;
741 pPool->cMsMinPushBack = cMsMinPushBack;
742 pPool->cMaxFreeRequests = cMaxThreads * 2;
743 pPool->hThreadTermEvt = NIL_RTSEMEVENTMULTI;
744 pPool->fDestructing = false;
745 pPool->cMsCurPushBack = 0;
746 pPool->cCurThreads = 0;
747 pPool->cThreadsCreated = 0;
748 pPool->uLastThreadCreateNanoTs = 0;
749 RTListInit(&pPool->WorkerThreads);
750 pPool->cReqProcessed = 0;
751 pPool->cNsTotalReqProcessing= 0;
752 pPool->cNsTotalReqQueued = 0;
753 pPool->cRefs = 1;
754 pPool->cIdleThreads = 0;
755 RTListInit(&pPool->IdleThreads);
756 pPool->pPendingRequests = NULL;
757 pPool->ppPendingRequests = &pPool->pPendingRequests;
758 pPool->cCurPendingRequests = 0;
759 pPool->cCurActiveRequests = 0;
760 pPool->cReqSubmitted = 0;
761 pPool->cReqCancelled = 0;
762 pPool->pFreeRequests = NULL;
763 pPool->cCurFreeRequests = 0;
764
765 int rc = RTSemEventMultiCreate(&pPool->hThreadTermEvt);
766 if (RT_SUCCESS(rc))
767 {
768 rc = RTCritSectInit(&pPool->CritSect);
769 if (RT_SUCCESS(rc))
770 {
771 *phPool = pPool;
772 return VINF_SUCCESS;
773 }
774
775 RTSemEventMultiDestroy(pPool->hThreadTermEvt);
776 }
777 pPool->u32Magic = RTREQPOOL_MAGIC_DEAD;
778 RTMemFree(pPool);
779 return rc;
780}
781
782
783
784RTDECL(int) RTReqPoolSetCfgVar(RTREQPOOL hPool, RTREQPOOLCFGVAR enmVar, uint64_t uValue)
785{
786 PRTREQPOOLINT pPool = hPool;
787 AssertPtrReturn(pPool, VERR_INVALID_HANDLE);
788 AssertReturn(pPool->u32Magic == RTREQPOOL_MAGIC, VERR_INVALID_HANDLE);
789 AssertReturn(enmVar > RTREQPOOLCFGVAR_INVALID && enmVar < RTREQPOOLCFGVAR_END, VERR_INVALID_PARAMETER);
790
791 RTCritSectEnter(&pPool->CritSect);
792
793 bool fWakeUpIdleThreads = false;
794 int rc = VINF_SUCCESS;
795 switch (enmVar)
796 {
797 case RTREQPOOLCFGVAR_THREAD_TYPE:
798 AssertMsgBreakStmt(uValue > (uint64_t)RTTHREADTYPE_INVALID && uValue < (uint64_t)RTTHREADTYPE_END,
799 ("%llu\n", uValue), rc = VERR_OUT_OF_RANGE);
800
801 pPool->enmThreadType = (RTTHREADTYPE)uValue;
802 break;
803
804 case RTREQPOOLCFGVAR_THREAD_FLAGS:
805 AssertMsgBreakStmt(!(uValue & ~(uint64_t)RTTHREADFLAGS_MASK) && !(uValue & RTTHREADFLAGS_WAITABLE),
806 ("%#llx\n", uValue), rc = VERR_INVALID_FLAGS);
807
808 pPool->fThreadFlags = (uint32_t)uValue;
809 break;
810
811 case RTREQPOOLCFGVAR_MIN_THREADS:
812 AssertMsgBreakStmt(uValue <= RTREQPOOL_MAX_THREADS, ("%llu\n", uValue), rc = VERR_OUT_OF_RANGE);
813 fWakeUpIdleThreads = pPool->cMinThreads > (uint32_t)uValue;
814 pPool->cMinThreads = (uint32_t)uValue;
815 if (pPool->cMinThreads > pPool->cMaxThreads)
816 pPool->cMaxThreads = pPool->cMinThreads;
817 if ( pPool->cThreadsPushBackThreshold < pPool->cMinThreads
818 || pPool->cThreadsPushBackThreshold > pPool->cMaxThreads)
819 pPool->cThreadsPushBackThreshold = pPool->cMinThreads + (pPool->cMaxThreads - pPool->cMinThreads) / 2;
820 rtReqPoolRecalcPushBack(pPool);
821 break;
822
823 case RTREQPOOLCFGVAR_MAX_THREADS:
824 AssertMsgBreakStmt(uValue <= RTREQPOOL_MAX_THREADS && uValue >= 1, ("%llu\n", uValue), rc = VERR_OUT_OF_RANGE);
825 pPool->cMaxThreads = (uint32_t)uValue;
826 if (pPool->cMaxThreads < pPool->cMinThreads)
827 {
828 pPool->cMinThreads = pPool->cMaxThreads;
829 fWakeUpIdleThreads = true;
830 }
831 if (pPool->cMaxThreads < pPool->cThreadsPushBackThreshold)
832 pPool->cThreadsPushBackThreshold = pPool->cMinThreads + (pPool->cMaxThreads - pPool->cMinThreads) / 2;
833 rtReqPoolRecalcPushBack(pPool);
834 break;
835
836 case RTREQPOOLCFGVAR_MS_MIN_IDLE:
837 AssertMsgBreakStmt(uValue < UINT32_MAX || uValue == RT_INDEFINITE_WAIT, ("%llu\n", uValue), rc = VERR_OUT_OF_RANGE);
838 if (uValue < UINT32_MAX && uValue != RT_INDEFINITE_WAIT)
839 {
840 fWakeUpIdleThreads = pPool->cMsMinIdle != (uint32_t)uValue;
841 pPool->cMsMinIdle = (uint32_t)uValue;
842 pPool->cNsMinIdle = pPool->cMsMinIdle * RT_NS_1MS_64;
843 if (pPool->cMsIdleSleep > pPool->cMsMinIdle)
844 pPool->cMsIdleSleep = RT_MAX(RT_MS_1SEC, pPool->cMsMinIdle);
845 }
846 else
847 {
848 pPool->cMsMinIdle = UINT32_MAX;
849 pPool->cNsMinIdle = UINT64_MAX;
850 pPool->cMsIdleSleep = RT_INDEFINITE_WAIT;
851 }
852 break;
853
854 case RTREQPOOLCFGVAR_MS_IDLE_SLEEP:
855 AssertMsgBreakStmt(uValue <= RT_INDEFINITE_WAIT, ("%llu\n", uValue), rc = VERR_OUT_OF_RANGE);
856 fWakeUpIdleThreads = pPool->cMsMinIdle > (RTMSINTERVAL)uValue;
857 pPool->cMsIdleSleep = (RTMSINTERVAL)uValue;
858 if (pPool->cMsIdleSleep == RT_INDEFINITE_WAIT)
859 {
860 pPool->cMsMinIdle = UINT32_MAX;
861 pPool->cNsMinIdle = UINT64_MAX;
862 }
863 break;
864
865 case RTREQPOOLCFGVAR_PUSH_BACK_THRESHOLD:
866 if (uValue == UINT64_MAX)
867 pPool->cThreadsPushBackThreshold = pPool->cMaxThreads;
868 else if (uValue == 0)
869 pPool->cThreadsPushBackThreshold = pPool->cMinThreads;
870 else
871 {
872 AssertMsgBreakStmt(uValue <= pPool->cMaxThreads, ("%llu\n", uValue), rc = VERR_OUT_OF_RANGE);
873 AssertMsgBreakStmt(uValue >= pPool->cMinThreads, ("%llu\n", uValue), rc = VERR_OUT_OF_RANGE);
874 pPool->cThreadsPushBackThreshold = (uint32_t)uValue;
875 }
876 break;
877
878 case RTREQPOOLCFGVAR_PUSH_BACK_MIN_MS:
879 if (uValue == UINT32_MAX || uValue == UINT64_MAX)
880 uValue = RTREQPOOL_PUSH_BACK_MAX_MS;
881 else
882 AssertMsgBreakStmt(uValue <= RTREQPOOL_PUSH_BACK_MAX_MS, ("%llu\n", uValue), rc = VERR_OUT_OF_RANGE);
883 pPool->cMsMinPushBack = (uint32_t)uValue;
884 if (pPool->cMsMaxPushBack < pPool->cMsMinPushBack)
885 pPool->cMsMaxPushBack = pPool->cMsMinPushBack;
886 rtReqPoolRecalcPushBack(pPool);
887 break;
888
889 case RTREQPOOLCFGVAR_PUSH_BACK_MAX_MS:
890 if (uValue == UINT32_MAX || uValue == UINT64_MAX)
891 uValue = RTREQPOOL_PUSH_BACK_MAX_MS;
892 else
893 AssertMsgBreakStmt(uValue <= RTREQPOOL_PUSH_BACK_MAX_MS, ("%llu\n", uValue), rc = VERR_OUT_OF_RANGE);
894 pPool->cMsMaxPushBack = (uint32_t)uValue;
895 if (pPool->cMsMinPushBack < pPool->cMsMaxPushBack)
896 pPool->cMsMinPushBack = pPool->cMsMaxPushBack;
897 rtReqPoolRecalcPushBack(pPool);
898 break;
899
900 case RTREQPOOLCFGVAR_MAX_FREE_REQUESTS:
901 if (uValue == UINT64_MAX)
902 {
903 pPool->cMaxFreeRequests = pPool->cMaxThreads * 2;
904 if (pPool->cMaxFreeRequests < 16)
905 pPool->cMaxFreeRequests = 16;
906 }
907 else
908 {
909 AssertMsgBreakStmt(uValue <= RTREQPOOL_MAX_FREE_REQUESTS, ("%llu\n", uValue), rc = VERR_OUT_OF_RANGE);
910 pPool->cMaxFreeRequests = (uint32_t)uValue;
911 }
912
913 while (pPool->cCurFreeRequests > pPool->cMaxFreeRequests)
914 {
915 PRTREQINT pReq = pPool->pFreeRequests;
916 pPool->pFreeRequests = pReq->pNext;
917 ASMAtomicDecU32(&pPool->cCurFreeRequests);
918 rtReqFreeIt(pReq);
919 }
920 break;
921
922 default:
923 AssertFailed();
924 rc = VERR_IPE_NOT_REACHED_DEFAULT_CASE;
925 }
926
927 /* Wake up all idle threads if required. */
928 if (fWakeUpIdleThreads)
929 {
930 Assert(rc == VINF_SUCCESS);
931 PRTREQPOOLTHREAD pThread;
932 RTListForEach(&pPool->WorkerThreads, pThread, RTREQPOOLTHREAD, ListNode)
933 {
934 RTThreadUserSignal(pThread->hThread);
935 }
936 }
937
938 RTCritSectLeave(&pPool->CritSect);
939
940 return rc;
941}
942RT_EXPORT_SYMBOL(RTReqPoolSetCfgVar);
943
944
945RTDECL(uint64_t) RTReqPoolGetCfgVar(RTREQPOOL hPool, RTREQPOOLCFGVAR enmVar)
946{
947 PRTREQPOOLINT pPool = hPool;
948 AssertPtrReturn(pPool, UINT64_MAX);
949 AssertReturn(pPool->u32Magic == RTREQPOOL_MAGIC, UINT64_MAX);
950 AssertReturn(enmVar > RTREQPOOLCFGVAR_INVALID && enmVar < RTREQPOOLCFGVAR_END, UINT64_MAX);
951
952 RTCritSectEnter(&pPool->CritSect);
953
954 uint64_t u64;
955 switch (enmVar)
956 {
957 case RTREQPOOLCFGVAR_THREAD_TYPE:
958 u64 = pPool->enmThreadType;
959 break;
960
961 case RTREQPOOLCFGVAR_THREAD_FLAGS:
962 u64 = pPool->fThreadFlags;
963 break;
964
965 case RTREQPOOLCFGVAR_MIN_THREADS:
966 u64 = pPool->cMinThreads;
967 break;
968
969 case RTREQPOOLCFGVAR_MAX_THREADS:
970 u64 = pPool->cMaxThreads;
971 break;
972
973 case RTREQPOOLCFGVAR_MS_MIN_IDLE:
974 u64 = pPool->cMsMinIdle;
975 break;
976
977 case RTREQPOOLCFGVAR_MS_IDLE_SLEEP:
978 u64 = pPool->cMsIdleSleep;
979 break;
980
981 case RTREQPOOLCFGVAR_PUSH_BACK_THRESHOLD:
982 u64 = pPool->cThreadsPushBackThreshold;
983 break;
984
985 case RTREQPOOLCFGVAR_PUSH_BACK_MIN_MS:
986 u64 = pPool->cMsMinPushBack;
987 break;
988
989 case RTREQPOOLCFGVAR_PUSH_BACK_MAX_MS:
990 u64 = pPool->cMsMaxPushBack;
991 break;
992
993 case RTREQPOOLCFGVAR_MAX_FREE_REQUESTS:
994 u64 = pPool->cMaxFreeRequests;
995 break;
996
997 default:
998 AssertFailed();
999 u64 = UINT64_MAX;
1000 break;
1001 }
1002
1003 RTCritSectLeave(&pPool->CritSect);
1004
1005 return u64;
1006}
1007RT_EXPORT_SYMBOL(RTReqGetQueryCfgVar);
1008
1009
1010RTDECL(uint64_t) RTReqPoolGetStat(RTREQPOOL hPool, RTREQPOOLSTAT enmStat)
1011{
1012 PRTREQPOOLINT pPool = hPool;
1013 AssertPtrReturn(pPool, UINT64_MAX);
1014 AssertReturn(pPool->u32Magic == RTREQPOOL_MAGIC, UINT64_MAX);
1015 AssertReturn(enmStat > RTREQPOOLSTAT_INVALID && enmStat < RTREQPOOLSTAT_END, UINT64_MAX);
1016
1017 RTCritSectEnter(&pPool->CritSect);
1018
1019 uint64_t u64;
1020 switch (enmStat)
1021 {
1022 case RTREQPOOLSTAT_THREADS: u64 = pPool->cCurThreads; break;
1023 case RTREQPOOLSTAT_THREADS_CREATED: u64 = pPool->cThreadsCreated; break;
1024 case RTREQPOOLSTAT_REQUESTS_PROCESSED: u64 = pPool->cReqProcessed; break;
1025 case RTREQPOOLSTAT_REQUESTS_SUBMITTED: u64 = pPool->cReqSubmitted; break;
1026 case RTREQPOOLSTAT_REQUESTS_CANCELLED: u64 = pPool->cReqCancelled; break;
1027 case RTREQPOOLSTAT_REQUESTS_PENDING: u64 = pPool->cCurPendingRequests; break;
1028 case RTREQPOOLSTAT_REQUESTS_ACTIVE: u64 = pPool->cCurActiveRequests; break;
1029 case RTREQPOOLSTAT_REQUESTS_FREE: u64 = pPool->cCurFreeRequests; break;
1030 case RTREQPOOLSTAT_NS_TOTAL_REQ_PROCESSING: u64 = pPool->cNsTotalReqProcessing; break;
1031 case RTREQPOOLSTAT_NS_TOTAL_REQ_QUEUED: u64 = pPool->cNsTotalReqQueued; break;
1032 case RTREQPOOLSTAT_NS_AVERAGE_REQ_PROCESSING: u64 = pPool->cNsTotalReqProcessing / RT_MAX(pPool->cReqProcessed, 1); break;
1033 case RTREQPOOLSTAT_NS_AVERAGE_REQ_QUEUED: u64 = pPool->cNsTotalReqQueued / RT_MAX(pPool->cReqProcessed, 1); break;
1034 default:
1035 AssertFailed();
1036 u64 = UINT64_MAX;
1037 break;
1038 }
1039
1040 RTCritSectLeave(&pPool->CritSect);
1041
1042 return u64;
1043}
1044RT_EXPORT_SYMBOL(RTReqPoolGetStat);
1045
1046
1047RTDECL(uint32_t) RTReqPoolRetain(RTREQPOOL hPool)
1048{
1049 PRTREQPOOLINT pPool = hPool;
1050 AssertPtrReturn(pPool, UINT32_MAX);
1051 AssertReturn(pPool->u32Magic == RTREQPOOL_MAGIC, UINT32_MAX);
1052
1053 return ASMAtomicIncU32(&pPool->cRefs);
1054}
1055RT_EXPORT_SYMBOL(RTReqPoolRetain);
1056
1057
1058RTDECL(uint32_t) RTReqPoolRelease(RTREQPOOL hPool)
1059{
1060 /*
1061 * Ignore NULL and validate the request.
1062 */
1063 if (!hPool)
1064 return 0;
1065 PRTREQPOOLINT pPool = hPool;
1066 AssertPtrReturn(pPool, UINT32_MAX);
1067 AssertReturn(pPool->u32Magic == RTREQPOOL_MAGIC, UINT32_MAX);
1068
1069 /*
1070 * Drop a reference, free it when it reaches zero.
1071 */
1072 uint32_t cRefs = ASMAtomicDecU32(&pPool->cRefs);
1073 if (cRefs == 0)
1074 {
1075 AssertReturn(ASMAtomicCmpXchgU32(&pPool->u32Magic, RTREQPOOL_MAGIC_DEAD, RTREQPOOL_MAGIC), UINT32_MAX);
1076
1077 RTCritSectEnter(&pPool->CritSect);
1078#ifdef RT_STRICT
1079 RTTHREAD const hSelf = RTThreadSelf();
1080#endif
1081
1082 /* Indicate to the worker threads that we're shutting down. */
1083 ASMAtomicWriteBool(&pPool->fDestructing, true);
1084 PRTREQPOOLTHREAD pThread;
1085 RTListForEach(&pPool->WorkerThreads, pThread, RTREQPOOLTHREAD, ListNode)
1086 {
1087 Assert(pThread->hThread != hSelf);
1088 RTThreadUserSignal(pThread->hThread);
1089 }
1090
1091 /* Cancel pending requests. */
1092 Assert(!pPool->pPendingRequests);
1093 while (pPool->pPendingRequests)
1094 {
1095 PRTREQINT pReq = pPool->pPendingRequests;
1096 pPool->pPendingRequests = pReq->pNext;
1097 rtReqPoolCancelReq(pReq);
1098 }
1099 pPool->ppPendingRequests = NULL;
1100 pPool->cCurPendingRequests = 0;
1101
1102 /* Wait for the workers to shut down. */
1103 while (!RTListIsEmpty(&pPool->WorkerThreads))
1104 {
1105 RTCritSectLeave(&pPool->CritSect);
1106 RTSemEventMultiWait(pPool->hThreadTermEvt, RT_MS_1MIN);
1107 RTCritSectEnter(&pPool->CritSect);
1108 /** @todo should we wait forever here? */
1109 }
1110
1111 /* Free recycled requests. */
1112 for (;;)
1113 {
1114 PRTREQINT pReq = pPool->pFreeRequests;
1115 if (!pReq)
1116 break;
1117 pPool->pFreeRequests = pReq->pNext;
1118 pPool->cCurFreeRequests--;
1119 rtReqFreeIt(pReq);
1120 }
1121
1122 /* Finally, free the critical section and pool instance. */
1123 RTSemEventMultiDestroy(pPool->hThreadTermEvt);
1124 RTCritSectLeave(&pPool->CritSect);
1125 RTCritSectDelete(&pPool->CritSect);
1126 RTMemFree(pPool);
1127 }
1128
1129 return cRefs;
1130}
1131RT_EXPORT_SYMBOL(RTReqPoolRelease);
1132
1133
1134RTDECL(int) RTReqPoolAlloc(RTREQPOOL hPool, RTREQTYPE enmType, PRTREQ *phReq)
1135{
1136 PRTREQPOOLINT pPool = hPool;
1137 AssertPtrReturn(pPool, VERR_INVALID_HANDLE);
1138 AssertReturn(pPool->u32Magic == RTREQPOOL_MAGIC, VERR_INVALID_HANDLE);
1139
1140 /*
1141 * Try recycle old requests.
1142 */
1143 if (ASMAtomicReadU32(&pPool->cCurFreeRequests) > 0)
1144 {
1145 RTCritSectEnter(&pPool->CritSect);
1146 PRTREQINT pReq = pPool->pFreeRequests;
1147 if (pReq)
1148 {
1149 ASMAtomicDecU32(&pPool->cCurFreeRequests);
1150 pPool->pFreeRequests = pReq->pNext;
1151
1152 RTCritSectLeave(&pPool->CritSect);
1153
1154 Assert(pReq->fPoolOrQueue);
1155 Assert(pReq->uOwner.hPool == pPool);
1156
1157 int rc = rtReqReInit(pReq, enmType);
1158 if (RT_SUCCESS(rc))
1159 {
1160 *phReq = pReq;
1161 LogFlow(("RTReqPoolAlloc: returns VINF_SUCCESS *phReq=%p recycled\n", pReq));
1162 return rc;
1163 }
1164 }
1165 else
1166 RTCritSectLeave(&pPool->CritSect);
1167 }
1168
1169 /*
1170 * Allocate a new request.
1171 */
1172 int rc = rtReqAlloc(enmType, true /*fPoolOrQueue*/, pPool, phReq);
1173 LogFlow(("RTReqPoolAlloc: returns %Rrc *phReq=%p\n", rc, *phReq));
1174 return rc;
1175}
1176RT_EXPORT_SYMBOL(RTReqPoolAlloc);
1177
1178
1179RTDECL(int) RTReqPoolCallEx( RTREQPOOL hPool, RTMSINTERVAL cMillies, PRTREQ *phReq, uint32_t fFlags, PFNRT pfnFunction, unsigned cArgs, ...)
1180{
1181 va_list va;
1182 va_start(va, cArgs);
1183 int rc = RTReqPoolCallExV(hPool, cMillies, phReq, fFlags, pfnFunction, cArgs, va);
1184 va_end(va);
1185 return rc;
1186}
1187RT_EXPORT_SYMBOL(RTReqPoolCallEx);
1188
1189
1190RTDECL(int) RTReqPoolCallExV(RTREQPOOL hPool, RTMSINTERVAL cMillies, PRTREQ *phReq, uint32_t fFlags, PFNRT pfnFunction, unsigned cArgs, va_list va)
1191{
1192 /*
1193 * Check input.
1194 */
1195 AssertPtrReturn(pfnFunction, VERR_INVALID_POINTER);
1196 AssertMsgReturn(!((uint32_t)fFlags & ~(uint32_t)(RTREQFLAGS_NO_WAIT | RTREQFLAGS_RETURN_MASK)), ("%#x\n", (uint32_t)fFlags), VERR_INVALID_PARAMETER);
1197 if (!(fFlags & RTREQFLAGS_NO_WAIT) || phReq)
1198 {
1199 AssertPtrReturn(phReq, VERR_INVALID_POINTER);
1200 *phReq = NIL_RTREQ;
1201 }
1202
1203 PRTREQINT pReq = NULL;
1204 AssertMsgReturn(cArgs * sizeof(uintptr_t) <= sizeof(pReq->u.Internal.aArgs), ("cArgs=%u\n", cArgs), VERR_TOO_MUCH_DATA);
1205
1206 /*
1207 * Allocate and initialize the request.
1208 */
1209 int rc = RTReqPoolAlloc(hPool, RTREQTYPE_INTERNAL, &pReq);
1210 if (RT_FAILURE(rc))
1211 return rc;
1212 pReq->fFlags = fFlags;
1213 pReq->u.Internal.pfn = pfnFunction;
1214 pReq->u.Internal.cArgs = cArgs;
1215 for (unsigned iArg = 0; iArg < cArgs; iArg++)
1216 pReq->u.Internal.aArgs[iArg] = va_arg(va, uintptr_t);
1217
1218 /*
1219 * Submit the request.
1220 */
1221 rc = RTReqSubmit(pReq, cMillies);
1222 if ( rc != VINF_SUCCESS
1223 && rc != VERR_TIMEOUT)
1224 {
1225 Assert(rc != VERR_INTERRUPTED);
1226 RTReqRelease(pReq);
1227 pReq = NULL;
1228 }
1229
1230 if (phReq)
1231 {
1232 *phReq = pReq;
1233 LogFlow(("RTReqPoolCallExV: returns %Rrc *phReq=%p\n", rc, pReq));
1234 }
1235 else
1236 {
1237 RTReqRelease(pReq);
1238 LogFlow(("RTReqPoolCallExV: returns %Rrc\n", rc));
1239 }
1240 return rc;
1241}
1242RT_EXPORT_SYMBOL(RTReqPoolCallExV);
1243
1244
1245RTDECL(int) RTReqPoolCallWait(RTREQPOOL hPool, PFNRT pfnFunction, unsigned cArgs, ...)
1246{
1247 PRTREQINT pReq;
1248 va_list va;
1249 va_start(va, cArgs);
1250 int rc = RTReqPoolCallExV(hPool, RT_INDEFINITE_WAIT, &pReq, RTREQFLAGS_IPRT_STATUS,
1251 pfnFunction, cArgs, va);
1252 va_end(va);
1253 if (RT_SUCCESS(rc))
1254 rc = pReq->iStatusX;
1255 RTReqRelease(pReq);
1256 return rc;
1257}
1258RT_EXPORT_SYMBOL(RTReqPoolCallWait);
1259
1260
1261RTDECL(int) RTReqPoolCallNoWait(RTREQPOOL hPool, PFNRT pfnFunction, unsigned cArgs, ...)
1262{
1263 va_list va;
1264 va_start(va, cArgs);
1265 int rc = RTReqPoolCallExV(hPool, 0, NULL, RTREQFLAGS_IPRT_STATUS | RTREQFLAGS_NO_WAIT,
1266 pfnFunction, cArgs, va);
1267 va_end(va);
1268 return rc;
1269}
1270RT_EXPORT_SYMBOL(RTReqPoolCallNoWait);
1271
1272
1273RTDECL(int) RTReqPoolCallVoidWait(RTREQPOOL hPool, PFNRT pfnFunction, unsigned cArgs, ...)
1274{
1275 PRTREQINT pReq;
1276 va_list va;
1277 va_start(va, cArgs);
1278 int rc = RTReqPoolCallExV(hPool, RT_INDEFINITE_WAIT, &pReq, RTREQFLAGS_VOID,
1279 pfnFunction, cArgs, va);
1280 va_end(va);
1281 if (RT_SUCCESS(rc))
1282 rc = pReq->iStatusX;
1283 RTReqRelease(pReq);
1284 return rc;
1285}
1286RT_EXPORT_SYMBOL(RTReqPoolCallVoidWait);
1287
1288
1289RTDECL(int) RTReqPoolCallVoidNoWait(RTREQPOOL hPool, PFNRT pfnFunction, unsigned cArgs, ...)
1290{
1291 va_list va;
1292 va_start(va, cArgs);
1293 int rc = RTReqPoolCallExV(hPool, 0, NULL, RTREQFLAGS_VOID | RTREQFLAGS_NO_WAIT,
1294 pfnFunction, cArgs, va);
1295 va_end(va);
1296 return rc;
1297}
1298RT_EXPORT_SYMBOL(RTReqPoolCallVoidNoWait);
1299
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette