2006-03-13 12:28:28 +01:00
|
|
|
/*
|
|
|
|
* Thread pooling
|
|
|
|
*
|
|
|
|
* Copyright (c) 2006 Robert Shearman
|
2016-08-20 21:22:29 +02:00
|
|
|
* Copyright (c) 2014-2016 Sebastian Lackner
|
2006-03-13 12:28:28 +01:00
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
2006-05-18 14:49:52 +02:00
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
2006-03-13 12:28:28 +01:00
|
|
|
*/
|
|
|
|
|
2006-03-21 14:44:14 +01:00
|
|
|
#include "config.h"
|
|
|
|
#include "wine/port.h"
|
|
|
|
|
2008-07-25 01:26:32 +02:00
|
|
|
#include <assert.h>
|
2006-03-13 12:28:28 +01:00
|
|
|
#include <stdarg.h>
|
2006-03-21 14:44:14 +01:00
|
|
|
#include <limits.h>
|
2006-03-13 12:28:28 +01:00
|
|
|
|
|
|
|
#define NONAMELESSUNION
|
|
|
|
#include "ntstatus.h"
|
|
|
|
#define WIN32_NO_STATUS
|
|
|
|
#include "winternl.h"
|
|
|
|
|
|
|
|
#include "wine/debug.h"
|
2006-03-21 14:44:14 +01:00
|
|
|
#include "wine/list.h"
|
|
|
|
|
2006-03-13 12:28:28 +01:00
|
|
|
#include "ntdll_misc.h"
|
|
|
|
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(threadpool);
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
/*
|
|
|
|
* Old thread pooling API
|
|
|
|
*/
|
|
|
|
|
2015-07-26 23:24:06 +02:00
|
|
|
struct rtl_work_item
|
|
|
|
{
|
|
|
|
PRTL_WORK_ITEM_ROUTINE function;
|
|
|
|
PVOID context;
|
|
|
|
};
|
|
|
|
|
2015-03-11 03:24:59 +01:00
|
|
|
#define EXPIRE_NEVER (~(ULONGLONG)0)
|
|
|
|
#define TIMER_QUEUE_MAGIC 0x516d6954 /* TimQ */
|
2006-03-21 14:44:14 +01:00
|
|
|
|
2015-03-05 14:25:32 +01:00
|
|
|
static RTL_CRITICAL_SECTION_DEBUG critsect_compl_debug;
|
2006-03-21 14:44:14 +01:00
|
|
|
|
2015-03-05 14:25:32 +01:00
|
|
|
static struct
|
|
|
|
{
|
|
|
|
HANDLE compl_port;
|
|
|
|
RTL_CRITICAL_SECTION threadpool_compl_cs;
|
|
|
|
}
|
|
|
|
old_threadpool =
|
|
|
|
{
|
|
|
|
NULL, /* compl_port */
|
|
|
|
{ &critsect_compl_debug, -1, 0, 0, 0, 0 }, /* threadpool_compl_cs */
|
|
|
|
};
|
2006-03-21 14:44:14 +01:00
|
|
|
|
2007-12-19 22:27:44 +01:00
|
|
|
static RTL_CRITICAL_SECTION_DEBUG critsect_compl_debug =
|
|
|
|
{
|
2015-03-05 14:25:32 +01:00
|
|
|
0, 0, &old_threadpool.threadpool_compl_cs,
|
2007-12-19 22:27:44 +01:00
|
|
|
{ &critsect_compl_debug.ProcessLocksList, &critsect_compl_debug.ProcessLocksList },
|
2015-07-03 01:58:29 +02:00
|
|
|
0, 0, { (DWORD_PTR)(__FILE__ ": threadpool_compl_cs") }
|
2007-12-19 22:27:44 +01:00
|
|
|
};
|
|
|
|
|
2015-03-11 03:24:59 +01:00
|
|
|
struct wait_work_item
|
|
|
|
{
|
|
|
|
HANDLE Object;
|
|
|
|
HANDLE CancelEvent;
|
|
|
|
WAITORTIMERCALLBACK Callback;
|
|
|
|
PVOID Context;
|
|
|
|
ULONG Milliseconds;
|
|
|
|
ULONG Flags;
|
|
|
|
HANDLE CompletionEvent;
|
|
|
|
LONG DeleteCount;
|
2018-10-28 22:14:14 +01:00
|
|
|
int CallbackInProgress;
|
2015-03-11 03:24:59 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
struct timer_queue;
|
|
|
|
struct queue_timer
|
|
|
|
{
|
|
|
|
struct timer_queue *q;
|
|
|
|
struct list entry;
|
|
|
|
ULONG runcount; /* number of callbacks pending execution */
|
|
|
|
RTL_WAITORTIMERCALLBACKFUNC callback;
|
|
|
|
PVOID param;
|
|
|
|
DWORD period;
|
|
|
|
ULONG flags;
|
|
|
|
ULONGLONG expire;
|
|
|
|
BOOL destroy; /* timer should be deleted; once set, never unset */
|
|
|
|
HANDLE event; /* removal event */
|
|
|
|
};
|
|
|
|
|
|
|
|
struct timer_queue
|
|
|
|
{
|
|
|
|
DWORD magic;
|
|
|
|
RTL_CRITICAL_SECTION cs;
|
|
|
|
struct list timers; /* sorted by expiration time */
|
|
|
|
BOOL quit; /* queue should be deleted; once set, never unset */
|
|
|
|
HANDLE event;
|
|
|
|
HANDLE thread;
|
|
|
|
};
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
/*
|
|
|
|
* Object-oriented thread pooling API
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define THREADPOOL_WORKER_TIMEOUT 5000
|
2015-07-05 02:22:52 +02:00
|
|
|
#define MAXIMUM_WAITQUEUE_OBJECTS (MAXIMUM_WAIT_OBJECTS - 1)
|
2015-07-01 02:55:27 +02:00
|
|
|
|
|
|
|
/* internal threadpool representation */
|
|
|
|
struct threadpool
|
|
|
|
{
|
|
|
|
LONG refcount;
|
|
|
|
LONG objcount;
|
|
|
|
BOOL shutdown;
|
|
|
|
CRITICAL_SECTION cs;
|
2019-03-13 07:28:45 +01:00
|
|
|
/* Pools of work items, locked via .cs, order matches TP_CALLBACK_PRIORITY - high, normal, low. */
|
|
|
|
struct list pools[3];
|
2015-07-01 02:55:27 +02:00
|
|
|
RTL_CONDITION_VARIABLE update_event;
|
|
|
|
/* information about worker threads, locked via .cs */
|
|
|
|
int max_workers;
|
|
|
|
int min_workers;
|
|
|
|
int num_workers;
|
|
|
|
int num_busy_workers;
|
2020-02-12 10:05:27 +01:00
|
|
|
TP_POOL_STACK_INFORMATION stack_info;
|
2015-07-01 02:55:27 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
enum threadpool_objtype
|
|
|
|
{
|
2015-07-01 02:57:57 +02:00
|
|
|
TP_OBJECT_TYPE_SIMPLE,
|
2015-07-03 01:57:54 +02:00
|
|
|
TP_OBJECT_TYPE_WORK,
|
2015-07-05 02:22:11 +02:00
|
|
|
TP_OBJECT_TYPE_TIMER,
|
|
|
|
TP_OBJECT_TYPE_WAIT
|
2015-07-01 02:55:27 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
/* internal threadpool object representation */
|
|
|
|
struct threadpool_object
|
|
|
|
{
|
|
|
|
LONG refcount;
|
|
|
|
BOOL shutdown;
|
|
|
|
/* read-only information */
|
|
|
|
enum threadpool_objtype type;
|
|
|
|
struct threadpool *pool;
|
2015-07-01 02:57:15 +02:00
|
|
|
struct threadpool_group *group;
|
2015-07-01 02:55:27 +02:00
|
|
|
PVOID userdata;
|
2015-07-01 22:52:52 +02:00
|
|
|
PTP_CLEANUP_GROUP_CANCEL_CALLBACK group_cancel_callback;
|
2015-07-01 22:53:26 +02:00
|
|
|
PTP_SIMPLE_CALLBACK finalization_callback;
|
2015-07-01 22:55:30 +02:00
|
|
|
BOOL may_run_long;
|
2015-07-01 22:53:44 +02:00
|
|
|
HMODULE race_dll;
|
2019-03-13 07:28:45 +01:00
|
|
|
TP_CALLBACK_PRIORITY priority;
|
2015-07-01 02:57:15 +02:00
|
|
|
/* information about the group, locked via .group->cs */
|
|
|
|
struct list group_entry;
|
|
|
|
BOOL is_group_member;
|
2015-07-01 02:55:27 +02:00
|
|
|
/* information about the pool, locked via .pool->cs */
|
|
|
|
struct list pool_entry;
|
2015-07-01 02:57:15 +02:00
|
|
|
RTL_CONDITION_VARIABLE finished_event;
|
2015-07-03 01:57:23 +02:00
|
|
|
RTL_CONDITION_VARIABLE group_finished_event;
|
2015-07-01 02:55:27 +02:00
|
|
|
LONG num_pending_callbacks;
|
|
|
|
LONG num_running_callbacks;
|
2015-07-03 01:57:23 +02:00
|
|
|
LONG num_associated_callbacks;
|
2015-07-01 02:55:27 +02:00
|
|
|
/* arguments for callback */
|
|
|
|
union
|
|
|
|
{
|
|
|
|
struct
|
|
|
|
{
|
|
|
|
PTP_SIMPLE_CALLBACK callback;
|
|
|
|
} simple;
|
2015-07-01 02:57:57 +02:00
|
|
|
struct
|
|
|
|
{
|
|
|
|
PTP_WORK_CALLBACK callback;
|
|
|
|
} work;
|
2015-07-03 01:57:54 +02:00
|
|
|
struct
|
|
|
|
{
|
|
|
|
PTP_TIMER_CALLBACK callback;
|
2015-07-03 01:58:29 +02:00
|
|
|
/* information about the timer, locked via timerqueue.cs */
|
|
|
|
BOOL timer_initialized;
|
|
|
|
BOOL timer_pending;
|
|
|
|
struct list timer_entry;
|
|
|
|
BOOL timer_set;
|
|
|
|
ULONGLONG timeout;
|
|
|
|
LONG period;
|
|
|
|
LONG window_length;
|
2015-07-03 01:57:54 +02:00
|
|
|
} timer;
|
2015-07-05 02:22:11 +02:00
|
|
|
struct
|
|
|
|
{
|
|
|
|
PTP_WAIT_CALLBACK callback;
|
2015-07-05 02:22:52 +02:00
|
|
|
LONG signaled;
|
|
|
|
/* information about the wait object, locked via waitqueue.cs */
|
|
|
|
struct waitqueue_bucket *bucket;
|
|
|
|
BOOL wait_pending;
|
|
|
|
struct list wait_entry;
|
|
|
|
ULONGLONG timeout;
|
|
|
|
HANDLE handle;
|
2015-07-05 02:22:11 +02:00
|
|
|
} wait;
|
2015-07-01 02:55:27 +02:00
|
|
|
} u;
|
|
|
|
};
|
|
|
|
|
2015-07-01 22:55:30 +02:00
|
|
|
/* internal threadpool instance representation */
|
|
|
|
struct threadpool_instance
|
|
|
|
{
|
|
|
|
struct threadpool_object *object;
|
|
|
|
DWORD threadid;
|
2015-07-03 01:57:23 +02:00
|
|
|
BOOL associated;
|
2015-07-01 22:55:30 +02:00
|
|
|
BOOL may_run_long;
|
2015-07-01 22:55:52 +02:00
|
|
|
struct
|
|
|
|
{
|
|
|
|
CRITICAL_SECTION *critical_section;
|
2015-07-01 22:56:14 +02:00
|
|
|
HANDLE mutex;
|
2015-07-01 22:56:51 +02:00
|
|
|
HANDLE semaphore;
|
|
|
|
LONG semaphore_count;
|
2015-07-01 22:57:53 +02:00
|
|
|
HANDLE event;
|
2015-07-01 22:58:23 +02:00
|
|
|
HMODULE library;
|
2015-07-01 22:55:52 +02:00
|
|
|
} cleanup;
|
2015-07-01 22:55:30 +02:00
|
|
|
};
|
|
|
|
|
2015-07-01 02:57:15 +02:00
|
|
|
/* internal threadpool group representation */
|
|
|
|
struct threadpool_group
|
|
|
|
{
|
|
|
|
LONG refcount;
|
|
|
|
BOOL shutdown;
|
|
|
|
CRITICAL_SECTION cs;
|
|
|
|
/* list of group members, locked via .cs */
|
|
|
|
struct list members;
|
|
|
|
};
|
|
|
|
|
2015-07-03 01:58:29 +02:00
|
|
|
/* global timerqueue object */
|
|
|
|
static RTL_CRITICAL_SECTION_DEBUG timerqueue_debug;
|
|
|
|
|
|
|
|
static struct
|
|
|
|
{
|
|
|
|
CRITICAL_SECTION cs;
|
|
|
|
LONG objcount;
|
|
|
|
BOOL thread_running;
|
|
|
|
struct list pending_timers;
|
|
|
|
RTL_CONDITION_VARIABLE update_event;
|
|
|
|
}
|
|
|
|
timerqueue =
|
|
|
|
{
|
|
|
|
{ &timerqueue_debug, -1, 0, 0, 0, 0 }, /* cs */
|
|
|
|
0, /* objcount */
|
|
|
|
FALSE, /* thread_running */
|
|
|
|
LIST_INIT( timerqueue.pending_timers ), /* pending_timers */
|
|
|
|
RTL_CONDITION_VARIABLE_INIT /* update_event */
|
|
|
|
};
|
|
|
|
|
|
|
|
static RTL_CRITICAL_SECTION_DEBUG timerqueue_debug =
|
|
|
|
{
|
|
|
|
0, 0, &timerqueue.cs,
|
|
|
|
{ &timerqueue_debug.ProcessLocksList, &timerqueue_debug.ProcessLocksList },
|
|
|
|
0, 0, { (DWORD_PTR)(__FILE__ ": timerqueue.cs") }
|
|
|
|
};
|
|
|
|
|
2015-07-05 02:22:52 +02:00
|
|
|
/* global waitqueue object */
|
|
|
|
static RTL_CRITICAL_SECTION_DEBUG waitqueue_debug;
|
|
|
|
|
|
|
|
static struct
|
|
|
|
{
|
|
|
|
CRITICAL_SECTION cs;
|
|
|
|
LONG num_buckets;
|
|
|
|
struct list buckets;
|
|
|
|
}
|
|
|
|
waitqueue =
|
|
|
|
{
|
|
|
|
{ &waitqueue_debug, -1, 0, 0, 0, 0 }, /* cs */
|
|
|
|
0, /* num_buckets */
|
|
|
|
LIST_INIT( waitqueue.buckets ) /* buckets */
|
|
|
|
};
|
|
|
|
|
|
|
|
static RTL_CRITICAL_SECTION_DEBUG waitqueue_debug =
|
|
|
|
{
|
|
|
|
0, 0, &waitqueue.cs,
|
|
|
|
{ &waitqueue_debug.ProcessLocksList, &waitqueue_debug.ProcessLocksList },
|
|
|
|
0, 0, { (DWORD_PTR)(__FILE__ ": waitqueue.cs") }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct waitqueue_bucket
|
|
|
|
{
|
|
|
|
struct list bucket_entry;
|
|
|
|
LONG objcount;
|
|
|
|
struct list reserved;
|
|
|
|
struct list waiting;
|
|
|
|
HANDLE update_event;
|
|
|
|
};
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
static inline struct threadpool *impl_from_TP_POOL( TP_POOL *pool )
|
|
|
|
{
|
|
|
|
return (struct threadpool *)pool;
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:57:57 +02:00
|
|
|
static inline struct threadpool_object *impl_from_TP_WORK( TP_WORK *work )
|
|
|
|
{
|
|
|
|
struct threadpool_object *object = (struct threadpool_object *)work;
|
|
|
|
assert( object->type == TP_OBJECT_TYPE_WORK );
|
|
|
|
return object;
|
|
|
|
}
|
|
|
|
|
2015-07-03 01:57:54 +02:00
|
|
|
static inline struct threadpool_object *impl_from_TP_TIMER( TP_TIMER *timer )
|
|
|
|
{
|
|
|
|
struct threadpool_object *object = (struct threadpool_object *)timer;
|
|
|
|
assert( object->type == TP_OBJECT_TYPE_TIMER );
|
|
|
|
return object;
|
|
|
|
}
|
|
|
|
|
2015-07-05 02:22:11 +02:00
|
|
|
static inline struct threadpool_object *impl_from_TP_WAIT( TP_WAIT *wait )
|
|
|
|
{
|
|
|
|
struct threadpool_object *object = (struct threadpool_object *)wait;
|
|
|
|
assert( object->type == TP_OBJECT_TYPE_WAIT );
|
|
|
|
return object;
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:57:15 +02:00
|
|
|
static inline struct threadpool_group *impl_from_TP_CLEANUP_GROUP( TP_CLEANUP_GROUP *group )
|
|
|
|
{
|
|
|
|
return (struct threadpool_group *)group;
|
|
|
|
}
|
|
|
|
|
2015-07-01 22:55:30 +02:00
|
|
|
static inline struct threadpool_instance *impl_from_TP_CALLBACK_INSTANCE( TP_CALLBACK_INSTANCE *instance )
|
|
|
|
{
|
|
|
|
return (struct threadpool_instance *)instance;
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
static void CALLBACK threadpool_worker_proc( void *param );
|
2015-07-05 02:22:52 +02:00
|
|
|
static void tp_object_submit( struct threadpool_object *object, BOOL signaled );
|
2016-08-20 21:22:29 +02:00
|
|
|
static void tp_object_prepare_shutdown( struct threadpool_object *object );
|
2015-07-01 02:57:15 +02:00
|
|
|
static BOOL tp_object_release( struct threadpool_object *object );
|
2015-07-01 02:55:27 +02:00
|
|
|
static struct threadpool *default_threadpool = NULL;
|
|
|
|
|
2007-03-17 11:28:32 +01:00
|
|
|
static inline LONG interlocked_inc( PLONG dest )
|
2006-03-21 14:44:14 +01:00
|
|
|
{
|
2008-01-12 17:05:50 +01:00
|
|
|
return interlocked_xchg_add( dest, 1 ) + 1;
|
2006-03-21 14:44:14 +01:00
|
|
|
}
|
|
|
|
|
2007-03-17 11:28:32 +01:00
|
|
|
static inline LONG interlocked_dec( PLONG dest )
|
2006-03-21 14:44:14 +01:00
|
|
|
{
|
2008-01-12 17:05:50 +01:00
|
|
|
return interlocked_xchg_add( dest, -1 ) - 1;
|
2006-03-21 14:44:14 +01:00
|
|
|
}
|
|
|
|
|
2015-07-26 23:24:06 +02:00
|
|
|
static void CALLBACK process_rtl_work_item( TP_CALLBACK_INSTANCE *instance, void *userdata )
|
2006-03-13 12:28:28 +01:00
|
|
|
{
|
2015-07-26 23:24:06 +02:00
|
|
|
struct rtl_work_item *item = userdata;
|
2006-03-13 12:28:28 +01:00
|
|
|
|
2015-07-26 23:24:06 +02:00
|
|
|
TRACE("executing %p(%p)\n", item->function, item->context);
|
|
|
|
item->function( item->context );
|
2006-03-13 12:28:28 +01:00
|
|
|
|
2015-07-26 23:24:06 +02:00
|
|
|
RtlFreeHeap( GetProcessHeap(), 0, item );
|
2006-03-13 12:28:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* RtlQueueWorkItem (NTDLL.@)
|
|
|
|
*
|
|
|
|
* Queues a work item into a thread in the thread pool.
|
|
|
|
*
|
|
|
|
* PARAMS
|
2015-07-26 23:24:06 +02:00
|
|
|
* function [I] Work function to execute.
|
|
|
|
* context [I] Context to pass to the work function when it is executed.
|
|
|
|
* flags [I] Flags. See notes.
|
2006-03-13 12:28:28 +01:00
|
|
|
*
|
|
|
|
* RETURNS
|
|
|
|
* Success: STATUS_SUCCESS.
|
|
|
|
* Failure: Any NTSTATUS code.
|
|
|
|
*
|
|
|
|
* NOTES
|
|
|
|
* Flags can be one or more of the following:
|
|
|
|
*|WT_EXECUTEDEFAULT - Executes the work item in a non-I/O worker thread.
|
|
|
|
*|WT_EXECUTEINIOTHREAD - Executes the work item in an I/O worker thread.
|
|
|
|
*|WT_EXECUTEINPERSISTENTTHREAD - Executes the work item in a thread that is persistent.
|
|
|
|
*|WT_EXECUTELONGFUNCTION - Hints that the execution can take a long time.
|
|
|
|
*|WT_TRANSFER_IMPERSONATION - Executes the function with the current access token.
|
|
|
|
*/
|
2015-07-26 23:24:06 +02:00
|
|
|
NTSTATUS WINAPI RtlQueueWorkItem( PRTL_WORK_ITEM_ROUTINE function, PVOID context, ULONG flags )
|
2006-03-13 12:28:28 +01:00
|
|
|
{
|
2015-07-26 23:24:06 +02:00
|
|
|
TP_CALLBACK_ENVIRON environment;
|
|
|
|
struct rtl_work_item *item;
|
2006-03-13 12:28:28 +01:00
|
|
|
NTSTATUS status;
|
|
|
|
|
2015-07-29 22:10:01 +02:00
|
|
|
TRACE( "%p %p %u\n", function, context, flags );
|
|
|
|
|
2015-07-26 23:24:06 +02:00
|
|
|
item = RtlAllocateHeap( GetProcessHeap(), 0, sizeof(*item) );
|
|
|
|
if (!item)
|
2006-03-13 12:28:28 +01:00
|
|
|
return STATUS_NO_MEMORY;
|
|
|
|
|
2015-07-26 23:24:06 +02:00
|
|
|
memset( &environment, 0, sizeof(environment) );
|
|
|
|
environment.Version = 1;
|
|
|
|
environment.u.s.LongFunction = (flags & WT_EXECUTELONGFUNCTION) != 0;
|
|
|
|
environment.u.s.Persistent = (flags & WT_EXECUTEINPERSISTENTTHREAD) != 0;
|
2006-03-13 12:28:28 +01:00
|
|
|
|
2015-07-26 23:24:06 +02:00
|
|
|
item->function = function;
|
|
|
|
item->context = context;
|
2006-03-13 12:28:28 +01:00
|
|
|
|
2015-07-26 23:24:06 +02:00
|
|
|
status = TpSimpleTryPost( process_rtl_work_item, item, &environment );
|
|
|
|
if (status) RtlFreeHeap( GetProcessHeap(), 0, item );
|
2014-03-09 04:04:53 +01:00
|
|
|
return status;
|
2006-03-13 12:28:28 +01:00
|
|
|
}
|
2007-12-19 22:27:44 +01:00
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* iocp_poller - get completion events and run callbacks
|
|
|
|
*/
|
|
|
|
static DWORD CALLBACK iocp_poller(LPVOID Arg)
|
|
|
|
{
|
2014-03-10 18:51:21 +01:00
|
|
|
HANDLE cport = Arg;
|
|
|
|
|
2007-12-19 22:27:44 +01:00
|
|
|
while( TRUE )
|
|
|
|
{
|
|
|
|
PRTL_OVERLAPPED_COMPLETION_ROUTINE callback;
|
|
|
|
LPVOID overlapped;
|
|
|
|
IO_STATUS_BLOCK iosb;
|
2014-03-10 18:51:21 +01:00
|
|
|
NTSTATUS res = NtRemoveIoCompletion( cport, (PULONG_PTR)&callback, (PULONG_PTR)&overlapped, &iosb, NULL );
|
2007-12-19 22:27:44 +01:00
|
|
|
if (res)
|
|
|
|
{
|
|
|
|
ERR("NtRemoveIoCompletion failed: 0x%x\n", res);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
DWORD transferred = 0;
|
|
|
|
DWORD err = 0;
|
|
|
|
|
|
|
|
if (iosb.u.Status == STATUS_SUCCESS)
|
|
|
|
transferred = iosb.Information;
|
|
|
|
else
|
|
|
|
err = RtlNtStatusToDosError(iosb.u.Status);
|
|
|
|
|
|
|
|
callback( err, transferred, overlapped );
|
|
|
|
}
|
|
|
|
}
|
2008-08-11 00:44:31 +02:00
|
|
|
return 0;
|
2007-12-19 22:27:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* RtlSetIoCompletionCallback (NTDLL.@)
|
|
|
|
*
|
|
|
|
* Binds a handle to a thread pool's completion port, and possibly
|
|
|
|
* starts a non-I/O thread to monitor this port and call functions back.
|
|
|
|
*
|
|
|
|
* PARAMS
|
|
|
|
* FileHandle [I] Handle to bind to a completion port.
|
|
|
|
* Function [I] Callback function to call on I/O completions.
|
|
|
|
* Flags [I] Not used.
|
|
|
|
*
|
|
|
|
* RETURNS
|
|
|
|
* Success: STATUS_SUCCESS.
|
|
|
|
* Failure: Any NTSTATUS code.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
NTSTATUS WINAPI RtlSetIoCompletionCallback(HANDLE FileHandle, PRTL_OVERLAPPED_COMPLETION_ROUTINE Function, ULONG Flags)
|
|
|
|
{
|
|
|
|
IO_STATUS_BLOCK iosb;
|
|
|
|
FILE_COMPLETION_INFORMATION info;
|
|
|
|
|
|
|
|
if (Flags) FIXME("Unknown value Flags=0x%x\n", Flags);
|
|
|
|
|
2015-03-05 14:25:32 +01:00
|
|
|
if (!old_threadpool.compl_port)
|
2007-12-19 22:27:44 +01:00
|
|
|
{
|
|
|
|
NTSTATUS res = STATUS_SUCCESS;
|
|
|
|
|
2015-03-05 14:25:32 +01:00
|
|
|
RtlEnterCriticalSection(&old_threadpool.threadpool_compl_cs);
|
|
|
|
if (!old_threadpool.compl_port)
|
2007-12-19 22:27:44 +01:00
|
|
|
{
|
|
|
|
HANDLE cport;
|
|
|
|
|
|
|
|
res = NtCreateIoCompletion( &cport, IO_COMPLETION_ALL_ACCESS, NULL, 0 );
|
|
|
|
if (!res)
|
|
|
|
{
|
|
|
|
/* FIXME native can start additional threads in case of e.g. hung callback function. */
|
2014-03-10 18:51:21 +01:00
|
|
|
res = RtlQueueWorkItem( iocp_poller, cport, WT_EXECUTEDEFAULT );
|
2007-12-19 22:27:44 +01:00
|
|
|
if (!res)
|
2015-03-05 14:25:32 +01:00
|
|
|
old_threadpool.compl_port = cport;
|
2007-12-19 22:27:44 +01:00
|
|
|
else
|
|
|
|
NtClose( cport );
|
|
|
|
}
|
|
|
|
}
|
2015-03-05 14:25:32 +01:00
|
|
|
RtlLeaveCriticalSection(&old_threadpool.threadpool_compl_cs);
|
2007-12-19 22:27:44 +01:00
|
|
|
if (res) return res;
|
|
|
|
}
|
|
|
|
|
2015-03-05 14:25:32 +01:00
|
|
|
info.CompletionPort = old_threadpool.compl_port;
|
2007-12-19 22:27:44 +01:00
|
|
|
info.CompletionKey = (ULONG_PTR)Function;
|
|
|
|
|
|
|
|
return NtSetInformationFile( FileHandle, &iosb, &info, sizeof(info), FileCompletionInformation );
|
|
|
|
}
|
2008-01-01 23:37:16 +01:00
|
|
|
|
|
|
|
static inline PLARGE_INTEGER get_nt_timeout( PLARGE_INTEGER pTime, ULONG timeout )
|
|
|
|
{
|
|
|
|
if (timeout == INFINITE) return NULL;
|
|
|
|
pTime->QuadPart = (ULONGLONG)timeout * -10000;
|
|
|
|
return pTime;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void delete_wait_work_item(struct wait_work_item *wait_work_item)
|
|
|
|
{
|
|
|
|
NtClose( wait_work_item->CancelEvent );
|
|
|
|
RtlFreeHeap( GetProcessHeap(), 0, wait_work_item );
|
|
|
|
}
|
|
|
|
|
|
|
|
static DWORD CALLBACK wait_thread_proc(LPVOID Arg)
|
|
|
|
{
|
|
|
|
struct wait_work_item *wait_work_item = Arg;
|
|
|
|
NTSTATUS status;
|
2012-08-09 09:32:54 +02:00
|
|
|
BOOLEAN alertable = (wait_work_item->Flags & WT_EXECUTEINIOTHREAD) != 0;
|
2008-01-01 23:37:16 +01:00
|
|
|
HANDLE handles[2] = { wait_work_item->Object, wait_work_item->CancelEvent };
|
|
|
|
LARGE_INTEGER timeout;
|
|
|
|
HANDLE completion_event;
|
|
|
|
|
|
|
|
TRACE("\n");
|
|
|
|
|
|
|
|
while (TRUE)
|
|
|
|
{
|
2014-09-01 08:40:42 +02:00
|
|
|
status = NtWaitForMultipleObjects( 2, handles, TRUE, alertable,
|
2008-01-01 23:37:16 +01:00
|
|
|
get_nt_timeout( &timeout, wait_work_item->Milliseconds ) );
|
|
|
|
if (status == STATUS_WAIT_0 || status == STATUS_TIMEOUT)
|
|
|
|
{
|
|
|
|
BOOLEAN TimerOrWaitFired;
|
|
|
|
|
|
|
|
if (status == STATUS_WAIT_0)
|
|
|
|
{
|
|
|
|
TRACE( "object %p signaled, calling callback %p with context %p\n",
|
|
|
|
wait_work_item->Object, wait_work_item->Callback,
|
|
|
|
wait_work_item->Context );
|
|
|
|
TimerOrWaitFired = FALSE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
TRACE( "wait for object %p timed out, calling callback %p with context %p\n",
|
|
|
|
wait_work_item->Object, wait_work_item->Callback,
|
|
|
|
wait_work_item->Context );
|
|
|
|
TimerOrWaitFired = TRUE;
|
|
|
|
}
|
2018-10-28 22:14:14 +01:00
|
|
|
interlocked_xchg( &wait_work_item->CallbackInProgress, TRUE );
|
|
|
|
if (wait_work_item->CompletionEvent)
|
|
|
|
{
|
|
|
|
TRACE( "Work has been canceled.\n" );
|
|
|
|
break;
|
|
|
|
}
|
2008-01-01 23:37:16 +01:00
|
|
|
wait_work_item->Callback( wait_work_item->Context, TimerOrWaitFired );
|
2018-10-28 22:14:14 +01:00
|
|
|
interlocked_xchg( &wait_work_item->CallbackInProgress, FALSE );
|
2008-01-01 23:37:16 +01:00
|
|
|
|
|
|
|
if (wait_work_item->Flags & WT_EXECUTEONLYONCE)
|
|
|
|
break;
|
|
|
|
}
|
2015-10-09 06:37:57 +02:00
|
|
|
else if (status != STATUS_USER_APC)
|
2008-01-01 23:37:16 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (interlocked_inc( &wait_work_item->DeleteCount ) == 2 )
|
2017-09-05 17:34:03 +02:00
|
|
|
{
|
|
|
|
completion_event = wait_work_item->CompletionEvent;
|
2008-01-01 23:37:16 +01:00
|
|
|
delete_wait_work_item( wait_work_item );
|
2018-10-28 22:14:14 +01:00
|
|
|
if (completion_event && completion_event != INVALID_HANDLE_VALUE)
|
|
|
|
NtSetEvent( completion_event, NULL );
|
2017-09-05 17:34:03 +02:00
|
|
|
}
|
2008-01-01 23:37:16 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* RtlRegisterWait (NTDLL.@)
|
|
|
|
*
|
|
|
|
* Registers a wait for a handle to become signaled.
|
|
|
|
*
|
|
|
|
* PARAMS
|
|
|
|
* NewWaitObject [I] Handle to the new wait object. Use RtlDeregisterWait() to free it.
|
|
|
|
* Object [I] Object to wait to become signaled.
|
|
|
|
* Callback [I] Callback function to execute when the wait times out or the handle is signaled.
|
|
|
|
* Context [I] Context to pass to the callback function when it is executed.
|
|
|
|
* Milliseconds [I] Number of milliseconds to wait before timing out.
|
|
|
|
* Flags [I] Flags. See notes.
|
|
|
|
*
|
|
|
|
* RETURNS
|
|
|
|
* Success: STATUS_SUCCESS.
|
|
|
|
* Failure: Any NTSTATUS code.
|
|
|
|
*
|
|
|
|
* NOTES
|
|
|
|
* Flags can be one or more of the following:
|
|
|
|
*|WT_EXECUTEDEFAULT - Executes the work item in a non-I/O worker thread.
|
|
|
|
*|WT_EXECUTEINIOTHREAD - Executes the work item in an I/O worker thread.
|
|
|
|
*|WT_EXECUTEINPERSISTENTTHREAD - Executes the work item in a thread that is persistent.
|
|
|
|
*|WT_EXECUTELONGFUNCTION - Hints that the execution can take a long time.
|
|
|
|
*|WT_TRANSFER_IMPERSONATION - Executes the function with the current access token.
|
|
|
|
*/
|
|
|
|
NTSTATUS WINAPI RtlRegisterWait(PHANDLE NewWaitObject, HANDLE Object,
|
|
|
|
RTL_WAITORTIMERCALLBACKFUNC Callback,
|
|
|
|
PVOID Context, ULONG Milliseconds, ULONG Flags)
|
|
|
|
{
|
|
|
|
struct wait_work_item *wait_work_item;
|
|
|
|
NTSTATUS status;
|
|
|
|
|
|
|
|
TRACE( "(%p, %p, %p, %p, %d, 0x%x)\n", NewWaitObject, Object, Callback, Context, Milliseconds, Flags );
|
|
|
|
|
|
|
|
wait_work_item = RtlAllocateHeap( GetProcessHeap(), 0, sizeof(*wait_work_item) );
|
|
|
|
if (!wait_work_item)
|
|
|
|
return STATUS_NO_MEMORY;
|
|
|
|
|
|
|
|
wait_work_item->Object = Object;
|
|
|
|
wait_work_item->Callback = Callback;
|
|
|
|
wait_work_item->Context = Context;
|
|
|
|
wait_work_item->Milliseconds = Milliseconds;
|
|
|
|
wait_work_item->Flags = Flags;
|
|
|
|
wait_work_item->CallbackInProgress = FALSE;
|
|
|
|
wait_work_item->DeleteCount = 0;
|
|
|
|
wait_work_item->CompletionEvent = NULL;
|
|
|
|
|
2010-01-18 20:34:54 +01:00
|
|
|
status = NtCreateEvent( &wait_work_item->CancelEvent, EVENT_ALL_ACCESS, NULL, NotificationEvent, FALSE );
|
2008-01-01 23:37:16 +01:00
|
|
|
if (status != STATUS_SUCCESS)
|
|
|
|
{
|
|
|
|
RtlFreeHeap( GetProcessHeap(), 0, wait_work_item );
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2010-11-11 16:31:56 +01:00
|
|
|
Flags = Flags & (WT_EXECUTEINIOTHREAD | WT_EXECUTEINPERSISTENTTHREAD |
|
|
|
|
WT_EXECUTELONGFUNCTION | WT_TRANSFER_IMPERSONATION);
|
|
|
|
status = RtlQueueWorkItem( wait_thread_proc, wait_work_item, Flags );
|
2008-01-01 23:37:16 +01:00
|
|
|
if (status != STATUS_SUCCESS)
|
|
|
|
{
|
|
|
|
delete_wait_work_item( wait_work_item );
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
*NewWaitObject = wait_work_item;
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* RtlDeregisterWaitEx (NTDLL.@)
|
|
|
|
*
|
|
|
|
* Cancels a wait operation and frees the resources associated with calling
|
|
|
|
* RtlRegisterWait().
|
|
|
|
*
|
|
|
|
* PARAMS
|
|
|
|
* WaitObject [I] Handle to the wait object to free.
|
|
|
|
*
|
|
|
|
* RETURNS
|
|
|
|
* Success: STATUS_SUCCESS.
|
|
|
|
* Failure: Any NTSTATUS code.
|
|
|
|
*/
|
|
|
|
NTSTATUS WINAPI RtlDeregisterWaitEx(HANDLE WaitHandle, HANDLE CompletionEvent)
|
|
|
|
{
|
|
|
|
struct wait_work_item *wait_work_item = WaitHandle;
|
2017-09-05 17:34:03 +02:00
|
|
|
NTSTATUS status;
|
|
|
|
HANDLE LocalEvent = NULL;
|
2018-10-28 22:14:14 +01:00
|
|
|
int CallbackInProgress;
|
2008-01-01 23:37:16 +01:00
|
|
|
|
2017-09-05 17:34:03 +02:00
|
|
|
TRACE( "(%p %p)\n", WaitHandle, CompletionEvent );
|
2008-01-01 23:37:16 +01:00
|
|
|
|
2016-08-19 23:24:44 +02:00
|
|
|
if (WaitHandle == NULL)
|
|
|
|
return STATUS_INVALID_HANDLE;
|
|
|
|
|
2018-10-28 22:14:14 +01:00
|
|
|
interlocked_xchg_ptr( &wait_work_item->CompletionEvent, INVALID_HANDLE_VALUE );
|
2017-09-05 17:34:03 +02:00
|
|
|
CallbackInProgress = wait_work_item->CallbackInProgress;
|
2018-10-28 22:14:14 +01:00
|
|
|
TRACE( "callback in progress %u\n", CallbackInProgress );
|
2017-09-05 17:34:03 +02:00
|
|
|
if (CompletionEvent == INVALID_HANDLE_VALUE || !CallbackInProgress)
|
2008-01-01 23:37:16 +01:00
|
|
|
{
|
2017-09-05 17:34:03 +02:00
|
|
|
status = NtCreateEvent( &LocalEvent, EVENT_ALL_ACCESS, NULL, NotificationEvent, FALSE );
|
|
|
|
if (status != STATUS_SUCCESS)
|
|
|
|
return status;
|
|
|
|
interlocked_xchg_ptr( &wait_work_item->CompletionEvent, LocalEvent );
|
|
|
|
}
|
|
|
|
else if (CompletionEvent != NULL)
|
|
|
|
{
|
|
|
|
interlocked_xchg_ptr( &wait_work_item->CompletionEvent, CompletionEvent );
|
2008-01-01 23:37:16 +01:00
|
|
|
}
|
|
|
|
|
2017-09-05 17:34:03 +02:00
|
|
|
NtSetEvent( wait_work_item->CancelEvent, NULL );
|
|
|
|
|
2008-01-01 23:37:16 +01:00
|
|
|
if (interlocked_inc( &wait_work_item->DeleteCount ) == 2 )
|
|
|
|
{
|
|
|
|
status = STATUS_SUCCESS;
|
|
|
|
delete_wait_work_item( wait_work_item );
|
|
|
|
}
|
2017-09-05 17:34:03 +02:00
|
|
|
else if (LocalEvent)
|
|
|
|
{
|
2018-10-28 22:14:14 +01:00
|
|
|
TRACE( "Waiting for completion event\n" );
|
2017-09-05 17:34:03 +02:00
|
|
|
NtWaitForSingleObject( LocalEvent, FALSE, NULL );
|
|
|
|
status = STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
status = STATUS_PENDING;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (LocalEvent)
|
|
|
|
NtClose( LocalEvent );
|
2008-01-01 23:37:16 +01:00
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* RtlDeregisterWait (NTDLL.@)
|
|
|
|
*
|
|
|
|
* Cancels a wait operation and frees the resources associated with calling
|
|
|
|
* RtlRegisterWait().
|
|
|
|
*
|
|
|
|
* PARAMS
|
|
|
|
* WaitObject [I] Handle to the wait object to free.
|
|
|
|
*
|
|
|
|
* RETURNS
|
|
|
|
* Success: STATUS_SUCCESS.
|
|
|
|
* Failure: Any NTSTATUS code.
|
|
|
|
*/
|
|
|
|
NTSTATUS WINAPI RtlDeregisterWait(HANDLE WaitHandle)
|
|
|
|
{
|
|
|
|
return RtlDeregisterWaitEx(WaitHandle, NULL);
|
|
|
|
}
|
2008-07-22 02:46:30 +02:00
|
|
|
|
|
|
|
|
|
|
|
/************************** Timer Queue Impl **************************/
|
|
|
|
|
2008-07-23 03:10:08 +02:00
|
|
|
static void queue_remove_timer(struct queue_timer *t)
|
|
|
|
{
|
2008-07-25 01:26:32 +02:00
|
|
|
/* We MUST hold the queue cs while calling this function. This ensures
|
|
|
|
that we cannot queue another callback for this timer. The runcount
|
|
|
|
being zero makes sure we don't have any already queued. */
|
|
|
|
struct timer_queue *q = t->q;
|
|
|
|
|
|
|
|
assert(t->runcount == 0);
|
|
|
|
assert(t->destroy);
|
|
|
|
|
2008-07-23 03:10:08 +02:00
|
|
|
list_remove(&t->entry);
|
2008-07-25 01:27:54 +02:00
|
|
|
if (t->event)
|
|
|
|
NtSetEvent(t->event, NULL);
|
2008-07-23 03:10:08 +02:00
|
|
|
RtlFreeHeap(GetProcessHeap(), 0, t);
|
2008-07-25 01:26:32 +02:00
|
|
|
|
2011-06-15 20:25:56 +02:00
|
|
|
if (q->quit && list_empty(&q->timers))
|
2008-07-25 01:26:32 +02:00
|
|
|
NtSetEvent(q->event, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void timer_cleanup_callback(struct queue_timer *t)
|
|
|
|
{
|
|
|
|
struct timer_queue *q = t->q;
|
|
|
|
RtlEnterCriticalSection(&q->cs);
|
|
|
|
|
|
|
|
assert(0 < t->runcount);
|
|
|
|
--t->runcount;
|
|
|
|
|
|
|
|
if (t->destroy && t->runcount == 0)
|
|
|
|
queue_remove_timer(t);
|
|
|
|
|
|
|
|
RtlLeaveCriticalSection(&q->cs);
|
|
|
|
}
|
|
|
|
|
|
|
|
static DWORD WINAPI timer_callback_wrapper(LPVOID p)
|
|
|
|
{
|
|
|
|
struct queue_timer *t = p;
|
|
|
|
t->callback(t->param, TRUE);
|
|
|
|
timer_cleanup_callback(t);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline ULONGLONG queue_current_time(void)
|
|
|
|
{
|
2013-02-05 18:01:04 +01:00
|
|
|
LARGE_INTEGER now, freq;
|
|
|
|
NtQueryPerformanceCounter(&now, &freq);
|
|
|
|
return now.QuadPart * 1000 / freq.QuadPart;
|
2008-07-25 01:26:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void queue_add_timer(struct queue_timer *t, ULONGLONG time,
|
|
|
|
BOOL set_event)
|
|
|
|
{
|
|
|
|
/* We MUST hold the queue cs while calling this function. */
|
|
|
|
struct timer_queue *q = t->q;
|
|
|
|
struct list *ptr = &q->timers;
|
|
|
|
|
|
|
|
assert(!q->quit || (t->destroy && time == EXPIRE_NEVER));
|
|
|
|
|
|
|
|
if (time != EXPIRE_NEVER)
|
|
|
|
LIST_FOR_EACH(ptr, &q->timers)
|
|
|
|
{
|
|
|
|
struct queue_timer *cur = LIST_ENTRY(ptr, struct queue_timer, entry);
|
|
|
|
if (time < cur->expire)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
list_add_before(ptr, &t->entry);
|
|
|
|
|
|
|
|
t->expire = time;
|
|
|
|
|
|
|
|
/* If we insert at the head of the list, we need to expire sooner
|
|
|
|
than expected. */
|
|
|
|
if (set_event && &t->entry == list_head(&q->timers))
|
|
|
|
NtSetEvent(q->event, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void queue_move_timer(struct queue_timer *t, ULONGLONG time,
|
|
|
|
BOOL set_event)
|
|
|
|
{
|
|
|
|
/* We MUST hold the queue cs while calling this function. */
|
|
|
|
list_remove(&t->entry);
|
|
|
|
queue_add_timer(t, time, set_event);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void queue_timer_expire(struct timer_queue *q)
|
|
|
|
{
|
2008-07-25 23:50:10 +02:00
|
|
|
struct queue_timer *t = NULL;
|
2008-07-25 01:26:32 +02:00
|
|
|
|
|
|
|
RtlEnterCriticalSection(&q->cs);
|
|
|
|
if (list_head(&q->timers))
|
|
|
|
{
|
2012-03-04 07:30:18 +01:00
|
|
|
ULONGLONG now, next;
|
2008-07-25 01:26:32 +02:00
|
|
|
t = LIST_ENTRY(list_head(&q->timers), struct queue_timer, entry);
|
2012-03-04 07:30:18 +01:00
|
|
|
if (!t->destroy && t->expire <= ((now = queue_current_time())))
|
2008-07-25 01:26:32 +02:00
|
|
|
{
|
|
|
|
++t->runcount;
|
2012-03-04 07:30:18 +01:00
|
|
|
if (t->period)
|
|
|
|
{
|
|
|
|
next = t->expire + t->period;
|
|
|
|
/* avoid trigger cascade if overloaded / hibernated */
|
|
|
|
if (next < now)
|
|
|
|
next = now + t->period;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
next = EXPIRE_NEVER;
|
|
|
|
queue_move_timer(t, next, FALSE);
|
2008-07-25 01:26:32 +02:00
|
|
|
}
|
2008-07-25 23:50:10 +02:00
|
|
|
else
|
|
|
|
t = NULL;
|
2008-07-25 01:26:32 +02:00
|
|
|
}
|
|
|
|
RtlLeaveCriticalSection(&q->cs);
|
|
|
|
|
|
|
|
if (t)
|
|
|
|
{
|
|
|
|
if (t->flags & WT_EXECUTEINTIMERTHREAD)
|
|
|
|
timer_callback_wrapper(t);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ULONG flags
|
|
|
|
= (t->flags
|
|
|
|
& (WT_EXECUTEINIOTHREAD | WT_EXECUTEINPERSISTENTTHREAD
|
|
|
|
| WT_EXECUTELONGFUNCTION | WT_TRANSFER_IMPERSONATION));
|
|
|
|
NTSTATUS status = RtlQueueWorkItem(timer_callback_wrapper, t, flags);
|
|
|
|
if (status != STATUS_SUCCESS)
|
|
|
|
timer_cleanup_callback(t);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static ULONG queue_get_timeout(struct timer_queue *q)
|
|
|
|
{
|
|
|
|
struct queue_timer *t;
|
|
|
|
ULONG timeout = INFINITE;
|
|
|
|
|
|
|
|
RtlEnterCriticalSection(&q->cs);
|
|
|
|
if (list_head(&q->timers))
|
|
|
|
{
|
|
|
|
t = LIST_ENTRY(list_head(&q->timers), struct queue_timer, entry);
|
|
|
|
assert(!t->destroy || t->expire == EXPIRE_NEVER);
|
|
|
|
|
|
|
|
if (t->expire != EXPIRE_NEVER)
|
|
|
|
{
|
|
|
|
ULONGLONG time = queue_current_time();
|
|
|
|
timeout = t->expire < time ? 0 : t->expire - time;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
RtlLeaveCriticalSection(&q->cs);
|
|
|
|
|
|
|
|
return timeout;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void WINAPI timer_queue_thread_proc(LPVOID p)
|
|
|
|
{
|
|
|
|
struct timer_queue *q = p;
|
|
|
|
ULONG timeout_ms;
|
|
|
|
|
|
|
|
timeout_ms = INFINITE;
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
LARGE_INTEGER timeout;
|
|
|
|
NTSTATUS status;
|
|
|
|
BOOL done = FALSE;
|
|
|
|
|
|
|
|
status = NtWaitForSingleObject(
|
|
|
|
q->event, FALSE, get_nt_timeout(&timeout, timeout_ms));
|
|
|
|
|
|
|
|
if (status == STATUS_WAIT_0)
|
|
|
|
{
|
|
|
|
/* There are two possible ways to trigger the event. Either
|
|
|
|
we are quitting and the last timer got removed, or a new
|
|
|
|
timer got put at the head of the list so we need to adjust
|
|
|
|
our timeout. */
|
|
|
|
RtlEnterCriticalSection(&q->cs);
|
2011-06-15 20:25:56 +02:00
|
|
|
if (q->quit && list_empty(&q->timers))
|
2008-07-25 01:26:32 +02:00
|
|
|
done = TRUE;
|
|
|
|
RtlLeaveCriticalSection(&q->cs);
|
|
|
|
}
|
|
|
|
else if (status == STATUS_TIMEOUT)
|
|
|
|
queue_timer_expire(q);
|
|
|
|
|
|
|
|
if (done)
|
|
|
|
break;
|
|
|
|
|
|
|
|
timeout_ms = queue_get_timeout(q);
|
|
|
|
}
|
|
|
|
|
|
|
|
NtClose(q->event);
|
|
|
|
RtlDeleteCriticalSection(&q->cs);
|
2013-08-20 20:13:58 +02:00
|
|
|
q->magic = 0;
|
2008-07-25 01:26:32 +02:00
|
|
|
RtlFreeHeap(GetProcessHeap(), 0, q);
|
2015-07-08 07:07:41 +02:00
|
|
|
RtlExitUserThread( 0 );
|
2008-07-25 01:26:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void queue_destroy_timer(struct queue_timer *t)
|
|
|
|
{
|
|
|
|
/* We MUST hold the queue cs while calling this function. */
|
|
|
|
t->destroy = TRUE;
|
|
|
|
if (t->runcount == 0)
|
|
|
|
/* Ensure a timer is promptly removed. If callbacks are pending,
|
|
|
|
it will be removed after the last one finishes by the callback
|
|
|
|
cleanup wrapper. */
|
|
|
|
queue_remove_timer(t);
|
|
|
|
else
|
|
|
|
/* Make sure no destroyed timer masks an active timer at the head
|
|
|
|
of the sorted list. */
|
|
|
|
queue_move_timer(t, EXPIRE_NEVER, FALSE);
|
2008-07-23 03:10:08 +02:00
|
|
|
}
|
|
|
|
|
2008-07-22 02:46:30 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* RtlCreateTimerQueue (NTDLL.@)
|
|
|
|
*
|
|
|
|
* Creates a timer queue object and returns a handle to it.
|
|
|
|
*
|
|
|
|
* PARAMS
|
|
|
|
* NewTimerQueue [O] The newly created queue.
|
|
|
|
*
|
|
|
|
* RETURNS
|
|
|
|
* Success: STATUS_SUCCESS.
|
|
|
|
* Failure: Any NTSTATUS code.
|
|
|
|
*/
|
|
|
|
NTSTATUS WINAPI RtlCreateTimerQueue(PHANDLE NewTimerQueue)
|
|
|
|
{
|
2008-07-25 01:26:32 +02:00
|
|
|
NTSTATUS status;
|
2008-07-22 02:46:30 +02:00
|
|
|
struct timer_queue *q = RtlAllocateHeap(GetProcessHeap(), 0, sizeof *q);
|
|
|
|
if (!q)
|
|
|
|
return STATUS_NO_MEMORY;
|
|
|
|
|
|
|
|
RtlInitializeCriticalSection(&q->cs);
|
|
|
|
list_init(&q->timers);
|
2008-07-25 01:26:32 +02:00
|
|
|
q->quit = FALSE;
|
2013-08-20 20:13:58 +02:00
|
|
|
q->magic = TIMER_QUEUE_MAGIC;
|
2010-01-18 20:34:54 +01:00
|
|
|
status = NtCreateEvent(&q->event, EVENT_ALL_ACCESS, NULL, SynchronizationEvent, FALSE);
|
2008-07-25 01:26:32 +02:00
|
|
|
if (status != STATUS_SUCCESS)
|
|
|
|
{
|
|
|
|
RtlFreeHeap(GetProcessHeap(), 0, q);
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
status = RtlCreateUserThread(GetCurrentProcess(), NULL, FALSE, NULL, 0, 0,
|
|
|
|
timer_queue_thread_proc, q, &q->thread, NULL);
|
|
|
|
if (status != STATUS_SUCCESS)
|
|
|
|
{
|
|
|
|
NtClose(q->event);
|
|
|
|
RtlFreeHeap(GetProcessHeap(), 0, q);
|
|
|
|
return status;
|
|
|
|
}
|
2008-07-22 02:46:30 +02:00
|
|
|
|
|
|
|
*NewTimerQueue = q;
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* RtlDeleteTimerQueueEx (NTDLL.@)
|
|
|
|
*
|
|
|
|
* Deletes a timer queue object.
|
|
|
|
*
|
|
|
|
* PARAMS
|
|
|
|
* TimerQueue [I] The timer queue to destroy.
|
|
|
|
* CompletionEvent [I] If NULL, return immediately. If INVALID_HANDLE_VALUE,
|
|
|
|
* wait until all timers are finished firing before
|
|
|
|
* returning. Otherwise, return immediately and set the
|
|
|
|
* event when all timers are done.
|
|
|
|
*
|
|
|
|
* RETURNS
|
|
|
|
* Success: STATUS_SUCCESS if synchronous, STATUS_PENDING if not.
|
|
|
|
* Failure: Any NTSTATUS code.
|
|
|
|
*/
|
|
|
|
NTSTATUS WINAPI RtlDeleteTimerQueueEx(HANDLE TimerQueue, HANDLE CompletionEvent)
|
|
|
|
{
|
|
|
|
struct timer_queue *q = TimerQueue;
|
2008-07-23 03:10:08 +02:00
|
|
|
struct queue_timer *t, *temp;
|
2008-07-29 23:36:41 +02:00
|
|
|
HANDLE thread;
|
2008-07-25 01:26:32 +02:00
|
|
|
NTSTATUS status;
|
2008-07-23 03:10:08 +02:00
|
|
|
|
2013-08-20 20:13:58 +02:00
|
|
|
if (!q || q->magic != TIMER_QUEUE_MAGIC)
|
2008-07-29 23:36:41 +02:00
|
|
|
return STATUS_INVALID_HANDLE;
|
|
|
|
|
|
|
|
thread = q->thread;
|
|
|
|
|
2008-07-23 03:10:08 +02:00
|
|
|
RtlEnterCriticalSection(&q->cs);
|
2008-07-25 01:26:32 +02:00
|
|
|
q->quit = TRUE;
|
|
|
|
if (list_head(&q->timers))
|
|
|
|
/* When the last timer is removed, it will signal the timer thread to
|
|
|
|
exit... */
|
|
|
|
LIST_FOR_EACH_ENTRY_SAFE(t, temp, &q->timers, struct queue_timer, entry)
|
|
|
|
queue_destroy_timer(t);
|
|
|
|
else
|
|
|
|
/* However if we have none, we must do it ourselves. */
|
|
|
|
NtSetEvent(q->event, NULL);
|
2008-07-23 03:10:08 +02:00
|
|
|
RtlLeaveCriticalSection(&q->cs);
|
2008-07-22 02:46:30 +02:00
|
|
|
|
|
|
|
if (CompletionEvent == INVALID_HANDLE_VALUE)
|
2008-07-25 01:26:32 +02:00
|
|
|
{
|
|
|
|
NtWaitForSingleObject(thread, FALSE, NULL);
|
|
|
|
status = STATUS_SUCCESS;
|
|
|
|
}
|
2008-07-22 02:46:30 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
if (CompletionEvent)
|
2008-07-25 01:26:32 +02:00
|
|
|
{
|
|
|
|
FIXME("asynchronous return on completion event unimplemented\n");
|
|
|
|
NtWaitForSingleObject(thread, FALSE, NULL);
|
2008-07-22 02:46:30 +02:00
|
|
|
NtSetEvent(CompletionEvent, NULL);
|
2008-07-25 01:26:32 +02:00
|
|
|
}
|
|
|
|
status = STATUS_PENDING;
|
2008-07-22 02:46:30 +02:00
|
|
|
}
|
2008-07-25 01:26:32 +02:00
|
|
|
|
|
|
|
NtClose(thread);
|
|
|
|
return status;
|
2008-07-22 02:46:30 +02:00
|
|
|
}
|
2008-07-23 03:10:08 +02:00
|
|
|
|
2008-07-29 23:36:41 +02:00
|
|
|
static struct timer_queue *get_timer_queue(HANDLE TimerQueue)
|
|
|
|
{
|
2015-03-11 03:24:59 +01:00
|
|
|
static struct timer_queue *default_timer_queue;
|
|
|
|
|
2008-07-29 23:36:41 +02:00
|
|
|
if (TimerQueue)
|
|
|
|
return TimerQueue;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!default_timer_queue)
|
|
|
|
{
|
|
|
|
HANDLE q;
|
|
|
|
NTSTATUS status = RtlCreateTimerQueue(&q);
|
|
|
|
if (status == STATUS_SUCCESS)
|
|
|
|
{
|
|
|
|
PVOID p = interlocked_cmpxchg_ptr(
|
|
|
|
(void **) &default_timer_queue, q, NULL);
|
|
|
|
if (p)
|
|
|
|
/* Got beat to the punch. */
|
2014-01-30 03:56:19 +01:00
|
|
|
RtlDeleteTimerQueueEx(q, NULL);
|
2008-07-29 23:36:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return default_timer_queue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-07-23 03:10:08 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* RtlCreateTimer (NTDLL.@)
|
|
|
|
*
|
|
|
|
* Creates a new timer associated with the given queue.
|
|
|
|
*
|
|
|
|
* PARAMS
|
|
|
|
* NewTimer [O] The newly created timer.
|
|
|
|
* TimerQueue [I] The queue to hold the timer.
|
|
|
|
* Callback [I] The callback to fire.
|
|
|
|
* Parameter [I] The argument for the callback.
|
|
|
|
* DueTime [I] The delay, in milliseconds, before first firing the
|
|
|
|
* timer.
|
|
|
|
* Period [I] The period, in milliseconds, at which to fire the timer
|
|
|
|
* after the first callback. If zero, the timer will only
|
|
|
|
* fire once. It still needs to be deleted with
|
|
|
|
* RtlDeleteTimer.
|
2012-05-15 10:15:32 +02:00
|
|
|
* Flags [I] Flags controlling the execution of the callback. In
|
2008-07-23 03:10:08 +02:00
|
|
|
* addition to the WT_* thread pool flags (see
|
|
|
|
* RtlQueueWorkItem), WT_EXECUTEINTIMERTHREAD and
|
|
|
|
* WT_EXECUTEONLYONCE are supported.
|
|
|
|
*
|
|
|
|
* RETURNS
|
|
|
|
* Success: STATUS_SUCCESS.
|
|
|
|
* Failure: Any NTSTATUS code.
|
|
|
|
*/
|
|
|
|
NTSTATUS WINAPI RtlCreateTimer(PHANDLE NewTimer, HANDLE TimerQueue,
|
|
|
|
RTL_WAITORTIMERCALLBACKFUNC Callback,
|
|
|
|
PVOID Parameter, DWORD DueTime, DWORD Period,
|
|
|
|
ULONG Flags)
|
|
|
|
{
|
2008-07-25 01:26:32 +02:00
|
|
|
NTSTATUS status;
|
2008-07-29 23:36:41 +02:00
|
|
|
struct queue_timer *t;
|
|
|
|
struct timer_queue *q = get_timer_queue(TimerQueue);
|
2013-08-20 20:13:58 +02:00
|
|
|
|
|
|
|
if (!q) return STATUS_NO_MEMORY;
|
|
|
|
if (q->magic != TIMER_QUEUE_MAGIC) return STATUS_INVALID_HANDLE;
|
2008-07-29 23:36:41 +02:00
|
|
|
|
|
|
|
t = RtlAllocateHeap(GetProcessHeap(), 0, sizeof *t);
|
2008-07-23 03:10:08 +02:00
|
|
|
if (!t)
|
|
|
|
return STATUS_NO_MEMORY;
|
|
|
|
|
2008-07-25 01:26:32 +02:00
|
|
|
t->q = q;
|
|
|
|
t->runcount = 0;
|
|
|
|
t->callback = Callback;
|
|
|
|
t->param = Parameter;
|
|
|
|
t->period = Period;
|
|
|
|
t->flags = Flags;
|
|
|
|
t->destroy = FALSE;
|
2008-07-25 01:27:54 +02:00
|
|
|
t->event = NULL;
|
2008-07-23 03:10:08 +02:00
|
|
|
|
2008-07-25 01:26:32 +02:00
|
|
|
status = STATUS_SUCCESS;
|
2008-07-23 03:10:08 +02:00
|
|
|
RtlEnterCriticalSection(&q->cs);
|
2008-07-25 01:26:32 +02:00
|
|
|
if (q->quit)
|
|
|
|
status = STATUS_INVALID_HANDLE;
|
|
|
|
else
|
|
|
|
queue_add_timer(t, queue_current_time() + DueTime, TRUE);
|
2008-07-23 03:10:08 +02:00
|
|
|
RtlLeaveCriticalSection(&q->cs);
|
|
|
|
|
2008-07-25 01:26:32 +02:00
|
|
|
if (status == STATUS_SUCCESS)
|
|
|
|
*NewTimer = t;
|
|
|
|
else
|
|
|
|
RtlFreeHeap(GetProcessHeap(), 0, t);
|
|
|
|
|
|
|
|
return status;
|
2008-07-23 03:10:08 +02:00
|
|
|
}
|
2008-07-25 01:27:03 +02:00
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* RtlUpdateTimer (NTDLL.@)
|
|
|
|
*
|
|
|
|
* Changes the time at which a timer expires.
|
|
|
|
*
|
|
|
|
* PARAMS
|
|
|
|
* TimerQueue [I] The queue that holds the timer.
|
|
|
|
* Timer [I] The timer to update.
|
|
|
|
* DueTime [I] The delay, in milliseconds, before next firing the timer.
|
|
|
|
* Period [I] The period, in milliseconds, at which to fire the timer
|
|
|
|
* after the first callback. If zero, the timer will not
|
|
|
|
* refire once. It still needs to be deleted with
|
|
|
|
* RtlDeleteTimer.
|
|
|
|
*
|
|
|
|
* RETURNS
|
|
|
|
* Success: STATUS_SUCCESS.
|
|
|
|
* Failure: Any NTSTATUS code.
|
|
|
|
*/
|
|
|
|
NTSTATUS WINAPI RtlUpdateTimer(HANDLE TimerQueue, HANDLE Timer,
|
|
|
|
DWORD DueTime, DWORD Period)
|
|
|
|
{
|
|
|
|
struct queue_timer *t = Timer;
|
2008-07-29 23:36:41 +02:00
|
|
|
struct timer_queue *q = t->q;
|
2008-07-25 01:27:03 +02:00
|
|
|
|
|
|
|
RtlEnterCriticalSection(&q->cs);
|
|
|
|
/* Can't change a timer if it was once-only or destroyed. */
|
|
|
|
if (t->expire != EXPIRE_NEVER)
|
|
|
|
{
|
|
|
|
t->period = Period;
|
|
|
|
queue_move_timer(t, queue_current_time() + DueTime, TRUE);
|
|
|
|
}
|
|
|
|
RtlLeaveCriticalSection(&q->cs);
|
|
|
|
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
2008-07-25 01:27:54 +02:00
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* RtlDeleteTimer (NTDLL.@)
|
|
|
|
*
|
|
|
|
* Cancels a timer-queue timer.
|
|
|
|
*
|
|
|
|
* PARAMS
|
|
|
|
* TimerQueue [I] The queue that holds the timer.
|
|
|
|
* Timer [I] The timer to update.
|
|
|
|
* CompletionEvent [I] If NULL, return immediately. If INVALID_HANDLE_VALUE,
|
|
|
|
* wait until the timer is finished firing all pending
|
|
|
|
* callbacks before returning. Otherwise, return
|
|
|
|
* immediately and set the timer is done.
|
|
|
|
*
|
|
|
|
* RETURNS
|
|
|
|
* Success: STATUS_SUCCESS if the timer is done, STATUS_PENDING if not,
|
|
|
|
or if the completion event is NULL.
|
|
|
|
* Failure: Any NTSTATUS code.
|
|
|
|
*/
|
|
|
|
NTSTATUS WINAPI RtlDeleteTimer(HANDLE TimerQueue, HANDLE Timer,
|
|
|
|
HANDLE CompletionEvent)
|
|
|
|
{
|
|
|
|
struct queue_timer *t = Timer;
|
2008-11-28 23:51:45 +01:00
|
|
|
struct timer_queue *q;
|
2008-07-25 01:27:54 +02:00
|
|
|
NTSTATUS status = STATUS_PENDING;
|
|
|
|
HANDLE event = NULL;
|
|
|
|
|
2008-11-28 23:51:45 +01:00
|
|
|
if (!Timer)
|
|
|
|
return STATUS_INVALID_PARAMETER_1;
|
|
|
|
q = t->q;
|
2008-07-25 01:27:54 +02:00
|
|
|
if (CompletionEvent == INVALID_HANDLE_VALUE)
|
2012-03-06 19:07:44 +01:00
|
|
|
{
|
2010-01-18 20:34:54 +01:00
|
|
|
status = NtCreateEvent(&event, EVENT_ALL_ACCESS, NULL, SynchronizationEvent, FALSE);
|
2012-03-06 19:07:44 +01:00
|
|
|
if (status == STATUS_SUCCESS)
|
|
|
|
status = STATUS_PENDING;
|
|
|
|
}
|
2008-07-25 01:27:54 +02:00
|
|
|
else if (CompletionEvent)
|
|
|
|
event = CompletionEvent;
|
|
|
|
|
|
|
|
RtlEnterCriticalSection(&q->cs);
|
|
|
|
t->event = event;
|
|
|
|
if (t->runcount == 0 && event)
|
|
|
|
status = STATUS_SUCCESS;
|
|
|
|
queue_destroy_timer(t);
|
|
|
|
RtlLeaveCriticalSection(&q->cs);
|
|
|
|
|
|
|
|
if (CompletionEvent == INVALID_HANDLE_VALUE && event)
|
|
|
|
{
|
|
|
|
if (status == STATUS_PENDING)
|
2012-03-06 19:07:44 +01:00
|
|
|
{
|
2008-07-25 01:27:54 +02:00
|
|
|
NtWaitForSingleObject(event, FALSE, NULL);
|
2012-03-06 19:07:44 +01:00
|
|
|
status = STATUS_SUCCESS;
|
|
|
|
}
|
2008-07-25 01:27:54 +02:00
|
|
|
NtClose(event);
|
|
|
|
}
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
2015-07-01 02:55:27 +02:00
|
|
|
|
2015-07-03 01:58:29 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* timerqueue_thread_proc (internal)
|
|
|
|
*/
|
|
|
|
static void CALLBACK timerqueue_thread_proc( void *param )
|
|
|
|
{
|
|
|
|
ULONGLONG timeout_lower, timeout_upper, new_timeout;
|
|
|
|
struct threadpool_object *other_timer;
|
|
|
|
LARGE_INTEGER now, timeout;
|
|
|
|
struct list *ptr;
|
|
|
|
|
|
|
|
TRACE( "starting timer queue thread\n" );
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &timerqueue.cs );
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
NtQuerySystemTime( &now );
|
|
|
|
|
|
|
|
/* Check for expired timers. */
|
|
|
|
while ((ptr = list_head( &timerqueue.pending_timers )))
|
|
|
|
{
|
|
|
|
struct threadpool_object *timer = LIST_ENTRY( ptr, struct threadpool_object, u.timer.timer_entry );
|
|
|
|
assert( timer->type == TP_OBJECT_TYPE_TIMER );
|
|
|
|
assert( timer->u.timer.timer_pending );
|
|
|
|
if (timer->u.timer.timeout > now.QuadPart)
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* Queue a new callback in one of the worker threads. */
|
|
|
|
list_remove( &timer->u.timer.timer_entry );
|
|
|
|
timer->u.timer.timer_pending = FALSE;
|
2015-07-05 02:22:52 +02:00
|
|
|
tp_object_submit( timer, FALSE );
|
2015-07-03 01:58:29 +02:00
|
|
|
|
2016-08-02 00:10:09 +02:00
|
|
|
/* Insert the timer back into the queue, except it's marked for shutdown. */
|
2015-07-03 01:58:29 +02:00
|
|
|
if (timer->u.timer.period && !timer->shutdown)
|
|
|
|
{
|
|
|
|
timer->u.timer.timeout += (ULONGLONG)timer->u.timer.period * 10000;
|
|
|
|
if (timer->u.timer.timeout <= now.QuadPart)
|
|
|
|
timer->u.timer.timeout = now.QuadPart + 1;
|
|
|
|
|
|
|
|
LIST_FOR_EACH_ENTRY( other_timer, &timerqueue.pending_timers,
|
|
|
|
struct threadpool_object, u.timer.timer_entry )
|
|
|
|
{
|
|
|
|
assert( other_timer->type == TP_OBJECT_TYPE_TIMER );
|
|
|
|
if (timer->u.timer.timeout < other_timer->u.timer.timeout)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
list_add_before( &other_timer->u.timer.timer_entry, &timer->u.timer.timer_entry );
|
|
|
|
timer->u.timer.timer_pending = TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
timeout_lower = TIMEOUT_INFINITE;
|
|
|
|
timeout_upper = TIMEOUT_INFINITE;
|
|
|
|
|
|
|
|
/* Determine next timeout and use the window length to optimize wakeup times. */
|
|
|
|
LIST_FOR_EACH_ENTRY( other_timer, &timerqueue.pending_timers,
|
|
|
|
struct threadpool_object, u.timer.timer_entry )
|
|
|
|
{
|
|
|
|
assert( other_timer->type == TP_OBJECT_TYPE_TIMER );
|
|
|
|
if (other_timer->u.timer.timeout >= timeout_upper)
|
|
|
|
break;
|
|
|
|
|
|
|
|
timeout_lower = other_timer->u.timer.timeout;
|
|
|
|
new_timeout = timeout_lower + (ULONGLONG)other_timer->u.timer.window_length * 10000;
|
|
|
|
if (new_timeout < timeout_upper)
|
|
|
|
timeout_upper = new_timeout;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Wait for timer update events or until the next timer expires. */
|
|
|
|
if (timerqueue.objcount)
|
|
|
|
{
|
|
|
|
timeout.QuadPart = timeout_lower;
|
|
|
|
RtlSleepConditionVariableCS( &timerqueue.update_event, &timerqueue.cs, &timeout );
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* All timers have been destroyed, if no new timers are created
|
|
|
|
* within some amount of time, then we can shutdown this thread. */
|
|
|
|
timeout.QuadPart = (ULONGLONG)THREADPOOL_WORKER_TIMEOUT * -10000;
|
|
|
|
if (RtlSleepConditionVariableCS( &timerqueue.update_event, &timerqueue.cs,
|
|
|
|
&timeout ) == STATUS_TIMEOUT && !timerqueue.objcount)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
timerqueue.thread_running = FALSE;
|
|
|
|
RtlLeaveCriticalSection( &timerqueue.cs );
|
|
|
|
|
|
|
|
TRACE( "terminating timer queue thread\n" );
|
2015-07-08 07:07:41 +02:00
|
|
|
RtlExitUserThread( 0 );
|
2015-07-03 01:58:29 +02:00
|
|
|
}
|
|
|
|
|
2016-08-22 22:39:10 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* tp_new_worker_thread (internal)
|
|
|
|
*
|
|
|
|
* Create and account a new worker thread for the desired pool.
|
|
|
|
*/
|
|
|
|
static NTSTATUS tp_new_worker_thread( struct threadpool *pool )
|
|
|
|
{
|
|
|
|
HANDLE thread;
|
|
|
|
NTSTATUS status;
|
|
|
|
|
|
|
|
status = RtlCreateUserThread( GetCurrentProcess(), NULL, FALSE, NULL, 0, 0,
|
|
|
|
threadpool_worker_proc, pool, &thread, NULL );
|
|
|
|
if (status == STATUS_SUCCESS)
|
|
|
|
{
|
|
|
|
interlocked_inc( &pool->refcount );
|
|
|
|
pool->num_workers++;
|
|
|
|
pool->num_busy_workers++;
|
|
|
|
NtClose( thread );
|
|
|
|
}
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2015-07-03 01:58:29 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* tp_timerqueue_lock (internal)
|
|
|
|
*
|
|
|
|
* Acquires a lock on the global timerqueue. When the lock is acquired
|
|
|
|
* successfully, it is guaranteed that the timer thread is running.
|
|
|
|
*/
|
|
|
|
static NTSTATUS tp_timerqueue_lock( struct threadpool_object *timer )
|
|
|
|
{
|
|
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
|
|
assert( timer->type == TP_OBJECT_TYPE_TIMER );
|
|
|
|
|
|
|
|
timer->u.timer.timer_initialized = FALSE;
|
|
|
|
timer->u.timer.timer_pending = FALSE;
|
|
|
|
timer->u.timer.timer_set = FALSE;
|
|
|
|
timer->u.timer.timeout = 0;
|
|
|
|
timer->u.timer.period = 0;
|
|
|
|
timer->u.timer.window_length = 0;
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &timerqueue.cs );
|
|
|
|
|
|
|
|
/* Make sure that the timerqueue thread is running. */
|
|
|
|
if (!timerqueue.thread_running)
|
|
|
|
{
|
|
|
|
HANDLE thread;
|
|
|
|
status = RtlCreateUserThread( GetCurrentProcess(), NULL, FALSE, NULL, 0, 0,
|
|
|
|
timerqueue_thread_proc, NULL, &thread, NULL );
|
|
|
|
if (status == STATUS_SUCCESS)
|
|
|
|
{
|
|
|
|
timerqueue.thread_running = TRUE;
|
|
|
|
NtClose( thread );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (status == STATUS_SUCCESS)
|
|
|
|
{
|
|
|
|
timer->u.timer.timer_initialized = TRUE;
|
|
|
|
timerqueue.objcount++;
|
|
|
|
}
|
|
|
|
|
|
|
|
RtlLeaveCriticalSection( &timerqueue.cs );
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* tp_timerqueue_unlock (internal)
|
|
|
|
*
|
|
|
|
* Releases a lock on the global timerqueue.
|
|
|
|
*/
|
|
|
|
static void tp_timerqueue_unlock( struct threadpool_object *timer )
|
|
|
|
{
|
|
|
|
assert( timer->type == TP_OBJECT_TYPE_TIMER );
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &timerqueue.cs );
|
|
|
|
if (timer->u.timer.timer_initialized)
|
|
|
|
{
|
|
|
|
/* If timer was pending, remove it. */
|
|
|
|
if (timer->u.timer.timer_pending)
|
|
|
|
{
|
|
|
|
list_remove( &timer->u.timer.timer_entry );
|
|
|
|
timer->u.timer.timer_pending = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If the last timer object was destroyed, then wake up the thread. */
|
|
|
|
if (!--timerqueue.objcount)
|
|
|
|
{
|
|
|
|
assert( list_empty( &timerqueue.pending_timers ) );
|
|
|
|
RtlWakeAllConditionVariable( &timerqueue.update_event );
|
|
|
|
}
|
|
|
|
|
|
|
|
timer->u.timer.timer_initialized = FALSE;
|
|
|
|
}
|
|
|
|
RtlLeaveCriticalSection( &timerqueue.cs );
|
|
|
|
}
|
|
|
|
|
2015-07-05 02:22:52 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* waitqueue_thread_proc (internal)
|
|
|
|
*/
|
|
|
|
static void CALLBACK waitqueue_thread_proc( void *param )
|
|
|
|
{
|
|
|
|
struct threadpool_object *objects[MAXIMUM_WAITQUEUE_OBJECTS];
|
|
|
|
HANDLE handles[MAXIMUM_WAITQUEUE_OBJECTS + 1];
|
|
|
|
struct waitqueue_bucket *bucket = param;
|
|
|
|
struct threadpool_object *wait, *next;
|
|
|
|
LARGE_INTEGER now, timeout;
|
|
|
|
DWORD num_handles;
|
|
|
|
NTSTATUS status;
|
|
|
|
|
|
|
|
TRACE( "starting wait queue thread\n" );
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &waitqueue.cs );
|
|
|
|
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
NtQuerySystemTime( &now );
|
|
|
|
timeout.QuadPart = TIMEOUT_INFINITE;
|
|
|
|
num_handles = 0;
|
|
|
|
|
|
|
|
LIST_FOR_EACH_ENTRY_SAFE( wait, next, &bucket->waiting, struct threadpool_object,
|
|
|
|
u.wait.wait_entry )
|
|
|
|
{
|
|
|
|
assert( wait->type == TP_OBJECT_TYPE_WAIT );
|
|
|
|
if (wait->u.wait.timeout <= now.QuadPart)
|
|
|
|
{
|
|
|
|
/* Wait object timed out. */
|
|
|
|
list_remove( &wait->u.wait.wait_entry );
|
|
|
|
list_add_tail( &bucket->reserved, &wait->u.wait.wait_entry );
|
|
|
|
tp_object_submit( wait, FALSE );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (wait->u.wait.timeout < timeout.QuadPart)
|
|
|
|
timeout.QuadPart = wait->u.wait.timeout;
|
|
|
|
|
|
|
|
assert( num_handles < MAXIMUM_WAITQUEUE_OBJECTS );
|
|
|
|
interlocked_inc( &wait->refcount );
|
|
|
|
objects[num_handles] = wait;
|
|
|
|
handles[num_handles] = wait->u.wait.handle;
|
|
|
|
num_handles++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!bucket->objcount)
|
|
|
|
{
|
|
|
|
/* All wait objects have been destroyed, if no new wait objects are created
|
|
|
|
* within some amount of time, then we can shutdown this thread. */
|
|
|
|
assert( num_handles == 0 );
|
|
|
|
RtlLeaveCriticalSection( &waitqueue.cs );
|
|
|
|
timeout.QuadPart = (ULONGLONG)THREADPOOL_WORKER_TIMEOUT * -10000;
|
|
|
|
status = NtWaitForMultipleObjects( 1, &bucket->update_event, TRUE, FALSE, &timeout );
|
|
|
|
RtlEnterCriticalSection( &waitqueue.cs );
|
|
|
|
|
|
|
|
if (status == STATUS_TIMEOUT && !bucket->objcount)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
handles[num_handles] = bucket->update_event;
|
|
|
|
RtlLeaveCriticalSection( &waitqueue.cs );
|
|
|
|
status = NtWaitForMultipleObjects( num_handles + 1, handles, TRUE, FALSE, &timeout );
|
|
|
|
RtlEnterCriticalSection( &waitqueue.cs );
|
|
|
|
|
|
|
|
if (status >= STATUS_WAIT_0 && status < STATUS_WAIT_0 + num_handles)
|
|
|
|
{
|
|
|
|
wait = objects[status - STATUS_WAIT_0];
|
|
|
|
assert( wait->type == TP_OBJECT_TYPE_WAIT );
|
|
|
|
if (wait->u.wait.bucket)
|
|
|
|
{
|
|
|
|
/* Wait object signaled. */
|
|
|
|
assert( wait->u.wait.bucket == bucket );
|
|
|
|
list_remove( &wait->u.wait.wait_entry );
|
|
|
|
list_add_tail( &bucket->reserved, &wait->u.wait.wait_entry );
|
|
|
|
tp_object_submit( wait, TRUE );
|
|
|
|
}
|
|
|
|
else
|
2016-08-19 01:03:31 +02:00
|
|
|
WARN("wait object %p triggered while object was destroyed\n", wait);
|
2015-07-05 02:22:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Release temporary references to wait objects. */
|
|
|
|
while (num_handles)
|
|
|
|
{
|
|
|
|
wait = objects[--num_handles];
|
|
|
|
assert( wait->type == TP_OBJECT_TYPE_WAIT );
|
|
|
|
tp_object_release( wait );
|
|
|
|
}
|
|
|
|
}
|
2015-07-05 02:25:32 +02:00
|
|
|
|
|
|
|
/* Try to merge bucket with other threads. */
|
|
|
|
if (waitqueue.num_buckets > 1 && bucket->objcount &&
|
|
|
|
bucket->objcount <= MAXIMUM_WAITQUEUE_OBJECTS * 1 / 3)
|
|
|
|
{
|
|
|
|
struct waitqueue_bucket *other_bucket;
|
|
|
|
LIST_FOR_EACH_ENTRY( other_bucket, &waitqueue.buckets, struct waitqueue_bucket, bucket_entry )
|
|
|
|
{
|
|
|
|
if (other_bucket != bucket && other_bucket->objcount &&
|
|
|
|
other_bucket->objcount + bucket->objcount <= MAXIMUM_WAITQUEUE_OBJECTS * 2 / 3)
|
|
|
|
{
|
|
|
|
other_bucket->objcount += bucket->objcount;
|
|
|
|
bucket->objcount = 0;
|
|
|
|
|
|
|
|
/* Update reserved list. */
|
|
|
|
LIST_FOR_EACH_ENTRY( wait, &bucket->reserved, struct threadpool_object, u.wait.wait_entry )
|
|
|
|
{
|
|
|
|
assert( wait->type == TP_OBJECT_TYPE_WAIT );
|
|
|
|
wait->u.wait.bucket = other_bucket;
|
|
|
|
}
|
|
|
|
list_move_tail( &other_bucket->reserved, &bucket->reserved );
|
|
|
|
|
|
|
|
/* Update waiting list. */
|
|
|
|
LIST_FOR_EACH_ENTRY( wait, &bucket->waiting, struct threadpool_object, u.wait.wait_entry )
|
|
|
|
{
|
|
|
|
assert( wait->type == TP_OBJECT_TYPE_WAIT );
|
|
|
|
wait->u.wait.bucket = other_bucket;
|
|
|
|
}
|
|
|
|
list_move_tail( &other_bucket->waiting, &bucket->waiting );
|
|
|
|
|
|
|
|
/* Move bucket to the end, to keep the probability of
|
|
|
|
* newly added wait objects as small as possible. */
|
|
|
|
list_remove( &bucket->bucket_entry );
|
|
|
|
list_add_tail( &waitqueue.buckets, &bucket->bucket_entry );
|
|
|
|
|
|
|
|
NtSetEvent( other_bucket->update_event, NULL );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-07-05 02:22:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Remove this bucket from the list. */
|
|
|
|
list_remove( &bucket->bucket_entry );
|
|
|
|
if (!--waitqueue.num_buckets)
|
|
|
|
assert( list_empty( &waitqueue.buckets ) );
|
|
|
|
|
|
|
|
RtlLeaveCriticalSection( &waitqueue.cs );
|
|
|
|
|
|
|
|
TRACE( "terminating wait queue thread\n" );
|
|
|
|
|
|
|
|
assert( bucket->objcount == 0 );
|
|
|
|
assert( list_empty( &bucket->reserved ) );
|
|
|
|
assert( list_empty( &bucket->waiting ) );
|
|
|
|
NtClose( bucket->update_event );
|
|
|
|
|
|
|
|
RtlFreeHeap( GetProcessHeap(), 0, bucket );
|
2015-07-08 07:07:41 +02:00
|
|
|
RtlExitUserThread( 0 );
|
2015-07-05 02:22:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* tp_waitqueue_lock (internal)
|
|
|
|
*/
|
|
|
|
static NTSTATUS tp_waitqueue_lock( struct threadpool_object *wait )
|
|
|
|
{
|
|
|
|
struct waitqueue_bucket *bucket;
|
|
|
|
NTSTATUS status;
|
|
|
|
HANDLE thread;
|
2015-07-11 15:33:04 +02:00
|
|
|
assert( wait->type == TP_OBJECT_TYPE_WAIT );
|
2015-07-05 02:22:52 +02:00
|
|
|
|
|
|
|
wait->u.wait.signaled = 0;
|
|
|
|
wait->u.wait.bucket = NULL;
|
|
|
|
wait->u.wait.wait_pending = FALSE;
|
|
|
|
wait->u.wait.timeout = 0;
|
|
|
|
wait->u.wait.handle = INVALID_HANDLE_VALUE;
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &waitqueue.cs );
|
|
|
|
|
|
|
|
/* Try to assign to existing bucket if possible. */
|
|
|
|
LIST_FOR_EACH_ENTRY( bucket, &waitqueue.buckets, struct waitqueue_bucket, bucket_entry )
|
|
|
|
{
|
|
|
|
if (bucket->objcount < MAXIMUM_WAITQUEUE_OBJECTS)
|
|
|
|
{
|
|
|
|
list_add_tail( &bucket->reserved, &wait->u.wait.wait_entry );
|
|
|
|
wait->u.wait.bucket = bucket;
|
|
|
|
bucket->objcount++;
|
|
|
|
|
|
|
|
status = STATUS_SUCCESS;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Create a new bucket and corresponding worker thread. */
|
|
|
|
bucket = RtlAllocateHeap( GetProcessHeap(), 0, sizeof(*bucket) );
|
|
|
|
if (!bucket)
|
|
|
|
{
|
|
|
|
status = STATUS_NO_MEMORY;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
bucket->objcount = 0;
|
|
|
|
list_init( &bucket->reserved );
|
|
|
|
list_init( &bucket->waiting );
|
|
|
|
|
|
|
|
status = NtCreateEvent( &bucket->update_event, EVENT_ALL_ACCESS,
|
|
|
|
NULL, SynchronizationEvent, FALSE );
|
|
|
|
if (status)
|
|
|
|
{
|
|
|
|
RtlFreeHeap( GetProcessHeap(), 0, bucket );
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
status = RtlCreateUserThread( GetCurrentProcess(), NULL, FALSE, NULL, 0, 0,
|
|
|
|
waitqueue_thread_proc, bucket, &thread, NULL );
|
|
|
|
if (status == STATUS_SUCCESS)
|
|
|
|
{
|
|
|
|
list_add_tail( &waitqueue.buckets, &bucket->bucket_entry );
|
|
|
|
waitqueue.num_buckets++;
|
|
|
|
|
|
|
|
list_add_tail( &bucket->reserved, &wait->u.wait.wait_entry );
|
|
|
|
wait->u.wait.bucket = bucket;
|
|
|
|
bucket->objcount++;
|
|
|
|
|
|
|
|
NtClose( thread );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
NtClose( bucket->update_event );
|
|
|
|
RtlFreeHeap( GetProcessHeap(), 0, bucket );
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
RtlLeaveCriticalSection( &waitqueue.cs );
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* tp_waitqueue_unlock (internal)
|
|
|
|
*/
|
|
|
|
static void tp_waitqueue_unlock( struct threadpool_object *wait )
|
|
|
|
{
|
|
|
|
assert( wait->type == TP_OBJECT_TYPE_WAIT );
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &waitqueue.cs );
|
|
|
|
if (wait->u.wait.bucket)
|
|
|
|
{
|
|
|
|
struct waitqueue_bucket *bucket = wait->u.wait.bucket;
|
|
|
|
assert( bucket->objcount > 0 );
|
|
|
|
|
|
|
|
list_remove( &wait->u.wait.wait_entry );
|
|
|
|
wait->u.wait.bucket = NULL;
|
|
|
|
bucket->objcount--;
|
|
|
|
|
|
|
|
NtSetEvent( bucket->update_event, NULL );
|
|
|
|
}
|
|
|
|
RtlLeaveCriticalSection( &waitqueue.cs );
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* tp_threadpool_alloc (internal)
|
|
|
|
*
|
|
|
|
* Allocates a new threadpool object.
|
|
|
|
*/
|
|
|
|
static NTSTATUS tp_threadpool_alloc( struct threadpool **out )
|
|
|
|
{
|
2020-02-12 10:05:27 +01:00
|
|
|
IMAGE_NT_HEADERS *nt = RtlImageNtHeader( NtCurrentTeb()->Peb->ImageBaseAddress );
|
2015-07-01 02:55:27 +02:00
|
|
|
struct threadpool *pool;
|
2019-03-13 07:28:45 +01:00
|
|
|
unsigned int i;
|
2015-07-01 02:55:27 +02:00
|
|
|
|
|
|
|
pool = RtlAllocateHeap( GetProcessHeap(), 0, sizeof(*pool) );
|
|
|
|
if (!pool)
|
|
|
|
return STATUS_NO_MEMORY;
|
|
|
|
|
|
|
|
pool->refcount = 1;
|
|
|
|
pool->objcount = 0;
|
|
|
|
pool->shutdown = FALSE;
|
|
|
|
|
|
|
|
RtlInitializeCriticalSection( &pool->cs );
|
|
|
|
pool->cs.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": threadpool.cs");
|
|
|
|
|
2019-03-13 07:28:45 +01:00
|
|
|
for (i = 0; i < ARRAY_SIZE(pool->pools); ++i)
|
|
|
|
list_init( &pool->pools[i] );
|
2015-07-01 02:55:27 +02:00
|
|
|
RtlInitializeConditionVariable( &pool->update_event );
|
|
|
|
|
2020-02-12 10:05:27 +01:00
|
|
|
pool->max_workers = 500;
|
|
|
|
pool->min_workers = 0;
|
|
|
|
pool->num_workers = 0;
|
|
|
|
pool->num_busy_workers = 0;
|
|
|
|
pool->stack_info.StackReserve = nt->OptionalHeader.SizeOfStackReserve;
|
|
|
|
pool->stack_info.StackCommit = nt->OptionalHeader.SizeOfStackCommit;
|
2015-07-01 02:55:27 +02:00
|
|
|
|
|
|
|
TRACE( "allocated threadpool %p\n", pool );
|
|
|
|
|
|
|
|
*out = pool;
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* tp_threadpool_shutdown (internal)
|
|
|
|
*
|
|
|
|
* Prepares the shutdown of a threadpool object and notifies all worker
|
|
|
|
* threads to terminate (after all remaining work items have been
|
|
|
|
* processed).
|
|
|
|
*/
|
|
|
|
static void tp_threadpool_shutdown( struct threadpool *pool )
|
|
|
|
{
|
|
|
|
assert( pool != default_threadpool );
|
|
|
|
|
|
|
|
pool->shutdown = TRUE;
|
|
|
|
RtlWakeAllConditionVariable( &pool->update_event );
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* tp_threadpool_release (internal)
|
|
|
|
*
|
|
|
|
* Releases a reference to a threadpool object.
|
|
|
|
*/
|
|
|
|
static BOOL tp_threadpool_release( struct threadpool *pool )
|
|
|
|
{
|
2019-03-13 07:28:45 +01:00
|
|
|
unsigned int i;
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
if (interlocked_dec( &pool->refcount ))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
TRACE( "destroying threadpool %p\n", pool );
|
|
|
|
|
|
|
|
assert( pool->shutdown );
|
|
|
|
assert( !pool->objcount );
|
2019-03-13 07:28:45 +01:00
|
|
|
for (i = 0; i < ARRAY_SIZE(pool->pools); ++i)
|
|
|
|
assert( list_empty( &pool->pools[i] ) );
|
2015-07-01 02:55:27 +02:00
|
|
|
|
|
|
|
pool->cs.DebugInfo->Spare[0] = 0;
|
|
|
|
RtlDeleteCriticalSection( &pool->cs );
|
|
|
|
|
|
|
|
RtlFreeHeap( GetProcessHeap(), 0, pool );
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* tp_threadpool_lock (internal)
|
|
|
|
*
|
|
|
|
* Acquires a lock on a threadpool, specified with an TP_CALLBACK_ENVIRON
|
|
|
|
* block. When the lock is acquired successfully, it is guaranteed that
|
|
|
|
* there is at least one worker thread to process tasks.
|
|
|
|
*/
|
|
|
|
static NTSTATUS tp_threadpool_lock( struct threadpool **out, TP_CALLBACK_ENVIRON *environment )
|
|
|
|
{
|
|
|
|
struct threadpool *pool = NULL;
|
|
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
|
|
|
|
|
|
if (environment)
|
2019-03-13 07:28:45 +01:00
|
|
|
{
|
|
|
|
/* Validate environment parameters. */
|
|
|
|
if (environment->Version == 3)
|
|
|
|
{
|
|
|
|
TP_CALLBACK_ENVIRON_V3 *environment3 = (TP_CALLBACK_ENVIRON_V3 *)environment;
|
|
|
|
|
|
|
|
switch (environment3->CallbackPriority)
|
|
|
|
{
|
|
|
|
case TP_CALLBACK_PRIORITY_HIGH:
|
|
|
|
case TP_CALLBACK_PRIORITY_NORMAL:
|
|
|
|
case TP_CALLBACK_PRIORITY_LOW:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
pool = (struct threadpool *)environment->Pool;
|
2019-03-13 07:28:45 +01:00
|
|
|
}
|
2015-07-01 02:55:27 +02:00
|
|
|
|
|
|
|
if (!pool)
|
|
|
|
{
|
|
|
|
if (!default_threadpool)
|
|
|
|
{
|
|
|
|
status = tp_threadpool_alloc( &pool );
|
|
|
|
if (status != STATUS_SUCCESS)
|
|
|
|
return status;
|
|
|
|
|
|
|
|
if (interlocked_cmpxchg_ptr( (void *)&default_threadpool, pool, NULL ) != NULL)
|
|
|
|
{
|
|
|
|
tp_threadpool_shutdown( pool );
|
|
|
|
tp_threadpool_release( pool );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pool = default_threadpool;
|
|
|
|
}
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &pool->cs );
|
|
|
|
|
|
|
|
/* Make sure that the threadpool has at least one thread. */
|
|
|
|
if (!pool->num_workers)
|
2016-08-22 22:39:10 +02:00
|
|
|
status = tp_new_worker_thread( pool );
|
2015-07-01 02:55:27 +02:00
|
|
|
|
|
|
|
/* Keep a reference, and increment objcount to ensure that the
|
|
|
|
* last thread doesn't terminate. */
|
|
|
|
if (status == STATUS_SUCCESS)
|
|
|
|
{
|
|
|
|
interlocked_inc( &pool->refcount );
|
|
|
|
pool->objcount++;
|
|
|
|
}
|
|
|
|
|
|
|
|
RtlLeaveCriticalSection( &pool->cs );
|
|
|
|
|
|
|
|
if (status != STATUS_SUCCESS)
|
|
|
|
return status;
|
|
|
|
|
|
|
|
*out = pool;
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* tp_threadpool_unlock (internal)
|
|
|
|
*
|
|
|
|
* Releases a lock on a threadpool.
|
|
|
|
*/
|
|
|
|
static void tp_threadpool_unlock( struct threadpool *pool )
|
|
|
|
{
|
|
|
|
RtlEnterCriticalSection( &pool->cs );
|
|
|
|
pool->objcount--;
|
|
|
|
RtlLeaveCriticalSection( &pool->cs );
|
|
|
|
tp_threadpool_release( pool );
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:57:15 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* tp_group_alloc (internal)
|
|
|
|
*
|
|
|
|
* Allocates a new threadpool group object.
|
|
|
|
*/
|
|
|
|
static NTSTATUS tp_group_alloc( struct threadpool_group **out )
|
|
|
|
{
|
|
|
|
struct threadpool_group *group;
|
|
|
|
|
|
|
|
group = RtlAllocateHeap( GetProcessHeap(), 0, sizeof(*group) );
|
|
|
|
if (!group)
|
|
|
|
return STATUS_NO_MEMORY;
|
|
|
|
|
|
|
|
group->refcount = 1;
|
|
|
|
group->shutdown = FALSE;
|
|
|
|
|
|
|
|
RtlInitializeCriticalSection( &group->cs );
|
|
|
|
group->cs.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": threadpool_group.cs");
|
|
|
|
|
|
|
|
list_init( &group->members );
|
|
|
|
|
|
|
|
TRACE( "allocated group %p\n", group );
|
|
|
|
|
|
|
|
*out = group;
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* tp_group_shutdown (internal)
|
|
|
|
*
|
|
|
|
* Marks the group object for shutdown.
|
|
|
|
*/
|
|
|
|
static void tp_group_shutdown( struct threadpool_group *group )
|
|
|
|
{
|
|
|
|
group->shutdown = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* tp_group_release (internal)
|
|
|
|
*
|
|
|
|
* Releases a reference to a group object.
|
|
|
|
*/
|
|
|
|
static BOOL tp_group_release( struct threadpool_group *group )
|
|
|
|
{
|
|
|
|
if (interlocked_dec( &group->refcount ))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
TRACE( "destroying group %p\n", group );
|
|
|
|
|
|
|
|
assert( group->shutdown );
|
|
|
|
assert( list_empty( &group->members ) );
|
|
|
|
|
|
|
|
group->cs.DebugInfo->Spare[0] = 0;
|
|
|
|
RtlDeleteCriticalSection( &group->cs );
|
|
|
|
|
|
|
|
RtlFreeHeap( GetProcessHeap(), 0, group );
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* tp_object_initialize (internal)
|
|
|
|
*
|
|
|
|
* Initializes members of a threadpool object.
|
|
|
|
*/
|
|
|
|
static void tp_object_initialize( struct threadpool_object *object, struct threadpool *pool,
|
|
|
|
PVOID userdata, TP_CALLBACK_ENVIRON *environment )
|
|
|
|
{
|
2015-07-01 02:57:15 +02:00
|
|
|
BOOL is_simple_callback = (object->type == TP_OBJECT_TYPE_SIMPLE);
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
object->refcount = 1;
|
|
|
|
object->shutdown = FALSE;
|
|
|
|
|
|
|
|
object->pool = pool;
|
2015-07-01 02:57:15 +02:00
|
|
|
object->group = NULL;
|
2015-07-01 02:55:27 +02:00
|
|
|
object->userdata = userdata;
|
2015-07-01 22:52:52 +02:00
|
|
|
object->group_cancel_callback = NULL;
|
2015-07-01 22:53:26 +02:00
|
|
|
object->finalization_callback = NULL;
|
2015-07-01 22:55:30 +02:00
|
|
|
object->may_run_long = 0;
|
2015-07-01 22:53:44 +02:00
|
|
|
object->race_dll = NULL;
|
2019-03-13 07:28:45 +01:00
|
|
|
object->priority = TP_CALLBACK_PRIORITY_NORMAL;
|
2015-07-01 02:55:27 +02:00
|
|
|
|
2015-07-01 02:57:15 +02:00
|
|
|
memset( &object->group_entry, 0, sizeof(object->group_entry) );
|
|
|
|
object->is_group_member = FALSE;
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
memset( &object->pool_entry, 0, sizeof(object->pool_entry) );
|
2015-07-01 02:57:15 +02:00
|
|
|
RtlInitializeConditionVariable( &object->finished_event );
|
2015-07-03 01:57:23 +02:00
|
|
|
RtlInitializeConditionVariable( &object->group_finished_event );
|
2015-07-01 02:55:27 +02:00
|
|
|
object->num_pending_callbacks = 0;
|
|
|
|
object->num_running_callbacks = 0;
|
2015-07-03 01:57:23 +02:00
|
|
|
object->num_associated_callbacks = 0;
|
2015-07-01 02:55:27 +02:00
|
|
|
|
|
|
|
if (environment)
|
2015-07-01 02:57:15 +02:00
|
|
|
{
|
2016-03-31 04:53:51 +02:00
|
|
|
if (environment->Version != 1 && environment->Version != 3)
|
2015-07-01 02:57:15 +02:00
|
|
|
FIXME( "unsupported environment version %u\n", environment->Version );
|
|
|
|
|
|
|
|
object->group = impl_from_TP_CLEANUP_GROUP( environment->CleanupGroup );
|
2015-07-01 22:52:52 +02:00
|
|
|
object->group_cancel_callback = environment->CleanupGroupCancelCallback;
|
2015-07-01 22:53:26 +02:00
|
|
|
object->finalization_callback = environment->FinalizationCallback;
|
2015-07-01 22:55:30 +02:00
|
|
|
object->may_run_long = environment->u.s.LongFunction != 0;
|
2015-07-01 22:53:44 +02:00
|
|
|
object->race_dll = environment->RaceDll;
|
2019-03-13 07:28:45 +01:00
|
|
|
if (environment->Version == 3)
|
|
|
|
{
|
|
|
|
TP_CALLBACK_ENVIRON_V3 *environment_v3 = (TP_CALLBACK_ENVIRON_V3 *)environment;
|
|
|
|
|
|
|
|
object->priority = environment_v3->CallbackPriority;
|
|
|
|
assert( object->priority < ARRAY_SIZE(pool->pools) );
|
|
|
|
}
|
2015-07-01 02:57:15 +02:00
|
|
|
|
2015-07-01 22:55:30 +02:00
|
|
|
if (environment->ActivationContext)
|
|
|
|
FIXME( "activation context not supported yet\n" );
|
|
|
|
|
|
|
|
if (environment->u.s.Persistent)
|
|
|
|
FIXME( "persistent threads not supported yet\n" );
|
2015-07-01 02:57:15 +02:00
|
|
|
}
|
2015-07-01 02:55:27 +02:00
|
|
|
|
2015-07-01 22:53:44 +02:00
|
|
|
if (object->race_dll)
|
|
|
|
LdrAddRefDll( 0, object->race_dll );
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
TRACE( "allocated object %p of type %u\n", object, object->type );
|
2015-07-01 02:57:15 +02:00
|
|
|
|
|
|
|
/* For simple callbacks we have to run tp_object_submit before adding this object
|
|
|
|
* to the cleanup group. As soon as the cleanup group members are released ->shutdown
|
|
|
|
* will be set, and tp_object_submit would fail with an assertion. */
|
|
|
|
|
|
|
|
if (is_simple_callback)
|
2015-07-05 02:22:52 +02:00
|
|
|
tp_object_submit( object, FALSE );
|
2015-07-01 02:57:15 +02:00
|
|
|
|
|
|
|
if (object->group)
|
|
|
|
{
|
|
|
|
struct threadpool_group *group = object->group;
|
|
|
|
interlocked_inc( &group->refcount );
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &group->cs );
|
|
|
|
list_add_tail( &group->members, &object->group_entry );
|
|
|
|
object->is_group_member = TRUE;
|
|
|
|
RtlLeaveCriticalSection( &group->cs );
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_simple_callback)
|
|
|
|
tp_object_release( object );
|
2015-07-01 02:55:27 +02:00
|
|
|
}
|
|
|
|
|
2019-03-13 07:28:45 +01:00
|
|
|
static void tp_object_prio_queue( struct threadpool_object *object )
|
|
|
|
{
|
|
|
|
list_add_tail( &object->pool->pools[object->priority], &object->pool_entry );
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* tp_object_submit (internal)
|
|
|
|
*
|
2016-05-18 12:10:32 +02:00
|
|
|
* Submits a threadpool object to the associated threadpool. This
|
2015-07-01 02:55:27 +02:00
|
|
|
* function has to be VOID because TpPostWork can never fail on Windows.
|
|
|
|
*/
|
2015-07-05 02:22:52 +02:00
|
|
|
static void tp_object_submit( struct threadpool_object *object, BOOL signaled )
|
2015-07-01 02:55:27 +02:00
|
|
|
{
|
|
|
|
struct threadpool *pool = object->pool;
|
|
|
|
NTSTATUS status = STATUS_UNSUCCESSFUL;
|
|
|
|
|
|
|
|
assert( !object->shutdown );
|
|
|
|
assert( !pool->shutdown );
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &pool->cs );
|
|
|
|
|
|
|
|
/* Start new worker threads if required. */
|
|
|
|
if (pool->num_busy_workers >= pool->num_workers &&
|
|
|
|
pool->num_workers < pool->max_workers)
|
2016-08-22 22:39:10 +02:00
|
|
|
status = tp_new_worker_thread( pool );
|
2015-07-01 02:55:27 +02:00
|
|
|
|
|
|
|
/* Queue work item and increment refcount. */
|
|
|
|
interlocked_inc( &object->refcount );
|
|
|
|
if (!object->num_pending_callbacks++)
|
2019-03-13 07:28:45 +01:00
|
|
|
tp_object_prio_queue( object );
|
2015-07-01 02:55:27 +02:00
|
|
|
|
2015-07-05 02:22:52 +02:00
|
|
|
/* Count how often the object was signaled. */
|
|
|
|
if (object->type == TP_OBJECT_TYPE_WAIT && signaled)
|
|
|
|
object->u.wait.signaled++;
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
/* No new thread started - wake up one existing thread. */
|
|
|
|
if (status != STATUS_SUCCESS)
|
|
|
|
{
|
|
|
|
assert( pool->num_workers > 0 );
|
|
|
|
RtlWakeConditionVariable( &pool->update_event );
|
|
|
|
}
|
|
|
|
|
|
|
|
RtlLeaveCriticalSection( &pool->cs );
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:57:15 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* tp_object_cancel (internal)
|
|
|
|
*
|
|
|
|
* Cancels all currently pending callbacks for a specific object.
|
|
|
|
*/
|
2016-08-20 21:23:48 +02:00
|
|
|
static void tp_object_cancel( struct threadpool_object *object )
|
2015-07-01 02:57:15 +02:00
|
|
|
{
|
|
|
|
struct threadpool *pool = object->pool;
|
|
|
|
LONG pending_callbacks = 0;
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &pool->cs );
|
|
|
|
if (object->num_pending_callbacks)
|
|
|
|
{
|
|
|
|
pending_callbacks = object->num_pending_callbacks;
|
|
|
|
object->num_pending_callbacks = 0;
|
|
|
|
list_remove( &object->pool_entry );
|
2015-07-05 02:22:52 +02:00
|
|
|
|
|
|
|
if (object->type == TP_OBJECT_TYPE_WAIT)
|
|
|
|
object->u.wait.signaled = 0;
|
2015-07-01 02:57:15 +02:00
|
|
|
}
|
|
|
|
RtlLeaveCriticalSection( &pool->cs );
|
|
|
|
|
|
|
|
while (pending_callbacks--)
|
|
|
|
tp_object_release( object );
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* tp_object_wait (internal)
|
|
|
|
*
|
|
|
|
* Waits until all pending and running callbacks of a specific object
|
|
|
|
* have been processed.
|
|
|
|
*/
|
2015-07-03 01:57:23 +02:00
|
|
|
static void tp_object_wait( struct threadpool_object *object, BOOL group_wait )
|
2015-07-01 02:57:15 +02:00
|
|
|
{
|
|
|
|
struct threadpool *pool = object->pool;
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &pool->cs );
|
2015-07-03 01:57:23 +02:00
|
|
|
if (group_wait)
|
|
|
|
{
|
|
|
|
while (object->num_pending_callbacks || object->num_running_callbacks)
|
|
|
|
RtlSleepConditionVariableCS( &object->group_finished_event, &pool->cs, NULL );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while (object->num_pending_callbacks || object->num_associated_callbacks)
|
|
|
|
RtlSleepConditionVariableCS( &object->finished_event, &pool->cs, NULL );
|
|
|
|
}
|
2015-07-01 02:57:15 +02:00
|
|
|
RtlLeaveCriticalSection( &pool->cs );
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
/***********************************************************************
|
2016-08-20 21:22:29 +02:00
|
|
|
* tp_object_prepare_shutdown (internal)
|
2015-07-01 02:55:27 +02:00
|
|
|
*
|
2016-08-20 21:22:29 +02:00
|
|
|
* Prepares a threadpool object for shutdown.
|
2015-07-01 02:55:27 +02:00
|
|
|
*/
|
2016-08-20 21:22:29 +02:00
|
|
|
static void tp_object_prepare_shutdown( struct threadpool_object *object )
|
2015-07-01 02:55:27 +02:00
|
|
|
{
|
2015-07-03 01:58:29 +02:00
|
|
|
if (object->type == TP_OBJECT_TYPE_TIMER)
|
|
|
|
tp_timerqueue_unlock( object );
|
2015-07-05 02:22:52 +02:00
|
|
|
else if (object->type == TP_OBJECT_TYPE_WAIT)
|
|
|
|
tp_waitqueue_unlock( object );
|
2015-07-01 02:55:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* tp_object_release (internal)
|
|
|
|
*
|
|
|
|
* Releases a reference to a threadpool object.
|
|
|
|
*/
|
|
|
|
static BOOL tp_object_release( struct threadpool_object *object )
|
|
|
|
{
|
|
|
|
if (interlocked_dec( &object->refcount ))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
TRACE( "destroying object %p of type %u\n", object, object->type );
|
|
|
|
|
|
|
|
assert( object->shutdown );
|
|
|
|
assert( !object->num_pending_callbacks );
|
|
|
|
assert( !object->num_running_callbacks );
|
2015-07-03 01:57:23 +02:00
|
|
|
assert( !object->num_associated_callbacks );
|
2015-07-01 02:55:27 +02:00
|
|
|
|
2015-07-01 02:57:15 +02:00
|
|
|
/* release reference to the group */
|
|
|
|
if (object->group)
|
|
|
|
{
|
|
|
|
struct threadpool_group *group = object->group;
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &group->cs );
|
|
|
|
if (object->is_group_member)
|
|
|
|
{
|
|
|
|
list_remove( &object->group_entry );
|
|
|
|
object->is_group_member = FALSE;
|
|
|
|
}
|
|
|
|
RtlLeaveCriticalSection( &group->cs );
|
|
|
|
|
|
|
|
tp_group_release( group );
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
tp_threadpool_unlock( object->pool );
|
|
|
|
|
2015-07-01 22:53:44 +02:00
|
|
|
if (object->race_dll)
|
|
|
|
LdrUnloadDll( object->race_dll );
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
RtlFreeHeap( GetProcessHeap(), 0, object );
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2019-03-13 07:28:45 +01:00
|
|
|
static struct list *threadpool_get_next_item( const struct threadpool *pool )
|
|
|
|
{
|
|
|
|
struct list *ptr;
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(pool->pools); ++i)
|
|
|
|
{
|
|
|
|
if ((ptr = list_head( &pool->pools[i] )))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* threadpool_worker_proc (internal)
|
|
|
|
*/
|
|
|
|
static void CALLBACK threadpool_worker_proc( void *param )
|
|
|
|
{
|
2015-07-01 22:55:30 +02:00
|
|
|
TP_CALLBACK_INSTANCE *callback_instance;
|
|
|
|
struct threadpool_instance instance;
|
2015-07-01 02:55:27 +02:00
|
|
|
struct threadpool *pool = param;
|
2015-07-05 02:22:52 +02:00
|
|
|
TP_WAIT_RESULT wait_result = 0;
|
2015-07-01 02:55:27 +02:00
|
|
|
LARGE_INTEGER timeout;
|
|
|
|
struct list *ptr;
|
2015-07-01 22:56:51 +02:00
|
|
|
NTSTATUS status;
|
2015-07-01 02:55:27 +02:00
|
|
|
|
|
|
|
TRACE( "starting worker thread for pool %p\n", pool );
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &pool->cs );
|
2015-07-29 22:10:01 +02:00
|
|
|
pool->num_busy_workers--;
|
2015-07-01 02:55:27 +02:00
|
|
|
for (;;)
|
|
|
|
{
|
2019-03-13 07:28:45 +01:00
|
|
|
while ((ptr = threadpool_get_next_item( pool )))
|
2015-07-01 02:55:27 +02:00
|
|
|
{
|
|
|
|
struct threadpool_object *object = LIST_ENTRY( ptr, struct threadpool_object, pool_entry );
|
|
|
|
assert( object->num_pending_callbacks > 0 );
|
|
|
|
|
|
|
|
/* If further pending callbacks are queued, move the work item to
|
|
|
|
* the end of the pool list. Otherwise remove it from the pool. */
|
|
|
|
list_remove( &object->pool_entry );
|
|
|
|
if (--object->num_pending_callbacks)
|
2019-03-13 07:28:45 +01:00
|
|
|
tp_object_prio_queue( object );
|
2015-07-01 02:55:27 +02:00
|
|
|
|
2015-07-05 02:22:52 +02:00
|
|
|
/* For wait objects check if they were signaled or have timed out. */
|
|
|
|
if (object->type == TP_OBJECT_TYPE_WAIT)
|
|
|
|
{
|
|
|
|
wait_result = object->u.wait.signaled ? WAIT_OBJECT_0 : WAIT_TIMEOUT;
|
|
|
|
if (wait_result == WAIT_OBJECT_0) object->u.wait.signaled--;
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
/* Leave critical section and do the actual callback. */
|
2015-07-03 01:57:23 +02:00
|
|
|
object->num_associated_callbacks++;
|
2015-07-01 02:55:27 +02:00
|
|
|
object->num_running_callbacks++;
|
|
|
|
pool->num_busy_workers++;
|
|
|
|
RtlLeaveCriticalSection( &pool->cs );
|
|
|
|
|
2015-07-01 22:55:30 +02:00
|
|
|
/* Initialize threadpool instance struct. */
|
|
|
|
callback_instance = (TP_CALLBACK_INSTANCE *)&instance;
|
|
|
|
instance.object = object;
|
|
|
|
instance.threadid = GetCurrentThreadId();
|
2015-07-03 01:57:23 +02:00
|
|
|
instance.associated = TRUE;
|
2015-07-01 22:55:30 +02:00
|
|
|
instance.may_run_long = object->may_run_long;
|
2015-07-01 22:55:52 +02:00
|
|
|
instance.cleanup.critical_section = NULL;
|
2015-07-01 22:56:14 +02:00
|
|
|
instance.cleanup.mutex = NULL;
|
2015-07-01 22:56:51 +02:00
|
|
|
instance.cleanup.semaphore = NULL;
|
|
|
|
instance.cleanup.semaphore_count = 0;
|
2015-07-01 22:57:53 +02:00
|
|
|
instance.cleanup.event = NULL;
|
2015-07-01 22:58:23 +02:00
|
|
|
instance.cleanup.library = NULL;
|
2015-07-01 22:55:30 +02:00
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
switch (object->type)
|
|
|
|
{
|
|
|
|
case TP_OBJECT_TYPE_SIMPLE:
|
|
|
|
{
|
2015-07-01 22:55:30 +02:00
|
|
|
TRACE( "executing simple callback %p(%p, %p)\n",
|
|
|
|
object->u.simple.callback, callback_instance, object->userdata );
|
|
|
|
object->u.simple.callback( callback_instance, object->userdata );
|
2015-07-01 02:55:27 +02:00
|
|
|
TRACE( "callback %p returned\n", object->u.simple.callback );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:57:57 +02:00
|
|
|
case TP_OBJECT_TYPE_WORK:
|
|
|
|
{
|
2015-07-01 22:55:30 +02:00
|
|
|
TRACE( "executing work callback %p(%p, %p, %p)\n",
|
|
|
|
object->u.work.callback, callback_instance, object->userdata, object );
|
|
|
|
object->u.work.callback( callback_instance, object->userdata, (TP_WORK *)object );
|
2015-07-01 02:57:57 +02:00
|
|
|
TRACE( "callback %p returned\n", object->u.work.callback );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-07-03 01:57:54 +02:00
|
|
|
case TP_OBJECT_TYPE_TIMER:
|
|
|
|
{
|
|
|
|
TRACE( "executing timer callback %p(%p, %p, %p)\n",
|
|
|
|
object->u.timer.callback, callback_instance, object->userdata, object );
|
|
|
|
object->u.timer.callback( callback_instance, object->userdata, (TP_TIMER *)object );
|
|
|
|
TRACE( "callback %p returned\n", object->u.timer.callback );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-07-05 02:22:11 +02:00
|
|
|
case TP_OBJECT_TYPE_WAIT:
|
|
|
|
{
|
|
|
|
TRACE( "executing wait callback %p(%p, %p, %p, %u)\n",
|
2015-07-05 02:22:52 +02:00
|
|
|
object->u.wait.callback, callback_instance, object->userdata, object, wait_result );
|
|
|
|
object->u.wait.callback( callback_instance, object->userdata, (TP_WAIT *)object, wait_result );
|
2015-07-05 02:22:11 +02:00
|
|
|
TRACE( "callback %p returned\n", object->u.wait.callback );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
default:
|
|
|
|
assert(0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-07-01 22:53:26 +02:00
|
|
|
/* Execute finalization callback. */
|
|
|
|
if (object->finalization_callback)
|
|
|
|
{
|
2015-07-01 22:55:30 +02:00
|
|
|
TRACE( "executing finalization callback %p(%p, %p)\n",
|
|
|
|
object->finalization_callback, callback_instance, object->userdata );
|
|
|
|
object->finalization_callback( callback_instance, object->userdata );
|
2015-07-01 22:53:26 +02:00
|
|
|
TRACE( "callback %p returned\n", object->finalization_callback );
|
|
|
|
}
|
|
|
|
|
2015-07-01 22:55:52 +02:00
|
|
|
/* Execute cleanup tasks. */
|
|
|
|
if (instance.cleanup.critical_section)
|
|
|
|
{
|
|
|
|
RtlLeaveCriticalSection( instance.cleanup.critical_section );
|
|
|
|
}
|
2015-07-01 22:56:14 +02:00
|
|
|
if (instance.cleanup.mutex)
|
|
|
|
{
|
2015-07-01 22:56:51 +02:00
|
|
|
status = NtReleaseMutant( instance.cleanup.mutex, NULL );
|
|
|
|
if (status != STATUS_SUCCESS) goto skip_cleanup;
|
|
|
|
}
|
|
|
|
if (instance.cleanup.semaphore)
|
|
|
|
{
|
2015-07-01 22:57:53 +02:00
|
|
|
status = NtReleaseSemaphore( instance.cleanup.semaphore, instance.cleanup.semaphore_count, NULL );
|
|
|
|
if (status != STATUS_SUCCESS) goto skip_cleanup;
|
|
|
|
}
|
|
|
|
if (instance.cleanup.event)
|
|
|
|
{
|
2015-07-01 22:58:23 +02:00
|
|
|
status = NtSetEvent( instance.cleanup.event, NULL );
|
|
|
|
if (status != STATUS_SUCCESS) goto skip_cleanup;
|
|
|
|
}
|
|
|
|
if (instance.cleanup.library)
|
|
|
|
{
|
|
|
|
LdrUnloadDll( instance.cleanup.library );
|
2015-07-01 22:56:14 +02:00
|
|
|
}
|
2015-07-01 22:55:52 +02:00
|
|
|
|
2015-07-01 22:56:51 +02:00
|
|
|
skip_cleanup:
|
2015-07-01 02:55:27 +02:00
|
|
|
RtlEnterCriticalSection( &pool->cs );
|
|
|
|
pool->num_busy_workers--;
|
2015-07-03 01:57:23 +02:00
|
|
|
|
2016-08-20 22:38:43 +02:00
|
|
|
/* Simple callbacks are automatically shutdown after execution. */
|
|
|
|
if (object->type == TP_OBJECT_TYPE_SIMPLE)
|
|
|
|
{
|
|
|
|
tp_object_prepare_shutdown( object );
|
|
|
|
object->shutdown = TRUE;
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
object->num_running_callbacks--;
|
2015-07-01 02:57:15 +02:00
|
|
|
if (!object->num_pending_callbacks && !object->num_running_callbacks)
|
2015-07-03 01:57:23 +02:00
|
|
|
RtlWakeAllConditionVariable( &object->group_finished_event );
|
|
|
|
|
|
|
|
if (instance.associated)
|
|
|
|
{
|
|
|
|
object->num_associated_callbacks--;
|
|
|
|
if (!object->num_pending_callbacks && !object->num_associated_callbacks)
|
|
|
|
RtlWakeAllConditionVariable( &object->finished_event );
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
tp_object_release( object );
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Shutdown worker thread if requested. */
|
|
|
|
if (pool->shutdown)
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* Wait for new tasks or until the timeout expires. A thread only terminates
|
|
|
|
* when no new tasks are available, and the number of threads can be
|
|
|
|
* decreased without violating the min_workers limit. An exception is when
|
|
|
|
* min_workers == 0, then objcount is used to detect if the last thread
|
|
|
|
* can be terminated. */
|
|
|
|
timeout.QuadPart = (ULONGLONG)THREADPOOL_WORKER_TIMEOUT * -10000;
|
|
|
|
if (RtlSleepConditionVariableCS( &pool->update_event, &pool->cs, &timeout ) == STATUS_TIMEOUT &&
|
2019-03-13 07:28:45 +01:00
|
|
|
!threadpool_get_next_item( pool ) && (pool->num_workers > max( pool->min_workers, 1 ) ||
|
2015-07-01 02:55:27 +02:00
|
|
|
(!pool->min_workers && !pool->objcount)))
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pool->num_workers--;
|
|
|
|
RtlLeaveCriticalSection( &pool->cs );
|
|
|
|
|
|
|
|
TRACE( "terminating worker thread for pool %p\n", pool );
|
|
|
|
tp_threadpool_release( pool );
|
2015-07-08 07:07:41 +02:00
|
|
|
RtlExitUserThread( 0 );
|
2015-07-01 02:55:27 +02:00
|
|
|
}
|
|
|
|
|
2015-07-01 02:57:15 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpAllocCleanupGroup (NTDLL.@)
|
|
|
|
*/
|
|
|
|
NTSTATUS WINAPI TpAllocCleanupGroup( TP_CLEANUP_GROUP **out )
|
|
|
|
{
|
|
|
|
TRACE( "%p\n", out );
|
|
|
|
|
|
|
|
return tp_group_alloc( (struct threadpool_group **)out );
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpAllocPool (NTDLL.@)
|
|
|
|
*/
|
|
|
|
NTSTATUS WINAPI TpAllocPool( TP_POOL **out, PVOID reserved )
|
|
|
|
{
|
|
|
|
TRACE( "%p %p\n", out, reserved );
|
|
|
|
|
|
|
|
if (reserved)
|
2015-08-07 11:55:51 +02:00
|
|
|
FIXME( "reserved argument is nonzero (%p)\n", reserved );
|
2015-07-01 02:55:27 +02:00
|
|
|
|
|
|
|
return tp_threadpool_alloc( (struct threadpool **)out );
|
|
|
|
}
|
|
|
|
|
2015-07-03 01:57:54 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpAllocTimer (NTDLL.@)
|
|
|
|
*/
|
|
|
|
NTSTATUS WINAPI TpAllocTimer( TP_TIMER **out, PTP_TIMER_CALLBACK callback, PVOID userdata,
|
|
|
|
TP_CALLBACK_ENVIRON *environment )
|
|
|
|
{
|
|
|
|
struct threadpool_object *object;
|
|
|
|
struct threadpool *pool;
|
|
|
|
NTSTATUS status;
|
|
|
|
|
|
|
|
TRACE( "%p %p %p %p\n", out, callback, userdata, environment );
|
|
|
|
|
|
|
|
object = RtlAllocateHeap( GetProcessHeap(), 0, sizeof(*object) );
|
|
|
|
if (!object)
|
|
|
|
return STATUS_NO_MEMORY;
|
|
|
|
|
|
|
|
status = tp_threadpool_lock( &pool, environment );
|
|
|
|
if (status)
|
|
|
|
{
|
|
|
|
RtlFreeHeap( GetProcessHeap(), 0, object );
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
object->type = TP_OBJECT_TYPE_TIMER;
|
|
|
|
object->u.timer.callback = callback;
|
2015-07-03 01:58:29 +02:00
|
|
|
|
|
|
|
status = tp_timerqueue_lock( object );
|
|
|
|
if (status)
|
|
|
|
{
|
|
|
|
tp_threadpool_unlock( pool );
|
|
|
|
RtlFreeHeap( GetProcessHeap(), 0, object );
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2015-07-03 01:57:54 +02:00
|
|
|
tp_object_initialize( object, pool, userdata, environment );
|
|
|
|
|
|
|
|
*out = (TP_TIMER *)object;
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2015-07-05 02:22:11 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpAllocWait (NTDLL.@)
|
|
|
|
*/
|
|
|
|
NTSTATUS WINAPI TpAllocWait( TP_WAIT **out, PTP_WAIT_CALLBACK callback, PVOID userdata,
|
|
|
|
TP_CALLBACK_ENVIRON *environment )
|
|
|
|
{
|
|
|
|
struct threadpool_object *object;
|
|
|
|
struct threadpool *pool;
|
|
|
|
NTSTATUS status;
|
|
|
|
|
|
|
|
TRACE( "%p %p %p %p\n", out, callback, userdata, environment );
|
|
|
|
|
|
|
|
object = RtlAllocateHeap( GetProcessHeap(), 0, sizeof(*object) );
|
|
|
|
if (!object)
|
|
|
|
return STATUS_NO_MEMORY;
|
|
|
|
|
|
|
|
status = tp_threadpool_lock( &pool, environment );
|
|
|
|
if (status)
|
|
|
|
{
|
|
|
|
RtlFreeHeap( GetProcessHeap(), 0, object );
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
object->type = TP_OBJECT_TYPE_WAIT;
|
|
|
|
object->u.wait.callback = callback;
|
2015-07-05 02:22:52 +02:00
|
|
|
|
|
|
|
status = tp_waitqueue_lock( object );
|
|
|
|
if (status)
|
|
|
|
{
|
|
|
|
tp_threadpool_unlock( pool );
|
|
|
|
RtlFreeHeap( GetProcessHeap(), 0, object );
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2015-07-05 02:22:11 +02:00
|
|
|
tp_object_initialize( object, pool, userdata, environment );
|
|
|
|
|
|
|
|
*out = (TP_WAIT *)object;
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:57:57 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpAllocWork (NTDLL.@)
|
|
|
|
*/
|
|
|
|
NTSTATUS WINAPI TpAllocWork( TP_WORK **out, PTP_WORK_CALLBACK callback, PVOID userdata,
|
|
|
|
TP_CALLBACK_ENVIRON *environment )
|
|
|
|
{
|
|
|
|
struct threadpool_object *object;
|
|
|
|
struct threadpool *pool;
|
|
|
|
NTSTATUS status;
|
|
|
|
|
|
|
|
TRACE( "%p %p %p %p\n", out, callback, userdata, environment );
|
|
|
|
|
|
|
|
object = RtlAllocateHeap( GetProcessHeap(), 0, sizeof(*object) );
|
|
|
|
if (!object)
|
|
|
|
return STATUS_NO_MEMORY;
|
|
|
|
|
|
|
|
status = tp_threadpool_lock( &pool, environment );
|
|
|
|
if (status)
|
|
|
|
{
|
|
|
|
RtlFreeHeap( GetProcessHeap(), 0, object );
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
object->type = TP_OBJECT_TYPE_WORK;
|
|
|
|
object->u.work.callback = callback;
|
|
|
|
tp_object_initialize( object, pool, userdata, environment );
|
|
|
|
|
|
|
|
*out = (TP_WORK *)object;
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2015-07-01 22:55:52 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpCallbackLeaveCriticalSectionOnCompletion (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpCallbackLeaveCriticalSectionOnCompletion( TP_CALLBACK_INSTANCE *instance, CRITICAL_SECTION *crit )
|
|
|
|
{
|
|
|
|
struct threadpool_instance *this = impl_from_TP_CALLBACK_INSTANCE( instance );
|
|
|
|
|
|
|
|
TRACE( "%p %p\n", instance, crit );
|
|
|
|
|
|
|
|
if (!this->cleanup.critical_section)
|
|
|
|
this->cleanup.critical_section = crit;
|
|
|
|
}
|
|
|
|
|
2015-07-01 22:55:30 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpCallbackMayRunLong (NTDLL.@)
|
|
|
|
*/
|
|
|
|
NTSTATUS WINAPI TpCallbackMayRunLong( TP_CALLBACK_INSTANCE *instance )
|
|
|
|
{
|
|
|
|
struct threadpool_instance *this = impl_from_TP_CALLBACK_INSTANCE( instance );
|
|
|
|
struct threadpool_object *object = this->object;
|
|
|
|
struct threadpool *pool;
|
|
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
|
|
|
|
|
|
TRACE( "%p\n", instance );
|
|
|
|
|
|
|
|
if (this->threadid != GetCurrentThreadId())
|
|
|
|
{
|
|
|
|
ERR("called from wrong thread, ignoring\n");
|
|
|
|
return STATUS_UNSUCCESSFUL; /* FIXME */
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this->may_run_long)
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
|
|
|
|
pool = object->pool;
|
|
|
|
RtlEnterCriticalSection( &pool->cs );
|
|
|
|
|
|
|
|
/* Start new worker threads if required. */
|
|
|
|
if (pool->num_busy_workers >= pool->num_workers)
|
|
|
|
{
|
|
|
|
if (pool->num_workers < pool->max_workers)
|
|
|
|
{
|
2016-08-22 22:39:10 +02:00
|
|
|
status = tp_new_worker_thread( pool );
|
2015-07-01 22:55:30 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
status = STATUS_TOO_MANY_THREADS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
RtlLeaveCriticalSection( &pool->cs );
|
|
|
|
this->may_run_long = TRUE;
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2015-07-01 22:56:14 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpCallbackReleaseMutexOnCompletion (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpCallbackReleaseMutexOnCompletion( TP_CALLBACK_INSTANCE *instance, HANDLE mutex )
|
|
|
|
{
|
|
|
|
struct threadpool_instance *this = impl_from_TP_CALLBACK_INSTANCE( instance );
|
|
|
|
|
|
|
|
TRACE( "%p %p\n", instance, mutex );
|
|
|
|
|
|
|
|
if (!this->cleanup.mutex)
|
|
|
|
this->cleanup.mutex = mutex;
|
|
|
|
}
|
|
|
|
|
2015-07-01 22:56:51 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpCallbackReleaseSemaphoreOnCompletion (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpCallbackReleaseSemaphoreOnCompletion( TP_CALLBACK_INSTANCE *instance, HANDLE semaphore, DWORD count )
|
|
|
|
{
|
|
|
|
struct threadpool_instance *this = impl_from_TP_CALLBACK_INSTANCE( instance );
|
|
|
|
|
|
|
|
TRACE( "%p %p %u\n", instance, semaphore, count );
|
|
|
|
|
|
|
|
if (!this->cleanup.semaphore)
|
|
|
|
{
|
|
|
|
this->cleanup.semaphore = semaphore;
|
|
|
|
this->cleanup.semaphore_count = count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-01 22:57:53 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpCallbackSetEventOnCompletion (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpCallbackSetEventOnCompletion( TP_CALLBACK_INSTANCE *instance, HANDLE event )
|
|
|
|
{
|
|
|
|
struct threadpool_instance *this = impl_from_TP_CALLBACK_INSTANCE( instance );
|
|
|
|
|
|
|
|
TRACE( "%p %p\n", instance, event );
|
|
|
|
|
|
|
|
if (!this->cleanup.event)
|
|
|
|
this->cleanup.event = event;
|
|
|
|
}
|
|
|
|
|
2015-07-01 22:58:23 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpCallbackUnloadDllOnCompletion (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpCallbackUnloadDllOnCompletion( TP_CALLBACK_INSTANCE *instance, HMODULE module )
|
|
|
|
{
|
|
|
|
struct threadpool_instance *this = impl_from_TP_CALLBACK_INSTANCE( instance );
|
|
|
|
|
|
|
|
TRACE( "%p %p\n", instance, module );
|
|
|
|
|
|
|
|
if (!this->cleanup.library)
|
|
|
|
this->cleanup.library = module;
|
|
|
|
}
|
|
|
|
|
2015-07-03 01:57:23 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpDisassociateCallback (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpDisassociateCallback( TP_CALLBACK_INSTANCE *instance )
|
|
|
|
{
|
|
|
|
struct threadpool_instance *this = impl_from_TP_CALLBACK_INSTANCE( instance );
|
|
|
|
struct threadpool_object *object = this->object;
|
|
|
|
struct threadpool *pool;
|
|
|
|
|
|
|
|
TRACE( "%p\n", instance );
|
|
|
|
|
|
|
|
if (this->threadid != GetCurrentThreadId())
|
|
|
|
{
|
|
|
|
ERR("called from wrong thread, ignoring\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!this->associated)
|
|
|
|
return;
|
|
|
|
|
|
|
|
pool = object->pool;
|
|
|
|
RtlEnterCriticalSection( &pool->cs );
|
|
|
|
|
|
|
|
object->num_associated_callbacks--;
|
|
|
|
if (!object->num_pending_callbacks && !object->num_associated_callbacks)
|
|
|
|
RtlWakeAllConditionVariable( &object->finished_event );
|
|
|
|
|
|
|
|
RtlLeaveCriticalSection( &pool->cs );
|
|
|
|
this->associated = FALSE;
|
|
|
|
}
|
|
|
|
|
2015-07-03 01:58:29 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpIsTimerSet (NTDLL.@)
|
|
|
|
*/
|
|
|
|
BOOL WINAPI TpIsTimerSet( TP_TIMER *timer )
|
|
|
|
{
|
|
|
|
struct threadpool_object *this = impl_from_TP_TIMER( timer );
|
|
|
|
|
|
|
|
TRACE( "%p\n", timer );
|
|
|
|
|
|
|
|
return this->u.timer.timer_set;
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:57:57 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpPostWork (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpPostWork( TP_WORK *work )
|
|
|
|
{
|
|
|
|
struct threadpool_object *this = impl_from_TP_WORK( work );
|
|
|
|
|
|
|
|
TRACE( "%p\n", work );
|
|
|
|
|
2015-07-05 02:22:52 +02:00
|
|
|
tp_object_submit( this, FALSE );
|
2015-07-01 02:57:57 +02:00
|
|
|
}
|
|
|
|
|
2015-07-01 02:57:15 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpReleaseCleanupGroup (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpReleaseCleanupGroup( TP_CLEANUP_GROUP *group )
|
|
|
|
{
|
|
|
|
struct threadpool_group *this = impl_from_TP_CLEANUP_GROUP( group );
|
|
|
|
|
|
|
|
TRACE( "%p\n", group );
|
|
|
|
|
|
|
|
tp_group_shutdown( this );
|
|
|
|
tp_group_release( this );
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* TpReleaseCleanupGroupMembers (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpReleaseCleanupGroupMembers( TP_CLEANUP_GROUP *group, BOOL cancel_pending, PVOID userdata )
|
|
|
|
{
|
|
|
|
struct threadpool_group *this = impl_from_TP_CLEANUP_GROUP( group );
|
|
|
|
struct threadpool_object *object, *next;
|
|
|
|
struct list members;
|
|
|
|
|
|
|
|
TRACE( "%p %u %p\n", group, cancel_pending, userdata );
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &this->cs );
|
|
|
|
|
|
|
|
/* Unset group, increase references, and mark objects for shutdown */
|
|
|
|
LIST_FOR_EACH_ENTRY_SAFE( object, next, &this->members, struct threadpool_object, group_entry )
|
|
|
|
{
|
|
|
|
assert( object->group == this );
|
|
|
|
assert( object->is_group_member );
|
|
|
|
|
2016-08-20 21:22:29 +02:00
|
|
|
if (interlocked_inc( &object->refcount ) == 1)
|
2015-07-01 02:57:15 +02:00
|
|
|
{
|
2016-08-20 21:22:29 +02:00
|
|
|
/* Object is basically already destroyed, but group reference
|
|
|
|
* was not deleted yet. We can safely ignore this object. */
|
|
|
|
interlocked_dec( &object->refcount );
|
|
|
|
list_remove( &object->group_entry );
|
|
|
|
object->is_group_member = FALSE;
|
|
|
|
continue;
|
2015-07-01 02:57:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
object->is_group_member = FALSE;
|
2016-08-20 21:22:29 +02:00
|
|
|
tp_object_prepare_shutdown( object );
|
2015-07-01 02:57:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Move members to a new temporary list */
|
|
|
|
list_init( &members );
|
|
|
|
list_move_tail( &members, &this->members );
|
|
|
|
|
|
|
|
RtlLeaveCriticalSection( &this->cs );
|
|
|
|
|
|
|
|
/* Cancel pending callbacks if requested */
|
|
|
|
if (cancel_pending)
|
|
|
|
{
|
|
|
|
LIST_FOR_EACH_ENTRY( object, &members, struct threadpool_object, group_entry )
|
|
|
|
{
|
2016-08-20 21:23:48 +02:00
|
|
|
tp_object_cancel( object );
|
2015-07-01 02:57:15 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Wait for remaining callbacks to finish */
|
|
|
|
LIST_FOR_EACH_ENTRY_SAFE( object, next, &members, struct threadpool_object, group_entry )
|
|
|
|
{
|
2015-07-03 01:57:23 +02:00
|
|
|
tp_object_wait( object, TRUE );
|
2016-08-20 21:22:29 +02:00
|
|
|
|
2016-08-20 22:38:43 +02:00
|
|
|
if (!object->shutdown)
|
2016-08-20 21:23:48 +02:00
|
|
|
{
|
2016-08-20 22:38:43 +02:00
|
|
|
/* Execute group cancellation callback if defined, and if this was actually a group cancel. */
|
|
|
|
if (cancel_pending && object->group_cancel_callback)
|
|
|
|
{
|
|
|
|
TRACE( "executing group cancel callback %p(%p, %p)\n",
|
|
|
|
object->group_cancel_callback, object->userdata, userdata );
|
|
|
|
object->group_cancel_callback( object->userdata, userdata );
|
|
|
|
TRACE( "callback %p returned\n", object->group_cancel_callback );
|
|
|
|
}
|
2016-08-20 21:23:48 +02:00
|
|
|
|
2016-08-20 22:38:43 +02:00
|
|
|
if (object->type != TP_OBJECT_TYPE_SIMPLE)
|
|
|
|
tp_object_release( object );
|
|
|
|
}
|
2016-08-20 21:22:29 +02:00
|
|
|
|
|
|
|
object->shutdown = TRUE;
|
2015-07-01 02:57:15 +02:00
|
|
|
tp_object_release( object );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpReleasePool (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpReleasePool( TP_POOL *pool )
|
|
|
|
{
|
|
|
|
struct threadpool *this = impl_from_TP_POOL( pool );
|
|
|
|
|
|
|
|
TRACE( "%p\n", pool );
|
|
|
|
|
|
|
|
tp_threadpool_shutdown( this );
|
|
|
|
tp_threadpool_release( this );
|
|
|
|
}
|
|
|
|
|
2015-07-03 01:57:54 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpReleaseTimer (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpReleaseTimer( TP_TIMER *timer )
|
|
|
|
{
|
|
|
|
struct threadpool_object *this = impl_from_TP_TIMER( timer );
|
|
|
|
|
|
|
|
TRACE( "%p\n", timer );
|
|
|
|
|
2016-08-20 21:22:29 +02:00
|
|
|
tp_object_prepare_shutdown( this );
|
|
|
|
this->shutdown = TRUE;
|
2015-07-03 01:57:54 +02:00
|
|
|
tp_object_release( this );
|
|
|
|
}
|
|
|
|
|
2015-07-05 02:22:11 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpReleaseWait (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpReleaseWait( TP_WAIT *wait )
|
|
|
|
{
|
|
|
|
struct threadpool_object *this = impl_from_TP_WAIT( wait );
|
|
|
|
|
|
|
|
TRACE( "%p\n", wait );
|
|
|
|
|
2016-08-20 21:22:29 +02:00
|
|
|
tp_object_prepare_shutdown( this );
|
|
|
|
this->shutdown = TRUE;
|
2015-07-05 02:22:11 +02:00
|
|
|
tp_object_release( this );
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:57:57 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpReleaseWork (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpReleaseWork( TP_WORK *work )
|
|
|
|
{
|
|
|
|
struct threadpool_object *this = impl_from_TP_WORK( work );
|
|
|
|
|
|
|
|
TRACE( "%p\n", work );
|
|
|
|
|
2016-08-20 21:22:29 +02:00
|
|
|
tp_object_prepare_shutdown( this );
|
|
|
|
this->shutdown = TRUE;
|
2015-07-01 02:57:57 +02:00
|
|
|
tp_object_release( this );
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:55:53 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpSetPoolMaxThreads (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpSetPoolMaxThreads( TP_POOL *pool, DWORD maximum )
|
|
|
|
{
|
|
|
|
struct threadpool *this = impl_from_TP_POOL( pool );
|
|
|
|
|
|
|
|
TRACE( "%p %u\n", pool, maximum );
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &this->cs );
|
|
|
|
this->max_workers = max( maximum, 1 );
|
|
|
|
this->min_workers = min( this->min_workers, this->max_workers );
|
|
|
|
RtlLeaveCriticalSection( &this->cs );
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:56:26 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpSetPoolMinThreads (NTDLL.@)
|
|
|
|
*/
|
|
|
|
BOOL WINAPI TpSetPoolMinThreads( TP_POOL *pool, DWORD minimum )
|
|
|
|
{
|
|
|
|
struct threadpool *this = impl_from_TP_POOL( pool );
|
|
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
|
|
|
|
|
|
TRACE( "%p %u\n", pool, minimum );
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &this->cs );
|
|
|
|
|
|
|
|
while (this->num_workers < minimum)
|
|
|
|
{
|
2016-08-22 22:39:10 +02:00
|
|
|
status = tp_new_worker_thread( this );
|
2015-07-01 02:56:26 +02:00
|
|
|
if (status != STATUS_SUCCESS)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (status == STATUS_SUCCESS)
|
|
|
|
{
|
|
|
|
this->min_workers = minimum;
|
|
|
|
this->max_workers = max( this->min_workers, this->max_workers );
|
|
|
|
}
|
|
|
|
|
|
|
|
RtlLeaveCriticalSection( &this->cs );
|
|
|
|
return !status;
|
|
|
|
}
|
|
|
|
|
2015-07-03 01:58:29 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpSetTimer (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpSetTimer( TP_TIMER *timer, LARGE_INTEGER *timeout, LONG period, LONG window_length )
|
|
|
|
{
|
|
|
|
struct threadpool_object *this = impl_from_TP_TIMER( timer );
|
|
|
|
struct threadpool_object *other_timer;
|
|
|
|
BOOL submit_timer = FALSE;
|
|
|
|
ULONGLONG timestamp;
|
|
|
|
|
|
|
|
TRACE( "%p %p %u %u\n", timer, timeout, period, window_length );
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &timerqueue.cs );
|
|
|
|
|
|
|
|
assert( this->u.timer.timer_initialized );
|
|
|
|
this->u.timer.timer_set = timeout != NULL;
|
|
|
|
|
|
|
|
/* Convert relative timeout to absolute timestamp and handle a timeout
|
|
|
|
* of zero, which means that the timer is submitted immediately. */
|
|
|
|
if (timeout)
|
|
|
|
{
|
|
|
|
timestamp = timeout->QuadPart;
|
|
|
|
if ((LONGLONG)timestamp < 0)
|
|
|
|
{
|
|
|
|
LARGE_INTEGER now;
|
|
|
|
NtQuerySystemTime( &now );
|
|
|
|
timestamp = now.QuadPart - timestamp;
|
|
|
|
}
|
|
|
|
else if (!timestamp)
|
|
|
|
{
|
|
|
|
if (!period)
|
|
|
|
timeout = NULL;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
LARGE_INTEGER now;
|
|
|
|
NtQuerySystemTime( &now );
|
|
|
|
timestamp = now.QuadPart + (ULONGLONG)period * 10000;
|
|
|
|
}
|
|
|
|
submit_timer = TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* First remove existing timeout. */
|
|
|
|
if (this->u.timer.timer_pending)
|
|
|
|
{
|
|
|
|
list_remove( &this->u.timer.timer_entry );
|
|
|
|
this->u.timer.timer_pending = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If the timer was enabled, then add it back to the queue. */
|
|
|
|
if (timeout)
|
|
|
|
{
|
|
|
|
this->u.timer.timeout = timestamp;
|
|
|
|
this->u.timer.period = period;
|
|
|
|
this->u.timer.window_length = window_length;
|
|
|
|
|
|
|
|
LIST_FOR_EACH_ENTRY( other_timer, &timerqueue.pending_timers,
|
|
|
|
struct threadpool_object, u.timer.timer_entry )
|
|
|
|
{
|
|
|
|
assert( other_timer->type == TP_OBJECT_TYPE_TIMER );
|
|
|
|
if (this->u.timer.timeout < other_timer->u.timer.timeout)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
list_add_before( &other_timer->u.timer.timer_entry, &this->u.timer.timer_entry );
|
|
|
|
|
|
|
|
/* Wake up the timer thread when the timeout has to be updated. */
|
|
|
|
if (list_head( &timerqueue.pending_timers ) == &this->u.timer.timer_entry )
|
|
|
|
RtlWakeAllConditionVariable( &timerqueue.update_event );
|
|
|
|
|
|
|
|
this->u.timer.timer_pending = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
RtlLeaveCriticalSection( &timerqueue.cs );
|
|
|
|
|
|
|
|
if (submit_timer)
|
2015-07-05 02:22:52 +02:00
|
|
|
tp_object_submit( this, FALSE );
|
2015-07-03 01:58:29 +02:00
|
|
|
}
|
|
|
|
|
2015-07-05 02:23:43 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpSetWait (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpSetWait( TP_WAIT *wait, HANDLE handle, LARGE_INTEGER *timeout )
|
|
|
|
{
|
|
|
|
struct threadpool_object *this = impl_from_TP_WAIT( wait );
|
|
|
|
ULONGLONG timestamp = TIMEOUT_INFINITE;
|
|
|
|
BOOL submit_wait = FALSE;
|
|
|
|
|
|
|
|
TRACE( "%p %p %p\n", wait, handle, timeout );
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &waitqueue.cs );
|
|
|
|
|
|
|
|
assert( this->u.wait.bucket );
|
|
|
|
this->u.wait.handle = handle;
|
|
|
|
|
|
|
|
if (handle || this->u.wait.wait_pending)
|
|
|
|
{
|
|
|
|
struct waitqueue_bucket *bucket = this->u.wait.bucket;
|
|
|
|
list_remove( &this->u.wait.wait_entry );
|
|
|
|
|
|
|
|
/* Convert relative timeout to absolute timestamp. */
|
|
|
|
if (handle && timeout)
|
|
|
|
{
|
|
|
|
timestamp = timeout->QuadPart;
|
|
|
|
if ((LONGLONG)timestamp < 0)
|
|
|
|
{
|
|
|
|
LARGE_INTEGER now;
|
|
|
|
NtQuerySystemTime( &now );
|
|
|
|
timestamp = now.QuadPart - timestamp;
|
|
|
|
}
|
|
|
|
else if (!timestamp)
|
|
|
|
{
|
|
|
|
submit_wait = TRUE;
|
|
|
|
handle = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Add wait object back into one of the queues. */
|
|
|
|
if (handle)
|
|
|
|
{
|
|
|
|
list_add_tail( &bucket->waiting, &this->u.wait.wait_entry );
|
|
|
|
this->u.wait.wait_pending = TRUE;
|
|
|
|
this->u.wait.timeout = timestamp;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
list_add_tail( &bucket->reserved, &this->u.wait.wait_entry );
|
|
|
|
this->u.wait.wait_pending = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Wake up the wait queue thread. */
|
|
|
|
NtSetEvent( bucket->update_event, NULL );
|
|
|
|
}
|
|
|
|
|
|
|
|
RtlLeaveCriticalSection( &waitqueue.cs );
|
|
|
|
|
|
|
|
if (submit_wait)
|
|
|
|
tp_object_submit( this, FALSE );
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:55:27 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpSimpleTryPost (NTDLL.@)
|
|
|
|
*/
|
|
|
|
NTSTATUS WINAPI TpSimpleTryPost( PTP_SIMPLE_CALLBACK callback, PVOID userdata,
|
|
|
|
TP_CALLBACK_ENVIRON *environment )
|
|
|
|
{
|
|
|
|
struct threadpool_object *object;
|
|
|
|
struct threadpool *pool;
|
|
|
|
NTSTATUS status;
|
|
|
|
|
|
|
|
TRACE( "%p %p %p\n", callback, userdata, environment );
|
|
|
|
|
|
|
|
object = RtlAllocateHeap( GetProcessHeap(), 0, sizeof(*object) );
|
|
|
|
if (!object)
|
|
|
|
return STATUS_NO_MEMORY;
|
|
|
|
|
|
|
|
status = tp_threadpool_lock( &pool, environment );
|
|
|
|
if (status)
|
|
|
|
{
|
|
|
|
RtlFreeHeap( GetProcessHeap(), 0, object );
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
object->type = TP_OBJECT_TYPE_SIMPLE;
|
|
|
|
object->u.simple.callback = callback;
|
|
|
|
tp_object_initialize( object, pool, userdata, environment );
|
|
|
|
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
2015-07-01 02:57:57 +02:00
|
|
|
|
2015-07-03 01:58:29 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpWaitForTimer (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpWaitForTimer( TP_TIMER *timer, BOOL cancel_pending )
|
|
|
|
{
|
|
|
|
struct threadpool_object *this = impl_from_TP_TIMER( timer );
|
|
|
|
|
|
|
|
TRACE( "%p %d\n", timer, cancel_pending );
|
|
|
|
|
|
|
|
if (cancel_pending)
|
2016-08-20 21:23:48 +02:00
|
|
|
tp_object_cancel( this );
|
2015-07-03 01:58:29 +02:00
|
|
|
tp_object_wait( this, FALSE );
|
|
|
|
}
|
|
|
|
|
2015-07-05 02:23:43 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpWaitForWait (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpWaitForWait( TP_WAIT *wait, BOOL cancel_pending )
|
|
|
|
{
|
|
|
|
struct threadpool_object *this = impl_from_TP_WAIT( wait );
|
|
|
|
|
|
|
|
TRACE( "%p %d\n", wait, cancel_pending );
|
|
|
|
|
|
|
|
if (cancel_pending)
|
2016-08-20 21:23:48 +02:00
|
|
|
tp_object_cancel( this );
|
2015-07-05 02:23:43 +02:00
|
|
|
tp_object_wait( this, FALSE );
|
|
|
|
}
|
|
|
|
|
2015-07-01 02:57:57 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* TpWaitForWork (NTDLL.@)
|
|
|
|
*/
|
|
|
|
VOID WINAPI TpWaitForWork( TP_WORK *work, BOOL cancel_pending )
|
|
|
|
{
|
|
|
|
struct threadpool_object *this = impl_from_TP_WORK( work );
|
|
|
|
|
|
|
|
TRACE( "%p %u\n", work, cancel_pending );
|
|
|
|
|
|
|
|
if (cancel_pending)
|
2016-08-20 21:23:48 +02:00
|
|
|
tp_object_cancel( this );
|
2015-07-03 01:57:23 +02:00
|
|
|
tp_object_wait( this, FALSE );
|
2015-07-01 02:57:57 +02:00
|
|
|
}
|
2020-02-12 10:05:27 +01:00
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* TpSetPoolStackInformation (NTDLL.@)
|
|
|
|
*/
|
|
|
|
NTSTATUS WINAPI TpSetPoolStackInformation( TP_POOL *pool, TP_POOL_STACK_INFORMATION *stack_info )
|
|
|
|
{
|
|
|
|
struct threadpool *this = impl_from_TP_POOL( pool );
|
|
|
|
|
|
|
|
TRACE( "%p %p\n", pool, stack_info );
|
|
|
|
|
|
|
|
if (!stack_info)
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &this->cs );
|
|
|
|
this->stack_info = *stack_info;
|
|
|
|
RtlLeaveCriticalSection( &this->cs );
|
|
|
|
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* TpQueryPoolStackInformation (NTDLL.@)
|
|
|
|
*/
|
|
|
|
NTSTATUS WINAPI TpQueryPoolStackInformation( TP_POOL *pool, TP_POOL_STACK_INFORMATION *stack_info )
|
|
|
|
{
|
|
|
|
struct threadpool *this = impl_from_TP_POOL( pool );
|
|
|
|
|
|
|
|
TRACE( "%p %p\n", pool, stack_info );
|
|
|
|
|
|
|
|
if (!stack_info)
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
|
|
|
|
|
|
RtlEnterCriticalSection( &this->cs );
|
|
|
|
*stack_info = this->stack_info;
|
|
|
|
RtlLeaveCriticalSection( &this->cs );
|
|
|
|
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
2020-02-12 10:05:29 +01:00
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* TpStartAsyncIoOperation (NTDLL.@)
|
|
|
|
*/
|
|
|
|
void WINAPI TpStartAsyncIoOperation( TP_IO *io )
|
|
|
|
{
|
|
|
|
FIXME( "%p\n", io );
|
|
|
|
}
|