diff options
author | Twinaphex | 2016-09-29 01:39:22 +0200 |
---|---|---|
committer | GitHub | 2016-09-29 01:39:22 +0200 |
commit | 8fc1be1840221e32534964dc35ebf046d6a2fcda (patch) | |
tree | 0c3f1fc8046828fbc4111325a8b2ff776ee09f60 /libco/scefiber.c | |
parent | e700ad78751555d0e7ec66c6b9b006853725fc19 (diff) | |
parent | 24cbebe2de483639f83c16452375831f7994fab8 (diff) | |
download | picogpsp-8fc1be1840221e32534964dc35ebf046d6a2fcda.tar.gz picogpsp-8fc1be1840221e32534964dc35ebf046d6a2fcda.tar.bz2 picogpsp-8fc1be1840221e32534964dc35ebf046d6a2fcda.zip |
Merge pull request #30 from frangarcj/master
(VITA) Change libco and some optimizations
Diffstat (limited to 'libco/scefiber.c')
-rw-r--r-- | libco/scefiber.c | 96 |
1 files changed, 96 insertions, 0 deletions
diff --git a/libco/scefiber.c b/libco/scefiber.c new file mode 100644 index 0000000..a233bec --- /dev/null +++ b/libco/scefiber.c @@ -0,0 +1,96 @@ +/* + libco.win (2016-09-06) + authors: frangarcj + license: public domain +*/ + +#define LIBCO_C +#include <libco.h> +#include <stdlib.h> +#include <psp2/sysmodule.h> + +#ifdef __cplusplus +extern "C" { +#endif + +static thread_local cothread_t co_active_ = 0; + +typedef struct SceFiber { + char reserved[128]; +} SceFiber __attribute__( ( aligned ( 8 ) ) ) ; + +int32_t _sceFiberInitializeImpl(SceFiber* fiber, char* name, void* entry, uint32_t argOnInitialize, void* addrContext, int32_t sizeContext, void* params); + +int32_t sceFiberFinalize(SceFiber* fiber); + +int32_t sceFiberRun(SceFiber* fiber, uint32_t argOnRunTo, uint32_t* argOnRun); + +int32_t sceFiberSwitch(SceFiber* fiber, uint32_t argOnRunTo, uint32_t* argOnRun); + +int32_t sceFiberReturnToThread(uint32_t argOnReturn, uint32_t* argOnRun); + +void co_thunk(uint32_t argOnInitialize, uint32_t argOnRun) +{ + ((void (*)(void))argOnInitialize)(); +} + +cothread_t co_active(void) +{ + if(!co_active_) + { + sceSysmoduleLoadModule(SCE_SYSMODULE_FIBER); + co_active_ = (cothread_t)1; + } + return co_active_; +} + +cothread_t co_create(unsigned int heapsize, void (*coentry)(void)) +{ + SceFiber* tailFiber = malloc(sizeof(SceFiber)); + char * m_contextBuffer = malloc(sizeof(char)*heapsize); + if(!co_active_) + { + sceSysmoduleLoadModule(SCE_SYSMODULE_FIBER); + co_active_ = (cothread_t)1; + } + + //_sceFiberInitializeImpl + int ret = _sceFiberInitializeImpl(tailFiber, "tailFiber", co_thunk, (uint32_t)coentry, (void*) m_contextBuffer, heapsize, NULL); + if(ret==0){ + return (cothread_t)tailFiber; + }else{ + return (cothread_t)ret; + } + +} + +void co_delete(cothread_t cothread) +{ + if(cothread == (cothread_t)1){ + return; + } + sceFiberFinalize((SceFiber*)cothread); +} + +void co_switch(cothread_t cothread) +{ + + uint32_t argOnReturn = 0; + if(cothread == (cothread_t)1){ + co_active_ = cothread; + sceFiberReturnToThread(0, NULL); + }else{ + SceFiber* theFiber = (SceFiber*)cothread; + if(co_active_ == (cothread_t)1){ + co_active_ = cothread; + sceFiberRun(theFiber, 0, &argOnReturn); + }else{ + co_active_ = cothread; + sceFiberSwitch(theFiber, 0, &argOnReturn); + } + } +} + +#ifdef __cplusplus +} +#endif |