blob: 190c897c0c06e40bff425226f3ce212db26279c7 [file] [log] [blame] [edit]
/* RTco.cc provides minimal access to thread primitives.
Copyright (C) 2019-2022 Free Software Foundation, Inc.
Contributed by Gaius Mulley <gaius.mulley@southwales.ac.uk>.
This file is part of GNU Modula-2.
GNU Modula-2 is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3, or (at your option)
any later version.
GNU Modula-2 is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
General Public License for more details.
Under Section 7 of GPL version 3, you are granted additional
permissions described in the GCC Runtime Library Exception, version
3.1, as published by the Free Software Foundation.
You should have received a copy of the GNU General Public License and
a copy of the GCC Runtime Library Exception along with this program;
see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
<http://www.gnu.org/licenses/>. */
#include "config.h"
#include <unistd.h>
#include <pthread.h>
#include <sys/select.h>
#include <stdlib.h>
#include <m2rts.h>
#include <cstdio>
#define EXPORT(FUNC) m2iso ## _RTco_ ## FUNC
#define M2EXPORT(FUNC) m2iso ## _M2_RTco_ ## FUNC
#define M2LIBNAME "m2iso"
/* This implementation of RTco.cc uses a single lock for mutex across
the whole module. It also forces context switching between threads
in transfer by combining an implementation of wait and signal.
All semaphores are implemented using the same mutex lock and
separate condition variables. */
#undef TRACEON
#define POOL
#define SEM_POOL 10000
#define THREAD_POOL 10000
#define _GTHREAD_USE_COND_INIT_FUNC
#include "gthr.h"
/* Ensure that ANSI conform stdio is used. This needs to be set
before any system header file is included. */
#if defined __MINGW32__
#define _POSIX 1
#define gm2_printf gnu_printf
#else
#define gm2_printf __printf__
#endif
#if defined(TRACEON)
#define tprintf printf
#else
#define tprintf(...)
#endif
typedef struct threadCB_s
{
void (*proc) (void);
pthread_t p;
int tid; /* The thread id. */
unsigned int interruptLevel;
__gthread_cond_t run_counter; /* Used to block the thread and force
a context switch. */
int value; /* Count 0 or 1. */
bool waiting; /* Is this thread waiting on the run_counter? */
} threadCB;
typedef struct threadSem_s
{
__gthread_cond_t counter;
bool waiting;
int sem_value;
} threadSem;
static unsigned int nThreads = 0;
static threadCB *threadArray = NULL;
static unsigned int nSemaphores = 0;
static threadSem **semArray = NULL;
/* These are used to lock the above module data structures. */
static __gthread_mutex_t lock; /* This is the only mutex for
the whole module. */
static bool initialized = false;
extern "C" int EXPORT(init) (void);
extern "C" void
M2EXPORT(dep) (void)
{
}
extern "C" void
M2EXPORT(init) (int argc, char *argv[], char *envp[])
{
}
extern "C" void
M2EXPORT(fini) (int argc, char *argv[], char *envp[])
{
}
static void
initSem (threadSem *sem, int value)
{
__GTHREAD_COND_INIT_FUNCTION (&sem->counter);
sem->waiting = false;
sem->sem_value = value;
}
static void
waitSem (threadSem *sem)
{
__gthread_mutex_lock (&lock);
if (sem->sem_value == 0)
{
sem->waiting = true;
__gthread_cond_wait (&sem->counter, &lock);
sem->waiting = false;
}
else
sem->sem_value--;
__gthread_mutex_unlock (&lock);
}
static void
signalSem (threadSem *sem)
{
__gthread_mutex_lock (&lock);
if (sem->waiting)
__gthread_cond_signal (&sem->counter);
else
sem->sem_value++;
__gthread_mutex_unlock (&lock);
}
extern "C" void
EXPORT(wait) (int sid)
{
EXPORT(init) ();
tprintf ("wait %d\n", sid);
waitSem (semArray[sid]);
}
extern "C" void
EXPORT(signal) (int sid)
{
EXPORT(init) ();
tprintf ("signal %d\n", sid);
signalSem (semArray[sid]);
}
static int
newSem (void)
{
#if defined(POOL)
semArray[nSemaphores]
= (threadSem *)malloc (sizeof (threadSem));
nSemaphores += 1;
if (nSemaphores == SEM_POOL)
m2iso_M2RTS_HaltC ("too many semaphores created",
__FILE__, __FUNCTION__, __LINE__);
#else
threadSem *sem
= (threadSem *)malloc (sizeof (threadSem));
/* We need to be careful when using realloc as the lock (semaphore)
operators use the semaphore address. So we keep an array of pointer
to semaphores. */
if (nSemaphores == 0)
{
semArray = (threadSem **)malloc (sizeof (sem));
nSemaphores = 1;
}
else
{
nSemaphores += 1;
semArray = (threadSem **)realloc (semArray,
sizeof (sem) * nSemaphores);
}
semArray[nSemaphores - 1] = sem;
#endif
return nSemaphores - 1;
}
static int
initSemaphore (int value)
{
int sid = newSem ();
initSem (semArray[sid], value);
tprintf ("%d = initSemaphore (%d)\n", sid, value);
return sid;
}
extern "C" int
EXPORT(initSemaphore) (int value)
{
int sid;
tprintf ("initSemaphore (%d) called\n", value);
EXPORT(init) ();
tprintf ("about to access lock\n");
__gthread_mutex_lock (&lock);
sid = initSemaphore (value);
__gthread_mutex_unlock (&lock);
return sid;
}
static int
currentThread (void)
{
int tid;
for (tid = 0; tid < nThreads; tid++)
if (pthread_self () == threadArray[tid].p)
return tid;
m2iso_M2RTS_HaltC ("failed to find currentThread",
__FILE__, __FUNCTION__, __LINE__);
}
extern "C" int
EXPORT(currentThread) (void)
{
int tid;
EXPORT(init) ();
__gthread_mutex_lock (&lock);
tid = currentThread ();
tprintf ("currentThread %d\n", tid);
__gthread_mutex_unlock (&lock);
return tid;
}
/* currentInterruptLevel returns the interrupt level of the current thread. */
extern "C" unsigned int
EXPORT(currentInterruptLevel) (void)
{
EXPORT(init) ();
__gthread_mutex_lock (&lock);
int current = currentThread ();
tprintf ("currentInterruptLevel %d\n",
threadArray[current].interruptLevel);
int level = threadArray[current].interruptLevel;
__gthread_mutex_unlock (&lock);
return level;
}
/* turninterrupts returns the old interrupt level and assigns the
interrupt level to newLevel. */
extern "C" unsigned int
EXPORT(turnInterrupts) (unsigned int newLevel)
{
EXPORT(init) ();
__gthread_mutex_lock (&lock);
int current = currentThread ();
unsigned int old = threadArray[current].interruptLevel;
tprintf ("turnInterrupts from %d to %d\n", old, newLevel);
threadArray[current].interruptLevel = newLevel;
__gthread_mutex_unlock (&lock);
return old;
}
static void
never (void)
{
m2iso_M2RTS_HaltC ("the main thread should never call here",
__FILE__, __FUNCTION__, __LINE__);
}
static void *
execThread (void *t)
{
threadCB *tp = (threadCB *)t;
tprintf ("exec thread tid = %d coming to life\n", tp->tid);
__gthread_mutex_lock (&lock);
tprintf ("exec thread tid = %d function = 0x%p arg = 0x%p\n", tp->tid,
tp->proc, t);
/* Has the thread been signalled? */
if (tp->value == 0)
{
/* Not been signalled therefore we force ourselves to block. */
tprintf ("%s: forcing thread tid = %d to wait\n",
__FUNCTION__, tp->tid);
tp->waiting = true; /* We are waiting. */
__gthread_cond_wait (&tp->run_counter, &lock);
tp->waiting = false; /* Running again. */
}
else
{
/* Yes signalled, therefore just take the recorded signal and continue. */
tprintf ("%s: no need for thread tid = %d to wait\n",
__FUNCTION__, tp->tid);
tp->value--;
}
tprintf (" running exec thread [%d] function = 0x%p arg = 0x%p\n", tp->tid,
tp->proc, t);
__gthread_mutex_unlock (&lock);
tp->proc (); /* Now execute user procedure. */
#if 0
m2iso_M2RTS_CoroutineException ( __FILE__, __LINE__, __COLUMN__, __FUNCTION__, "coroutine finishing");
#endif
m2iso_M2RTS_HaltC ("execThread should never finish",
__FILE__, __FUNCTION__, __LINE__);
return NULL;
}
static int
newThread (void)
{
#if defined(POOL)
nThreads += 1;
if (nThreads == THREAD_POOL)
m2iso_M2RTS_HaltC ("too many threads created",
__FILE__, __FUNCTION__, __LINE__);
return nThreads - 1;
#else
if (nThreads == 0)
{
threadArray = (threadCB *)malloc (sizeof (threadCB));
nThreads = 1;
}
else
{
nThreads += 1;
threadArray
= (threadCB *)realloc (threadArray, sizeof (threadCB) * nThreads);
}
return nThreads - 1;
#endif
}
static int
initThread (void (*proc) (void), unsigned int stackSize,
unsigned int interrupt)
{
int tid = newThread ();
pthread_attr_t attr;
int result;
threadArray[tid].proc = proc;
threadArray[tid].tid = tid;
/* Initialize the thread run_counter used to force a context switch. */
__GTHREAD_COND_INIT_FUNCTION (&threadArray[tid].run_counter);
threadArray[tid].interruptLevel = interrupt;
threadArray[tid].waiting = false; /* The thread is running. */
threadArray[tid].value = 0; /* No signal has been seen yet. */
/* Set thread creation attributes. */
result = pthread_attr_init (&attr);
if (result != 0)
m2iso_M2RTS_HaltC ("failed to create thread attribute",
__FILE__, __FUNCTION__, __LINE__);
if (stackSize > 0)
{
result = pthread_attr_setstacksize (&attr, stackSize);
if (result != 0)
m2iso_M2RTS_HaltC ("failed to set stack size attribute",
__FILE__, __FUNCTION__, __LINE__);
}
tprintf ("initThread [%d] function = 0x%p (arg = 0x%p)\n", tid, proc,
(void *)&threadArray[tid]);
result = pthread_create (&threadArray[tid].p, &attr, execThread,
(void *)&threadArray[tid]);
if (result != 0)
m2iso_M2RTS_HaltC ("thread_create failed",
__FILE__, __FUNCTION__, __LINE__);
tprintf (" created thread [%d] function = 0x%p 0x%p\n", tid, proc,
(void *)&threadArray[tid]);
return tid;
}
extern "C" int
EXPORT(initThread) (void (*proc) (void), unsigned int stackSize,
unsigned int interrupt)
{
int tid;
EXPORT(init) ();
__gthread_mutex_lock (&lock);
tid = initThread (proc, stackSize, interrupt);
__gthread_mutex_unlock (&lock);
return tid;
}
/* transfer unlocks thread p2 and locks the current thread. p1 is
updated with the current thread id.
The implementation of transfer uses a combined wait/signal. */
extern "C" void
EXPORT(transfer) (int *p1, int p2)
{
__gthread_mutex_lock (&lock);
{
int current = currentThread ();
if (!initialized)
m2iso_M2RTS_HaltC ("cannot transfer to a process before the process has been created",
__FILE__, __FUNCTION__, __LINE__);
if (current == p2)
{
/* Error. */
m2iso_M2RTS_HaltC ("attempting to transfer to ourself",
__FILE__, __FUNCTION__, __LINE__);
}
else
{
*p1 = current;
int old = current;
tprintf ("start, context switching from: %d to %d\n", current, p2);
/* Perform signal (p2 sem). Without the mutex lock as we have
already obtained it above. */
if (threadArray[p2].waiting)
{
/* p2 is blocked on the condition variable, release it. */
tprintf ("p1 = %d cond_signal to p2 (%d)\n", current, p2);
__gthread_cond_signal (&threadArray[p2].run_counter);
tprintf ("after p1 = %d cond_signal to p2 (%d)\n", current, p2);
}
else
{
/* p2 hasn't reached the condition variable, so bump value
ready for p2 to test. */
tprintf ("no need for thread %d to cond_signal - bump %d value (pre) = %d\n",
current, p2, threadArray[p2].value);
threadArray[p2].value++;
}
/* Perform wait (old sem). Again without obtaining mutex as
we've already claimed it. */
if (threadArray[old].value == 0)
{
/* Record we are about to wait on the condition variable. */
threadArray[old].waiting = true;
__gthread_cond_wait (&threadArray[old].run_counter, &lock);
threadArray[old].waiting = false;
/* We are running again. */
}
else
{
tprintf ("(currentThread = %d) no need for thread %d to cond_wait - taking value (pre) = %d\n",
current, old, threadArray[old].value);
/* No need to block as we have been told a signal has
effectively already been recorded. We remove the signal
notification without blocking. */
threadArray[old].value--;
}
tprintf ("end, context back to %d\n", current);
if (current != old)
m2iso_M2RTS_HaltC ("wrong process id",
__FILE__, __FUNCTION__, __LINE__);
}
}
__gthread_mutex_unlock (&lock);
}
extern "C" int
EXPORT(select) (int p1, fd_set *p2, fd_set *p3, fd_set *p4, const timespec *p5)
{
EXPORT(init) ();
tprintf ("[%x] RTco.select (...)\n", pthread_self ());
return pselect (p1, p2, p3, p4, p5, NULL);
}
extern "C" int
EXPORT(init) (void)
{
tprintf ("checking init\n");
if (! initialized)
{
initialized = true;
tprintf ("RTco initialized\n");
__GTHREAD_MUTEX_INIT_FUNCTION (&lock);
__gthread_mutex_lock (&lock);
/* Create initial thread container. */
#if defined(POOL)
threadArray = (threadCB *)malloc (sizeof (threadCB) * THREAD_POOL);
semArray = (threadSem **)malloc (sizeof (threadSem *) * SEM_POOL);
#endif
/* Create a thread control block for the main program (or process). */
int tid = newThread (); /* For the current initial thread. */
threadArray[tid].p = pthread_self ();
threadArray[tid].tid = tid;
__GTHREAD_COND_INIT_FUNCTION (&threadArray[tid].run_counter);
threadArray[tid].interruptLevel = 0;
/* The line below shouldn't be necessary as we are already running. */
threadArray[tid].proc = never;
threadArray[tid].waiting = false; /* We are running. */
threadArray[tid].value = 0; /* No signal from anyone yet. */
tprintf ("RTco initialized completed\n");
__gthread_mutex_unlock (&lock);
}
return 0;
}
extern "C" void __attribute__((__constructor__))
M2EXPORT(ctor) (void)
{
m2iso_M2RTS_RegisterModule ("RTco", M2LIBNAME,
M2EXPORT(init), M2EXPORT(fini),
M2EXPORT(dep));
}