// 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_