Skip to content

Commit 90e0906

Browse files
committed
Add a simple scheduler
1 parent 3ef1c0c commit 90e0906

File tree

7 files changed

+113
-7
lines changed

7 files changed

+113
-7
lines changed

Makefile

+8-5
Original file line numberDiff line numberDiff line change
@@ -14,9 +14,12 @@ all: centurion
1414
CFLAGS = -g3 -Wall -pedantic
1515

1616
centurion: centurion.o cpu6.o disassemble.o dsk.o hawk.o math128.o mux.o \
17-
cbin.o cbin_load.o $(SYS_OBJS)
17+
cbin.o cbin_load.o scheduler.o $(SYS_OBJS)
1818

19-
centurion.o: centurion.c centurion.h console.h cpu6.h disassemble.h dma.h dsk.h math128.o mux.h
19+
centurion.o: centurion.c centurion.h console.h cpu6.h disassemble.h dma.h \
20+
dsk.h math128.o mux.h scheduler.h
21+
22+
scheduler.o: scheduler.c scheduler.h cpu6.h
2023

2124
console.o : console.c console.h mux.h
2225

@@ -26,17 +29,17 @@ cpu6.o : cpu6.c cpu6.h
2629

2730
disassemble.o: disassemble.c disassemble.h cpu6.h
2831

29-
dsk.o: dsk.c dsk.h hawk.h dma.h
32+
dsk.o: dsk.c dsk.h hawk.h dma.h scheduler.h cpu6.h
3033

31-
hawk.o: hawk.c hawk.h
34+
hawk.o: hawk.c hawk.h scheduler.h
3235

3336
cbin.o: cbin.h
3437

3538
cbin_load.o: cpu6.h cbin.h
3639

3740
math128.o: math128.h
3841

39-
mux.o : centurion.h mux.h console.h cpu6.h
42+
mux.o : centurion.h mux.h console.h cpu6.h scheduler.h
4043

4144
clean:
4245
rm -f centurion *.o *~

centurion.c

+2
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@
1717
#include "dsk.h"
1818
#include "mux.h"
1919
#include "cbin_load.h"
20+
#include "scheduler.h"
2021

2122
static unsigned finch; /* Finch or original FDC */
2223

@@ -849,6 +850,7 @@ int main(int argc, char *argv[])
849850
/* Update peripherals state */
850851
mux_poll(trace & TRACE_MUX);
851852

853+
run_scheduler(cpu_timestamp_ns);
852854
throttle_emulation(cpu_timestamp_ns);
853855

854856
instruction_count++;

cpu6.h

-2
Original file line numberDiff line numberDiff line change
@@ -26,8 +26,6 @@
2626
#define C 12 /* Flags ? */
2727
#define P 14 /* PC */
2828

29-
#define ONE_SECOND_NS 1000000000.0
30-
3129
extern uint8_t mem_read8(uint32_t addr);
3230
extern uint8_t mem_read8_debug(uint32_t addr);
3331
extern uint16_t mem_read16_debug(uint32_t addr);

dsk.c

+1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@
77
#include "dma.h"
88
#include "dsk.h"
99
#include "hawk.h"
10+
#include "scheduler.h"
1011

1112
#ifndef O_BINARY
1213
#define O_BINARY 0

mux.c

+1
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@
88
#include "console.h"
99
#include "cpu6.h"
1010
#include "mux.h"
11+
#include "scheduler.h"
1112

1213
struct MuxUnit mux[NUM_MUX_UNITS];
1314
// 0 disables interrupts

scheduler.c

+72
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
#include "scheduler.h"
2+
#include "cpu6.h"
3+
4+
#include <stdlib.h>
5+
#include <assert.h>
6+
7+
static struct event_t* event_list = NULL;
8+
static uint64_t next_event = UINT64_MAX;
9+
10+
void schedule_event(struct event_t *event)
11+
{
12+
uint64_t now = get_current_time();
13+
int64_t scheduled = get_current_time() + event->delta_ns;
14+
assert(scheduled > now);
15+
16+
if (event->next != NULL)
17+
cancel_event(event);
18+
19+
event->scheduled_ns = scheduled;
20+
21+
if (!event_list) {
22+
event->next = NULL;
23+
event_list = event;
24+
next_event = scheduled;
25+
return;
26+
}
27+
28+
struct event_t* item = event_list;
29+
30+
while (item && item->scheduled_ns < scheduled) {
31+
item = item->next;
32+
}
33+
event->next = item->next;
34+
item->next = event;
35+
}
36+
37+
void run_scheduler(uint64_t current_time)
38+
{
39+
if (next_event > current_time)
40+
return;
41+
42+
assert(event_list);
43+
44+
while (next_event <= current_time) {
45+
// Pop event
46+
struct event_t* event = event_list;
47+
event_list = event_list->next;
48+
event->next = NULL;
49+
50+
// Update next_event
51+
next_event = (event_list == NULL) ? UINT64_MAX : event_list->scheduled_ns;
52+
53+
// Run callback
54+
event->callback(event, current_time - event->scheduled_ns);
55+
}
56+
}
57+
58+
59+
void cancel_event(struct event_t *event)
60+
{
61+
struct event_t** next_ptr = &event_list;
62+
63+
while(*next_ptr != NULL) {
64+
struct event_t* next = *next_ptr;
65+
if (next == event) {
66+
*next_ptr = next->next;
67+
event->next = NULL;
68+
return;
69+
}
70+
next_ptr = &next->next;
71+
}
72+
}

scheduler.h

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
2+
#pragma once
3+
4+
#include <stdint.h>
5+
6+
#define ONE_SECOND_NS 1000000000.0
7+
#define ONE_MILISECOND_NS 1000000.0
8+
9+
10+
struct event_t;
11+
12+
// event callbacks get called with their event and how many ns have passed
13+
// since their scheduled event time.
14+
// If event was dynamically allocated, the callback should free it.
15+
typedef void (*callback_t)(struct event_t *event, int64_t late_ns);
16+
17+
struct event_t {
18+
int64_t delta_ns;
19+
callback_t callback;
20+
21+
// internal state
22+
struct event_t *next;
23+
int64_t scheduled_ns;
24+
};
25+
26+
void schedule_event(struct event_t *event);
27+
void cancel_event(struct event_t *event);
28+
void run_scheduler(uint64_t current_time);
29+
uint64_t get_current_time();

0 commit comments

Comments
 (0)