2016-09-06 00:07:34 +02:00
|
|
|
/*
|
|
|
|
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_)
|
2016-09-06 09:26:25 +02:00
|
|
|
{
|
2016-09-06 00:07:34 +02:00
|
|
|
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;
|
|
|
|
}
|
2016-09-06 09:26:25 +02:00
|
|
|
|
2016-09-06 00:07:34 +02:00
|
|
|
//_sceFiberInitializeImpl
|
2016-09-10 01:20:34 +02:00
|
|
|
int ret = _sceFiberInitializeImpl(tailFiber, "tailFiber", co_thunk, (uint32_t)coentry, (void*) m_contextBuffer, heapsize, NULL);
|
2016-09-06 00:07:34 +02:00
|
|
|
if(ret==0){
|
|
|
|
return (cothread_t)tailFiber;
|
|
|
|
}else{
|
|
|
|
return (cothread_t)ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void co_delete(cothread_t cothread)
|
|
|
|
{
|
2016-09-10 01:20:34 +02:00
|
|
|
if(cothread == (cothread_t)1){
|
|
|
|
return;
|
|
|
|
}
|
2016-09-06 00:07:34 +02:00
|
|
|
sceFiberFinalize((SceFiber*)cothread);
|
|
|
|
}
|
|
|
|
|
|
|
|
void co_switch(cothread_t cothread)
|
|
|
|
{
|
2016-09-06 09:26:25 +02:00
|
|
|
|
2016-09-06 00:07:34 +02:00
|
|
|
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){
|
2016-09-06 09:26:25 +02:00
|
|
|
co_active_ = cothread;
|
2016-09-06 00:07:34 +02:00
|
|
|
sceFiberRun(theFiber, 0, &argOnReturn);
|
|
|
|
}else{
|
2016-09-06 09:26:25 +02:00
|
|
|
co_active_ = cothread;
|
2016-09-06 00:07:34 +02:00
|
|
|
sceFiberSwitch(theFiber, 0, &argOnReturn);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|