-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcontrol.luc
316 lines (278 loc) · 13.5 KB
/
control.luc
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
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
module control (
input clk, // clock
input rst, // reset
output ra_address[3],
output rb_address[3],
output write_address[3],
output we,
output alufn[6],
output bsel[2],
output asel[2],
output wdsel[2],
output random_value[3], // 100: BLUE ON, 010 RED ON, 001 GREEN ON. Only one light will be ON at a time
output current_button_press[16], // last 3 bits signifies current player's button press, e.g: 010 (middle button RED is pressed, the other two isn't)
input rb_data[16],
input p1_buttons_press[3],
input p2_buttons_press[3],
input start_button_press,
input p1_buttons_press_only_one_button,
input p2_buttons_press_only_one_button
) {
pn_gen random_generator(.clk(clk), .rst(rst));
dff ctr[32](.clk(clk), .rst(rst));
.clk(clk){
.rst(rst){
fsm game_state={
START,
INIT_SEQ,
IDLE,
CHECK_P1BUTTONINPUT_PART1,
CHECK_P1BUTTONINPUT_PART2,
CHECK_P2BUTTONINPUT_PART1,
CHECK_P2BUTTONINPUT_PART2,
INCREASE_P1SCORE,
INCREASE_P2SCORE,
RESET_P1,
RESET_P2,
ENDGAME_1, // p1 wins state
ENDGAME_2, // p2 wins state
CHECK_GAME_OVER_P1_PART1,
CHECK_GAME_OVER_P1_PART2,
CHECK_GAME_OVER_P2_PART1,
CHECK_GAME_OVER_P2_PART2,
RESET_EVERYTHING_PART1, // set 0 to R2, led reg
RESET_EVERYTHING_PART2, // set 0 to p1 score reg, R0
RESET_EVERYTHING_PART3 // set 0 to p2 score reg, R1
};
}
}
always {
ctr.d = ctr.q + 1;
random_generator.next = 1; // continuously generate random numbers
random_generator.seed = ctr.q; // seed is always changed based on counter clock
ra_address = 0;
rb_address = d2; // always receiving the led value
write_address = 0;
we = 0;
alufn = 0;
bsel = 0;
asel = 0;
wdsel = 0;
random_value = 0;
current_button_press = 0;
case (game_state.q){
game_state.START:
if (start_button_press){
game_state.d = game_state.RESET_EVERYTHING_PART1;
random_generator.seed = ctr.q; // use different seed so we have different sequence each time
}
game_state.INIT_SEQ:
we = 1;
write_address = 3d2; // write to R2, LED REG
asel = b11;
if(random_generator.num[0]){
random_value = b001; // GREEN
}
else if(random_generator.num[1]){
random_value = b010; // RED
}
else if(random_generator.num[2]){
random_value = b100; // BLUE
}
else{
random_value = b100; // BLUE for everything else
}
alufn = b011010; // A
wdsel = b00; // pass ALU output
game_state.d = game_state.IDLE;
game_state.IDLE:
// check for button presses here
if(p1_buttons_press){
if(p1_buttons_press_only_one_button){
// store p1 press into temp reg R3
we=1;
wdsel = b10; // pass current_button_press output to write_data regfile
write_address = b011; // write to temp reg R3
current_button_press = p1_buttons_press;
game_state.d = game_state.CHECK_P1BUTTONINPUT_PART1;
}
else{
game_state.d = game_state.RESET_P1;
}
}
else if(p2_buttons_press){
if(p2_buttons_press_only_one_button){
// store p2 press into temp reg R3
we=1;
wdsel = b10;
write_address = b011; // write to temp reg R3
current_button_press = p2_buttons_press;
game_state.d = game_state.CHECK_P2BUTTONINPUT_PART1;
}
else{
game_state.d = game_state.RESET_P2;
}
}
else if(start_button_press){
game_state.d = game_state.RESET_EVERYTHING_PART1;
}
else{
game_state.d = game_state.IDLE;
}
game_state.CHECK_P1BUTTONINPUT_PART1:
alufn = b110011; // CMPEQ
ra_address = b10; // Check R2 led == R3 (current player input)
rb_address = b11; // R3
asel = 0;
bsel = 0;
wdsel = 0;
write_address = b100; // store CMPEQ result to R4
we = 1;
game_state.d = game_state.CHECK_P1BUTTONINPUT_PART2;
game_state.CHECK_P1BUTTONINPUT_PART2:
rb_address = b100; // check the CMPEQ result
if (rb_data[0]){
game_state.d = game_state.INCREASE_P1SCORE;
}
else{
game_state.d = game_state.RESET_P1;
}
game_state.INCREASE_P1SCORE:
alufn = b000000; //ADD
asel = b10; // constant 1
bsel = b00;
rb_address = b000; // select R0
wdsel = b00;
write_address = b000;
we = 1;
game_state.d = game_state.CHECK_GAME_OVER_P1_PART1;
game_state.CHECK_GAME_OVER_P1_PART1:
alufn = b110011; // CMPEQ
bsel = b11; // 99
asel = b00;
ra_address = b000; // p1 score
wdsel = b00;
write_address = b101; // temp reg R5
we=1;
game_state.d = game_state.CHECK_GAME_OVER_P1_PART2;
game_state.CHECK_GAME_OVER_P1_PART2:
rb_address = b101; // inspect R5
if (rb_data[0]){
// means p0 score is 99
game_state.d = game_state.ENDGAME_1; // p1 wins
}
else{
game_state.d = game_state.INIT_SEQ;
}
game_state.ENDGAME_1:
// p1 wins
// light up 2 bulbs
current_button_press = b011;
wdsel = b10;
// pass current_button_press to wdsel
write_address = b10; // write 011 to R2, the led reg, so this lights up 2 bulbs, which differentiates from game time
we = 1;
game_state.d = game_state.START;
game_state.RESET_P1:
we=1;
wdsel=b11;
write_address=b000; // write 0 to R0
game_state.d = game_state.IDLE; // go back to idle to wait for correct input
game_state.RESET_EVERYTHING_PART1:
// reset the LED reg
we=1;
wdsel=b11;
write_address=b010; // write 0 to R2, the led reg
game_state.d = game_state.RESET_EVERYTHING_PART2;
game_state.RESET_EVERYTHING_PART2:
// reset the LED reg
we=1;
wdsel=b11;
write_address=b000; // write 0 to R0, the p0 score
game_state.d = game_state.RESET_EVERYTHING_PART3;
game_state.RESET_EVERYTHING_PART3:
// reset the LED reg
we=1;
wdsel=b11;
write_address=b001; // write 0 to R1, the p1 score
game_state.d = game_state.INIT_SEQ;
//Player 2 here
game_state.CHECK_P2BUTTONINPUT_PART1:
alufn = b110011; // CMPEQ
ra_address = b10; // Check R2 led == R3 (current player input)
rb_address = b11; // R3
asel = 0;
bsel = 0;
wdsel = 0;
write_address = b100; // store CMPEQ result to R4
we = 1;
game_state.d = game_state.CHECK_P2BUTTONINPUT_PART2;
game_state.CHECK_P2BUTTONINPUT_PART2:
rb_address = b100; // check the CMPEQ result
if (rb_data[0]){
game_state.d = game_state.INCREASE_P2SCORE;
}
else{
game_state.d = game_state.RESET_P2;
}
game_state.INCREASE_P2SCORE:
alufn = b000000; //ADD
asel = b10; // constant 1
bsel = b00;
rb_address = b001; // select R1: P2 score
wdsel = b00;
write_address = b001;
we = 1;
game_state.d = game_state.CHECK_GAME_OVER_P2_PART1;
game_state.CHECK_GAME_OVER_P2_PART1:
alufn = b110011; // CMPEQ
bsel = b11; // 99
asel = b00;
ra_address = b000; // p1 score
wdsel = b00;
write_address = b101; // temp reg R5
we=1;
game_state.d = game_state.CHECK_GAME_OVER_P2_PART2;
game_state.CHECK_GAME_OVER_P2_PART2:
rb_address = b101; // inspect R5
if (rb_data[0]){
// means p2 score is 99
game_state.d = game_state.ENDGAME_2; // p2 wins
}
else{
game_state.d = game_state.INIT_SEQ;
}
game_state.ENDGAME_2:
// p1 wins
// light up 3 bulbs
current_button_press = b111;
wdsel = b10;
write_address = b10; // write 011 to R2, the led reg, so this lights up 2 bulbs, which differentiates from game time
we = 1;
game_state.d = game_state.START;
game_state.RESET_P2:
we=1;
wdsel=b11;
write_address=b001; // write 0 to R1, P2's score
game_state.d = game_state.IDLE; // go back to idle to wait for correct input
game_state.RESET_EVERYTHING_PART1:
// reset the LED reg
we=1;
wdsel=b11;
write_address=b010; // write 0 to R2, the led reg
game_state.d = game_state.RESET_EVERYTHING_PART2;
game_state.RESET_EVERYTHING_PART2:
// reset the LED reg
we=1;
wdsel=b11;
write_address=b000; // write 0 to R0, the p0 score
game_state.d = game_state.RESET_EVERYTHING_PART3;
game_state.RESET_EVERYTHING_PART3:
// reset the LED reg
we=1;
wdsel=b11;
write_address=b001; // write 0 to R1, the p1 score
game_state.d = game_state.INIT_SEQ;
}
}
}