Newer
Older
UbixOS / sys / kernel / ubthread.c
@Charlie Root Charlie Root on 22 Jan 2018 4 KB Whys isn't VM86 working
/*-
 * Copyright (c) 2002-2018 The UbixOS Project.
 * All rights reserved.
 *
 * This was developed by Christopher W. Olsen for the UbixOS Project.
 *
 * Redistribution and use in source and binary forms, with or without modification, are permitted
 * provided that the following conditions are met:
 *
 * 1) Redistributions of source code must retain the above copyright notice, this list of
 *    conditions, the following disclaimer and the list of authors.
 * 2) Redistributions in binary form must reproduce the above copyright notice, this list of
 *    conditions, the following disclaimer and the list of authors in the documentation and/or
 *    other materials provided with the distribution.
 * 3) Neither the name of the UbixOS Project nor the names of its contributors may be used to
 *    endorse or promote products derived from this software without specific prior written
 *    permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED
 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
 * OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

/* All these must be converted to be done atomically */

#include <ubixos/ubthread.h>
#include <ubixos/exec.h>
#include <ubixos/sched.h>
#include <ubixos/time.h>
#include <ubixos/spinlock.h>
#include <ubixos/vitals.h>
#include <lib/kmalloc.h>
#include <lib/kprintf.h>
#include <ubixos/kpanic.h>
#include <sys/stdatomic.h>

struct ubthread_cond_list *conds = 0x0;
struct ubthread_mutex_list *mutex = 0x0;

kTask_t *ubthread_self() {
  return (_current);
}

int ubthread_cond_init(ubthread_cond_t *cond, const uint32_t attr) {
  ubthread_cond_t ubcond = kmalloc(sizeof(struct ubthread_cond));
  memset(ubcond, 0x0, sizeof(struct ubthread_cond));

  ubcond->id = (int) cond;
  ubcond->lock = ATOMIC_VAR_INIT(0);

  *cond = ubcond;
  return (0x0);
}

int ubthread_mutex_init(ubthread_mutex_t *mutex, const uint32_t attr) {
  ubthread_mutex_t ubmutex = kmalloc(sizeof(struct ubthread_mutex));
  memset(ubmutex, 0x0, sizeof(struct ubthread_mutex));

  ubmutex->id = (int) mutex;
  ubmutex->lock = ATOMIC_VAR_INIT(0);

  *mutex = ubmutex;
  return (0x0);
}

int ubthread_cond_destroy(ubthread_cond_t *cond) {
  kfree(*cond);
  *cond = 0x0;
  return (0x0);
}

int ubthread_mutex_destroy(ubthread_mutex_t *mutex) {
  kfree(*mutex);
  *mutex = 0x0;
  return (0x0);
}

int ubthread_create(kTask_t **thread, const uInt32 *attr, void (*tproc)(void), void *arg) {
  *thread = (void *) execThread(tproc,0x2000, arg);
  return (0x0);
}

int ubthread_mutex_lock(ubthread_mutex_t *mutex) {
  ubthread_mutex_t ubmutex = *mutex;

  if (ubmutex->lock == TRUE && ubmutex->pid == _current->id) {
    kprintf("Mutex Already Locked By This Thread");
    kpanic("WHY?");
    return (0x0);
  }

  while (1) {
    if (xchg_32(&ubmutex->lock, TRUE) == FALSE)
      break;

    while (ubmutex->lock == TRUE)
      sched_yield();
  }

  ubmutex->pid = _current->id;
  return (0x0);
}

int ubthread_mutex_unlock(ubthread_mutex_t *mutex) {
  ubthread_mutex_t ubmutex = *mutex;

  if (ubmutex->lock != TRUE)
    kpanic("NOT LOCKED?");

  if (ubmutex->pid != _current->id)
    kprintf("Trying To Unlock Mutex From No Locking Thread[%i - %i:0x%X]\n", ubmutex->pid, _current->id, *ubmutex);

  while (1) {
    if (xchg_32(&ubmutex->lock, FALSE) == TRUE)
      break;
    while (ubmutex->lock == FALSE)
      sched_yield();
  }

  ubmutex->pid = 0x0;
  return (0x0);
}

int ubthread_cond_timedwait(ubthread_cond_t *cond, ubthread_mutex_t *mutex, const struct timespec *abstime) {
  ubthread_cond_t ubcond = *cond;
  ubthread_mutex_t ubmutex = *mutex;

  uint32_t enterTime = systemVitals->sysUptime + 20;

  ubthread_mutex_unlock(mutex);

  while (enterTime > systemVitals->sysUptime) {
    if (ubcond->lock == FALSE)
      break;
    sched_yield();
  }

  ubthread_mutex_lock(mutex);

  return (0x0);
}

int ubthread_cond_wait(ubthread_cond_t *cond, ubthread_mutex_t *mutex) {
  ubthread_cond_t ubcond = *cond;
  ubthread_mutex_unlock(mutex);
  while (ubcond->lock == TRUE)
    sched_yield();
  ubthread_mutex_lock(mutex);
  return (0x0);
}

int ubthread_cond_signal(ubthread_cond_t *cond) {
  ubthread_cond_t ubcond = *cond;
  while (xchg_32(&ubcond->lock, FALSE))
    sched_yield();
  return (0x0);
}

int ubthread_cond_broadcast(ubthread_cond_t *cond) {
  ubthread_cond_t ubcond = *cond;
  while (xchg_32(&ubcond->lock, FALSE))
    sched_yield();
  return (0x0);
}