/* 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. */ #include #include #include #include #include #include #include /* global per-cpu structure */ cpu_ent cpu[MAX_BOOT_CPUS]; int cpu_init(kernel_args *ka) { 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); } int cpu_preboot_init(kernel_args *ka) { return arch_cpu_preboot_init(ka); } int user_atomic_add(int *uval, int incr) { int val; int ret; if((addr)uval >= KERNEL_BASE && (addr)uval <= KERNEL_TOP) goto error; if(user_memcpy(&val, uval, sizeof(val)) < 0) goto error; ret = val; val += incr; if(user_memcpy(uval, &val, sizeof(val)) < 0) goto error; return ret; error: // XXX kill the app return -1; } int user_atomic_and(int *uval, int incr) { int val; int ret; if((addr)uval >= KERNEL_BASE && (addr)uval <= KERNEL_TOP) goto error; if(user_memcpy(&val, uval, sizeof(val)) < 0) goto error; ret = val; val &= incr; if(user_memcpy(uval, &val, sizeof(val)) < 0) goto error; return ret; error: // XXX kill the app return -1; } int user_atomic_or(int *uval, int incr) { int val; int ret; if((addr)uval >= KERNEL_BASE && (addr)uval <= KERNEL_TOP) goto error; if(user_memcpy(&val, uval, sizeof(val)) < 0) goto error; ret = val; val |= incr; if(user_memcpy(uval, &val, sizeof(val)) < 0) goto error; return ret; error: // XXX kill the app return -1; } int user_atomic_set(int *uval, int set_to) { int val; int ret; if((addr)uval >= KERNEL_BASE && (addr)uval <= KERNEL_TOP) goto error; if(user_memcpy(&val, uval, sizeof(val)) < 0) goto error; ret = val; val = set_to; if(user_memcpy(uval, &val, sizeof(val)) < 0) goto error; return ret; error: // XXX kill the app return -1; } int user_test_and_set(int *uval, int set_to, int test_val) { int val; int ret; if((addr)uval >= KERNEL_BASE && (addr)uval <= KERNEL_TOP) goto error; if(user_memcpy(&val, uval, sizeof(val)) < 0) goto error; ret = val; if(val == test_val) { val = set_to; if(user_memcpy(uval, &val, sizeof(val)) < 0) goto error; } return ret; error: // XXX kill the app return -1; }