forked from tobyp/kthw
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmemory.c
141 lines (117 loc) · 4.85 KB
/
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
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
/* KTHW - Hardware Clone of Keep Talking and Nobody Explodes
Copyright (C) 2017 Toby P., Thomas H.
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, version 3.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>. */
#include "memory.h"
/* Memory has one display and 4 buttons that all show a number from 1 to 4.
Depending on which of the 5 stages the module is in, there are different rules to decide
which button should be pressed depending on the display number. In later stages,
the rules may require knowledge of what was pressed in earlier stages (hence the name).
Get one button wrong and you have to start over. */
enum {
MEM_POS, // the button in the Nth position
MEM_LBL, // the button with the label N
MEM_SAME_POS, // the button in the same position as in stage N
MEM_SAME_LBL, // the button with the same label as in stage N
};
static struct stage_rule {
struct stage_num_rule {
uint8_t rule; uint8_t n;
} display_values[4];
} rules[] = {
{{{MEM_POS, 1}, {MEM_POS, 1}, {MEM_POS, 2}, {MEM_POS, 3},}},
{{{MEM_LBL, 3}, {MEM_SAME_POS, 0}, {MEM_POS, 0}, {MEM_SAME_POS, 0},}},
{{{MEM_SAME_LBL, 1}, {MEM_SAME_LBL, 0}, {MEM_POS, 2}, {MEM_LBL, 3},}},
{{{MEM_SAME_POS, 0}, {MEM_POS, 0}, {MEM_SAME_POS, 1}, {MEM_SAME_POS, 1},}},
{{{MEM_SAME_LBL, 0}, {MEM_SAME_LBL, 1}, {MEM_SAME_LBL, 3}, {MEM_SAME_LBL, 2},}},
};
static inline uint8_t expected_lbl(struct memory * memory) {
struct stage_num_rule * rule = &rules[memory->stage].display_values[memory->display];
switch (rule->rule) {
case MEM_POS: return memory->buttons[rule->n];
case MEM_LBL: return rule->n;
case MEM_SAME_POS: return memory->buttons[memory->pos_hist[rule->n]];
case MEM_SAME_LBL:
default: return memory->lbl_hist[rule->n];
}
}
static inline void populate(struct memory * memory) {
memory->display = rnd_range(0, 4);
memory->buttons[0] = 0;
memory->buttons[1] = 1;
memory->buttons[2] = 2;
memory->buttons[3] = 3;
/* Unrolled Fisher-Yates Shuffle */
uint8_t j = rnd_range(0, 4);
uint8_t temp = memory->buttons[j];
memory->buttons[j] = memory->buttons[3];
memory->buttons[3] = temp;
j = rnd_range(0, 3);
temp = memory->buttons[j];
memory->buttons[j] = memory->buttons[2];
memory->buttons[2] = temp;
j = rnd_range(0, 2);
temp = memory->buttons[j];
memory->buttons[j] = memory->buttons[1];
memory->buttons[1] = temp;
printf("[%s] stage=%d display=%d buttons=[%d,%d,%d,%d] expect=lbl%d\n", memory->module.name, memory->stage, memory->display + 1, memory->buttons[0] + 1, memory->buttons[1] + 1, memory->buttons[2] + 1, memory->buttons[3] + 1, expected_lbl(memory) + 1);
memory->sr_display->value = sevenseg_digits[memory->display + 1];
memory->sr_btn_lbls[0]->value = sevenseg_digits[memory->buttons[0] + 1];
memory->sr_btn_lbls[1]->value = sevenseg_digits[memory->buttons[1] + 1];
memory->sr_btn_lbls[2]->value = sevenseg_digits[memory->buttons[2] + 1];
memory->sr_btn_lbls[3]->value = sevenseg_digits[memory->buttons[3] + 1];
}
void memory_prepare_tick(struct bomb * bomb, struct module * module) {
struct memory * memory = (struct memory *)module;
memory->ticks = 0;
memory->stage = 0;
populate(memory);
module->flags |= MF_READY;
}
void memory_tick(struct bomb * bomb, struct module * module) {
struct memory * memory = (struct memory *)module;
uint32_t poll_index = memory->ticks;
uint8_t poll_mask = 1 << poll_index;
uint8_t poll_value = (gpio_get(memory->in_btn)) ? poll_mask : 0;
if ((memory->btn_cache & poll_mask) ^ poll_value) {
memory->btn_cache = (memory->btn_cache & ~poll_mask) | poll_value;
if (poll_value) { //button is now pushed
if (memory->buttons[poll_index] == expected_lbl(memory)) {
memory->lbl_hist[memory->stage] = memory->buttons[poll_index];
memory->pos_hist[memory->stage] = poll_index;
memory->stage++;
if (memory->stage == 5) {
module->flags |= MF_COMPLETE;
return;
}
}
else {
strike(bomb, module);
memory->stage = 0;
}
populate(memory);
}
}
memory->ticks = (memory->ticks + 1) % 4;
poll_index = memory->ticks;
poll_mask = 1 << poll_index;
memory->sr_btn->value = poll_mask;
memory->sr_stage->value = (1 << (memory->stage + 1)) - 1;
}
void memory_reset(struct bomb * bomb, struct module * module) {
struct memory * memory = (struct memory *)module;
memory->sr_display->value = 0;
memory->sr_stage->value = 0;
memory->sr_btn->value = 0;
memory->sr_btn_lbls[0]->value = 0;
memory->sr_btn_lbls[1]->value = 0;
memory->sr_btn_lbls[2]->value = 0;
memory->sr_btn_lbls[3]->value = 0;
}