Um Zufallszahlen mit der rand-Funktion (aus stdlib.h) zu erzeugen, ist ein Startwert (Seed) notwendig. Dieser wird durch srand(seed) gesetzt. Bei gleichem Seed ist die Folge immer gleich. Um unterschiedliche Folgen zu erhalten, ist ein veränderlicher Seed erforderlich. Dazu gibt es mehrere Ansätze:
Legt man eine Konstante mit
const uint8_t constant = 2;
an, wird der Wert aus dem Flash in den RAM geladen und belegt dort unnötig Platz, denn verändert werden darf der Wert der Konstante sowieso nicht. Soll das automatische Laden in das RAM verhindert werden, muss das Schlüsselword PROGMEM verwendet werden:
const PROGMEM uint8_t constant = 2;
Diese Werte müssen dann mit pgm_read_byte()
, pgm_read_word()
, usw. gelesen werden. Soll eine Funktion Werte aus RAM und Flash verarbeiten, müssen zwei Versionen programmiert werden, denn innerhalb einer Funktion ist nicht zu erkennen, ob die Daten in RAM oder Flash liegen. Es ist üblich das Funktionen, die aus dem Flash lesen auf „_P“ enden:
const char stringInRAM[] = "Teststring im RAM"; const PROGMEM char stringInFlash[] = "Teststring im Flash"; lcd_string(stringInRAM); lcd_string_P(stringInFlash);
Häufig werden Strings inline definiert. Diese werden auch vom Flash ins RAM geladen. Um das zu verhindern, aber die Inline-Definition beizubehalten, gibt es das PSTR-Makro:
lcd_string("String im RAM"); lcd_string_P(PSTR("String im Flash");
Prinizipiell können alle Datentypen im Flash abgelegt werden. Bei structs sind einige Besonderheiten zu beachten.
typedef struct{ char text[20]; uint8_t index; } struct1; struct1 PROGMEM teststruct = { .text = "String in RAM", .index = 0 };
Hierbei liegt die Variable index und der Pointer auf den String im Flash, der String selbst liegt aber noch im RAM. Um das zu verhindern, muss er wie folgt definiert werden (eine Verwendung des PSTR-Makros ist hier nicht möglich):
typedef struct{ const char* text; uint8_t index; } struct1; const PROGMEM char stringInFlash[] = "String in Flash"; struct1 PROGMEM teststruct = { .text = stringInFlash, .index = 0 };
Der Zugriff auf die Elemente des Structs erfolgt dann so:
lcd_string_P((char*) pgm_read_word(&teststruct->text)); //Cast zu Char Pointer if (pgm_read_byte(&(teststruct->index))==curIndex) return true;
Da eine Compileunit in C immer nur eine Datei umfasst, kann der GCC im ganzen Projekt nicht benutzte Funktionen nicht erkennen und herausoptimieren. Man kann aber Funktionen mit
CFLAGS += -ffunctions-sections
in eine eigene Section legen. Durch
LDFLAGS += -W1,--gc-sections
kann man dann den Linker anweisen, unbenutzte Sections zu entfernen, so dass das entstehende Binary kleiner wird.
Achtung Damit Interrupthandler nicht entfernt werden, müssen sie mit
__attribute__ ((used))
gekennzeichnet werden.
Um einen Pin am AVR zu konfigurieren und zu benutzen, benötigt man die drei Variablen PINX, PORTX, DDRX. Um dies konfigurierbar zu machen, kann man drei defines verwenden:
#define SENSOR_DDR DDRA #define SENSOR_PORT PORTA #define SENSOR_PIN PINA
Möchte man den Port z.B. von A auf B ändern, muss man drei defines anpassen. Mit einem Makrotrick kann man das vermeiden. PINX, PORTX, und DDRX werden so definiert:
#define SENSOR_DDR __CONCAT(DDR, SENSOR_PORT_LETTER) #define SENSOR_PORT __CONCAT(PORT, SENSOR_PORT_LETTER) #define SENSOR_PIN __CONCAT(PIN, SENSOR_PORT_LETTER)
Mit
#define SENSOR_PORT_LETTER A
lässt sich dann der Buchstabe für alle drei Variablen anpassen. Das Makro __CONCAT ist dabei wie folgt definiert:
#define __CONCATenate(left, right) left ## right #define __CONCAT(left, right) __CONCATenate(left, right)
Man kann diese Makros selbst definieren oder die Definition aus dem Header stdint.h benutzen, da man diesen sowieso fast immer inkludiert.