|
- // Copyright 2009 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/>.
- //
- // -----------------------------------------------------------------------------
- //
- // Important: All buffer sizes are expected to be less than 256! (fit in 8
- // bits), and must be powers of 2.
-
- #ifndef AVRLIB_AVRLIB_H_
- #define AVRLIB_AVRLIB_H_
-
- #include <avr/io.h>
-
- #include "avrlib/base.h"
- #include "avrlib/size_to_type.h"
-
- namespace avrlib {
-
- enum DataOrder {
- MSB_FIRST = 0,
- LSB_FIRST = 1
- };
-
- enum DigitalValue {
- LOW = 0,
- HIGH = 1
- };
-
- // <avr/io.h> is full of useful defines, but they cannot be used as template
- // arguments because they are of the form: (*(volatile uint8_t *)(0x80))
- // The following define wraps this reference into a class to make it easier to
- // pass it as a template argument.
- #define IORegister(reg) struct reg##Register { \
- static volatile uint8_t* ptr() { return ® } \
- reg##Register& operator=(const uint8_t& value) { *ptr() = value; } \
- uint8_t operator()(const uint8_t& value) { return *ptr(); } \
- };
-
- #define IORegister16(reg) struct reg##Register { \
- static volatile uint16_t* ptr() { return ® } \
- reg##Register& operator=(const uint16_t& value) { *ptr() = value; } \
- uint16_t operator()(const uint16_t& value) { return *ptr(); } \
- };
-
- #define SpecialFunctionRegister(reg) struct reg##Register { \
- static volatile uint8_t* ptr() { return &_SFR_BYTE(reg); } \
- reg##Register& operator=(const uint8_t& value) { *ptr() = value; } \
- uint8_t operator()(const uint8_t& value) { return *ptr(); } \
- };
-
- #define SpecialFunctionRegister16(reg) struct reg##Register { \
- static volatile uint16_t* ptr() { return &_SFR_WORD(reg); } \
- reg##Register& operator=(const uint16_t& value) { *ptr() = value; } \
- uint16_t operator()(const uint16_t& value) { return *ptr(); } \
- };
-
-
- // Represents a bit in an i/o port register.
- template<typename Register, uint8_t bit>
- struct BitInRegister {
- static void clear() {
- *Register::ptr() &= ~_BV(bit);
- }
- static void set() {
- *Register::ptr() |= _BV(bit);
- }
- static void toggle() {
- *Register::ptr() ^= _BV(bit);
- }
- static uint8_t value() {
- return *Register::ptr() & _BV(bit) ? 1 : 0;
- }
- };
-
- // These classes implement/define the basic input/output interface. The default
- // implementation is that of an infinite stream of incoming/outgoing 0s.
- struct Input {
- enum {
- buffer_size = 0, // Recommended buffer size, when using buffered input.
- data_size = 0, // 0 for disabled port, 1 for digital, 8 for byte.
- };
- typedef uint8_t Value;
-
- // Blocking!
- static inline Value Read() { while (!readable()); return ImmediateRead(); }
-
- // Number of bytes available for read.
- static inline uint8_t readable() { return 1; }
-
- // A byte, or -1 if reading failed.
- static inline int16_t NonBlockingRead() { return readable() ? Read() : -1; }
-
- // No check for ready state.
- static inline Value ImmediateRead() { return 0; }
-
- // Called in data reception interrupt.
- static inline void Received() { }
- };
-
- struct Output {
- enum {
- buffer_size = 0, // Recommended buffer size, when using buffered output.
- data_size = 0 // 0 for disabled port, 1 for digital, 8 for byte.
- };
- typedef uint8_t Value;
-
- // Blocking!
- static inline void Write(Value v) { while (!writable()); Overwrite(v); }
-
- // Number of bytes that can be fed.
- static inline uint8_t writable() { return 1; }
-
- // 1 if success.
- static inline uint8_t NonBlockingWrite(Value v) {
- if (!writable()) {
- return 0;
- }
- Overwrite(v);
- return 1;
- }
-
- // No check for ready state.
- static inline void Overwrite(Value) { return; }
-
- // Called in data emission interrupt.
- static inline Value Requested() { return 0; }
- };
-
- // An object capable both of input and output, composed from an Input and an
- // Output implementation.
- template<typename I, typename O>
- struct InputOutput {
- typedef I Input;
- typedef O Output;
-
- static inline void Write(typename O::Value v) { O::Write(v); }
- static inline uint8_t writable() { return O::writable(); }
- static inline uint8_t NonBlockingWrite(typename O::Value v ) {
- return O::NonBlockingWrite(v);
- }
- static inline void Overwrite(typename O::Value v) { O::Overwrite(v); }
- static inline typename O::Value Requested() { return O::Requested(); }
- static inline typename I::Value Read() { return I::Read(); }
- static inline uint8_t readable() { return I::readable(); }
- static inline int16_t NonBlockingRead() { return I::NonBlockingRead(); }
- static inline typename I::Value ImmediateRead() { return I::ImmediateRead(); }
- static inline void Received() { I::Received(); }
- };
-
-
- // Dummy class that can be passed whenever we expect Input/Output types, and
- // which do not perform any IO.
- typedef Input DisabledInput;
- typedef Output DisabledOutput;
- typedef InputOutput<DisabledInput, DisabledOutput> DisabledInputOutput;
-
- enum PortMode {
- DISABLED = 0,
- POLLED = 1,
- BUFFERED = 2
- };
-
- // Some classes (SPI, shift register) have a notion of communication session -
- // Begin is called, several R/W are done, and then End is called to pull high
- // a chip select or latch line. This template ensures that any path leaving a
- // block of code will release the resource.
- template<typename T>
- class scoped_resource {
- public:
- scoped_resource() {
- T::Begin();
- }
-
- ~scoped_resource() {
- T::End();
- }
- };
-
- } // namespace avrlib
-
- #endif // AVRLIB_AVRLIB_H_
|