// 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 .
//
// -----------------------------------------------------------------------------
//
// Audio output. Supports PWM (through a PwmOutput object) and DAC (through a
// Dac object, for example the one defined in mcp492x.h).
#ifndef AVRLIB_AUDIO_OUTPUT_H_
#define AVRLIB_AUDIO_OUTPUT_H_
#include "avrlib/base.h"
#include "avrlib/avrlib.h"
#include "avrlib/ring_buffer.h"
namespace avrlib {
enum UnderrunPolicy {
  EMIT_CLICK = 0,
  HOLD_SAMPLE = 1
};
template
class AudioOutput {
 public:
  AudioOutput() { }
  enum {
    buffer_size = buffer_size_,
    data_size = OutputPort::data_size,
  };
  typedef AudioOutput Me;
  typedef typename DataTypeForSize::Type Value;
  typedef RingBuffer OutputBuffer;
  static inline void Init() {
    OutputPort::Init();
  }
  static inline void Write(Value v) { while (!writable()); Overwrite(v); }
  static inline void Overwrite(Value v) { OutputBuffer::Overwrite(v); }
  static inline uint8_t writable() { return OutputBuffer::writable(); }
  static inline uint8_t writable_block() {
    return OutputBuffer::writable() >= block_size;
  }
  static inline uint8_t NonBlockingWrite(Value v) {
    if (!writable()) {
      return 0;
    }
    Overwrite(v);
    return 1;
  }
  
  static inline void DiscardSample() {
    OutputBuffer::ImmediateRead();
  }
  // Called from data emission interrupt.
  static inline void EmitSample() {
    if (OutputBuffer::readable()) {
      OutputPort::Write(Value(OutputBuffer::ImmediateRead()));
    } else {
      ++num_glitches_;
      if (underrun_policy == EMIT_CLICK) {
        // Introduces clicks to allow underruns to be easily detected.
        OutputPort::Write(0);
      }
    }
  }
  static inline uint8_t num_glitches() { return num_glitches_; }
  static inline void ResetGlitchCounter() { num_glitches_ = 0; }
 private:
  static uint8_t num_glitches_;
  DISALLOW_COPY_AND_ASSIGN(AudioOutput);
};
/* static */
template
uint8_t AudioOutput::num_glitches_ = 0;
}  // namespace avrlib
#endif  // AVRLIB_AUDIO_OUTPUT_H_