4 Copyright 2008, 2009 Michel Pollet <buserror@gmail.com>
6 This file is part of simavr.
8 simavr is free software: you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation, either version 3 of the License, or
11 (at your option) any later version.
13 simavr is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
18 You should have received a copy of the GNU General Public License
19 along with simavr. If not, see <http://www.gnu.org/licenses/>.
22 #ifndef __AVR_MCU_SECTION_H__
23 #define __AVR_MCU_SECTION_H__
26 * This header is used to pass "parameters" to the programmer or the simulator,
27 * it tags the ELF file with a section that contains parameters about the physical
28 * AVR this was compiled for, including the speed, model, and signature bytes.
30 * A programmer software can read this and verify fuses values for example, and a
31 * simulator can instanciate the proper "model" of AVR, the speed and so on without
32 * command line parameters.
36 * #include "avr_mcu_section.h"
37 * AVR_MCU(F_CPU, "atmega88");
50 AVR_MMCU_TAG_FREQUENCY,
57 AVR_MMCU_TAG_SIGNATURE,
58 AVR_MMCU_TAG_SIMAVR_COMMAND,
59 AVR_MMCU_TAG_VCD_FILENAME,
60 AVR_MMCU_TAG_VCD_PERIOD,
61 AVR_MMCU_TAG_VCD_TRACE,
66 SIMAVR_CMD_VCD_START_TRACE,
67 SIMAVR_CMD_VCD_STOP_TRACE,
68 SIMAVR_CMD_UART_LOOPBACK,
73 #define _MMCU_ __attribute__((section(".mmcu")))
74 struct avr_mmcu_long_t {
78 } __attribute__((__packed__));
80 struct avr_mmcu_string_t {
84 } __attribute__((__packed__));
86 struct avr_mmcu_addr_t {
90 } __attribute__((__packed__));
92 struct avr_mmcu_vcd_trace_t {
98 } __attribute__((__packed__));
100 #define AVR_MCU_STRING(_tag, _str) \
101 const struct avr_mmcu_string_t _##_tag _MMCU_ = {\
103 .len = sizeof(_str),\
107 #define AVR_MCU_LONG(_tag, _val) \
108 const struct avr_mmcu_long_t _##_tag _MMCU_ = {\
110 .len = sizeof(uint32_t),\
114 #define AVR_MCU_BYTE(_tag, _val) \
115 const uint8_t _##_tag _MMCU_ = { _tag, 1, _val }
117 #define AVR_MCU_VCD_SYMBOL(_name) \
118 .tag = AVR_MMCU_TAG_VCD_TRACE, \
119 .len = sizeof(struct avr_mmcu_vcd_trace_t) - 2 + sizeof(_name),\
122 // specified the name and wanted period (usec) for a VCD file
123 // thid is not mandatory, a default one will be created if
124 // symbols are declared themselves
125 #define AVR_MCU_VCD_FILE(_name, _period) \
126 AVR_MCU_STRING(AVR_MMCU_TAG_VCD_FILENAME, _name);\
127 AVR_MCU_LONG(AVR_MMCU_TAG_VCD_PERIOD, _period)
129 // It is possible to send "commands" to simavr from the
130 // firmware itself. For this to work you need to specify
131 // an IO register that is to be used for a write-only
132 // bridge. A favourite is one of the usual "GPIO register"
133 // that most (all ?) AVR have
134 #define AVR_MCU_SIMAVR_COMMAND(_register) \
135 const struct avr_mmcu_addr_t _simavr_command_register _MMCU_ = {\
136 .tag = AVR_MMCU_TAG_SIMAVR_COMMAND,\
137 .len = sizeof(void *),\
138 .what = (void*)_register, \
142 * This the has to be used if you want to add other tags to the .mmcu section
143 * the _mmcu symbol is used as an anchor to make sure it stays linked in.
145 #define AVR_MCU(_speed, _name) \
146 const uint8_t _mmcu[2] _MMCU_ = { AVR_MMCU_TAG, 0 }; \
147 AVR_MCU_STRING(AVR_MMCU_TAG_NAME, _name);\
148 AVR_MCU_LONG(AVR_MMCU_TAG_FREQUENCY, _speed)
150 #define AVR_MCU_VOLTAGES(_vcc, _avcc, _aref) \
151 AVR_MCU_LONG(AVR_MMCU_TAG_VCC, (_vcc));\
152 AVR_MCU_LONG(AVR_MMCU_TAG_AVCC, (_avcc));\
153 AVR_MCU_LONG(AVR_MMCU_TAG_AREF, (_aref));