Commit | Line | Data |
---|---|---|
f00ff009 LOK |
1 | #pragma once |
2 | /* | |
3 | * This file is part of the libCEC(R) library. | |
4 | * | |
b492c10e | 5 | * libCEC(R) is Copyright (C) 2011-2012 Pulse-Eight Limited. All rights reserved. |
f00ff009 LOK |
6 | * libCEC(R) is an original work, containing original code. |
7 | * | |
8 | * libCEC(R) is a trademark of Pulse-Eight Limited. | |
9 | * | |
10 | * This program is dual-licensed; you can redistribute it and/or modify | |
11 | * it under the terms of the GNU General Public License as published by | |
12 | * the Free Software Foundation; either version 2 of the License, or | |
13 | * (at your option) any later version. | |
14 | * | |
15 | * This program is distributed in the hope that it will be useful, | |
16 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
17 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
18 | * GNU General Public License for more details. | |
19 | * | |
20 | * You should have received a copy of the GNU General Public License | |
21 | * along with this program; if not, write to the Free Software | |
22 | * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. | |
23 | * | |
24 | * | |
25 | * Alternatively, you can license this library under a commercial license, | |
26 | * please contact Pulse-Eight Licensing for more information. | |
27 | * | |
28 | * For more information contact: | |
29 | * Pulse-Eight Licensing <license@pulse-eight.com> | |
30 | * http://www.pulse-eight.com/ | |
31 | * http://www.pulse-eight.net/ | |
32 | */ | |
33 | ||
34 | #include "mutex.h" | |
35 | ||
36 | namespace PLATFORM | |
37 | { | |
38 | class CThread | |
39 | { | |
40 | public: | |
41 | CThread(void) : | |
42 | m_bStop(false), | |
960f33c6 | 43 | m_bRunning(false), |
99aeafb9 | 44 | m_bStopped(false), |
622f7c33 | 45 | m_thread(INVALID_THREAD_VALUE) {} |
f00ff009 LOK |
46 | |
47 | virtual ~CThread(void) | |
48 | { | |
002fa4ea | 49 | StopThread(0); |
5806561e | 50 | void *retVal = NULL; |
622f7c33 | 51 | if (m_thread != INVALID_THREAD_VALUE) |
dbe84b6e | 52 | if (ThreadsWait(m_thread, &retVal)){}; // silence unused warning |
f00ff009 LOK |
53 | } |
54 | ||
55 | static void *ThreadHandler(CThread *thread) | |
56 | { | |
57 | void *retVal = NULL; | |
58 | ||
59 | if (thread) | |
60 | { | |
960f33c6 LOK |
61 | { |
62 | CLockObject lock(thread->m_threadMutex); | |
63 | thread->m_bRunning = true; | |
64 | thread->m_bStopped = false; | |
65 | thread->m_threadCondition.Broadcast(); | |
66 | } | |
f00ff009 LOK |
67 | |
68 | retVal = thread->Process(); | |
69 | ||
960f33c6 LOK |
70 | { |
71 | CLockObject lock(thread->m_threadMutex); | |
72 | thread->m_bRunning = false; | |
73 | thread->m_bStopped = true; | |
74 | thread->m_threadCondition.Broadcast(); | |
75 | } | |
f00ff009 LOK |
76 | } |
77 | ||
78 | return retVal; | |
79 | } | |
80 | ||
c12ee8e6 | 81 | virtual bool IsRunning(void) |
f00ff009 LOK |
82 | { |
83 | CLockObject lock(m_threadMutex); | |
84 | return m_bRunning; | |
85 | } | |
86 | ||
c12ee8e6 | 87 | virtual bool IsStopped(void) |
f00ff009 LOK |
88 | { |
89 | CLockObject lock(m_threadMutex); | |
90 | return m_bStop; | |
91 | } | |
92 | ||
c12ee8e6 | 93 | virtual bool CreateThread(bool bWait = true) |
f00ff009 | 94 | { |
ce37a9b1 LOK |
95 | bool bReturn(false); |
96 | CLockObject lock(m_threadMutex); | |
97 | if (!IsRunning()) | |
98 | { | |
99 | m_bStop = false; | |
100 | if (ThreadsCreate(m_thread, CThread::ThreadHandler, ((void*)static_cast<CThread *>(this)))) | |
f00ff009 | 101 | { |
ce37a9b1 LOK |
102 | if (bWait) |
103 | m_threadCondition.Wait(m_threadMutex, m_bRunning); | |
104 | bReturn = true; | |
f00ff009 | 105 | } |
ce37a9b1 | 106 | } |
f00ff009 LOK |
107 | return bReturn; |
108 | } | |
109 | ||
78d138bd LOK |
110 | /*! |
111 | * @brief Stop the thread | |
112 | * @param iWaitMs negative = don't wait, 0 = infinite, or the amount of ms to wait | |
113 | */ | |
114 | virtual bool StopThread(int iWaitMs = 5000) | |
f00ff009 LOK |
115 | { |
116 | bool bReturn(true); | |
117 | bool bRunning(false); | |
118 | { | |
119 | CLockObject lock(m_threadMutex); | |
120 | bRunning = IsRunning(); | |
121 | m_bStop = true; | |
f00ff009 LOK |
122 | } |
123 | ||
78d138bd | 124 | if (bRunning && iWaitMs >= 0) |
f00ff009 | 125 | { |
78d138bd LOK |
126 | CLockObject lock(m_threadMutex); |
127 | bReturn = m_threadCondition.Wait(m_threadMutex, m_bStopped, iWaitMs); | |
128 | } | |
129 | else | |
130 | { | |
131 | bReturn = true; | |
f00ff009 | 132 | } |
78d138bd LOK |
133 | |
134 | return bReturn; | |
f00ff009 LOK |
135 | } |
136 | ||
c12ee8e6 | 137 | virtual bool Sleep(uint32_t iTimeout) |
f00ff009 LOK |
138 | { |
139 | CLockObject lock(m_threadMutex); | |
960f33c6 | 140 | return m_bStop ? false : m_threadCondition.Wait(m_threadMutex, m_bStopped, iTimeout); |
f00ff009 LOK |
141 | } |
142 | ||
143 | virtual void *Process(void) = 0; | |
144 | ||
145 | protected: | |
146 | void SetRunning(bool bSetTo); | |
0b714871 | 147 | CMutex m_threadMutex; |
f00ff009 LOK |
148 | |
149 | private: | |
8f6e48cd LOK |
150 | bool m_bStop; |
151 | bool m_bRunning; | |
152 | bool m_bStopped; | |
153 | CCondition<bool> m_threadCondition; | |
8f6e48cd | 154 | thread_t m_thread; |
f00ff009 LOK |
155 | }; |
156 | }; |