bla
This commit is contained in:
parent
703e365da4
commit
f03c8fc9ae
2 changed files with 861 additions and 0 deletions
212
firmware/applications/awake.c
Normal file
212
firmware/applications/awake.c
Normal file
|
@ -0,0 +1,212 @@
|
||||||
|
#include <sysinit.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include "basic/basic.h"
|
||||||
|
#include "funk/nrf24l01p.h"
|
||||||
|
//#include "usetable.h"
|
||||||
|
|
||||||
|
|
||||||
|
#define AWAKE_PACKET_RETRIES 20
|
||||||
|
#define AWAKE_MIN_WINNERS 3
|
||||||
|
#define AWAKE_FONT_HEIGHT 8
|
||||||
|
|
||||||
|
|
||||||
|
typedef enum packet_type_e {
|
||||||
|
PCKT_WINNER0,
|
||||||
|
PCKT_WINNER1,
|
||||||
|
PCKT_WINNER2,
|
||||||
|
PCKT_WINNER3,
|
||||||
|
PCKT_WINNER4,
|
||||||
|
PCKT_WINNER5,
|
||||||
|
PCKT_WINNER6,
|
||||||
|
PCKT_WINNER7,
|
||||||
|
PCKT_WAKEUP,
|
||||||
|
PCKT_STANDBY,
|
||||||
|
PCKT_NONE
|
||||||
|
} packet_type_t;
|
||||||
|
|
||||||
|
|
||||||
|
static char const *const awake_gPackets[] = {
|
||||||
|
"winner0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||||
|
"winner1\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||||
|
"winner2\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||||
|
"winner3\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||||
|
"winner4\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||||
|
"winner5\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||||
|
"winner6\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||||
|
"winner7\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||||
|
"wake up\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||||
|
"standby\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0"
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initializes wireless stuff.
|
||||||
|
*/
|
||||||
|
static void awake_initNrf(void) {
|
||||||
|
nrf_init();
|
||||||
|
static struct NRF_CFG config = {
|
||||||
|
.channel = 81,
|
||||||
|
.txmac = "\x1\x2\x3\x2\x1",
|
||||||
|
.nrmacs = 1,
|
||||||
|
.mac0 = "\x1\x2\x3\x2\x1",
|
||||||
|
.maclen = "\x20"
|
||||||
|
};
|
||||||
|
nrf_config_set(&config);
|
||||||
|
nrf_set_strength(3);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Polls for a special packet which indicates the success of a player.
|
||||||
|
*/
|
||||||
|
static packet_type_t awake_waitForPacket(void) {
|
||||||
|
uint8_t buffer[32];
|
||||||
|
if (nrf_rcv_pkt_time(100, 32, buffer) == 32) {
|
||||||
|
for (packet_type_t p = PCKT_WINNER0; p < PCKT_NONE; ++p) {
|
||||||
|
unsigned int bEqual = 1;
|
||||||
|
for (unsigned int i = 0; i < 8; ++i) {
|
||||||
|
if (buffer[i] != awake_gPackets[p][i]) {
|
||||||
|
bEqual = 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (bEqual) {
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return PCKT_NONE;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Waits for either pushed buttons or a signal on the hacker bus.
|
||||||
|
*/
|
||||||
|
static void awake_waitForKeyPressOrHackerbus(void) {
|
||||||
|
gpioSetDir(RB_HB0, gpioDirection_Input);
|
||||||
|
// watch out for pushed buttons and/or hacker bus activity
|
||||||
|
while (gpioGetValue(RB_HB0) != 0)
|
||||||
|
{
|
||||||
|
if (getInput() != BTN_NONE)
|
||||||
|
{
|
||||||
|
getInputWaitRelease();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
while (gpioGetValue(RB_HB0) == 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The greeter!
|
||||||
|
*/
|
||||||
|
static void awake_promptUserBegin(void) {
|
||||||
|
gpioSetDir(RB_HB0, gpioDirection_Input);
|
||||||
|
|
||||||
|
lcdFill(0);
|
||||||
|
DoString(0, 0, "Eine Taste");
|
||||||
|
DoString(0, 8, "druecken um");
|
||||||
|
DoString(0,16, "das Spiel");
|
||||||
|
DoString(0,24, "zu starten!");
|
||||||
|
lcdDisplay();
|
||||||
|
|
||||||
|
awake_waitForKeyPressOrHackerbus();
|
||||||
|
|
||||||
|
lcdFill(0);
|
||||||
|
DoString(0, 0, "Spiel laeuft!");
|
||||||
|
DoString(0, 16, "FEUERTASTE");
|
||||||
|
DoString(0, 24, "fuer Standby");
|
||||||
|
lcdDisplay();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Informs about the happy end!
|
||||||
|
*/
|
||||||
|
static void awake_promptUserEnd(void){
|
||||||
|
lcdFill(0);
|
||||||
|
DoString(0, 0, "Es wurden ge-");
|
||||||
|
DoString(0, 8, "nug Gewinner ");
|
||||||
|
DoString(0, 16, "ermittelt! ");
|
||||||
|
lcdDisplay();
|
||||||
|
|
||||||
|
// toggle RB_HB1 pin from 0V to 3,3 for 200ms
|
||||||
|
GPIO_GPIO0DATA |= (1u << 10);
|
||||||
|
delayms_queue(200);
|
||||||
|
GPIO_GPIO0DATA &= ~(1u << 10);
|
||||||
|
|
||||||
|
awake_waitForKeyPressOrHackerbus();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Explains a brief moment of unresponsiveness.
|
||||||
|
*/
|
||||||
|
static void awake_promptStandby(void){
|
||||||
|
lcdFill(0);
|
||||||
|
DoString(0, 0, "Bitte warten!");
|
||||||
|
DoString(0, 8, "Sende Standby");
|
||||||
|
DoString(0, 16, "Pakete... ");
|
||||||
|
lcdDisplay();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void awake_initHackerBus() {
|
||||||
|
// set RB_HB1 to GPIO mode (output)
|
||||||
|
#define IOCON_PIO0_10 (*((REG32*) (0x40044068)))
|
||||||
|
IOCON_PIO0_10 &= ~(00000007);
|
||||||
|
IOCON_PIO0_10 |= 0x00000001;
|
||||||
|
GPIO_GPIO0DIR |= (1 << 10);
|
||||||
|
GPIO_GPIO0DATA &= ~(1 << 10);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Main function of the l0dable.
|
||||||
|
*/
|
||||||
|
void main_awake(void) {
|
||||||
|
awake_initNrf();
|
||||||
|
awake_initHackerBus();
|
||||||
|
|
||||||
|
while (1) {
|
||||||
|
awake_promptUserBegin();
|
||||||
|
uint8_t joyinput = BTN_NONE;
|
||||||
|
unsigned int nWinnerFlags = 0, nWinnerCount = 0;
|
||||||
|
while ((joyinput != BTN_ENTER) && (nWinnerCount < AWAKE_MIN_WINNERS)) {
|
||||||
|
if ((joyinput = getInput()) != BTN_NONE)
|
||||||
|
{
|
||||||
|
getInputWaitRelease();
|
||||||
|
}
|
||||||
|
|
||||||
|
// send a "wake up" packet every loop cycle so that in case a r0ket
|
||||||
|
// is rebooted, the player can continue the game
|
||||||
|
uint8_t packet[32];
|
||||||
|
memcpy(packet, awake_gPackets[PCKT_WAKEUP], 32);
|
||||||
|
delayms(10);
|
||||||
|
nrf_snd_pkt_crc(32, packet);
|
||||||
|
|
||||||
|
// watch out for winners!
|
||||||
|
packet_type_t const ePacket = awake_waitForPacket();
|
||||||
|
unsigned int const nWinnerMask = (1 << ePacket) & 0xFF;
|
||||||
|
if ((ePacket <= PCKT_WINNER7) && !(nWinnerFlags & nWinnerMask)) {
|
||||||
|
nWinnerFlags |= nWinnerMask;
|
||||||
|
++nWinnerCount;
|
||||||
|
DoIntX(0, 32, nWinnerMask);
|
||||||
|
lcdDisplay();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (nWinnerCount >= AWAKE_MIN_WINNERS) {
|
||||||
|
awake_promptUserEnd();
|
||||||
|
}
|
||||||
|
|
||||||
|
awake_promptStandby();
|
||||||
|
uint8_t packet[32];
|
||||||
|
for (int i = 0; i < AWAKE_PACKET_RETRIES; ++i) {
|
||||||
|
delayms_queue(50);
|
||||||
|
memcpy(packet, awake_gPackets[PCKT_STANDBY], 32);
|
||||||
|
nrf_snd_pkt_crc(32, packet);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
649
firmware/applications/poem.c
Normal file
649
firmware/applications/poem.c
Normal file
|
@ -0,0 +1,649 @@
|
||||||
|
#include <sysinit.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include "basic/basic.h"
|
||||||
|
#include "basic/random.h"
|
||||||
|
#include "basic/config.h"
|
||||||
|
#include "lcd/render.h"
|
||||||
|
#include "lcd/backlight.h"
|
||||||
|
#include "lcd/allfonts.h"
|
||||||
|
#include "funk/nrf24l01p.h"
|
||||||
|
|
||||||
|
|
||||||
|
#define POEM_ID 4
|
||||||
|
|
||||||
|
#define BTN_WAKEUP (1 << 5)
|
||||||
|
#define BTN_STANDBY (1 << 6)
|
||||||
|
|
||||||
|
#define POEM_FONT_WIDTH 5
|
||||||
|
#define POEM_FONT_HEIGHT 8
|
||||||
|
#define POEM_FONT Font_5x8
|
||||||
|
|
||||||
|
#define POEM_COUNT 5u
|
||||||
|
#define POEM_VERSE_COUNT 4u
|
||||||
|
#define POEM_MAX_LINE_COUNT (RESY / POEM_FONT_HEIGHT)
|
||||||
|
#define POEM_LINE_LIMIT ((RESX - 4) / POEM_FONT_WIDTH)
|
||||||
|
|
||||||
|
|
||||||
|
#define POEM_PACKET_RETRIES 10
|
||||||
|
|
||||||
|
typedef enum packet_type_e {
|
||||||
|
PCKT_WINNER0,
|
||||||
|
PCKT_WINNER1,
|
||||||
|
PCKT_WINNER2,
|
||||||
|
PCKT_WINNER3,
|
||||||
|
PCKT_WINNER4,
|
||||||
|
PCKT_WINNER5,
|
||||||
|
PCKT_WINNER6,
|
||||||
|
PCKT_WINNER7,
|
||||||
|
PCKT_WAKEUP,
|
||||||
|
PCKT_STANDBY,
|
||||||
|
PCKT_NONE
|
||||||
|
} packet_type_t;
|
||||||
|
|
||||||
|
|
||||||
|
static char const *const poem_gPackets[10] = {
|
||||||
|
"winner0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||||
|
"winner1\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||||
|
"winner2\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||||
|
"winner3\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||||
|
"winner4\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||||
|
"winner5\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||||
|
"winner6\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||||
|
"winner7\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||||
|
"wake up\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0",
|
||||||
|
"standby\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0"
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
typedef enum poem_state_e {
|
||||||
|
POEM_STATE_STANDBY,
|
||||||
|
POEM_STATE_USAGE,
|
||||||
|
POEM_STATE_PLAYING,
|
||||||
|
POEM_STATE_WON
|
||||||
|
} poem_state_t;
|
||||||
|
|
||||||
|
|
||||||
|
typedef struct poem_break_marker_s {
|
||||||
|
unsigned int nStart;
|
||||||
|
unsigned int nStop;
|
||||||
|
} poem_break_marker_t;
|
||||||
|
|
||||||
|
|
||||||
|
typedef struct poem_s {
|
||||||
|
char const * const pszVerses[POEM_VERSE_COUNT];
|
||||||
|
unsigned int nLineCounts[POEM_VERSE_COUNT];
|
||||||
|
unsigned int nPermut[POEM_VERSE_COUNT];
|
||||||
|
poem_break_marker_t pszMarkers[POEM_VERSE_COUNT][POEM_MAX_LINE_COUNT];
|
||||||
|
} poem_t;
|
||||||
|
|
||||||
|
|
||||||
|
static poem_t g_poem_aPoems[POEM_COUNT]=
|
||||||
|
{
|
||||||
|
#if POEM_ID == 0
|
||||||
|
// Goethe - An den Mond, 3. Vers
|
||||||
|
{{"Jeden Nachklang fuehlt mein Herz",
|
||||||
|
"Froh und trueber Zeit,",
|
||||||
|
"Wandle zwischen Freud' und Schmerz",
|
||||||
|
"In der Einsamkeit."}},
|
||||||
|
|
||||||
|
// Goethe - Gefunden, 3. Vers
|
||||||
|
{{"Ich wollt es brechen,",
|
||||||
|
"Da sagt es fein:",
|
||||||
|
"Soll ich zum Welken",
|
||||||
|
"Gebrochen sein?"}},
|
||||||
|
|
||||||
|
// Grillparzer - In der Fremde, 2. Vers
|
||||||
|
{{"So willst Du denn nach Hause?",
|
||||||
|
"O nein! Nur nicht nach Haus!",
|
||||||
|
"Dort stirbt des Lebens Leben",
|
||||||
|
"Im Einerlei mir aus."}},
|
||||||
|
|
||||||
|
// Friedrich Schiller - Pilgrim, 1. Vers
|
||||||
|
{{"Noch in meines Lebens Lenze",
|
||||||
|
"War ich, und ich wandert' aus,",
|
||||||
|
"Und der Jugend frohe Taenze",
|
||||||
|
"Liess ich in des Vaters Haus."}},
|
||||||
|
|
||||||
|
// Frank Wedekind - Erdgeist, 2. Vers
|
||||||
|
{{"Meide nicht die ird'schen Schaetze:",
|
||||||
|
"Wo sie liegen, nimm sie mit.",
|
||||||
|
"Hat die Welt doch nur Gesetze,",
|
||||||
|
"Dass man sie mit Fuessen tritt."}},
|
||||||
|
#elif POEM_ID == 1
|
||||||
|
// Clemens Brentano - Loreley
|
||||||
|
{{"Zu Bacharach am Rheine",
|
||||||
|
"Wohnt eine Zauberin,",
|
||||||
|
"Die war so schoen und feine",
|
||||||
|
"Und riss viel Herzen hin."}},
|
||||||
|
|
||||||
|
// Wilhelm Busch - Frueher, da ich unerfahren
|
||||||
|
{{"Frueher, da ich unerfahren",
|
||||||
|
"Und bescheidner war als heute,",
|
||||||
|
"Hatten meine hoechste Achtung",
|
||||||
|
"Andre Leute."}},
|
||||||
|
|
||||||
|
// Heinz Erhardt - Warum die Zitronen sauer wurden
|
||||||
|
{{"Bis sie einst sprachen: 'Wir Zitronen,",
|
||||||
|
"wir wollen gross sein wie Melonen!",
|
||||||
|
"Auch finden wir das Gelb abscheulich,",
|
||||||
|
"wir wollen rot sein oder blaeulich!'"}},
|
||||||
|
|
||||||
|
// Joseph von Eichendorff - Mondnacht
|
||||||
|
{{"Die Luft ging durch die Felder,",
|
||||||
|
"Die Aehren wogten sacht,",
|
||||||
|
"Es rauschten leis' die Waelder,",
|
||||||
|
"So sternklar war die Nacht."}},
|
||||||
|
|
||||||
|
// Theodor Fontane - Trost
|
||||||
|
{{"Harre, hoffe. Nicht vergebens",
|
||||||
|
"zaehlest du der Stunden Schlag:",
|
||||||
|
"Wechsel ist das Los des Lebens,",
|
||||||
|
"Und - es kommt ein andrer Tag."}},
|
||||||
|
#elif POEM_ID == 2
|
||||||
|
// Goethe - Der Zauberlehrling
|
||||||
|
{{"Ach, da kommt der Meister!",
|
||||||
|
"Herr, die Not ist gross!",
|
||||||
|
"Die ich rief, die Geister,",
|
||||||
|
"Werd ich nun nicht los."}},
|
||||||
|
|
||||||
|
// Schiller - Das Lied von der Glocke
|
||||||
|
{{"Fest gemauert in der Erden",
|
||||||
|
"Steht die Form, aus Lehm gebrannt.",
|
||||||
|
"Heute muss die Glocke werden!",
|
||||||
|
"Frisch, Gesellen, seid zur Hand!"}},
|
||||||
|
|
||||||
|
// Hermann Hesse - Im Nebel
|
||||||
|
{{"Voll von Freunden war mir die Welt",
|
||||||
|
"Als noch mein Leben licht war;",
|
||||||
|
"Nun, da der Nebel faellt,",
|
||||||
|
"Ist keiner mehr sichtbar."}},
|
||||||
|
|
||||||
|
// irgend ein Kinderlied
|
||||||
|
{{"Schoen ist der Zylinderhut",
|
||||||
|
"Wenn man ihn besitzen tut",
|
||||||
|
"Doch von ganz besondrer Guete",
|
||||||
|
"Sind stets zwei Zylinderhuete"}},
|
||||||
|
|
||||||
|
// Conrad Ferdinand Meyer - Alles war ein Spiel
|
||||||
|
{{"In diesen Liedern suche du",
|
||||||
|
"Nach keinem ernsten Ziel!",
|
||||||
|
"Ein wenig Schmerz, ein wenig Lust,",
|
||||||
|
"Und alles war ein Spiel."}},
|
||||||
|
#elif POEM_ID == 3
|
||||||
|
// Friedrich Nietzsche - Vereinsamt
|
||||||
|
{{"Die Kraehen schrein",
|
||||||
|
"Und ziehen schwirren Flugs zur Stadt:",
|
||||||
|
"Bald wird es schnein, -",
|
||||||
|
"Wohl dem, der jetzt noch - Heimat hat!"}},
|
||||||
|
|
||||||
|
// Christian Morgenstern - Der Werwolf
|
||||||
|
{{"Dem Werwolf schmeichelten die Faelle,",
|
||||||
|
"er rollte seine Augenbaelle.",
|
||||||
|
"Indessen, bat er, fuege doch",
|
||||||
|
"zur Einzahl auch die Mehrzahl noch!"}},
|
||||||
|
|
||||||
|
// Wilhelm Busch - Dummheit, die man bei den anderen sieht
|
||||||
|
{{"Wenn andere klueger sind als wir,",
|
||||||
|
"Das macht uns selten nur Plaesier,",
|
||||||
|
"Doch die Gewissheit, dass sie duemmer,",
|
||||||
|
"Erfreut fast immer."}},
|
||||||
|
|
||||||
|
// August von Kotzebue - Gesellschaftslied, 2. Vers
|
||||||
|
{{"Wir sitzen so froehlich beisammen",
|
||||||
|
"Wir haben uns alle so lieb,",
|
||||||
|
"Wir heitern einander das Leben,",
|
||||||
|
"Ach wenn es doch immer so blieb'!"}},
|
||||||
|
|
||||||
|
// Gotthold Ephraim Lessing - Antwort eines trunknen Dichters, 1. Vers
|
||||||
|
{{"Ein trunkner Dichter leerte",
|
||||||
|
"Sein Glas auf jeden Zug;",
|
||||||
|
"Ihn warnte sein Gefaehrte:",
|
||||||
|
"Hoer' auf! du hast genug."}},
|
||||||
|
#elif POEM_ID == 4
|
||||||
|
// Hermann von Lingg - Das Krokodil, 1. Vers
|
||||||
|
{{"Im heil'gen Teich zu Singapur,",
|
||||||
|
"Da liegt ein altes Krokodil",
|
||||||
|
"Von aeusserst graemlicher Natur",
|
||||||
|
"Und kaut an einem Lotosstiel."}},
|
||||||
|
|
||||||
|
// Hermann Loens - Wegewarte, 2. Vers
|
||||||
|
{{"Ich stand an dem Wege,",
|
||||||
|
"Hielt auf meine Hand,",
|
||||||
|
"Du hast Deine Augen",
|
||||||
|
"Von mir abgewandt."}},
|
||||||
|
|
||||||
|
// Christian Morgenstern - An meine Taschenuhr, 1. und einziger Vers
|
||||||
|
{{"Du schlimme Uhr, du gehst mir viel zu schnell;",
|
||||||
|
"und doch - dich schauend, sah ich selber hell.",
|
||||||
|
"Unschuldig Raederwerk, was schalt ich dich?",
|
||||||
|
"Ich geh zu langsam, ach zu langsam - ich."}},
|
||||||
|
|
||||||
|
// Eduard Moerike - Jaegerlied, 2.Vers
|
||||||
|
{{"In die Luefte hoch der Reiher steigt,",
|
||||||
|
"dahin weder Pfeil noch Kugel fleugt:",
|
||||||
|
"Tausendmal so hoch und so geschwind",
|
||||||
|
"die Gedanken treuer Liebe sind."}},
|
||||||
|
|
||||||
|
// Erich Muehsam - Liebesweh, 3. Vers
|
||||||
|
{{"Ach, es ist der Traum der Liebe,",
|
||||||
|
"den ich durch die Seele siebe.",
|
||||||
|
"Ach, es ist der Liebe Weh,",
|
||||||
|
"das mich zwickt vom Kopf zur Zeh."}},
|
||||||
|
#elif POEM_ID == 5
|
||||||
|
// Wilhelm Mueller - Der Glockenguss zu Breslau, 6. Vers
|
||||||
|
{{"Wie hat der gute Meister",
|
||||||
|
"So treu das Werk bedacht!",
|
||||||
|
"Wie hat er seine Haende",
|
||||||
|
"Geruehrt bei Tag und Nacht!"}},
|
||||||
|
|
||||||
|
// Ludwig Pfau - Der Geiger von Oppenau, 2. Vers
|
||||||
|
{{"Wo seine Fiedel geklungen,",
|
||||||
|
"Da konnte kein Fuss mehr stehn,",
|
||||||
|
"Da sprangen die Alten und Jungen,",
|
||||||
|
"Die Stube fing an zu drehn."}},
|
||||||
|
|
||||||
|
// Robert Reinick - Der Faule, 2. Vers
|
||||||
|
{{"Doch die Zeit wird lang mir werden,",
|
||||||
|
"Und wie bring' ich sie herum?",
|
||||||
|
"Spitz! komm her! dich will ich lehren",
|
||||||
|
"Hund, du bist mir viel zu dumm!"}},
|
||||||
|
|
||||||
|
// Rainer Maria Rilke - Herbsttag, 2. Vers
|
||||||
|
{{"Befiehl den letzten Fruechten voll zu sein;",
|
||||||
|
"gib ihnen noch zwei suedlichere Tage,",
|
||||||
|
"draenge sie zur Vollendung hin und jage",
|
||||||
|
"die letzte Suesse in den schweren Wein."}},
|
||||||
|
|
||||||
|
// Joachim Ringelnatz - Ehrgeiz
|
||||||
|
{{"Ich habe meinen Soldaten aus Blei",
|
||||||
|
"Als Kind Verdienstkreuzchen eingeritzt.",
|
||||||
|
"Mir selber ging alle Ehre vorbei,",
|
||||||
|
"Bis auf zwei Orden, die jeder besitzt."}}
|
||||||
|
#endif
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prepares markers for line wrapping and determines the number of lines
|
||||||
|
* required to display a verse.
|
||||||
|
*/
|
||||||
|
static void poem_prepareLineWrappingMarkers(void) {
|
||||||
|
for (unsigned int iPoem = 0; iPoem < POEM_COUNT; ++iPoem) {
|
||||||
|
for (unsigned int iVerse = 0; iVerse < POEM_VERSE_COUNT; ++iVerse) {
|
||||||
|
char const * const psz = g_poem_aPoems[iPoem].pszVerses[iVerse];
|
||||||
|
size_t const nLen = strlen(psz);
|
||||||
|
unsigned int nPos = 0, nCharCnt = 0, nLineCount = 0;
|
||||||
|
int nLastSpc = -1;
|
||||||
|
|
||||||
|
poem_break_marker_t *pMarker =
|
||||||
|
&g_poem_aPoems[iPoem].pszMarkers[iVerse][nLineCount];
|
||||||
|
while ((nPos < nLen) && (nLineCount < POEM_MAX_LINE_COUNT)) {
|
||||||
|
if (nCharCnt < POEM_LINE_LIMIT && (nPos != (nLen - 1))) {
|
||||||
|
if (psz[nPos] == ' ') {
|
||||||
|
nLastSpc = nPos;
|
||||||
|
}
|
||||||
|
++nCharCnt;
|
||||||
|
} else {
|
||||||
|
pMarker->nStart = nPos - nCharCnt;
|
||||||
|
if (nPos == (nLen - 1)) {
|
||||||
|
pMarker->nStop = nPos;
|
||||||
|
} else if (nLastSpc < (nPos - nCharCnt)) {
|
||||||
|
pMarker->nStop = nPos - 1;
|
||||||
|
} else {
|
||||||
|
pMarker->nStop = nLastSpc;
|
||||||
|
nPos = nLastSpc;
|
||||||
|
}
|
||||||
|
pMarker = &g_poem_aPoems[iPoem].pszMarkers[iVerse][++nLineCount];
|
||||||
|
nCharCnt = 0;
|
||||||
|
}
|
||||||
|
++nPos;
|
||||||
|
}
|
||||||
|
g_poem_aPoems[iPoem].nLineCounts[iVerse] = nLineCount;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prints a physical line (that is one row of displayed characters) from a given
|
||||||
|
* poem and verse.
|
||||||
|
* @param pPoem The poem where to look for that line.
|
||||||
|
* @param nVerse The verse where the to be printed line resides.
|
||||||
|
* @param nLine The position (counting from 0) of that line within that verse.
|
||||||
|
* @param y The vertical offset where text should be rendered.
|
||||||
|
*/
|
||||||
|
static void poem_printLine(poem_t const *const pPoem, unsigned int nVerse,
|
||||||
|
unsigned int nLine, int y) {
|
||||||
|
char pszLine[POEM_LINE_LIMIT + 1] = { 0 };
|
||||||
|
poem_break_marker_t const * const pMarker =
|
||||||
|
&pPoem->pszMarkers[nVerse][nLine];
|
||||||
|
for (unsigned int k = pMarker->nStart; k <= pMarker->nStop; ++k) {
|
||||||
|
pszLine[k - pMarker->nStart] = pPoem->pszVerses[nVerse][k];
|
||||||
|
}
|
||||||
|
DoString(2, y, pszLine);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Shuffles the verses of a given poem.
|
||||||
|
* @param pPoem The poem whose verses need to be shuffled.
|
||||||
|
*/
|
||||||
|
static void poem_initPermutation(poem_t *const pPoem) {
|
||||||
|
for (unsigned int i = POEM_VERSE_COUNT; i--;) {
|
||||||
|
pPoem->nPermut[i] = i;
|
||||||
|
}
|
||||||
|
while ((pPoem->nPermut[0] < pPoem->nPermut[1])
|
||||||
|
&& (pPoem->nPermut[1] < pPoem->nPermut[2])
|
||||||
|
&& (pPoem->nPermut[2] < pPoem->nPermut[3])) {
|
||||||
|
for (unsigned int i = 4; i--;) {
|
||||||
|
unsigned int const nIndexA = getRandom() % POEM_VERSE_COUNT;
|
||||||
|
unsigned int const nIndexB = getRandom() % POEM_VERSE_COUNT;
|
||||||
|
|
||||||
|
unsigned int nSwap = pPoem->nPermut[nIndexA];
|
||||||
|
pPoem->nPermut[nIndexA] = pPoem->nPermut[nIndexB];
|
||||||
|
pPoem->nPermut[nIndexB] = nSwap;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t poem_identifyPacket(uint8_t *packet){
|
||||||
|
uint8_t input = BTN_NONE;
|
||||||
|
if (strncmp((char *)packet, poem_gPackets[PCKT_STANDBY], 8) == 0) {
|
||||||
|
input = BTN_STANDBY;
|
||||||
|
} else if (strncmp((char *)packet, poem_gPackets[PCKT_WAKEUP], 8) == 0) {
|
||||||
|
input = BTN_WAKEUP;
|
||||||
|
}
|
||||||
|
return input;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Watches for both joystick movements and standby NRF packets. It's blocks the
|
||||||
|
* control flow as long as there are neither appropriate joystick activities nor
|
||||||
|
* standby/wake up packets received.
|
||||||
|
* @param filter A mask of events to which this function should react.
|
||||||
|
* @return One of BTN_(UP|DOWN|LEFT|RIGHT|ENTER|STANDBY|WAKEUP).
|
||||||
|
*/
|
||||||
|
uint8_t poem_getInputBlocking(uint8_t filter) {
|
||||||
|
uint8_t pkt[32];
|
||||||
|
uint8_t input = BTN_NONE;
|
||||||
|
nrf_rcv_pkt_start();
|
||||||
|
do {
|
||||||
|
if ((nrf_rcv_pkt_poll(sizeof(pkt), pkt) != 32) ||
|
||||||
|
!((input = poem_identifyPacket(pkt)) & filter)) {
|
||||||
|
input = getInput();
|
||||||
|
}
|
||||||
|
} while (!(filter & input));
|
||||||
|
nrf_rcv_pkt_end();
|
||||||
|
return input;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Watches for both joystick movements and standby NRF packets. It returns
|
||||||
|
* BTN_NONE if there's neither a NRF packet nor joystick input.
|
||||||
|
* @return One of BTN_(NONE|UP|DOWN|LEFT|RIGHT|ENTER|STANDBY|WAKEUP).
|
||||||
|
*/
|
||||||
|
uint8_t poem_getInputNonBlocking(void) {
|
||||||
|
uint8_t pkt[32];
|
||||||
|
uint8_t input;
|
||||||
|
if (nrf_rcv_pkt_poll(sizeof(pkt), pkt) == 32) {
|
||||||
|
input = poem_identifyPacket(pkt);
|
||||||
|
if (input == BTN_NONE)
|
||||||
|
input == getInput();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
input = getInput();
|
||||||
|
}
|
||||||
|
return input;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Displays a small description of the game.
|
||||||
|
*/
|
||||||
|
static void poem_printUsage(void) {
|
||||||
|
// show usage
|
||||||
|
int dy = 0;
|
||||||
|
lcdFill(0);
|
||||||
|
DoString(0, dy, "Sortiere die Verse "); dy += POEM_FONT_HEIGHT;
|
||||||
|
DoString(0, dy, "in die richtige "); dy += POEM_FONT_HEIGHT;
|
||||||
|
DoString(0, dy, "Reihenfolge! Der "); dy += POEM_FONT_HEIGHT;
|
||||||
|
DoString(0, dy, "Knopf ist ein Joy- "); dy += POEM_FONT_HEIGHT;
|
||||||
|
DoString(0, dy, "stick. Waehle einen"); dy += POEM_FONT_HEIGHT;
|
||||||
|
DoString(0, dy, "Vers mit OBEN oder "); dy += POEM_FONT_HEIGHT;
|
||||||
|
DoString(0, dy, "UNTEN. Verschiebe "); dy += POEM_FONT_HEIGHT;
|
||||||
|
DoString(0, dy, "ihn per FEUERTASTE."); dy += POEM_FONT_HEIGHT;
|
||||||
|
lcdDisplay();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initializes the wireless stuff.
|
||||||
|
*/
|
||||||
|
static void poem_initRadio(void) {
|
||||||
|
nrf_init();
|
||||||
|
static struct NRF_CFG config = {
|
||||||
|
.channel = 81,
|
||||||
|
.txmac = "\x1\x2\x3\x2\x1",
|
||||||
|
.nrmacs = 1,
|
||||||
|
.mac0 = "\x1\x2\x3\x2\x1",
|
||||||
|
.maclen = "\x20",
|
||||||
|
};
|
||||||
|
nrf_config_set(&config);
|
||||||
|
delayms(50);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Turns on the backlight and switches to dark text on a light background.
|
||||||
|
*/
|
||||||
|
static void poem_turnOnBacklight(void) {
|
||||||
|
GLOBAL(daytrig) = 255;
|
||||||
|
GLOBAL(daytrighyst) = 50;
|
||||||
|
GLOBAL(dayinvert) = 0;
|
||||||
|
GLOBAL(lcdinvert) = 1;
|
||||||
|
backlightSetBrightness(100);
|
||||||
|
lcdSetInvert(1);
|
||||||
|
lcdFill(0);
|
||||||
|
lcdDisplay();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Turns off the backlight and switches to light text on a dark background.
|
||||||
|
*/
|
||||||
|
static void poem_turnOffBacklight(void) {
|
||||||
|
GLOBAL(daytrig) = 0;
|
||||||
|
GLOBAL(daytrighyst) = 50;
|
||||||
|
GLOBAL(dayinvert) = 0;
|
||||||
|
GLOBAL(lcdinvert) = 0;
|
||||||
|
backlightSetBrightness(0);
|
||||||
|
lcdSetInvert(0);
|
||||||
|
lcdFill(0);
|
||||||
|
lcdDisplay();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Highlights the specified item by drawing a bounding box around it (to
|
||||||
|
* indicate a selection) or by inverting its representation (dragging).
|
||||||
|
* @param bIsDragging Whether the item is just selected or about to be dragged.
|
||||||
|
* @param nStartY The (absolute) start offset (in pixels) of the item.
|
||||||
|
* @param nStopY The (absolute) stop offset (in pixels) of the item.
|
||||||
|
* @param nScroll The vertical scrolling offset.
|
||||||
|
*/
|
||||||
|
static void poem_hightlightSelectedItem(unsigned int const bIsDragging,
|
||||||
|
int nStartY, int const nStopY, int const nScroll) {
|
||||||
|
// invert or draw a bounding box around the selected verse
|
||||||
|
if (bIsDragging) {
|
||||||
|
// dragging mode: invert the to be dragged item
|
||||||
|
for (int y = nStartY; y < nStopY; ++y) {
|
||||||
|
for (int x = 0; x < RESX; ++x) {
|
||||||
|
lcdSetPixel(x, y - nScroll, !lcdGetPixel(x, y - nScroll));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// selection mode: draw a border around the selected item
|
||||||
|
for (int x = 0; x < RESX; ++x) {
|
||||||
|
lcdSetPixel(x, nStartY - nScroll, 1);
|
||||||
|
lcdSetPixel(x, nStopY - nScroll, 1);
|
||||||
|
}
|
||||||
|
for (int y = nStartY; y < nStopY; ++y) {
|
||||||
|
lcdSetPixel(0, y - nScroll, 1);
|
||||||
|
lcdSetPixel(RESX - 1, y - nScroll, 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Informs the master device that the player has accomplished the task and
|
||||||
|
* blocks as long as there are no "standby" or "wakeup" packets.
|
||||||
|
*/
|
||||||
|
static void poem_sendSuccessMessage(packet_type_t pktWinner) {
|
||||||
|
// send our winner information packet
|
||||||
|
static uint8_t pkt[32];
|
||||||
|
|
||||||
|
nrf_rcv_pkt_start();
|
||||||
|
while (1) {
|
||||||
|
memcpy(pkt, poem_gPackets[pktWinner], sizeof(pkt));
|
||||||
|
nrf_snd_pkt_crc(sizeof(pkt), pkt);
|
||||||
|
delayms_queue(80 + (getRandom() % 50));
|
||||||
|
uint8_t input = poem_getInputNonBlocking();
|
||||||
|
if (input == BTN_STANDBY) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
nrf_rcv_pkt_end();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Starts a game where the user has to sort verses of a poem into the right
|
||||||
|
* order.
|
||||||
|
*/
|
||||||
|
void main_poem(void) {
|
||||||
|
packet_type_t pktWinner = POEM_ID;
|
||||||
|
|
||||||
|
// display initialization
|
||||||
|
font = &POEM_FONT;
|
||||||
|
|
||||||
|
// init wireless stuff
|
||||||
|
poem_initRadio();
|
||||||
|
poem_prepareLineWrappingMarkers();
|
||||||
|
|
||||||
|
poem_state_t state = POEM_STATE_STANDBY;
|
||||||
|
|
||||||
|
while (1) {
|
||||||
|
switch (state) {
|
||||||
|
case POEM_STATE_STANDBY:
|
||||||
|
poem_turnOffBacklight();
|
||||||
|
lcdFill(0);
|
||||||
|
DoInt(0, 0, pktWinner);
|
||||||
|
lcdDisplay();
|
||||||
|
poem_getInputBlocking(BTN_WAKEUP);
|
||||||
|
state = POEM_STATE_USAGE;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case POEM_STATE_USAGE:
|
||||||
|
poem_turnOnBacklight();
|
||||||
|
poem_printUsage();
|
||||||
|
{
|
||||||
|
uint8_t input = poem_getInputBlocking(BTN_UP | BTN_DOWN |
|
||||||
|
BTN_LEFT | BTN_RIGHT | BTN_ENTER | BTN_STANDBY);
|
||||||
|
if (input == BTN_STANDBY) {
|
||||||
|
state = POEM_STATE_STANDBY;
|
||||||
|
}
|
||||||
|
else if (input != BTN_WAKEUP) {
|
||||||
|
state = POEM_STATE_PLAYING;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
DoString(20,0, "bla");
|
||||||
|
lcdDisplay();
|
||||||
|
break;
|
||||||
|
|
||||||
|
case POEM_STATE_PLAYING:
|
||||||
|
{
|
||||||
|
// print the verses
|
||||||
|
poem_t *const pPoem = &g_poem_aPoems[getRandom() % POEM_COUNT];
|
||||||
|
poem_initPermutation(pPoem);
|
||||||
|
unsigned int nSelectedVerse = 0, nNextSelected = 0;
|
||||||
|
unsigned int bIsDragging = 0;
|
||||||
|
while (state == POEM_STATE_PLAYING) {
|
||||||
|
// Calculate position and size of the bounding box of the selected
|
||||||
|
// verse. Also, calculate a vertical scrolling offset to ensure
|
||||||
|
// that the selected verse is visible.
|
||||||
|
int nStartY = 0;
|
||||||
|
for (unsigned int i = 0; i < nSelectedVerse; ++i) {
|
||||||
|
unsigned int index = pPoem->nPermut[i];
|
||||||
|
nStartY += POEM_FONT_HEIGHT * pPoem->nLineCounts[index] + 3;
|
||||||
|
}
|
||||||
|
const int nStopY = 2 + nStartY + (POEM_FONT_HEIGHT
|
||||||
|
* pPoem->nLineCounts[pPoem->nPermut[nSelectedVerse]]);
|
||||||
|
int nScroll = nStopY >= RESY ? nStopY - RESY + 1 : 0;
|
||||||
|
// display verses
|
||||||
|
lcdFill(0);
|
||||||
|
int dy = 2;
|
||||||
|
for (unsigned int i = 0; i < POEM_VERSE_COUNT; ++i) {
|
||||||
|
unsigned int const index = pPoem->nPermut[i];
|
||||||
|
for (unsigned int j = 0; j < pPoem->nLineCounts[index]; ++j) {
|
||||||
|
poem_printLine(pPoem, index, j, dy - nScroll);
|
||||||
|
dy += POEM_FONT_HEIGHT;
|
||||||
|
}
|
||||||
|
dy += 3;
|
||||||
|
}
|
||||||
|
// draw a bounding box around the selected verse (or invert it)
|
||||||
|
poem_hightlightSelectedItem(bIsDragging, nStartY, nStopY, nScroll);
|
||||||
|
// flush display buffer
|
||||||
|
lcdDisplay();
|
||||||
|
|
||||||
|
// both query joystick and listen for "standby" packets;
|
||||||
|
switch (poem_getInputBlocking(BTN_UP | BTN_DOWN |
|
||||||
|
BTN_ENTER | BTN_STANDBY)) {
|
||||||
|
case BTN_ENTER:
|
||||||
|
bIsDragging = !bIsDragging;
|
||||||
|
if (!bIsDragging && pPoem->nPermut[0] < pPoem->nPermut[1]
|
||||||
|
&& pPoem->nPermut[1] < pPoem->nPermut[2]
|
||||||
|
&& pPoem->nPermut[2] < pPoem->nPermut[3]) {
|
||||||
|
state = POEM_STATE_WON;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case BTN_UP:
|
||||||
|
nNextSelected = nSelectedVerse > 0 ?
|
||||||
|
nSelectedVerse - 1 : nSelectedVerse;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case BTN_DOWN:
|
||||||
|
nNextSelected = nSelectedVerse < (POEM_VERSE_COUNT - 1) ?
|
||||||
|
nSelectedVerse + 1 : nSelectedVerse;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case BTN_STANDBY:
|
||||||
|
state = POEM_STATE_STANDBY;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
if (nSelectedVerse != nNextSelected && bIsDragging) {
|
||||||
|
unsigned int temp = pPoem->nPermut[nSelectedVerse];
|
||||||
|
pPoem->nPermut[nSelectedVerse] = pPoem->nPermut[nNextSelected];
|
||||||
|
pPoem->nPermut[nNextSelected] = temp;
|
||||||
|
}
|
||||||
|
nSelectedVerse = nNextSelected;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case POEM_STATE_WON:
|
||||||
|
lcdFill(0);
|
||||||
|
DoString(22, 30, "Geschafft!");
|
||||||
|
lcdDisplay();
|
||||||
|
poem_sendSuccessMessage(pktWinner);
|
||||||
|
state = POEM_STATE_STANDBY;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue