LUMIERA.clone/tests/components/proc/play/diagnostic-output-slot.hpp

320 lines
8.2 KiB
C++
Raw Normal View History

/*
DIAGNOSTIC-OUTPUT-SLOT.hpp - helper for testing against the OutputSlot interface
Copyright (C) Lumiera.org
2011, Hermann Vosseler <Ichthyostega@web.de>
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 2 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, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
/** @file diagnostic-output-slot.hpp
** An facility for writing unit-tests against the OutputSlot interface.
**
** @see output-slot-protocol-test.cpp
*/
#ifndef PROC_PLAY_DIAGNOSTIC_OUTPUT_SLOT_H
#define PROC_PLAY_DIAGNOSTIC_OUTPUT_SLOT_H
#include "lib/error.hpp"
#include "include/logging.h"
#include "proc/play/output-slot.hpp"
#include "proc/play/output-slot-connection.hpp"
#include "proc/engine/buffhandle.hpp"
#include "proc/engine/tracking-heap-block-provider.hpp"
#include "lib/scoped-ptrvect.hpp"
#include "lib/iter-source.hpp"
#include "proc/engine/testframe.hpp"
//#include "lib/sync.hpp"
#include <boost/noncopyable.hpp>
//#include <string>
//#include <vector>
#include <tr1/memory>
//#include <boost/scoped_ptr.hpp>
namespace proc {
namespace play {
//using std::string;
using proc::engine::BufferDescriptor;
using proc::engine::test::TestFrame;
using proc::engine::TrackingHeapBlockProvider;
namespace diagn = proc::engine::diagn;
//using std::vector;
using std::tr1::shared_ptr;
//using boost::scoped_ptr;
class TrackingInMemoryBlockSequence
: public OutputSlot::Connection
{
shared_ptr<TrackingHeapBlockProvider> buffProvider_;
BufferDescriptor bufferType_;
/* === Connection API === */
BuffHandle
claimBufferFor(FrameID frameNr)
{
return buffProvider_->lockBuffer (bufferType_);
}
bool
isTimely (FrameID frameNr, TimeValue currentTime)
{
if (Time::MAX == currentTime)
return true;
UNIMPLEMENTED ("find out about timings");
return false;
}
void
transfer (BuffHandle const& filledBuffer)
{
pushout (filledBuffer);
}
void
pushout (BuffHandle const& data4output)
{
buffProvider_->emitBuffer (data4output);
buffProvider_->releaseBuffer(data4output);
}
void
discard (BuffHandle const& superseededData)
{
buffProvider_->releaseBuffer (superseededData);
}
void
shutDown ()
{
buffProvider_.reset();
}
public:
TrackingInMemoryBlockSequence()
: buffProvider_(new TrackingHeapBlockProvider())
, bufferType_(buffProvider_->getDescriptor<TestFrame>())
{
INFO (engine_dbg, "building in-memory diagnostic output sequence");
}
virtual
~TrackingInMemoryBlockSequence()
{
INFO (engine_dbg, "releasing diagnostic output sequence");
}
/* === Diagnostic API === */
TestFrame*
accessEmittedFrame (uint frameNo) const
{
REQUIRE (buffProvider_);
if (frameNo <= buffProvider_->emittedCnt())
return &buffProvider_->accessAs<TestFrame> (frameNo);
else
return 0; ////////////////////////////////TICKET #856
}
diagn::Block*
accessEmittedBuffer (uint bufferNo) const
{
REQUIRE (buffProvider_);
if (bufferNo <= buffProvider_->emittedCnt())
return buffProvider_->access_emitted (bufferNo);
else
return 0;
}
};
class SimulatedOutputSequences
: public ConnectionStateManager<TrackingInMemoryBlockSequence>
{
TrackingInMemoryBlockSequence
buildConnection()
{
return TrackingInMemoryBlockSequence();
}
public:
SimulatedOutputSequences (uint numChannels)
{
init (numChannels);
}
};
/********************************************************************
* Helper for unit tests: Mock output sink.
*
* @todo write type comment
*/
class DiagnosticOutputSlot
: public OutputSlot
{
static const uint MAX_CHANNELS = 5;
/** hook into the OutputSlot frontend */
ConnectionState*
buildState()
{
return new SimulatedOutputSequences(MAX_CHANNELS);
}
/** @internal is self-managed and non-copyable.
* Clients use #build() to get an instance */
DiagnosticOutputSlot() { }
/** @internal access the implementation object
* representing a single stream connection
*/
TrackingInMemoryBlockSequence const&
accessSequence (uint channel)
{
REQUIRE (!isFree(), "diagnostic OutputSlot not (yet) connected");
REQUIRE (channel <= MAX_CHANNELS);
return static_cast<SimulatedOutputSequences&> (*state_).at(channel);
}
public:
2011-08-18 12:31:26 +02:00
/** build a new Diagnostic Output Slot instance,
* discard the existing one. Use the static query API
* for investigating collected data. */
static OutputSlot&
build()
{
static lib::ScopedPtrVect<OutputSlot> diagnosticSlots;
return diagnosticSlots.manage(new DiagnosticOutputSlot);
2011-08-18 12:31:26 +02:00
}
static DiagnosticOutputSlot&
access (OutputSlot& to_investigate)
{
return dynamic_cast<DiagnosticOutputSlot&> (to_investigate);
}
/* === diagnostics API === */
/**
* diagnostic facility to verify test data frames
* written to this Test/Dummy "output". It exposes
* the emitted Data as a sequence of TestFrame objects.
*/
class OutputFramesLog
: public lib::IterSource<TestFrame>
, boost::noncopyable
{
TrackingInMemoryBlockSequence const& outSeq_;
uint currentFrame_;
virtual Pos
firstResult ()
{
REQUIRE (0 == currentFrame_);
return outSeq_.accessEmittedFrame (currentFrame_);
}
virtual void
nextResult (Pos& pos)
{
++currentFrame_;
pos = outSeq_.accessEmittedFrame(currentFrame_);
}
public:
OutputFramesLog (TrackingInMemoryBlockSequence const& bs)
: outSeq_(bs)
, currentFrame_(0)
{ }
};
typedef OutputFramesLog::iterator OutFrames;
OutFrames
getChannel (uint channel)
{
REQUIRE (channel < MAX_CHANNELS);
return OutputFramesLog::build(
new OutputFramesLog (
accessSequence(channel)));
}
bool
buffer_was_used (uint channel, FrameID frame)
{
diagn::Block* block = accessSequence(channel)
.accessEmittedBuffer(frame);
return block
&& block->was_used();
}
bool
buffer_unused (uint channel, FrameID frame)
{
return !buffer_was_used(channel, frame);
}
bool
buffer_was_closed (uint channel, FrameID frame)
{
UNIMPLEMENTED ("determine if the specified buffer was indeed closed properly");
}
bool
emitted (uint channel, FrameID frame)
{
UNIMPLEMENTED ("determine if the specivied buffer was indeed handed over for emitting output");
}
private:
};
}} // namespace proc::play
#endif