diff --git a/cwIo.cpp b/cwIo.cpp index e25dad1..10b2879 100644 --- a/cwIo.cpp +++ b/cwIo.cpp @@ -9,6 +9,7 @@ #include "cwText.h" #include "cwTextBuf.h" +#include "cwThread.h" #include "cwIo.h" #include "cwMidi.h" @@ -17,7 +18,6 @@ #include "cwObject.h" -#include "cwThread.h" #include "cwThreadMach.h" #include "cwMutex.h" @@ -115,6 +115,18 @@ namespace cw unsigned sockA_index; unsigned userId; } socket_t; + + typedef struct thread_once_str + { + struct io_str* p; + thread::handle_t threadH; + void* arg; + thread_once_func_t cbFunc; + unsigned id; + struct thread_once_str* link; + bool asyncFl; + bool doneFl; + } thread_once_t; typedef struct io_str { @@ -175,6 +187,8 @@ namespace cw sample_t latency_meas_thresh_lin; bool latency_meas_enable_fl; latency_meas_result_t latency_meas_result; + + thread_once_t* threadOnceList; } io_t; @@ -270,7 +284,7 @@ namespace cw { rc_t rc = kOkRC; thread_t* t = (thread_t*)arg; - thread_msg_t tm = { .id=t->id, .arg=t->arg }; + thread_msg_t tm = { .id=t->id, .arg=t->arg, .rc=kOkRC }; msg_t m; m.tid = kThreadTId; @@ -293,6 +307,62 @@ namespace cw } } + bool _thread_once_func( void* arg ) + { + rc_t rc = kOkRC; + thread_once_t* t = (thread_once_t*)arg; + + rc = t->cbFunc(t->arg); + + thread_msg_t tm = { .id=t->id, .arg=t->arg, .rc=rc }; + msg_t m; + + m.tid = kThreadTId; + m.u.thread = &tm; + + if((rc = _ioCallback( t->p, t->asyncFl, &m )) != kOkRC ) + cwLogError(rc,"Thread app callback failed."); + + t->doneFl = true; + + return false; + } + + void _thread_once_cleanup( io_t* p, bool force_clean_fl=false ) + { + thread_once_t* t0 = nullptr; + thread_once_t* t = p->threadOnceList; + while( t != nullptr ) + { + thread_once_t* t_next = t->link; + + if( t->doneFl || force_clean_fl ) + { + thread_once_t* tt = t; + + // point the previous link around this link + if( t0 != nullptr ) + { + t0->link = t_next; + } + else + { // first link is being removed + p->threadOnceList = t_next; + } + + thread::destroy(tt->threadH); + mem::release(tt); + + } + else + { + t0 = t; + } + + t = t_next; + } + } + //---------------------------------------------------------------------------------------------------------- // @@ -2334,6 +2404,9 @@ namespace cw if((rc = thread_mach::destroy(p->threadMachH)) != kOkRC ) return rc; + + _thread_once_cleanup(p,true); + for(unsigned i=0; itimerN; ++i) mem::release(p->timerA[i].label); @@ -2365,7 +2438,6 @@ namespace cw p->cfg->free(); - mem::release(p); return rc; @@ -2564,6 +2636,9 @@ cw::rc_t cw::io::exec( handle_t h, unsigned timeOutMs, void* execCbArg ) if( p->audioMeterDevEnabledN ) _audioDeviceProcessMeters(p); + if( p->threadOnceList != nullptr ) + _thread_once_cleanup(p); + msg_t m; m.tid = kExecTId; m.u.exec.execArg = execCbArg; @@ -2627,7 +2702,7 @@ cw::rc_t cw::io::threadCreate( handle_t h, unsigned id, bool asyncFl, void* arg, { rc_t rc = kOkRC; io_t* p = _handleToPtr(h); - thread_t* t = mem::allocZ(1); + thread_t* t = mem::allocZ(1); // BUG BUG BUG: where does this get released() t->id = id; t->asyncFl = asyncFl; @@ -2642,6 +2717,46 @@ cw::rc_t cw::io::threadCreate( handle_t h, unsigned id, bool asyncFl, void* arg, return rc; } +cw::rc_t cw::io::threadRunOnce( handle_t h, unsigned id, bool asyncFl, thread_once_func_t func, void* arg, const char* thread_label ) +{ + + rc_t rc = kOkRC; + io_t* p = _handleToPtr(h); + thread_once_t* t = mem::allocZ(); + + t->p = p; + t->arg = arg; + t->cbFunc = func; + t->asyncFl = asyncFl; + t->doneFl = false; + t->id = id; + t->link = p->threadOnceList; + p->threadOnceList = t; + + if((rc = thread::create(t->threadH,_thread_once_func,t,thread_label)) != kOkRC ) + { + rc = cwLogError(rc,"One-time thread '%s' create failed.",cwStringNullGuard(thread_label)); + goto errLabel; + } + + sleepMs(10); + + if((rc = thread::unpause(t->threadH)) != kOkRC ) + { + rc = cwLogError(rc,"One-time '%s' thread start failed.",cwStringNullGuard(thread_label)); + goto errLabel; + } + +errLabel: + if( rc != kOkRC ) + { + thread::destroy(t->threadH); + mem::release(t); + } + return rc; +} + + //---------------------------------------------------------------------------------------------------------- // // Timer diff --git a/cwIo.h b/cwIo.h index 9199cc5..2f5a038 100644 --- a/cwIo.h +++ b/cwIo.h @@ -43,8 +43,9 @@ namespace cw typedef struct thread_msg_str { - unsigned id; - void* arg; + unsigned id; // user supplied thread id + void* arg; // user supplied arg passed to threadRunOnce() thread func + rc_t rc; // rc returned from user supplied threadRunOncec() thread func } thread_msg_t; typedef struct timer_msg_str @@ -186,6 +187,10 @@ namespace cw // rc_t threadCreate( handle_t h, unsigned id, bool asyncFl, void* arg, const char* label ); + // Run 'func(arg)' in a thread and callback the app with a thread_msg_t when the func is complete. + typedef rc_t (*thread_once_func_t)(void* arg); + rc_t threadRunOnce( handle_t h, unsigned id, bool asyncFl, thread_once_func_t func, void* arg, const char* thread_label ); + //---------------------------------------------------------------------------------------------------------- // // Timer