Ich habe hier mal in einem einzigen Demo-C-Source das geballte Knowhow zusammengefasst, wie man für einen Rechner mit Memorymapped I/O (z.B. ARM oder Motorola 68000/ColdFire) korrekt den Zugriff auf ein Hardwareregister via Pointer realisiert:
/* * Memory mapped I/O made easy with ANSI C * commented by Matthias Arndt <marndt@asmsoftware.de> */ #include <stdint.h>
/* temporary valid location for demonstration purposes */ uint8_t storage;
/* Now the magic declaration pointer to a hw register: * We point it to some known storage but ofcourse for pointing * to a real I/O hardware register, one would supply a constant * register address. * * a) declare it volatile because hardware I/O locations may change inside * a different context (interrupt and/or hardware event) * b) make the pointer address const between register name and the type * definition so that noone may modify the pointer * c) adding const before the declaration will declare a read/only register */
volatile uint8_t * const HWREG = &storage; /* to point to real I/O you would use the following syntax: */ /* volatile uint8_t * const HWREG = (uint8_t *)0xF000; */
/* just some demo calls */ void task(void); uint8_t access(void);
void task() { *HWREG = 0x5a; /* set I/O for demo purpose */
/* access to alter the pointer is forbidden! Uncomment to try! */ /* HWREG = (uint8_t *) 0xaaaa; */ }
uint8_t access() { return(*HWREG); /* read I/O */ }
Für 8051 oder andere Plattformen, die ähnlich verquerte Speicherbereiche haben, sollten zusätzliche Angaben verwendet werden, die den Zugriff auf den richtigen Speicherbereich mappen, etwa XRAM beim 8051.