File: eeprom-1.c

package info (click to toggle)
avr-libc 1%3A1.6.2.cvs20080610-2
  • links: PTS
  • area: main
  • in suites: lenny
  • size: 14,848 kB
  • ctags: 55,619
  • sloc: ansic: 92,267; asm: 6,692; sh: 4,131; makefile: 2,481; python: 976; pascal: 426; perl: 116
file content (113 lines) | stat: -rw-r--r-- 2,941 bytes parent folder | download
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
/* $Id: eeprom-1.c,v 1.2.2.3 2008/04/04 08:14:45 dmix Exp $	*/

#include <avr/eeprom.h>
#include <stdlib.h>
#include <string.h>

/* Note: GCC permits arithmetic with 'void *'.	*/

int main ()
{
    void *p;
#define MAXLEN	(RAMEND > 0x15F ? 260 : (RAMEND - 0x60 - 30))
    unsigned char s[E2END > MAXLEN ? MAXLEN : E2END];

    /* Fill all EEPROM.	*/
    for (p = 0; p <= (void *)E2END; p++)
	eeprom_write_byte (p, ~(int)p);

    /* Read bytes.	*/
    for (p = 0; p <= (void *)E2END; p++)
	if (eeprom_read_byte (p) != (~(int)p & 0xFF))
	    exit (__LINE__);

    /* Read words.	*/
    for (p = 0; p <= (void *)E2END - 1; p += 2) {
	if (eeprom_read_word (p)
	    != ((~(unsigned)p & 0xFF) | (~(unsigned)(p+1) & 0xFF) << 8))
	{
	    exit (__LINE__);
	}
    }

    /* Read double words.	*/
    for (p = 0; p <= (void *)E2END - 3; p += 4) {
	if (eeprom_read_dword (p)
	    != ((~(unsigned)p & 0xFF)
		| (~(unsigned)(p+1) & 0xFF) << 8
		| (unsigned long)(~(unsigned)(p+2) & 0xFF) << 16
		| (unsigned long)(~(unsigned)(p+3) & 0xFF) << 24))
	{
	    exit (__LINE__);
	}
    }

    /* Read blocks.	*/
    for (p = 0; p <= (void *)E2END; ) {
	size_t i, n;
	if ( (n = E2END + 1 - (unsigned)p) > sizeof(s))
	    n = sizeof(s);
	eeprom_read_block (s, p, n);
	for (i = 0; i < n; i++) {
	    if (s[i] != (~(int)p & 0xFF))
		exit (__LINE__);
	    p += 1;
	}
    }

    /* Write only 1 byte.	*/
    eeprom_write_byte ((unsigned char *)5, 1);
    for (p = 0; p <= (void *)E2END; p++) {
	if ((unsigned)p != 5
	    && eeprom_read_byte (p) != (~(int)p & 0xFF))
	{
	    exit (__LINE__);
	}
    }
    if (eeprom_read_byte ((unsigned char *)5) != 1)
	exit (__LINE__);

    /* Write only 1 word.	*/
    /* Avr-gcc 4.2.2 produces incorrect code: the comparison of address
       with 5 and 6 values is omited. In result exit with error.
       Versions 3.3.6, 3.4.6, 4.0.4, 4.1.2, 4.2.3, 4.3.0(pre) give
       correct code and are simulated OK.	*/
    eeprom_write_word ((unsigned *)5, 0x1234);
    for (p = 0; p <= (void *)E2END; p++) {
	if ((unsigned)p != 5
	    && (unsigned)p != 6
	    && eeprom_read_byte (p) != (~(int)p & 0xFF))
	{
	    exit (__LINE__);
	}
    }
    if (eeprom_read_word ((unsigned *)5) != 0x1234)
	exit (__LINE__);

    /* Write only 1 double word.	*/
    eeprom_write_dword ((unsigned long *)5, 0x12345678);
    for (p = 0; p <= (void *)E2END; p++) {
	if (((unsigned)p < 5 || (unsigned)p > 8)
	    && eeprom_read_byte (p) != (~(int)p & 0xFF))
	{
	    exit (__LINE__);
	}
    }
    if (eeprom_read_dword ((unsigned long *)5) != 0x12345678)
	exit (__LINE__);

    /* Write block.	*/
    eeprom_write_block ("\x90\xAB\xCD\xEF\x01", (void *)5, 5);
    for (p = 0; p <= (void *)E2END; p++) {
	if (((unsigned)p < 5 || (unsigned)p > 9)
	    && eeprom_read_byte (p) != (~(int)p & 0xFF))
	{
	    exit (__LINE__);
	}
    }
    eeprom_read_block (s, (const void *)5, 5);
    if (memcmp (s, "\x90\xAB\xCD\xEF\x01", 3))
	exit (__LINE__);

    return 0;
}