16 #ifndef INCLUDE_NN_NLIB_THREADING_THREAD_H_ 17 #define INCLUDE_NN_NLIB_THREADING_THREAD_H_ 20 #include "nn/nlib/Swap.h" 33 if (is_initialized_) {
43 NLIB_THREAD_ATTR_KEY_DETACHSTATE,
53 NLIB_THREAD_ATTR_KEY_DETACHSTATE,
57 return (detached != 0);
63 NLIB_THREAD_ATTR_KEY_STACKSIZE,
72 NLIB_THREAD_ATTR_KEY_STACKSIZE,
82 NLIB_THREAD_ATTR_KEY_PRIORITY,
91 NLIB_THREAD_ATTR_KEY_PRIORITY,
98 return is_initialized_ ? &attr_ : NULL;
101 return is_initialized_ ? &attr_ : NULL;
106 if (!is_initialized_) {
108 NLIB_ASSERT_NOERR(e);
110 is_initialized_ =
true;
113 mutable bool is_initialized_;
117 template <
class T1 = None,
class T2 = None,
class T3 = None,
class T4 = None,
class T5 = None>
121 typedef void (*Func)(ArgType& ptr);
124 : func(func_), arg1(arg1_), arg2(arg2_), arg3(arg3_), arg4(arg4_), arg5(arg5_) {}
125 static void Call(ArgType& ptr) { ptr->func(ptr); }
136 struct ThreadArg<None, None, None, None, None> {
139 typedef void (*Func)();
143 struct ThreadArg<T1, None, None, None, None> {
146 typedef void (*Func)(ArgType& ptr);
148 NLIB_CEXPR ThreadArg(Func func_, T1 arg1_) : func(func_), arg1(arg1_) {}
149 static void Call(ArgType& ptr) { ptr->func(ptr); }
155 template <
class T1,
class T2>
156 struct ThreadArg<T1, T2, None, None, None> {
159 typedef void (*Func)(ArgType& ptr);
161 NLIB_CEXPR ThreadArg(Func func_, T1 arg1_, T2 arg2_) : func(func_), arg1(arg1_), arg2(arg2_) {}
162 static void Call(ArgType& ptr) { ptr->func(ptr); }
169 template <
class T1,
class T2,
class T3>
170 struct ThreadArg<T1, T2, T3, None, None> {
173 typedef void (*Func)(ArgType& ptr);
175 NLIB_CEXPR ThreadArg(Func func_, T1 arg1_, T2 arg2_, T3 arg3_)
176 : func(func_), arg1(arg1_), arg2(arg2_), arg3(arg3_) {}
177 static void Call(ArgType& ptr) { ptr->func(ptr); }
185 template <
class T1,
class T2,
class T3,
class T4>
189 typedef void (*Func)(ArgType& ptr);
191 NLIB_CEXPR ThreadArg(Func func_, T1 arg1_, T2 arg2_, T3 arg3_, T4 arg4_)
192 : func(func_), arg1(arg1_), arg2(arg2_), arg3(arg3_), arg4(arg4_) {}
193 static void Call(ArgType& ptr) { ptr->func(ptr); }
203 #pragma warning(push) 204 #pragma warning(disable : 4180) 214 typedef void (*ThreadFunc)(
void* arg);
219 NLIB_MOVE_MEMBER_HELPER_1(
Thread, thread_id_)
223 if (!thread_id_)
return ESRCH;
250 swap(thread_id_, rhs.thread_id_);
266 errno_t e = this->StartRaw(Thread::Exec, reinterpret_cast<void*>(func), settings);
270 errno_t e = this->StartRaw(Thread::Exec, reinterpret_cast<void*>(func));
273 template <
class ThArg>
276 if (!ptr)
return EINVAL;
277 errno_t e = this->StartRaw((ThreadFunc)Thread::Exec<ThArg>,
278 reinterpret_cast<void*>(ptr.get()), settings);
279 if (e == 0) ptr.release();
282 template <
class ThArg>
284 if (!ptr)
return EINVAL;
285 errno_t e = this->StartRaw((ThreadFunc)Thread::Exec<ThArg>,
286 reinterpret_cast<void*>(ptr.get()));
287 if (e == 0) ptr.release();
292 static void Exec(
void* p) {
297 static void Exec(
void* p) {
302 #define NLIB_THFUNC_USESWAP_NO(tp) \ 304 !IsSame<None, typename RemoveCv<B>::type>::value && \ 305 (IsSame<FalseType, typename RemoveCv<B>::type>::value || !IsSwappable<tp>::value), \ 307 #define NLIB_THFUNC_USESWAP_YES(tp) \ 308 typename EnableIf<!IsSame<None, typename RemoveCv<B>::type>::value && \ 309 IsSame<TrueType, typename RemoveCv<B>::type>::value && \ 310 IsSwappable<tp>::value, \ 314 struct ArgType :
public Conditional<
315 IsArithmetic<T>::value || IsPointer<T>::value || IsMemberPointer<T>::value,
316 FalseType, TrueType> {};
318 template <
class FUNC>
320 typedef typename RemoveRef<FUNC>::type FUNC_;
322 Args0(NLIB_THFUNC_USESWAP_NO(FUNC_) func_, B)
325 Args0(NLIB_THFUNC_USESWAP_YES(FUNC_) func_, B)
328 swap(const_cast<FUNC&>(func_), func);
331 Args0(
typename EnableIf<IsSame<
typename RemoveCv<B>::type, None>::value,
const FUNC&>::type
340 template <
class FUNC,
class T1>
341 struct Args1 :
public Args0<FUNC> {
342 typedef Args0<FUNC> BaseType;
343 typedef typename RemoveRef<T1>::type T1_;
345 Args1(
const FUNC& func_, NLIB_THFUNC_USESWAP_NO(T1) arg1_, B)
346 : BaseType(func_,
typename IsSwappable<FUNC>::type()), arg1(arg1_) {}
348 Args1(
const FUNC& func_, NLIB_THFUNC_USESWAP_YES(T1) arg1_, B)
349 : BaseType(func_,
typename IsSwappable<FUNC>::type()), arg1() {
351 swap(const_cast<T1&>(arg1_), arg1);
354 Args1(
typename EnableIf<IsSame<
typename RemoveCv<B>::type, None>::value,
const FUNC&>::type
356 const T1& arg1_, B b)
357 : BaseType(func_, b), arg1(arg1_) {}
360 template <
class FUNC,
class T1,
class T2>
361 struct Args2 :
public Args1<FUNC, T1> {
362 typedef Args1<FUNC, T1> BaseType;
363 typedef typename RemoveRef<T2>::type T2_;
365 Args2(
const FUNC& func_,
const T1& arg1_, NLIB_THFUNC_USESWAP_NO(T2) arg2_, B)
366 : BaseType(func_, arg1_,
typename ArgType<T1>::type()), arg2(arg2_) {}
368 Args2(
const FUNC& func_,
const T1& arg1_, NLIB_THFUNC_USESWAP_YES(T2) arg2_, B)
369 : BaseType(func_, arg1_,
typename ArgType<T1>::type()), arg2() {
371 swap(const_cast<T2&>(arg2_), arg2);
374 Args2(
typename EnableIf<IsSame<
typename RemoveCv<B>::type, None>::value,
const FUNC&>::type
376 const T1& arg1_,
const T2& arg2_, B b)
377 : BaseType(func_, arg1_, b), arg2(arg2_) {}
380 template <
class FUNC,
class T1,
class T2,
class T3>
381 struct Args3 :
public Args2<FUNC, T1, T2> {
382 typedef Args2<FUNC, T1, T2> BaseType;
383 typedef typename RemoveRef<T3>::type T3_;
385 Args3(
const FUNC& func_,
const T1& arg1_,
const T2& arg2_, NLIB_THFUNC_USESWAP_NO(T3) arg3_,
387 : BaseType(func_, arg1_, arg2_,
typename ArgType<T2>::type()), arg3(arg3_) {}
389 Args3(
const FUNC& func,
const T1& arg1_,
const T2& arg2_, NLIB_THFUNC_USESWAP_YES(T3) arg3_,
391 : BaseType(func, arg1_, arg2_,
typename ArgType<T2>::type()), arg3() {
393 swap(const_cast<T3&>(arg3_), arg3);
396 Args3(
typename EnableIf<IsSame<
typename RemoveCv<B>::type, None>::value,
const FUNC&>::type
398 const T1& arg1_,
const T2& arg2_,
const T3& arg3_, B b)
399 : BaseType(func_, arg1_, arg2_, b), arg3(arg3_) {}
402 template <
class FUNC,
class T1,
class T2,
class T3,
class T4>
403 struct Args4 :
public Args3<FUNC, T1, T2, T3> {
404 typedef Args3<FUNC, T1, T2, T3> BaseType;
405 typedef typename RemoveRef<T4>::type T4_;
407 Args4(
const FUNC& func_,
const T1& arg1_,
const T2& arg2_,
const T3& arg3_,
408 NLIB_THFUNC_USESWAP_NO(T4) arg4_, B)
409 : BaseType(func_, arg1_, arg2_, arg3_,
typename ArgType<T3>::type()), arg4(arg4_) {}
411 Args4(
const FUNC& func_,
const T1& arg1_,
const T2& arg2_,
const T3& arg3_,
412 NLIB_THFUNC_USESWAP_YES(T4) arg4_, B)
413 : BaseType(func_, arg1_, arg2_, arg3_,
typename ArgType<T3>::type()), arg4() {
415 swap(const_cast<T4&>(arg4_), arg4);
418 Args4(
typename EnableIf<IsSame<
typename RemoveCv<B>::type, None>::value,
const FUNC&>::type
420 const T1& arg1_,
const T2& arg2_,
const T3& arg3_,
const T4& arg4_, B b)
421 : BaseType(func_, arg1_, arg2_, arg3_, b), arg4(arg4_) {}
424 template <
class FUNC,
class T1,
class T2,
class T3,
class T4,
class T5>
425 struct Args5 :
public Args4<FUNC, T1, T2, T3, T4> {
426 typedef Args4<FUNC, T1, T2, T3, T4> BaseType;
427 typedef typename RemoveRef<T5>::type T5_;
429 Args5(
const FUNC& func_,
const T1& arg1_,
const T2& arg2_,
const T3& arg3_,
const T4& arg4_,
430 NLIB_THFUNC_USESWAP_NO(T5) arg5_, B)
431 : BaseType(func_, arg1_, arg2_, arg3_, arg4_,
typename ArgType<T4>::type()),
434 Args5(
const FUNC& func_,
const T1& arg1_,
const T2& arg2_,
const T3& arg3_,
const T4& arg4_,
435 NLIB_THFUNC_USESWAP_YES(T4) arg5_, B)
436 : BaseType(func_, arg1_, arg2_, arg3_, arg4_,
typename ArgType<T4>::type()), arg5() {
438 swap(const_cast<T5&>(arg5_), arg5);
441 Args5(
typename EnableIf<IsSame<
typename RemoveCv<B>::type, None>::value,
const FUNC&>::type
443 const T1& arg1_,
const T2& arg2_,
const T3& arg3_,
const T4& arg4_,
const T5& arg5_,
445 : BaseType(func_, arg1_, arg2_, arg3_, arg4_, b), arg5(arg5_) {}
448 #undef NLIB_THFUNC_USESWAP_NO 449 #undef NLIB_THFUNC_USESWAP_YES 451 template <
class FUNC>
452 struct ThreadFuncX0 :
public Args0<FUNC> {
453 typedef ThreadFuncX0<FUNC> ThisType;
454 typedef Args0<FUNC> BaseType;
456 ThreadFuncX0(
const FUNC& func_, B b)
457 : BaseType(func_, b) {}
458 explicit ThreadFuncX0(
const FUNC& func_) : BaseType(func_, None()) {}
463 template <
class FUNC,
class T1>
464 struct ThreadFuncX1 :
public Args1<FUNC, T1> {
465 typedef ThreadFuncX1<FUNC, T1> ThisType;
466 typedef Args1<FUNC, T1> BaseType;
468 ThreadFuncX1(
const FUNC& func_,
const T1& arg1_, B b)
469 : BaseType(func_, arg1_, b) {}
470 ThreadFuncX1(
const FUNC& func_,
const T1& arg1_) : BaseType(func_, arg1_, None()) {}
472 ptr->func(NLIB_MOVE(ptr->arg1));
475 template <
class FUNC,
class T1,
class T2>
476 struct ThreadFuncX2 :
public Args2<FUNC, T1, T2> {
477 typedef ThreadFuncX2<FUNC, T1, T2> ThisType;
478 typedef Args2<FUNC, T1, T2> BaseType;
480 ThreadFuncX2(
const FUNC& func_,
const T1& arg1_,
const T2& arg2_, B b)
481 : BaseType(func_, arg1_, arg2_, b) {}
482 ThreadFuncX2(
const FUNC& func_,
const T1& arg1_,
const T2& arg2_)
483 : BaseType(func_, arg1_, arg2_, None()) {}
485 ptr->func(NLIB_MOVE(ptr->arg1), NLIB_MOVE(ptr->arg2));
488 template <
class FUNC,
class T1,
class T2,
class T3>
489 struct ThreadFuncX3 :
public Args3<FUNC, T1, T2, T3> {
490 typedef ThreadFuncX3<FUNC, T1, T2, T3> ThisType;
491 typedef Args3<FUNC, T1, T2, T3> BaseType;
493 ThreadFuncX3(
const FUNC& func_,
const T1& arg1_,
const T2& arg2_,
const T3& arg3_, B b)
494 : BaseType(func_, arg1_, arg2_, arg3_, b) {}
495 ThreadFuncX3(
const FUNC& func_,
const T1& arg1_,
const T2& arg2_,
const T3& arg3_)
496 : BaseType(func_, arg1_, arg2_, arg3_, None()) {}
498 ptr->func(NLIB_MOVE(ptr->arg1), NLIB_MOVE(ptr->arg2), NLIB_MOVE(ptr->arg3));
501 template <
class FUNC,
class T1,
class T2,
class T3,
class T4>
502 struct ThreadFuncX4 :
public Args4<FUNC, T1, T2, T3, T4> {
503 typedef ThreadFuncX4<FUNC, T1, T2, T3, T4> ThisType;
504 typedef Args4<FUNC, T1, T2, T3, T4> BaseType;
506 ThreadFuncX4(
const FUNC& func_,
const T1& arg1_,
const T2& arg2_,
const T3& arg3_,
507 const T4& arg4_, B b)
508 : BaseType(func_, arg1_, arg2_, arg3_, arg4_, b) {}
509 ThreadFuncX4(
const FUNC& func_,
const T1& arg1_,
const T2& arg2_,
const T3& arg3_,
511 : BaseType(func_, arg1_, arg2_, arg3_, arg4_, None()) {}
513 ptr->func(NLIB_MOVE(ptr->arg1), NLIB_MOVE(ptr->arg2), NLIB_MOVE(ptr->arg3),
514 NLIB_MOVE(ptr->arg4));
517 template <
class FUNC,
class T1,
class T2,
class T3,
class T4,
class T5>
518 struct ThreadFuncX5 :
public Args5<FUNC, T1, T2, T3, T4, T5> {
519 typedef ThreadFuncX5<FUNC, T1, T2, T3, T4, T5> ThisType;
520 typedef Args5<FUNC, T1, T2, T3, T4, T5> BaseType;
522 ThreadFuncX5(
const FUNC& func_,
const T1& arg1_,
const T2& arg2_,
const T3& arg3_,
523 const T4& arg4_,
const T5& arg5_, B b)
524 : BaseType(func_, arg1_, arg2_, arg3_, arg4_, arg5_, b) {}
525 ThreadFuncX5(
const FUNC& func_,
const T1& arg1_,
const T2& arg2_,
const T3& arg3_,
526 const T4& arg4_,
const T5& arg5_)
527 : BaseType(func_, arg1_, arg2_, arg3_, arg4_, arg5_, None()) {}
529 ptr->func(NLIB_MOVE(ptr->arg1), NLIB_MOVE(ptr->arg2), NLIB_MOVE(ptr->arg3),
530 NLIB_MOVE(ptr->arg4), NLIB_MOVE(ptr->arg5));
535 template <
class FUNC>
538 new (std::nothrow) ThreadFuncX0<FUNC>(f,
typename IsSwappable<FUNC>::type()));
539 if (!ptr)
return ENOMEM;
540 return this->StartRaw(settings, ptr);
542 template <
class FUNC>
545 new (std::nothrow) ThreadFuncX0<FUNC>(f,
typename IsSwappable<FUNC>::type()));
546 if (!ptr)
return ENOMEM;
547 return this->StartRaw(ptr);
549 template <
class FUNC,
class T1>
552 new (std::nothrow) ThreadFuncX1<FUNC, T1>(f, arg1,
typename ArgType<T1>::type()));
553 if (!ptr)
return ENOMEM;
554 return this->StartRaw(settings, ptr);
556 template <
class FUNC,
class T1>
559 new (std::nothrow) ThreadFuncX1<FUNC, T1>(f, arg1,
typename ArgType<T1>::type()));
560 if (!ptr)
return ENOMEM;
561 return this->StartRaw(ptr);
563 template <
class FUNC,
class T1,
class T2>
567 f, arg1, arg2,
typename ArgType<T2>::type()));
568 if (!ptr)
return ENOMEM;
569 return this->StartRaw(settings, ptr);
571 template <
class FUNC,
class T1,
class T2>
574 f, arg1, arg2,
typename ArgType<T2>::type()));
575 if (!ptr)
return ENOMEM;
576 return this->StartRaw(ptr);
578 template <
class FUNC,
class T1,
class T2,
class T3>
583 ThreadFuncX3<FUNC, T1, T2, T3>(f, arg1, arg2, arg3,
typename ArgType<T3>::type()));
584 if (!ptr)
return ENOMEM;
585 return this->StartRaw(settings, ptr);
587 template <
class FUNC,
class T1,
class T2,
class T3>
591 ThreadFuncX3<FUNC, T1, T2, T3>(f, arg1, arg2, arg3,
typename ArgType<T3>::type()));
592 if (!ptr)
return ENOMEM;
593 return this->StartRaw(ptr);
595 template <
class FUNC,
class T1,
class T2,
class T3,
class T4>
597 const T3& arg3,
const T4& arg4,
move_tag) {
599 new (std::nothrow) ThreadFuncX4<FUNC, T1, T2, T3, T4>(f, arg1, arg2, arg3, arg4,
600 typename ArgType<T4>::type()));
601 if (!ptr)
return ENOMEM;
602 return this->StartRaw(settings, ptr);
604 template <
class FUNC,
class T1,
class T2,
class T3,
class T4>
605 errno_t Start(
const FUNC& f,
const T1& arg1,
const T2& arg2,
const T3& arg3,
const T4& arg4,
608 new (std::nothrow) ThreadFuncX4<FUNC, T1, T2, T3, T4>(f, arg1, arg2, arg3, arg4,
609 typename ArgType<T4>::type()));
610 if (!ptr)
return ENOMEM;
611 return this->StartRaw(ptr);
613 template <
class FUNC,
class T1,
class T2,
class T3,
class T4,
class T5>
615 const T3& arg3,
const T4& arg4,
const T5& arg5,
move_tag) {
617 new (std::nothrow) ThreadFuncX5<FUNC, T1, T2, T3, T4, T5>(
618 f, arg1, arg2, arg3, arg4, arg5,
typename ArgType<T5>::type()));
619 if (!ptr)
return ENOMEM;
620 return this->StartRaw(settings, ptr);
622 template <
class FUNC,
class T1,
class T2,
class T3,
class T4,
class T5>
623 errno_t Start(
const FUNC& f,
const T1& arg1,
const T2& arg2,
const T3& arg3,
const T4& arg4,
626 new (std::nothrow) ThreadFuncX5<FUNC, T1, T2, T3, T4, T5>(
627 f, arg1, arg2, arg3, arg4, arg5,
typename ArgType<T5>::type()));
628 if (!ptr)
return ENOMEM;
629 return this->StartRaw(ptr);
631 template <
class FUNC>
634 if (!ptr)
return ENOMEM;
635 return this->StartRaw(settings, ptr);
637 template <
class FUNC>
640 if (!ptr)
return ENOMEM;
641 return this->StartRaw(ptr);
643 template <
class FUNC,
class T1>
646 if (!ptr)
return ENOMEM;
647 return this->StartRaw(settings, ptr);
649 template <
class FUNC,
class T1>
652 if (!ptr)
return ENOMEM;
653 return this->StartRaw(ptr);
655 template <
class FUNC,
class T1,
class T2>
658 ThreadFuncX2<FUNC, T1, T2>(f, arg1, arg2));
659 if (!ptr)
return ENOMEM;
660 return this->StartRaw(settings, ptr);
662 template <
class FUNC,
class T1,
class T2>
665 ThreadFuncX2<FUNC, T1, T2>(f, arg1, arg2));
666 if (!ptr)
return ENOMEM;
667 return this->StartRaw(ptr);
669 template <
class FUNC,
class T1,
class T2,
class T3>
673 new (std::nothrow) ThreadFuncX3<FUNC, T1, T2, T3>(f, arg1, arg2, arg3));
674 if (!ptr)
return ENOMEM;
675 return this->StartRaw(settings, ptr);
677 template <
class FUNC,
class T1,
class T2,
class T3>
678 errno_t Start(
const FUNC& f,
const T1& arg1,
const T2& arg2,
const T3& arg3) {
680 new (std::nothrow) ThreadFuncX3<FUNC, T1, T2, T3>(f, arg1, arg2, arg3));
681 if (!ptr)
return ENOMEM;
682 return this->StartRaw(ptr);
684 template <
class FUNC,
class T1,
class T2,
class T3,
class T4>
686 const T3& arg3,
const T4& arg4) {
688 new (std::nothrow) ThreadFuncX4<FUNC, T1, T2, T3, T4>(f, arg1, arg2, arg3, arg4));
689 if (!ptr)
return ENOMEM;
690 return this->StartRaw(settings, ptr);
692 template <
class FUNC,
class T1,
class T2,
class T3,
class T4>
693 errno_t Start(
const FUNC& f,
const T1& arg1,
const T2& arg2,
const T3& arg3,
const T4& arg4) {
695 new (std::nothrow) ThreadFuncX4<FUNC, T1, T2, T3, T4>(f, arg1, arg2, arg3, arg4));
696 if (!ptr)
return ENOMEM;
697 return this->StartRaw(ptr);
699 template <
class FUNC,
class T1,
class T2,
class T3,
class T4,
class T5>
701 const T3& arg3,
const T4& arg4,
const T5& arg5) {
703 std::nothrow) ThreadFuncX5<FUNC, T1, T2, T3, T4, T5>(f, arg1, arg2, arg3, arg4, arg5));
704 if (!ptr)
return ENOMEM;
705 return this->StartRaw(settings, ptr);
707 template <
class FUNC,
class T1,
class T2,
class T3,
class T4,
class T5>
708 errno_t Start(
const FUNC& f,
const T1& arg1,
const T2& arg2,
const T3& arg3,
const T4& arg4,
711 std::nothrow) ThreadFuncX5<FUNC, T1, T2, T3, T4, T5>(f, arg1, arg2, arg3, arg4, arg5));
712 if (!ptr)
return ENOMEM;
713 return this->StartRaw(ptr);
726 struct Thread::Args0<void()> {
728 Args0(
void (*func_)(), X)
730 Args0(
void (*func_)()) : func(func_) {}
738 struct Thread::Args0<void(T1)> {
740 Args0(
void (*func_)(T1), X)
742 Args0(
void (*func_)(T1)) : func(func_) {}
749 template <
class T1,
class T2>
750 struct Thread::Args0<void(T1, T2)> {
752 Args0(
void (*func_)(T1, T2), X)
754 Args0(
void (*func_)(T1, T2)) : func(func_) {}
755 void (*func)(T1, T2);
761 template <
class T1,
class T2,
class T3>
762 struct Thread::Args0<void(T1, T2, T3)> {
764 Args0(
void (*func_)(T1, T2, T3), X)
766 Args0(
void (*func_)(T1, T2, T3)) : func(func_) {}
767 void (*func)(T1, T2, T3);
773 template <
class T1,
class T2,
class T3,
class T4>
774 struct Thread::Args0<void(T1, T2, T3, T4)> {
776 Args0(
void (*func_)(T1, T2, T3, T4), X)
778 Args0(
void (*func_)(T1, T2, T3, T4)) : func(func_) {}
779 void (*func)(T1, T2, T3, T4);
785 template <
class T1,
class T2,
class T3,
class T4,
class T5>
786 struct Thread::Args0<void(T1, T2, T3, T4, T5)> {
788 Args0(
void (*func_)(T1, T2, T3, T4, T5), X)
790 Args0(
void (*func_)(T1, T2, T3, T4, T5)) : func(func_) {}
791 void (*func)(T1, T2, T3, T4, T5);
798 unsigned int num_cpu;
803 namespace this_thread {
814 return e == 0 ? id : -1;
820 if (e != 0)
return e;
831 NLIB_DEFINE_STD_SWAP(::nlib_ns::threading::Thread)
833 #endif // INCLUDE_NN_NLIB_THREADING_THREAD_H_ errno_t GetCpu(int *cpuid) noexcept
呼び出したスレッドが実行されているCPUを取得します。
errno_t YieldThread() noexcept
他のスレッドに制御を譲ります。
errno_t Start(const ThreadSettings &settings, const FUNC &f, move_tag)
スレッドの起動オプションを指定できること以外はそうでないバージョンと同じです。
errno_t Start(const FUNC &f, const T1 &arg1, const T2 &arg2, move_tag)
引数を2つもつ関数を別スレッドで実行開始します。 引数を2つ持つこと以外は、引数が1つのバージョンと同じ...
nlib_thread GetNativeHandle() const noexcept
実装依存のスレッド識別子を取得します。
static errno_t Sleep(const TimeSpan &span) noexcept
span の期間だけスリープします。
constexpr Thread() noexcept
デフォルトコンストラクタです。
int GetStackSize() const noexcept
設定されているスタックサイズを返します。
errno_t Sleep(const TimeSpan &span) noexcept
スリープします。
constexpr ThreadArg(Func func_, T1 arg1_, T2 arg2_, T3 arg3_, T4 arg4_, T5 arg5_)
コンストラクタで構造体のフィールドを初期化します。
C++11の標準ヘッダとなるtype_traitsの代用定義です。 コンパイラや標準ライブラリによってサポートされてい...
errno_t Start(const ThreadSettings &settings, const FUNC &f, const T1 &arg1, const T2 &arg2, const T3 &arg3)
スレッドの起動オプションを指定できること以外はそうでないバージョンと同じです。
bool operator!=(const Thread &rhs) noexcept
nlib_thread_equal()を用いてスレッドを比較します。
constexpr ThreadArg()
デフォルトコンストラクタです。
#define NLIB_DISALLOW_COPY_AND_ASSIGN(TypeName)
TypeName で指定されたクラスのコピーコンストラクタと代入演算子を禁止します。
T1 arg1
別スレッドで実行する関数に渡されるオブジェクトです。
errno_t Detach() noexcept
スレッドをデタッチします。
Func func
別スレッドで実行する関数へのポインタです。
bool GetDetachState() const noexcept
デタッチ状態でスレッドを起動する設定かどうかを返します。
errno_t GetPriority(int32_t *priority) noexcept
スレッドの優先度を取得します。
T3 arg3
別スレッドで実行する関数に渡されるオブジェクトです。
errno_t Start(const FUNC &f, const T1 &arg1, const T2 &arg2, const T3 &arg3, const T4 &arg4, const T5 &arg5, move_tag)
引数を5つもつ関数を別スレッドで実行開始します。引数を5つ持つこと以外は、引数が1つのバージョンと同じで...
~Thread() noexcept
デストラクタです。スレッドがJoinされていない場合はJoinします。
errno_t ChangePriority(int32_t priority) noexcept
スレッドの優先度を設定します。
T5 arg5
別スレッドで実行する関数に渡されるオブジェクトです。
void SetDetachState(bool detached) noexcept
デタッチした状態でスレッドを起動するかどうかを設定します。
errno_t Start(const FUNC &f, const T1 &arg1, const T2 &arg2)
引数を2つもつ関数を別スレッドで実行開始します。引数を2つ持つこと以外は、引数が1つのバージョンと同じで...
errno_t Start(const FUNC &f, const T1 &arg1, const T2 &arg2, const T3 &arg3)
引数を3つもつ関数を別スレッドで実行開始します。引数を3つ持つこと以外は、引数が1つのバージョンと同じで...
UniquePtrはポインタの所有権を保持し、UniquePtrがスコープから出るときにデストラクタでポインタをDELで指...
errno_t Start(const FUNC &f, const T1 &arg1, const T2 &arg2, const T3 &arg3, const T4 &arg4, const T5 &arg5)
引数を5つもつ関数を別スレッドで実行開始します。引数を5つ持つこと以外は、引数が1つのバージョンと同じで...
std::unique_ptrに相当するクラスが定義されています。
Threadクラスでスレッドを実行するために利用できる構造体です。
void swap(Thread &rhs) noexcept
オブジェクトの内容をスワップします。
errno_t Start(const FUNC &f, const T1 &arg1, const T2 &arg2, const T3 &arg3, move_tag)
引数を3つもつ関数を別スレッドで実行開始します。 引数を3つ持つこと以外は、引数が1つのバージョンと同じ...
int GetPriority() const noexcept
スレッドの優先度を取得します。
bool IsJoinable() const noexcept
join可能かどうか調べます。
errno_t SetStackSize(int size) noexcept
スタックサイズを設定します。
errno_t Start(const FUNC &f, const T1 &arg1, move_tag)
引数を1つもつ関数を別スレッドで実行開始します。
errno_t StartRaw(UniquePtr< ThArg > &ptr) noexcept
別スレッドの実行を開始します。
errno_t SetName(const char *literal_string) noexcept
スレッドに名前をつけます。
nlib_thread_attrをラップするクラスです。必要に応じて自動的にnlib_thread_attr_init()とnlib_thread_attr...
errno_t StartRaw(const ThreadSettings &settings, UniquePtr< ThArg > &ptr) noexcept
別スレッドの実行を開始します。
static errno_t YieldThread() noexcept
nlib_yield()を呼び出します。
errno_t Join() noexcept
スレッドの実行完了を待ちます。
errno_t Start(const ThreadSettings &settings, const FUNC &f, const T1 &arg1, const T2 &arg2, const T3 &arg3, move_tag)
スレッドの起動オプションを指定できること以外はそうでないバージョンと同じです。
空の構造体で、関数の引数をムーブすべきことを示すために利用されます。
errno_t Start(const ThreadSettings &settings, const FUNC &f, const T1 &arg1)
スレッドの起動オプションを指定できること以外はそうでないバージョンと同じです。
bool operator==(const Thread &rhs) noexcept
nlib_thread_equal()を用いてスレッドを比較します。
errno_t Start(const FUNC &f, const T1 &arg1, const T2 &arg2, const T3 &arg3, const T4 &arg4)
引数を4つもつ関数を別スレッドで実行開始します。引数を4つ持つこと以外は、引数が1つのバージョンと同じで...
errno_t Start(const FUNC &f, const T1 &arg1)
引数を1つもつ関数を別スレッドで実行開始します。
errno_t Start(const ThreadSettings &settings, const FUNC &f)
スレッドの起動オプションを指定できること以外はそうでないバージョンと同じです。
errno_t SetPriority(int priority) noexcept
スレッドの優先度を設定します。優先度の値はプラットフォーム依存です。
nlib_thread_id GetId() noexcept
カレントスレッドのIDを取得します。
#define NLIB_NOEXCEPT
環境に合わせてnoexcept 又は同等の定義がされます。
#define NLIB_CEXPR
利用可能であればconstexprが定義されます。そうでない場合は空文字列です。
errno_t Start(const FUNC &f)
引数を持たない関数を別スレッドで実行開始します。
errno_t Start(const ThreadSettings &settings, const FUNC &f, const T1 &arg1, const T2 &arg2, const T3 &arg3, const T4 &arg4)
スレッドの起動オプションを指定できること以外はそうでないバージョンと同じです。
size_t GetHardwareConcurrency() noexcept
ハードウェアスレッドの数を返します。
errno_t SetAffinity(uint32_t affinity) noexcept
スレッドのプロセッサアフィニティマスクを設定します。
ThreadArg< T1, T2, T3, T4, T5 > ThisType
この構造体の型へのtypedefです。
errno_t StartRaw(const ThreadSettings &settings, ThreadArg<>::Func func) noexcept
引数を持たない関数を別スレッドで実行開始します。
UniquePtr< ThisType > ArgType
スレッド関数の引数型のtypedefです。
errno_t Start(const ThreadSettings &settings, const FUNC &f, const T1 &arg1, const T2 &arg2, const T3 &arg3, const T4 &arg4, move_tag)
スレッドの起動オプションを指定できること以外はそうでないバージョンと同じです。
#define NLIB_FINAL
利用可能であればfinalが定義されます。そうでない場合は空文字列です。
errno_t Start(const ThreadSettings &settings, const FUNC &f, const T1 &arg1, const T2 &arg2, const T3 &arg3, const T4 &arg4, const T5 &arg5)
スレッドの起動オプションを指定できること以外はそうでないバージョンと同じです。
errno_t Start(const ThreadSettings &settings, const FUNC &f, const T1 &arg1, const T2 &arg2, move_tag)
スレッドの起動オプションを指定できること以外はそうでないバージョンと同じです。
errno_t Start(const ThreadSettings &settings, const FUNC &f, const T1 &arg1, const T2 &arg2, const T3 &arg3, const T4 &arg4, const T5 &arg5, move_tag)
スレッドの起動オプションを指定できること以外はそうでないバージョンと同じです。
errno_t SleepMilliSeconds(unsigned int millisec) noexcept
スリープします。
errno_t StartRaw(ThreadArg<>::Func func) noexcept
引数を持たない関数を別スレッドで実行開始します。
T4 arg4
別スレッドで実行する関数に渡されるオブジェクトです。
errno_t GetPriority(int32_t *priority) noexcept
nlib_thread_getpriority()を呼び出して、スレッドの実行優先順位を取得します。
errno_t Start(const ThreadSettings &settings, const FUNC &f, const T1 &arg1, move_tag)
スレッドの起動オプションを指定できること以外はそうでないバージョンと同じです。
T2 arg2
別スレッドで実行する関数に渡されるオブジェクトです。
errno_t Start(const FUNC &f, move_tag)
引数を持たない関数を別スレッドで実行開始します。 引数を持たないこと以外は、引数が1つのバージョンと同...
errno_t Start(const FUNC &f, const T1 &arg1, const T2 &arg2, const T3 &arg3, const T4 &arg4, move_tag)
引数を4つもつ関数を別スレッドで実行開始します。 引数を4つ持つこと以外は、引数が1つのバージョンと同じ...
errno_t ChangePriority(int32_t priority) noexcept
nlib_thread_setpriority()を呼び出して、スレッドの実行優先順位を設定します。
errno_t Start(const ThreadSettings &settings, const FUNC &f, const T1 &arg1, const T2 &arg2)
スレッドの起動オプションを指定できること以外はそうでないバージョンと同じです。