2002-07-09 16:24:59 +04:00
|
|
|
/* This file contains the atomic_* and cpu functions (init, etc). */
|
|
|
|
|
|
|
|
/*
|
|
|
|
** Copyright 2002, Travis Geiselbrecht. All rights reserved.
|
|
|
|
** Distributed under the terms of the NewOS License.
|
|
|
|
*/
|
2003-05-03 20:20:38 +04:00
|
|
|
|
|
|
|
// ToDo: fix the atomic_*() functions and possibly move them elsewhere;
|
|
|
|
// they are exported in libroot.so anyway, so the best place would
|
|
|
|
// probably there.
|
|
|
|
|
|
|
|
|
2002-07-09 16:24:59 +04:00
|
|
|
#include <kernel.h>
|
|
|
|
#include <cpu.h>
|
|
|
|
#include <vm.h>
|
|
|
|
#include <arch/cpu.h>
|
2003-05-03 20:20:38 +04:00
|
|
|
#include <boot/stage2.h>
|
2002-07-09 16:24:59 +04:00
|
|
|
#include <atomic.h>
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
/* global per-cpu structure */
|
|
|
|
cpu_ent cpu[MAX_BOOT_CPUS];
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
|
|
|
|
int
|
|
|
|
cpu_init(kernel_args *ka)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
memset(cpu, 0, sizeof(cpu));
|
|
|
|
for(i = 0; i < MAX_BOOT_CPUS; i++) {
|
|
|
|
cpu[i].info.cpu_num = i;
|
|
|
|
}
|
|
|
|
|
|
|
|
return arch_cpu_init(ka);
|
|
|
|
}
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
|
|
|
|
int
|
|
|
|
cpu_preboot_init(kernel_args *ka)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
return arch_cpu_preboot_init(ka);
|
|
|
|
}
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
|
|
|
|
int32
|
|
|
|
user_atomic_add(vint32 *uval, int32 incr)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2002-09-03 06:19:22 +04:00
|
|
|
int32 val;
|
|
|
|
int32 ret;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
if ((addr)uval >= KERNEL_BASE && (addr)uval <= KERNEL_TOP)
|
2002-07-09 16:24:59 +04:00
|
|
|
goto error;
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
if (user_memcpy(&val, (int32 *)uval, sizeof(val)) < 0)
|
2002-07-09 16:24:59 +04:00
|
|
|
goto error;
|
|
|
|
|
2002-09-03 06:19:22 +04:00
|
|
|
// XXX broken on non SH4-systems, or when interrupts are enabled
|
|
|
|
// XXX x86 must use the assembly functions directly in userspace and not this ones
|
2002-07-09 16:24:59 +04:00
|
|
|
ret = val;
|
|
|
|
val += incr;
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
if (user_memcpy((int32 *)uval, &val, sizeof(val)) < 0)
|
2002-07-09 16:24:59 +04:00
|
|
|
goto error;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
error:
|
|
|
|
// XXX kill the app
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
|
|
|
|
int32
|
|
|
|
user_atomic_and(vint32 *uval, int32 incr)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
int val;
|
|
|
|
int ret;
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
if ((addr)uval >= KERNEL_BASE && (addr)uval <= KERNEL_TOP)
|
2002-07-09 16:24:59 +04:00
|
|
|
goto error;
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
if (user_memcpy(&val, (int32 *)uval, sizeof(val)) < 0)
|
2002-07-09 16:24:59 +04:00
|
|
|
goto error;
|
|
|
|
|
2002-09-03 06:19:22 +04:00
|
|
|
// XXX broken on non SH4-systems, or when interrupts are enabled
|
|
|
|
// XXX x86 must use the assembly functions directly in userspace and not this ones
|
2002-07-09 16:24:59 +04:00
|
|
|
ret = val;
|
|
|
|
val &= incr;
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
if (user_memcpy((int32 *)uval, &val, sizeof(val)) < 0)
|
2002-07-09 16:24:59 +04:00
|
|
|
goto error;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
error:
|
|
|
|
// XXX kill the app
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
|
|
|
|
int32
|
|
|
|
user_atomic_or(vint32 *uval, int32 incr)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
int val;
|
|
|
|
int ret;
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
if ((addr)uval >= KERNEL_BASE && (addr)uval <= KERNEL_TOP)
|
2002-07-09 16:24:59 +04:00
|
|
|
goto error;
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
if (user_memcpy(&val, (int32 *)uval, sizeof(val)) < 0)
|
2002-07-09 16:24:59 +04:00
|
|
|
goto error;
|
|
|
|
|
2002-09-03 06:19:22 +04:00
|
|
|
// XXX broken on non SH4-systems, or when interrupts are enabled
|
|
|
|
// XXX x86 must use the assembly functions directly in userspace and not this ones
|
2002-07-09 16:24:59 +04:00
|
|
|
ret = val;
|
|
|
|
val |= incr;
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
if (user_memcpy((int32 *)uval, &val, sizeof(val)) < 0)
|
2002-07-09 16:24:59 +04:00
|
|
|
goto error;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
error:
|
|
|
|
// XXX kill the app
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
|
|
|
|
int32
|
|
|
|
user_atomic_set(vint32 *uval, int32 set_to)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
int val;
|
|
|
|
int ret;
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
if ((addr)uval >= KERNEL_BASE && (addr)uval <= KERNEL_TOP)
|
2002-07-09 16:24:59 +04:00
|
|
|
goto error;
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
if (user_memcpy(&val, (int32 *)uval, sizeof(val)) < 0)
|
2002-07-09 16:24:59 +04:00
|
|
|
goto error;
|
|
|
|
|
2002-09-03 06:19:22 +04:00
|
|
|
// XXX broken on non SH4-systems, or when interrupts are enabled
|
|
|
|
// XXX x86 must use the assembly functions directly in userspace and not this ones
|
2002-07-09 16:24:59 +04:00
|
|
|
ret = val;
|
|
|
|
val = set_to;
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
if (user_memcpy((int32 *)uval, &val, sizeof(val)) < 0)
|
2002-07-09 16:24:59 +04:00
|
|
|
goto error;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
error:
|
|
|
|
// XXX kill the app
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
|
|
|
|
int32
|
|
|
|
user_test_and_set(vint32 *uval, int32 set_to, int32 test_val)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
int val;
|
|
|
|
int ret;
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
if ((addr)uval >= KERNEL_BASE && (addr)uval <= KERNEL_TOP)
|
2002-07-09 16:24:59 +04:00
|
|
|
goto error;
|
|
|
|
|
2003-05-03 20:20:38 +04:00
|
|
|
if (user_memcpy(&val, (int32 *)uval, sizeof(val)) < 0)
|
2002-07-09 16:24:59 +04:00
|
|
|
goto error;
|
|
|
|
|
2002-09-03 06:19:22 +04:00
|
|
|
// XXX broken on non SH4-systems, or when interrupts are enabled
|
|
|
|
// XXX x86 must use the assembly functions directly in userspace and not this ones
|
2002-07-09 16:24:59 +04:00
|
|
|
ret = val;
|
2003-05-03 20:20:38 +04:00
|
|
|
if (val == test_val) {
|
2002-07-09 16:24:59 +04:00
|
|
|
val = set_to;
|
2003-05-03 20:20:38 +04:00
|
|
|
if (user_memcpy((int32 *)uval, &val, sizeof(val)) < 0)
|
2002-07-09 16:24:59 +04:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
error:
|
|
|
|
// XXX kill the app
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|