Memory Architectures
Harvard vs Princeton
In the early days of electronic computing, two different processor/memory architectures emerged:
The Von Neumann (http://adafru.it/coe) (a.k.a.
(http://adafru.it/coe)Princeton (http://adafru.it/coe))
(http://adafru.it/coe)architecture (http://adafru.it/coe) developed for the ENIAC
(http://adafru.it/cof)uses the same memory and data paths for both program and data storage.
The Harvard architecture (http://adafru.it/cog) characterized by the the Harvard Mark
Adafruit
https://learn.adafruit.com/memories-of-an-arduino Page 5 of 26
Industries
1 (http://adafru.it/coh) (http://adafru.it/coh)used physically separate memory and data paths for
program and memory.
Which is better?
Each architecture has its advantages: All else being equal, the Harvard model has the edge in
performance. The Von Neumann model is more flexible.
Modern Hybrids
These days, most general purpose computers (PC's Mac's etc.) are hybrid designs that give you the
best of both architectures. Deep within the CPU they operate on the Harvard model using separate
caches for instructions and data to maximize performance. But the instruction and data caches are
both loaded automatically from a common memory space. From a programming perspective, these
computers appear to be pure Von Neumann machines with many gigabytes of virtual storage.
Microcontrollers
Microcontrollers such as the ones that power the Arduinos are designed for embedded applications.
Unlike general purpose computers, an embedded processor typically has a well defined task that it
must perform reliably and efficiently - and at minimal cost, Microcontroller designs tend to be rather
spartan. They forego the luxuries of multi-layer caching and disk-based virtual memory systems and
stick to what is essential to the task.
The Harvard model turns out to be a good match for embedded applications and the Atmega 328
Adafruit
https://learn.adafruit.com/memories-of-an-arduino Page 6 of 26
Industries
used in the Arduino UNO use a relatively pure Harvard architecture. Programs are stored in Flash
memory and data is stored in SRAM.
For the most part, the compiler and run-time systems take care of managing these for you, but when
things start getting tight, it helps to be aware of how things work under the hood. And things start
getting tight much quicker on these tiny machines!
A completely different Scale
The biggest difference between these microcontrollers and your general purpose computer is the
sheer amount of memory available. The Arduino UNO has only 32K bytes of Flash memory and 2K
bytes of SRAM. That is more than 100,000 times LESS physical memory than a low-end PC! And
that's not even counting the disk drive!
Working in this minimalist environment, you must use your resources wisely.
Adafruit
https://learn.adafruit.com/memories-of-an-arduino Page 7 of 26
Industries
Arduino Memories
There are 3 types of memory in an Arduino:
Flash or Program Memory
SRAM
EEPROM
Flash Memory
Flash memory is used to store your program image and any initialized data. You can execute program
code from flash, but you can't modify data in flash memory from your executing code. To modify the
data, it must first be copied into SRAM
Flash memory is the same technology used for thumb-drives and SD cards. It is non-volatile, so your
program will still be there when the system is powered off.
Flash memory has a finite lifetime of about 100,000 write cycles. So if you upload 10 programs a day,
every day for the next 27 years, you might wear it out.
SRAM
SRAM or Static Random Access Memory, can be read and written from your executing program.
SRAM memory is used for several purposes by a running program:
Static Data - This is a block of reserved space in SRAM for all the global and static variables
from your program. For variables with initial values, the runtime system copies the initial value
from Flash when the program starts.
Heap - The heap is for dynamically allocated data items. The heap grows from the top of the
static data area up as data items are allocated.
Stack - The stack is for local variables and for maintaining a record of interrupts and function
calls. The stack grows from the top of memory down towards the heap. Every interrupt, function
call and/or local variable allocation causes the stack to grow. Returning from an interrupt or
function call will reclaim all stack space used by that interrupt or function.
Adafruit
https://learn.adafruit.com/memories-of-an-arduino Page 8 of 26
Industries
Most memory problems occur when the stack and the heap collide. When this happens, one or both of
these memory areas will be corrupted with unpredictable results. In some cases it will cause an
immediate crash. In others, the effects of the corruption may not be noticed until much later.
EEPROM
EEPROM is another form of non-volatile memory that can be read or written from your executing
program. It can only be read byte-by-byte, so it can be a little awkward to use. It is also slower than
SRAM and has a finite lifetime of about 100,000 write cycles (you can read it as many times as you
want).
While it can't take the place of precious SRAM, there are times when it can be very useful!
Adafruit
https://learn.adafruit.com/memories-of-an-arduino Page 9 of 26
Industries
Adafruit
https://learn.adafruit.com/memories-of-an-arduino Page 10 of 26
Industries
Arduino Memory Comparision
The following chart shows the amounts of each type of memory for several Arduino and Arduino
compatible boards.
"...that's not got much SRAM in it."
Adafruit
https://learn.adafruit.com/memories-of-an-arduino Page 11 of 26
Industries
Measuring Memory Usage
One way to diagnose memory problems is to measure how much memory is in use.
Flash
Measuring Flash memory usage is trivial. The compiler does that for you, every time you compile!
EEPROM
You are 100% in control of EEPROM usage. You have to read and write each byte to a specific
address, so there is no excuse for not knowing exactly which bytes are in use!
// ************************************************
// Write floating point values to EEPROM
// ************************************************
void EEPROM_writeDouble(int address, double value)
{
byte* p = (byte*)(void*)&value;
for (int i = 0; i < sizeof(value); i++)
{
EEPROM.write(address++, *p++);
}
}
// ************************************************
// Read floating point values from EEPROM
// ************************************************
double EEPROM_readDouble(int address)
{
double value = 0.0;
byte* p = (byte*)(void*)&value;
for (int i = 0; i < sizeof(value); i++)
{
Adafruit
https://learn.adafruit.com/memories-of-an-arduino Page 12 of 26
Industries
*p++ = EEPROM.read(address++);
}
return value;
}
SRAM
SRAM usage is more dynamic and therefore more difficult to measure. The free_ram() function below
is one way to do this. You can add this function definition to your code, then call it from various places
in your code to report the amount of free SRAM.
SRAM utilization is dynamic and will change over time. So It is important to call free_ram() at various
times and from various places in your sketch to see how it changes over time.
int freeRam ()
{
extern int __heap_start, *__brkval;
int v;
return (int) &v - (__brkval == 0 ? (int) &__heap_start : (int) __brkval);
}
What free_ram() is actually reporting is the space between the heap and the stack. it does not report
any de-allocated memory that is buried in the heap. Buried heap space is not usable by the stack, and
may be fragmented enough that it is not usable for many heap allocations either. The space between
the heap and the stack is what you really need to monitor if you are trying to avoid stack crashes.
Adafruit
https://learn.adafruit.com/memories-of-an-arduino Page 13 of 26
Industries
Adafruit
https://learn.adafruit.com/memories-of-an-arduino Page 14 of 26
Industries
Large Memory Consumers
There are devices and drivers which require large amounts of SRAM to operate. Some of the largest
memory consumers are:
SD Cards
Anything with an SD or Micro-SD interface requires a 512 byte SRAM buffer to
communicate with the card.
Pixels
Each pixel requires just 3 bytes of SRAM to store the color. But those bytes
start to add up when you have many meters of strip or a large array.
On an Uno, you might be able to drive as many as 500 pixels - assuming you
don't use much SRAM for anything else.
Adafruit
https://learn.adafruit.com/memories-of-an-arduino Page 15 of 26
Industries