-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathprogram_memory.c
99 lines (88 loc) · 4.36 KB
/
program_memory.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
/********************************************************************************
* program_memory.c: Contains function definitions and macro definitions for
* implementation of a 0.75 kB program memory, capable of
* storing up to 256 24-bit instructions. Since C doesn't
* support unsigned 24-bit integers (without using structs or
* unions), the program memory is set to 32 bits data width,
* but only 24 bits are used.
********************************************************************************/
#include "program_memory.h"
/* Macro definitions: */
#define main 4 /* Start address for subroutine main. */
#define main_loop 5 /* Start address for loop in subroutine main. */
#define setup 6 /* Start address for subroutine setup. */
#define ISR_PCINT 15 /* Start address for subroutine ISR_PCINT. */
#define ISR_PCINT_end 19 /* Start address for subroutine ISR_PCINT_end. */
#define LED1 PORTA8 /* LED 1 connected to pin 8 (PORTB0). */
#define LED2 PORTA9
#define LED3 PORTA10
#define BUTTON1 PORTA11
#define BUTTON2 PORTA12
#define BUTTON3 PORTA13 /* Button 1 connected to pin 13 (PORTB5). */
/* Static functions: */
static inline uint64_t assemble(const uint16_t op_code,
const uint16_t op1,
const uint32_t op2);
/* Static variables: */
static uint64_t program_memory[PROGRAM_MEMORY_ADDRESS_WIDTH]; /* 0.75 kB program memory. */
/********************************************************************************
* program_memory_write: Writes machine code to the program memory. This function
* should be called once when the program starts.
********************************************************************************/
void program_memory_write(void)
{
static bool program_memory_initialized = false;
if (program_memory_initialized) return;
program_memory[0] = assemble(JMP, main, 0x00);
program_memory[1] = assemble(NOP, 0x00, 0x00);
program_memory[2] = assemble(JMP, ISR_PCINT, 0x00);
program_memory[3] = assemble(NOP, 0x00, 0x00);
program_memory[4] = assemble(CALL, setup, 0x00);
program_memory[5] = assemble(JMP, main_loop, 0x00);
program_memory[6] = assemble(LDI, R16, (1 << LED1));
program_memory[7] = assemble(OUT, DDRA, R16);
program_memory[8] = assemble(LDI, R17, (1 << BUTTON1));
program_memory[9] = assemble(OUT, PORTA, R17);
program_memory[10] = assemble(SEI, 0x00, 0x00);
program_memory[11] = assemble(LDI, R24, (1 << PCIEA));
program_memory[12] = assemble(OUT, ICR, R24);
program_memory[13] = assemble(OUT, PCMSKA, R17);
program_memory[14] = assemble(RET, 0x00, 0x00);
program_memory[15] = assemble(IN, R24, PINA);
program_memory[16] = assemble(ANDI, R24, (1 << BUTTON1));
program_memory[17] = assemble(BREQ, ISR_PCINT_end, 0x00);
program_memory[18] = assemble(OUT, PINA, R16);
program_memory[19] = assemble(RETI, 0x00, 0x00);
program_memory_initialized = true;
return;
}
/********************************************************************************
* program_memory_read: Returns the instruction at specified address. If an
* invalid address is specified (should be impossible as
* long as the program memory address width isn't increased)
* no operation (0x00) is returned.
*
* - address: Address to instruction in program memory.
********************************************************************************/
uint64_t program_memory_read(const uint16_t address)
{
if (address < PROGRAM_MEMORY_ADDRESS_WIDTH)
{
return program_memory[address];
}
else
{
return 0x00;
}
}
/********************************************************************************
* assemble: Returns instruction assembled to machine code.
********************************************************************************/
static inline uint64_t assemble(const uint16_t op_code,
const uint16_t op1,
const uint32_t op2)
{
return ((uint64_t)op_code << 48) | ((uint64_t)op1 << 32) | op2;
//const uint32_t instruction = ((uint64_t)(op_code) << 48) | ((uint64_t)(op1) << 32) | op2;
//return instruction;
}