|
- // Copyright 2011 Olivier Gillet.
- //
- // Author: Olivier Gillet (ol.gillet@gmail.com)
- //
- // This program is free software: you can redistribute it and/or modify
- // it under the terms of the GNU General Public License as published by
- // the Free Software Foundation, either version 3 of the License, or
- // (at your option) any later version.
- // This program is distributed in the hope that it will be useful,
- // but WITHOUT ANY WARRANTY; without even the implied warranty of
- // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- // GNU General Public License for more details.
- // You should have received a copy of the GNU General Public License
- // along with this program. If not, see <http://www.gnu.org/licenses/>.
-
- #ifndef AVRLIBX_IO_GPIO_H_
- #define AVRLIBX_IO_GPIO_H_
-
- #include <avr/io.h>
-
- #include "avrlibx/avrlibx.h"
-
- namespace avrlibx {
-
- enum PortDirection {
- INPUT,
- OUTPUT
- };
-
- enum PortMode {
- PORT_MODE_TOTEM_POLE,
- PORT_MODE_BUS_KEEPER,
- PORT_MODE_PULL_UP,
- PORT_MODE_PULL_DOWN
- };
-
- enum SenseMode {
- SENSE_MODE_BOTH_EDGES,
- SENSE_MODE_RISING,
- SENSE_MODE_FALLING,
- SENSE_MODE_LOW_LEVEL
- };
-
- #define WRAP_PORT(letter) \
- struct Port ## letter { \
- static inline PORT_t& port_t() { return PORT ## letter; } \
- static inline void dir(uint8_t value) { PORT ## letter ## _DIR = value; } \
- static inline void dir_clr(uint8_t value) { PORT ## letter ## _DIRCLR = value; } \
- static inline void dir_set(uint8_t value) { PORT ## letter ## _DIRSET = value; } \
- static inline void out(uint8_t value) { PORT ## letter ## _OUT = value; } \
- static inline void out_set(uint8_t value) { PORT ## letter ## _OUTSET = value; } \
- static inline void out_clr(uint8_t value) { PORT ## letter ## _OUTCLR = value; } \
- static inline void out_tgl(uint8_t value) { PORT ## letter ## _OUTTGL = value; } \
- static inline uint8_t in() { return PORT ## letter ## _IN; } \
- static inline uint8_t dir() { return PORT ## letter ## _DIR; } \
- static inline uint8_t out() { return PORT ## letter ## _OUT; } \
- static inline uint8_t event_base() { return EVSYS_CHMUX_PORT ## letter ## _PIN0_gc; } \
- };
-
- WRAP_PORT(A)
- WRAP_PORT(B)
- WRAP_PORT(C)
- WRAP_PORT(D)
- WRAP_PORT(E)
- #ifdef PORTF
- WRAP_PORT(F)
- #endif
-
- template<typename Port, uint8_t bit>
- struct Pin {
- static inline void set_control(uint8_t value) { }
- };
-
- #define SPECIALIZE_PIN(id) \
- template<typename Port> \
- struct Pin<Port, id> { \
- static inline void set_control(uint8_t value) { \
- Port::port_t().PIN ## id ## CTRL = value; \
- } \
- };
-
- SPECIALIZE_PIN(0)
- SPECIALIZE_PIN(1)
- SPECIALIZE_PIN(2)
- SPECIALIZE_PIN(3)
- SPECIALIZE_PIN(4)
- SPECIALIZE_PIN(5)
- SPECIALIZE_PIN(6)
- SPECIALIZE_PIN(7)
-
- template<typename Port, uint8_t bit, bool slow = true>
- struct Gpio {
- static inline void set_direction(PortDirection direction) {
- if (direction == INPUT) {
- Port::dir_clr(_BV(bit));
- } else {
- Port::dir_set(_BV(bit));
- if (slow) {
- Pin<Port, bit>::set_control(PORT_SRLEN_bm);
- }
- }
- }
-
- static inline void set_mode(PortMode mode) {
- if (mode == PORT_MODE_TOTEM_POLE) {
- Pin<Port, bit>::set_control(PORT_OPC_TOTEM_gc);
- } else if (mode == PORT_MODE_BUS_KEEPER) {
- Pin<Port, bit>::set_control(PORT_OPC_BUSKEEPER_gc);
- } else if (mode == PORT_MODE_PULL_UP) {
- Pin<Port, bit>::set_control(PORT_OPC_PULLUP_gc);
- } else if (mode == PORT_MODE_PULL_DOWN) {
- Pin<Port, bit>::set_control(PORT_OPC_PULLDOWN_gc);
- }
- }
-
- static inline void set_sense(SenseMode mode) {
- if (mode == SENSE_MODE_BOTH_EDGES) {
- Pin<Port, bit>::set_control(PORT_ISC_BOTHEDGES_gc);
- } else if (mode == SENSE_MODE_RISING) {
- Pin<Port, bit>::set_control(PORT_ISC_RISING_gc);
- } else if (mode == SENSE_MODE_FALLING) {
- Pin<Port, bit>::set_control(PORT_ISC_FALLING_gc);
- } else if (mode == SENSE_MODE_LOW_LEVEL) {
- Pin<Port, bit>::set_control(PORT_ISC_LEVEL_gc);
- }
- }
-
- static inline uint8_t event() {
- return Port::event_base() + bit;
- }
-
- static inline void High() {
- Port::out_set(_BV(bit));
- }
- static inline void Low() {
- Port::out_clr(_BV(bit));
- }
- static inline void Toggle() {
- Port::out_tgl(_BV(bit));
- }
-
- static inline void set_value(uint8_t value) {
- if (value) {
- Port::out_set(_BV(bit));
- } else {
- Port::out_clr(_BV(bit));
- }
- }
-
- static inline uint8_t value() {
- return Port::in() & _BV(bit) ? HIGH : LOW;
- }
-
- static inline void Write(uint8_t value) { set_value(value); }
- static inline uint8_t Read() { return value(); }
- };
-
- struct DummyGpio {
- static inline void set_direction(PortDirection direction) { }
- static inline void set_mode(PortDirection direction) { }
-
- static inline void High() { }
- static inline void Low() { }
- static inline void Toggle() { }
-
- static inline void set_value(uint8_t value) { }
- static inline uint8_t value() { return 0; }
- static inline void Write(uint8_t value) { set_value(value); }
- static inline uint8_t Read() { return value(); }
- };
-
- template<typename Gpio>
- struct Inverter {
- static inline void set_direction(PortDirection direction) {
- Gpio::set_direction(direction);
- }
- static inline void set_mode(PortDirection direction) {
- Gpio::set_mode(direction);
- }
-
- static inline void High() { Gpio::Low(); }
- static inline void Low() { Gpio::High(); }
- static inline void Toggle() { Gpio::Toggle(); }
-
- static inline void set_value(uint8_t value) { Gpio::set_value(!value); }
- static inline uint8_t value() { return !Gpio::value(); }
- static inline void Write(uint8_t value) { set_value(value); }
- static inline uint8_t Read() { return value(); }
- };
-
- } // namespace avrlibx
-
- #endif // AVRLIBX_IO_GPIO_H_
|