// 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 . // // ----------------------------------------------------------------------------- // // 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 #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 }; // 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 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 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 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 class scoped_resource { public: scoped_resource() { T::Begin(); } ~scoped_resource() { T::End(); } }; } // namespace avrlib #endif // AVRLIB_AVRLIB_H_