all repos — mgba @ 4c38f769565e8ddd7d3a8eef1a41975206c129a0

mGBA Game Boy Advance Emulator

src/platform/3ds/threading.h (view raw)

  1/* Copyright (c) 2013-2015 Jeffrey Pfau
  2 *
  3 * This Source Code Form is subject to the terms of the Mozilla Public
  4 * License, v. 2.0. If a copy of the MPL was not distributed with this
  5 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
  6#ifndef N3DS_THREADING_H
  7#define N3DS_THREADING_H
  8
  9#include "util/common.h"
 10
 11#include <3ds.h>
 12#include <malloc.h>
 13
 14#ifdef _3DS
 15// ctrulib already has a type called Thread
 16#define Thread CustomThread
 17#endif
 18
 19#define THREAD_ENTRY void
 20typedef ThreadFunc ThreadEntry;
 21
 22typedef struct {
 23	Handle handle;
 24	u8* stack;
 25} Thread;
 26typedef LightLock Mutex;
 27typedef struct {
 28	Mutex mutex;
 29	Handle semaphore;
 30	u32 waiting;
 31} Condition;
 32
 33static inline int MutexInit(Mutex* mutex) {
 34	LightLock_Init(mutex);
 35	return 0;
 36}
 37
 38static inline int MutexDeinit(Mutex* mutex) {
 39	UNUSED(mutex);
 40	return 0;
 41}
 42
 43static inline int MutexLock(Mutex* mutex) {
 44	LightLock_Lock(mutex);
 45	return 0;
 46}
 47
 48static inline int MutexTryLock(Mutex* mutex) {
 49	return LightLock_TryLock(mutex);
 50}
 51
 52static inline int MutexUnlock(Mutex* mutex) {
 53	LightLock_Unlock(mutex);
 54	return 0;
 55}
 56
 57static inline int ConditionInit(Condition* cond) {
 58	Result res = MutexInit(&cond->mutex);
 59	if (res) {
 60		return res;
 61	}
 62	res = svcCreateSemaphore(&cond->semaphore, 0, 1);
 63	cond->waiting = 0;
 64	return res;
 65}
 66
 67static inline int ConditionDeinit(Condition* cond) {
 68	return svcCloseHandle(cond->semaphore);
 69}
 70
 71static inline int ConditionWait(Condition* cond, Mutex* mutex) {
 72	MutexLock(&cond->mutex);
 73	++cond->waiting;
 74	MutexUnlock(mutex);
 75	MutexUnlock(&cond->mutex);
 76	svcWaitSynchronization(cond->semaphore, U64_MAX);
 77	MutexLock(mutex);
 78	return 0;
 79}
 80
 81static inline int ConditionWaitTimed(Condition* cond, Mutex* mutex, int32_t timeoutMs) {
 82	MutexLock(&cond->mutex);
 83	++cond->waiting;
 84	MutexUnlock(mutex);
 85	MutexUnlock(&cond->mutex);
 86	svcWaitSynchronization(cond->semaphore, timeoutMs * 10000000LL);
 87	MutexLock(mutex);
 88	return 0;
 89}
 90
 91static inline int ConditionWake(Condition* cond) {
 92	MutexLock(&cond->mutex);
 93	if (cond->waiting) {
 94		--cond->waiting;
 95		s32 count = 0;
 96		svcReleaseSemaphore(&count, cond->semaphore, 1);
 97	}
 98	MutexUnlock(&cond->mutex);
 99	return 0;
100}
101
102static inline int ThreadCreate(Thread* thread, ThreadEntry entry, void* context) {
103	if (!entry || !thread) {
104		return 1;
105	}
106	thread->stack = memalign(8, 0x8000);
107	if (!thread->stack) {
108		return 1;
109	}
110	bool isNew3DS;
111	APT_CheckNew3DS(&isNew3DS);
112	if (isNew3DS && svcCreateThread(&thread->handle, entry, (u32) context, (u32*) &thread->stack[0x8000], 0x18, 2) == 0) {
113		return 0;
114	}
115	return svcCreateThread(&thread->handle, entry, (u32) context, (u32*) &thread->stack[0x8000], 0x18, -1);
116}
117
118static inline int ThreadJoin(Thread thread) {
119	svcWaitSynchronization(thread.handle, U64_MAX);
120	free(thread.stack);
121	return 0;
122}
123
124static inline void ThreadSetName(const char* name) {
125	UNUSED(name);
126	// Unimplemented
127}
128
129#endif