Skip to content

Commit b22853b

Browse files
Add a SMP example
This just prints "hello world" on two cores. It contains an example of how to initialize a multi-core system using IPIs, and a simple spin lock.
1 parent c08ce47 commit b22853b

File tree

5 files changed

+126
-1
lines changed

5 files changed

+126
-1
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -13,3 +13,4 @@ software/hello/hello
1313
software/led_fade/led_fade
1414
software/local_interrupts/local_interrupts
1515
software/performance_counters/performance_counters
16+
software/smp/smp

bsp/env/start.S

+50-1
Original file line numberDiff line numberDiff line change
@@ -1,18 +1,27 @@
11
// See LICENSE for license details.
2+
#include <sifive/smp.h>
23

3-
// See LICENSE for license details.
4+
/* This is defined in sifive/platform.h, but that can't be included from
5+
* assembly. */
6+
#define CLINT_CTRL_ADDR 0x02000000
47

58
.section .init
69
.globl _start
710
.type _start,@function
811

912
_start:
13+
.cfi_startproc
14+
.cfi_undefined ra
1015
.option push
1116
.option norelax
1217
la gp, __global_pointer$
1318
.option pop
1419
la sp, _sp
1520

21+
#if defined(ENABLE_SMP)
22+
smp_pause(t0, t1)
23+
#endif
24+
1625
/* Load data section */
1726
la a0, _data_lma
1827
la a1, _data
@@ -52,11 +61,51 @@ _start:
5261
1:
5362
#endif
5463

64+
#if defined(ENABLE_SMP)
65+
smp_resume(t0, t1)
66+
67+
csrr a0, mhartid
68+
bnez a0, 2f
69+
#endif
70+
71+
auipc ra, 0
72+
addi sp, sp, -16
73+
#if __riscv_xlen == 32
74+
sw ra, 8(sp)
75+
#else
76+
sd ra, 8(sp)
77+
#endif
78+
5579
/* argc = argv = 0 */
5680
li a0, 0
5781
li a1, 0
5882
call main
5983
tail exit
84+
1:
85+
j 1b
86+
87+
#if defined(ENABLE_SMP)
88+
2:
89+
la t0, trap_entry
90+
csrw mtvec, t0
91+
92+
csrr a0, mhartid
93+
la t1, _sp
94+
slli t0, a0, 10
95+
sub sp, t1, t0
96+
97+
auipc ra, 0
98+
addi sp, sp, -16
99+
#if __riscv_xlen == 32
100+
sw ra, 8(sp)
101+
#else
102+
sd ra, 8(sp)
103+
#endif
104+
105+
call secondary_main
106+
tail exit
60107

61108
1:
62109
j 1b
110+
#endif
111+
.cfi_endproc

software/smp/Makefile

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
TARGET = smp
2+
C_SRCS += smp.c
3+
CFLAGS += -O2 -fno-builtin-printf -DENABLE_SMP
4+
5+
BSP_BASE = ../../bsp
6+
include $(BSP_BASE)/env/common.mk

software/smp/atomic.h

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
#ifndef SIFIVE_ATOMIC_H
2+
#define SIFIVE_ATOMIC_H
3+
4+
#define ATOMIC_INIT(x) \
5+
{ \
6+
.counter = (x), \
7+
}
8+
9+
typedef struct {
10+
int counter;
11+
} atomic_t;
12+
13+
static inline int atomic_xchg(atomic_t *v, int n)
14+
{
15+
register int c;
16+
17+
__asm__ __volatile__ (
18+
"amoswap.w.aqrl %0, %2, %1"
19+
: "=r" (c), "+A" (v->counter)
20+
: "r" (n));
21+
return c;
22+
}
23+
24+
static inline void mb(void)
25+
{
26+
__asm__ __volatile__ ("fence");
27+
}
28+
29+
#endif

software/smp/smp.c

+40
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
#include <stdio.h>
2+
#include <unistd.h>
3+
#include "atomic.h"
4+
void write_hex(int, unsigned long);
5+
6+
atomic_t tty_lock = ATOMIC_INIT(0);
7+
8+
void get_lock(atomic_t *lock)
9+
{
10+
while (atomic_xchg(lock, 1) == 1);
11+
mb();
12+
}
13+
14+
void put_lock(atomic_t *lock)
15+
{
16+
mb();
17+
atomic_xchg(lock, 0);
18+
}
19+
20+
int secondary_main(int hartid)
21+
{
22+
volatile int counter;
23+
24+
while (1) {
25+
get_lock(&tty_lock);
26+
write(1, "hello world from hart ", 22);
27+
char s[] = {'0', '\n', '\0'};
28+
s[0] += hartid;
29+
write(1, s, 2);
30+
put_lock(&tty_lock);
31+
32+
for (counter = 0; counter < 10000 + 100*hartid; ++counter)
33+
mb();
34+
}
35+
}
36+
37+
int main()
38+
{
39+
return secondary_main(0);
40+
}

0 commit comments

Comments
 (0)