#include <avr/io.h> // this contains all the IO port definitions
#include <avr/interrupt.h>
#include <avr/signal.h>
#include <avr/pgmspace.h>
#include <util/delay.h>
void delay_ms( uint16_t milliseconds)
{
for( ; milliseconds > 0; milliseconds--)
{
_delay_ms( 1);
}
}
#define TIMER1_PRESCALE_1 1
#define TIMER1_PRESCALE_8 2
#define TIMER1_PRESCALE_64 3
#define TIMER1_PRESCALE_256 4
#define TIMER1_PRESCALE_1024 5
// We use these macros because binary constants arent always supported. ugh.
#define HEX__(n) 0x##n##UL
#define B8__(x) ((x&0x0000000FLU)?1:0) \
+((x&0x000000F0LU)?2:0) \
+((x&0x00000F00LU)?4:0) \
+((x&0x0000F000LU)?8:0) \
+((x&0x000F0000LU)?16:0) \
+((x&0x00F00000LU)?32:0) \
+((x&0x0F000000LU)?64:0) \
+((x&0xF0000000LU)?128:0)
#define B8(d) ((unsigned char)B8__(HEX__(d)))
// store all the image data in program memory (ROM)
// instead of RAM (the default)
const uint8_t large_image[] PROGMEM = {
B8(00000000),
B8(01110000),
B8(10101000),
B8(10101000),
B8(10101000),
B8(10110000),
B8(00000000),
B8(00000000),
B8(10001000),
B8(11011000),
B8(00100000),
B8(11011000),
B8(10001000),
B8(00000000),
B8(00000000),
B8(10001000),
B8(10001000),
B8(11111010),
B8(10000000),
B8(10000000),
B8(00000000),
B8(00001000),
B8(01111100),
B8(10001000),
B8(10001000),
B8(10001000),
B8(01000000),
B8(00000000),
B8(00000000),
};
// special pointer for reading from ROM memory
PGM_P largeimage_p PROGMEM = large_image;
#define NUM_ELEM(x) (sizeof (x) / sizeof (*(x)))
int imagesize = NUM_ELEM(large_image);
// this function is called when timer1 compare matches OCR1A
uint8_t j = 0;
SIGNAL( SIG_TIMER1_COMPA ) {
if (j >= imagesize)
j = 0;
// read the image data from ROM
PORTB = pgm_read_byte(largeimage_p + j);
j++;
}
int main(void) {
DDRB = 0xFF; // set all 8 pins on port B to outputs
/*
the frequency of the interrupt overflow is determined by the
prescaler and overflow value.
freq = clock_frequency / ( 2 * prescaler * overflow_val)
where prescaler can be 1, 8, 64, 256, or 1024
clock_freq is 8MHz
and overflow_val is 16bit
the overflow value is placed in OCR1A, the prescale is set in TCCR1B
so for example:
A good POV frequency is around 400Hz
desired freq = 400Hz
clock freq = 8MHz
8MHz / (400Hz * 2) = 10000
since 10000 is less than 655536 (largest 16 bit number)
OCR1A = 10000 and the prescale is 1
*/
TCCR1B = (1 << WGM12) | TIMER1_PRESCALE_1;
OCR1A = (uint16_t)10000;
TIMSK |= 1 << OCIE1A; // Output Compare Interrupt Enable (timer 1, OCR1A)
sei(); // Set Enable Interrupts
while (1);
}
I2luY2x1ZGUgPGF2ci9pby5oPiAgICAgIC8vIHRoaXMgY29udGFpbnMgYWxsIHRoZSBJTyBwb3J0IGRlZmluaXRpb25zCiNpbmNsdWRlIDxhdnIvaW50ZXJydXB0Lmg+CiNpbmNsdWRlIDxhdnIvc2lnbmFsLmg+CiNpbmNsdWRlIDxhdnIvcGdtc3BhY2UuaD4KI2luY2x1ZGUgPHV0aWwvZGVsYXkuaD4KIAp2b2lkIGRlbGF5X21zKCB1aW50MTZfdCBtaWxsaXNlY29uZHMpCnsKICAgZm9yKCA7IG1pbGxpc2Vjb25kcyA+IDA7IG1pbGxpc2Vjb25kcy0tKQogICB7CiAgICAgIF9kZWxheV9tcyggMSk7CiAgIH0KfSAKIAojZGVmaW5lIFRJTUVSMV9QUkVTQ0FMRV8xIDEKI2RlZmluZSBUSU1FUjFfUFJFU0NBTEVfOCAyCiNkZWZpbmUgVElNRVIxX1BSRVNDQUxFXzY0IDMKI2RlZmluZSBUSU1FUjFfUFJFU0NBTEVfMjU2IDQKI2RlZmluZSBUSU1FUjFfUFJFU0NBTEVfMTAyNCA1CiAKIAovLyBXZSB1c2UgdGhlc2UgbWFjcm9zIGJlY2F1c2UgYmluYXJ5IGNvbnN0YW50cyBhcmVudCBhbHdheXMgc3VwcG9ydGVkLiB1Z2guCiNkZWZpbmUgSEVYX18obikgMHgjI24jI1VMCiNkZWZpbmUgQjhfXyh4KSAoKHgmMHgwMDAwMDAwRkxVKT8xOjApICBcCiAgICAgICAgICAgICAgICsoKHgmMHgwMDAwMDBGMExVKT8yOjApICBcCiAgICAgICAgICAgICAgICsoKHgmMHgwMDAwMEYwMExVKT80OjApICBcCiAgICAgICAgICAgICAgICsoKHgmMHgwMDAwRjAwMExVKT84OjApICBcCiAgICAgICAgICAgICAgICsoKHgmMHgwMDBGMDAwMExVKT8xNjowKSBcCiAgICAgICAgICAgICAgICsoKHgmMHgwMEYwMDAwMExVKT8zMjowKSBcCiAgICAgICAgICAgICAgICsoKHgmMHgwRjAwMDAwMExVKT82NDowKSBcCiAgICAgICAgICAgICAgICsoKHgmMHhGMDAwMDAwMExVKT8xMjg6MCkKI2RlZmluZSBCOChkKSAoKHVuc2lnbmVkIGNoYXIpQjhfXyhIRVhfXyhkKSkpCiAKIAovLyBzdG9yZSBhbGwgdGhlIGltYWdlIGRhdGEgaW4gcHJvZ3JhbSBtZW1vcnkgKFJPTSkKLy8gaW5zdGVhZCBvZiBSQU0gKHRoZSBkZWZhdWx0KQpjb25zdCB1aW50OF90IGxhcmdlX2ltYWdlW10gUFJPR01FTSA9IHsKQjgoMDAwMDAwMDApLApCOCgwMTExMDAwMCksCkI4KDEwMTAxMDAwKSwKQjgoMTAxMDEwMDApLApCOCgxMDEwMTAwMCksCkI4KDEwMTEwMDAwKSwKQjgoMDAwMDAwMDApLApCOCgwMDAwMDAwMCksCkI4KDEwMDAxMDAwKSwKQjgoMTEwMTEwMDApLApCOCgwMDEwMDAwMCksCkI4KDExMDExMDAwKSwKQjgoMTAwMDEwMDApLApCOCgwMDAwMDAwMCksCkI4KDAwMDAwMDAwKSwKQjgoMTAwMDEwMDApLApCOCgxMDAwMTAwMCksCkI4KDExMTExMDEwKSwKQjgoMTAwMDAwMDApLApCOCgxMDAwMDAwMCksCkI4KDAwMDAwMDAwKSwKQjgoMDAwMDEwMDApLApCOCgwMTExMTEwMCksCkI4KDEwMDAxMDAwKSwKQjgoMTAwMDEwMDApLApCOCgxMDAwMTAwMCksCkI4KDAxMDAwMDAwKSwKQjgoMDAwMDAwMDApLApCOCgwMDAwMDAwMCksCn07CiAKLy8gc3BlY2lhbCBwb2ludGVyIGZvciByZWFkaW5nIGZyb20gUk9NIG1lbW9yeQpQR01fUCBsYXJnZWltYWdlX3AgUFJPR01FTSA9IGxhcmdlX2ltYWdlOwogCiNkZWZpbmUgTlVNX0VMRU0oeCkgKHNpemVvZiAoeCkgLyBzaXplb2YgKCooeCkpKQppbnQgaW1hZ2VzaXplID0gTlVNX0VMRU0obGFyZ2VfaW1hZ2UpOwogCiAKIAovLyB0aGlzIGZ1bmN0aW9uIGlzIGNhbGxlZCB3aGVuIHRpbWVyMSBjb21wYXJlIG1hdGNoZXMgT0NSMUEKdWludDhfdCBqID0gMDsKU0lHTkFMKCBTSUdfVElNRVIxX0NPTVBBICkgewogIGlmIChqID49IGltYWdlc2l6ZSkgCiAgICBqID0gMDsKIAogIC8vIHJlYWQgdGhlIGltYWdlIGRhdGEgZnJvbSBST00KICBQT1JUQiA9IHBnbV9yZWFkX2J5dGUobGFyZ2VpbWFnZV9wICsgaik7IAogCiAgaisrOwp9CiAKaW50IG1haW4odm9pZCkgewogCiAgRERSQiA9IDB4RkY7ICAgICAgIC8vIHNldCBhbGwgOCBwaW5zIG9uIHBvcnQgQiB0byBvdXRwdXRzCiAKICAvKgogICAgdGhlIGZyZXF1ZW5jeSBvZiB0aGUgaW50ZXJydXB0IG92ZXJmbG93IGlzIGRldGVybWluZWQgYnkgdGhlIAogICAgcHJlc2NhbGVyIGFuZCBvdmVyZmxvdyB2YWx1ZS4KICAgIGZyZXEgPSBjbG9ja19mcmVxdWVuY3kgLyAoIDIgKiBwcmVzY2FsZXIgKiBvdmVyZmxvd192YWwpCiAgICB3aGVyZSBwcmVzY2FsZXIgY2FuIGJlIDEsIDgsIDY0LCAyNTYsIG9yIDEwMjQKICAgIGNsb2NrX2ZyZXEgaXMgOE1IegogICAgYW5kIG92ZXJmbG93X3ZhbCBpcyAxNmJpdAogCiAgICB0aGUgb3ZlcmZsb3cgdmFsdWUgaXMgcGxhY2VkIGluIE9DUjFBLCB0aGUgcHJlc2NhbGUgaXMgc2V0IGluIFRDQ1IxQgogICAgc28gZm9yIGV4YW1wbGU6CiAgICBBIGdvb2QgUE9WIGZyZXF1ZW5jeSBpcyBhcm91bmQgNDAwSHoKICAgIGRlc2lyZWQgZnJlcSA9IDQwMEh6CiAgICBjbG9jayBmcmVxID0gOE1IegogICAgOE1IeiAvICg0MDBIeiAqIDIpID0gMTAwMDAKICAgIHNpbmNlIDEwMDAwIGlzIGxlc3MgdGhhbiA2NTU1MzYgKGxhcmdlc3QgMTYgYml0IG51bWJlcikKICAgIE9DUjFBID0gMTAwMDAgYW5kIHRoZSBwcmVzY2FsZSBpcyAxCiAgKi8KIAogIFRDQ1IxQiA9ICgxIDw8IFdHTTEyKSB8IFRJTUVSMV9QUkVTQ0FMRV8xOwogIE9DUjFBID0gKHVpbnQxNl90KTEwMDAwOwogCiAgVElNU0sgfD0gMSA8PCBPQ0lFMUE7ICAgLy8gT3V0cHV0IENvbXBhcmUgSW50ZXJydXB0IEVuYWJsZSAodGltZXIgMSwgT0NSMUEpIAogCiAgc2VpKCk7ICAgICAgICAgICAgICAgICAvLyBTZXQgRW5hYmxlIEludGVycnVwdHMKIAogIHdoaWxlICgxKTsKfQ==