— функции создания и уничтожения контекста потока, которые вызываются при создании и уничтожении каждого потока пула. Функция создания контекста потока ответственна за индивидуальные настройки создаваемого потока. Она возвращает «указатель на контекст» типа
THREAD_POOL_PARAM_T
. Однако системе такой тип неизвестен:
#ifndef THREAD_POOL_PARAM_T
#define THREAD_POOL_PARAM_T void
#endif
В качестве
контекста может использоваться любой пользовательский тип, и он будет передаваться последовательно в качестве параметра (
ctp
) во все последующие функции обслуживания потока.
block_func
— функция блокирования, которая вызывается в потоке сразу же после
context_alloc
или после очередного этапа выполнения потоком функции обработчика
handler_func
. Функция блокирования получает и возвращает далее обработчику (возможно, после модификации) структуру контекста (в приведенном выше примере контекстом является
int
— значение присоединенного TCP-сокета).
handler_func
— это, собственно, и есть аналог потоковой функции, в которой выполняется вся полезная работа потока. Функция вызывается библиотекой после выхода потока из блокирующей функции
block_func
, при этом функция-обработчик
handler_func
получит параметр контекста, возвращенный
block_func
.
Примечание
В текущей реализации
handler_func
должна возвращать 0; все другие значения зарезервированы для дальнейших расширений. Аналогично определенная в атрибутной записи функция
unblock_func
зарезервирована для дальнейших расширений, и вместо ее адреса следует устанавливать
NULL
.
2. После создания атрибутной записи пула, определяющей всю функциональность его дальнейшего поведения, можно приступать к непосредственному созданию пула потоков:
thread_pool_t* thread_pool_create(
thread_pool_attr_t* attr, unsigned flags);
где
attr
— подробно рассмотренная (и созданная) ранее атрибутная запись пула;
flags
— флаг, определяющий поведение вызывающего потока после последующего вызова
thread_pool_start
. В документации описано два возможных значения флага:
•
POOL_FLAG_EXIT_SELF
— после старта пула поток, вызвавший
thread_pool_start
(часто это главный поток приложения), завершается;
•
POOL_FLAG_USE_SELF
— после старта пула поток, вызвавший
thread_pool_start
, включается в пул в качестве одного из его потоков.
И в том и в другом случае в типовом фрагменте (как и в показанном выше примере):
thread_pool_start(tpp);
exit(EXIT_SUCCESS);
управление никогда не дойдет до выполнения exit. Но существует еще третье допустимое значение, прямо не указанное в документации, но мельком упоминаемое в других местах документации:
•
0
— после старта пула поток, вызвавший
thread_pool_start
, продолжает свое естественное выполнение.
Например, некоторый фрагмент кода мог бы выглядеть
При успешном завершении (которого почти никогда не происходит, за исключением значения флага
0
; об этом см. выше) функция возвращает
EOK
, в противном случае (что происходит гораздо чаще) — значение
– 1
.
4. Другие, относящиеся к библиотеке динамического пула потоков функции, которые целесообразно посмотреть в документации QNX (но которые в силу различных обстоятельств используются гораздо реже):
40
Вы спросите, почему указатель, возвращаемый
thread_pool_create
, имеет тип
thread_pool_t*
, а получающий его параметр
thread_pool_start
определен как
void*
? Это или неаккуратность разработчиков QNX, или глубокая сермяжная правда, которую мы пока не понимаем.
int thread_pool_destroy(thread_pool_t* pool);
int thread_pool_control(thread_pool_t* pool, thread_pool_attr_t* attr,
_Uint16t lower, _Uint16t upper, unsigned flags);
int thread_pool_limits(thread_pool_t* pool,
int lowater, int hiwater, int maximum, int increment, unsigned flags);