Add ARM spinlock
This commit is contained in:
11
Makefile
11
Makefile
@@ -1,11 +1,14 @@
|
||||
CC = arm-none-eabi-gcc
|
||||
AS = arm-none-eabi-as
|
||||
LD = arm-none-eabi-ld
|
||||
TOOLCHAIN = arm-none-eabi
|
||||
|
||||
CC = $(TOOLCHAIN)-gcc
|
||||
AS = $(TOOLCHAIN)-as
|
||||
LD = $(TOOLCHAIN)-ld
|
||||
CPU = -mcpu=cortex-m4
|
||||
THUMB = -mthumb
|
||||
|
||||
LDFLAGS = -T linker.ld
|
||||
CFLAGS = $(CPU) $(THUMB) -I libc/ -I ./
|
||||
CC_SYMBOLS = -DTOOLCHAIN_GCC_ARM
|
||||
|
||||
QEMU = qemu-system-arm
|
||||
BOARD = netduinoplus2
|
||||
@@ -13,7 +16,7 @@ BOARD = netduinoplus2
|
||||
# Output file
|
||||
TARGET = main.elf
|
||||
# Source files
|
||||
SRCS = os/main.c os/process.c os/scheduler.c os/alloc.c os/delay.c os/driver/usart.c libc/library.c tasks/tasks.c
|
||||
SRCS = os/main.c os/process.c os/scheduler.c os/alloc.c os/delay.c os/driver/usart.c os/sync/spinlock.c libc/library.c tasks/tasks.c
|
||||
# Object files
|
||||
OBJS = $(SRCS:.c=.o)
|
||||
|
||||
|
||||
22
os/Makefile
22
os/Makefile
@@ -1,22 +0,0 @@
|
||||
CC = arm-none-eabi-gcc
|
||||
AS = arm-none-eabi-as
|
||||
LD = arm-none-eabi-ld
|
||||
CPU = -mcpu=cortex-m4
|
||||
THUMB = -mthumb
|
||||
|
||||
LDFLAGS = -T linker.ld
|
||||
CFLAGS = -c $(CPU) $(THUMB) -g
|
||||
|
||||
all: main.o process.o scheduler.o
|
||||
|
||||
main.o: main.c
|
||||
$(CC) $(CFLAGS) main.c -o main.o
|
||||
|
||||
scheduler.o: scheduler.c
|
||||
$(CC) $(CFLAGS) scheduler.c -o scheduler.o
|
||||
|
||||
process.o: process.c
|
||||
$(CC) $(CFLAGS) process.c -o process.o
|
||||
|
||||
clean:
|
||||
rm -f *.elf *.o *.i
|
||||
34
os/sync/spinlock.c
Normal file
34
os/sync/spinlock.c
Normal file
@@ -0,0 +1,34 @@
|
||||
// sync_primitives.c
|
||||
#include "spinlock.h"
|
||||
|
||||
// Initialize the spinlock
|
||||
void spinlock_init(Spinlock *spinlock) {
|
||||
spinlock->lock = 0; // Unlocked state
|
||||
}
|
||||
|
||||
// Acquire the spinlock
|
||||
void spinlock_acquire(Spinlock *spinlock) {
|
||||
int status;
|
||||
do {
|
||||
// Wait until the lock is available
|
||||
do {
|
||||
__asm__ volatile ("ldrex %0, [%1]"
|
||||
: "=&r" (status)
|
||||
: "r" (&spinlock->lock)
|
||||
: "memory");
|
||||
} while (status != 0);
|
||||
|
||||
// Try to acquire the lock using strex
|
||||
__asm__ volatile ("strex %0, %2, [%1]"
|
||||
: "=&r" (status)
|
||||
: "r" (&spinlock->lock), "r" (1)
|
||||
: "memory");
|
||||
} while (status != 0); // Retry if strex failed
|
||||
}
|
||||
|
||||
// Release the spinlock
|
||||
void spinlock_release(Spinlock *spinlock) {
|
||||
// Use a memory barrier to ensure proper ordering
|
||||
__asm__ volatile ("dmb" ::: "memory");
|
||||
spinlock->lock = 0; // Release the lock
|
||||
}
|
||||
16
os/sync/spinlock.h
Normal file
16
os/sync/spinlock.h
Normal file
@@ -0,0 +1,16 @@
|
||||
#ifndef SPINLOCK_H
|
||||
#define SPINLOCK_H
|
||||
|
||||
#include <stdint.h>
|
||||
|
||||
// Spinlock
|
||||
typedef struct {
|
||||
volatile int lock;
|
||||
} Spinlock;
|
||||
|
||||
void spinlock_init(Spinlock *spinlock);
|
||||
void spinlock_acquire(Spinlock *spinlock);
|
||||
void spinlock_release(Spinlock *spinlock);
|
||||
|
||||
|
||||
#endif
|
||||
Reference in New Issue
Block a user