2007-09-03 02:33:47 +02:00
|
|
|
|
/*
|
2024-12-06 23:43:18 +01:00
|
|
|
|
NodeFeed(Test) - verify render node data feeds
|
2010-12-17 23:28:49 +01:00
|
|
|
|
|
Copyright: clarify and simplify the file headers
* Lumiera source code always was copyrighted by individual contributors
* there is no entity "Lumiera.org" which holds any copyrights
* Lumiera source code is provided under the GPL Version 2+
== Explanations ==
Lumiera as a whole is distributed under Copyleft, GNU General Public License Version 2 or above.
For this to become legally effective, the ''File COPYING in the root directory is sufficient.''
The licensing header in each file is not strictly necessary, yet considered good practice;
attaching a licence notice increases the likeliness that this information is retained
in case someone extracts individual code files. However, it is not by the presence of some
text, that legally binding licensing terms become effective; rather the fact matters that a
given piece of code was provably copyrighted and published under a license. Even reformatting
the code, renaming some variables or deleting parts of the code will not alter this legal
situation, but rather creates a derivative work, which is likewise covered by the GPL!
The most relevant information in the file header is the notice regarding the
time of the first individual copyright claim. By virtue of this initial copyright,
the first author is entitled to choose the terms of licensing. All further
modifications are permitted and covered by the License. The specific wording
or format of the copyright header is not legally relevant, as long as the
intention to publish under the GPL remains clear. The extended wording was
based on a recommendation by the FSF. It can be shortened, because the full terms
of the license are provided alongside the distribution, in the file COPYING.
2024-11-17 23:42:55 +01:00
|
|
|
|
Copyright (C)
|
2024-12-06 23:43:18 +01:00
|
|
|
|
2025, Hermann Vosseler <Ichthyostega@web.de>
|
2010-12-17 23:28:49 +01:00
|
|
|
|
|
Copyright: clarify and simplify the file headers
* Lumiera source code always was copyrighted by individual contributors
* there is no entity "Lumiera.org" which holds any copyrights
* Lumiera source code is provided under the GPL Version 2+
== Explanations ==
Lumiera as a whole is distributed under Copyleft, GNU General Public License Version 2 or above.
For this to become legally effective, the ''File COPYING in the root directory is sufficient.''
The licensing header in each file is not strictly necessary, yet considered good practice;
attaching a licence notice increases the likeliness that this information is retained
in case someone extracts individual code files. However, it is not by the presence of some
text, that legally binding licensing terms become effective; rather the fact matters that a
given piece of code was provably copyrighted and published under a license. Even reformatting
the code, renaming some variables or deleting parts of the code will not alter this legal
situation, but rather creates a derivative work, which is likewise covered by the GPL!
The most relevant information in the file header is the notice regarding the
time of the first individual copyright claim. By virtue of this initial copyright,
the first author is entitled to choose the terms of licensing. All further
modifications are permitted and covered by the License. The specific wording
or format of the copyright header is not legally relevant, as long as the
intention to publish under the GPL remains clear. The extended wording was
based on a recommendation by the FSF. It can be shortened, because the full terms
of the license are provided alongside the distribution, in the file COPYING.
2024-11-17 23:42:55 +01:00
|
|
|
|
**Lumiera** 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. See the file COPYING for further details.
|
2010-12-17 23:28:49 +01:00
|
|
|
|
|
Copyright: clarify and simplify the file headers
* Lumiera source code always was copyrighted by individual contributors
* there is no entity "Lumiera.org" which holds any copyrights
* Lumiera source code is provided under the GPL Version 2+
== Explanations ==
Lumiera as a whole is distributed under Copyleft, GNU General Public License Version 2 or above.
For this to become legally effective, the ''File COPYING in the root directory is sufficient.''
The licensing header in each file is not strictly necessary, yet considered good practice;
attaching a licence notice increases the likeliness that this information is retained
in case someone extracts individual code files. However, it is not by the presence of some
text, that legally binding licensing terms become effective; rather the fact matters that a
given piece of code was provably copyrighted and published under a license. Even reformatting
the code, renaming some variables or deleting parts of the code will not alter this legal
situation, but rather creates a derivative work, which is likewise covered by the GPL!
The most relevant information in the file header is the notice regarding the
time of the first individual copyright claim. By virtue of this initial copyright,
the first author is entitled to choose the terms of licensing. All further
modifications are permitted and covered by the License. The specific wording
or format of the copyright header is not legally relevant, as long as the
intention to publish under the GPL remains clear. The extended wording was
based on a recommendation by the FSF. It can be shortened, because the full terms
of the license are provided alongside the distribution, in the file COPYING.
2024-11-17 23:42:55 +01:00
|
|
|
|
* *****************************************************************/
|
2007-09-03 02:33:47 +02:00
|
|
|
|
|
2024-12-06 23:43:18 +01:00
|
|
|
|
/** @file node-feed-test.cpp
|
|
|
|
|
|
** Feeding into and retrieving data from render nodes is covered by \ref NodeFeed_test.
|
2016-11-03 18:20:10 +01:00
|
|
|
|
*/
|
|
|
|
|
|
|
2007-09-03 02:33:47 +02:00
|
|
|
|
|
2008-12-18 04:47:41 +01:00
|
|
|
|
#include "lib/test/run.hpp"
|
2024-12-26 21:42:32 +01:00
|
|
|
|
#include "steam/engine/proc-node.hpp"
|
|
|
|
|
|
#include "steam/engine/node-builder.hpp"
|
2024-12-28 21:41:08 +01:00
|
|
|
|
#include "steam/engine/media-weaving-pattern.hpp"
|
2024-12-28 23:16:55 +01:00
|
|
|
|
#include "steam/engine/param-weaving-pattern.hpp"
|
2024-12-26 21:42:32 +01:00
|
|
|
|
#include "steam/engine/turnout-system.hpp"
|
|
|
|
|
|
#include "steam/engine/turnout.hpp"
|
|
|
|
|
|
#include "steam/engine/diagnostic-buffer-provider.hpp"
|
2024-12-29 03:23:59 +01:00
|
|
|
|
#include "steam/asset/meta/time-grid.hpp"
|
2024-12-26 21:42:32 +01:00
|
|
|
|
#include "lib/several-builder.hpp"
|
2024-12-29 03:23:59 +01:00
|
|
|
|
#include "lib/time/timecode.hpp"
|
2024-12-30 01:56:18 +01:00
|
|
|
|
#include "lib/test/test-helper.hpp"
|
2024-12-26 21:42:32 +01:00
|
|
|
|
#include "lib/test/diagnostic-output.hpp"/////////////////////TODO
|
2008-12-18 04:47:41 +01:00
|
|
|
|
//#include "lib/util.hpp"
|
2007-09-03 02:33:47 +02:00
|
|
|
|
|
|
|
|
|
|
|
2016-01-07 03:58:29 +01:00
|
|
|
|
//using std::string;
|
2024-12-26 21:42:32 +01:00
|
|
|
|
using lib::Several;
|
|
|
|
|
|
using lib::makeSeveral;
|
2024-12-29 03:23:59 +01:00
|
|
|
|
using lib::time::Time;
|
|
|
|
|
|
using lib::time::FSecs;
|
2024-12-30 01:56:18 +01:00
|
|
|
|
using lib::time::FrameNr;
|
|
|
|
|
|
using lib::test::showType;
|
2007-09-03 02:33:47 +02:00
|
|
|
|
|
|
|
|
|
|
|
2018-11-15 23:55:13 +01:00
|
|
|
|
namespace steam {
|
2009-08-31 00:49:08 +02:00
|
|
|
|
namespace engine{
|
|
|
|
|
|
namespace test {
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-12-07 18:15:44 +01:00
|
|
|
|
/************************************************************************//**
|
|
|
|
|
|
* @test demonstrate how to feed data into, through and out of render nodes.
|
2009-08-31 00:49:08 +02:00
|
|
|
|
*/
|
2024-12-06 23:43:18 +01:00
|
|
|
|
class NodeFeed_test : public Test
|
2007-09-03 02:33:47 +02:00
|
|
|
|
{
|
2024-12-07 18:15:44 +01:00
|
|
|
|
virtual void
|
|
|
|
|
|
run (Arg)
|
2009-08-31 00:49:08 +02:00
|
|
|
|
{
|
2024-12-30 01:56:18 +01:00
|
|
|
|
seedRand();
|
2024-12-07 18:15:44 +01:00
|
|
|
|
feedParam();
|
2024-12-28 21:41:08 +01:00
|
|
|
|
feedParamNode();
|
2018-11-15 21:13:52 +01:00
|
|
|
|
UNIMPLEMENTED ("render node pulling source data from vault");
|
2024-12-07 18:15:44 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-12-26 21:42:32 +01:00
|
|
|
|
|
|
|
|
|
|
/** @test demonstrate internal setup to invoke a simple output-only function,
|
|
|
|
|
|
* passing an additional invocation parameter generated from a parameter-functor
|
|
|
|
|
|
* - embed the processing-functor and parameter-functor into a FeedPrototype
|
|
|
|
|
|
* - construct the type of the »Weaving Pattern« to use for invocation
|
|
|
|
|
|
* - setup an empty wiring (output-only, thus no predecessor ports)
|
|
|
|
|
|
* - setup a single BuffDesrc for a result puffer to pass to the processing-functor
|
|
|
|
|
|
* - create a Turnout, which implements the Port interface, using the Weaving-Pattern
|
|
|
|
|
|
* - for the actual invocation, setup a TurnoutSystem, initialised with a nominal time
|
|
|
|
|
|
* - invoke the Port::weave() function and retrieve the result from the buffer.
|
|
|
|
|
|
* @remark this is a semi-integrated setup to demonstrate the interplay of the
|
|
|
|
|
|
* internal components within a Render Node, without the _outer shell_
|
|
|
|
|
|
* provided by the NodeBuilder and the ProcNode itself
|
|
|
|
|
|
*/
|
2024-12-07 18:15:44 +01:00
|
|
|
|
void
|
|
|
|
|
|
feedParam()
|
2024-12-26 21:42:32 +01:00
|
|
|
|
{
|
|
|
|
|
|
auto procFun = [](ushort param, uint* buff){ *buff = param; };
|
|
|
|
|
|
auto paramFun = [](TurnoutSystem&){ return LIFE_AND_UNIVERSE_4EVER; };
|
|
|
|
|
|
|
|
|
|
|
|
auto feedPrototype = FeedPrototype{move(procFun), move(paramFun)};
|
|
|
|
|
|
using Prototype = decltype(feedPrototype);
|
|
|
|
|
|
using WeavingPattern = MediaWeavingPattern<Prototype>;
|
|
|
|
|
|
using TurnoutWeaving = Turnout<WeavingPattern>;
|
|
|
|
|
|
|
|
|
|
|
|
BufferProvider& provider = DiagnosticBufferProvider::build();
|
|
|
|
|
|
|
|
|
|
|
|
Several<PortRef> noLeadPorts; // ◁————————— empty predecessor-port-sequence
|
|
|
|
|
|
Several<BuffDescr> outBuffDescr = makeSeveral({provider.getDescriptor<uint>()})
|
|
|
|
|
|
.build(); // ◁————————— a single output buffer to hold an `uint`
|
|
|
|
|
|
uint resultSlot{0};
|
|
|
|
|
|
|
|
|
|
|
|
TurnoutWeaving port{ProcID::describe ("SimpleNode","procFun()")
|
|
|
|
|
|
, move (noLeadPorts)
|
|
|
|
|
|
, move (outBuffDescr)
|
|
|
|
|
|
, resultSlot
|
|
|
|
|
|
, move (feedPrototype)
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// setup for invocation...
|
|
|
|
|
|
Time nomTime =Time::ZERO;
|
|
|
|
|
|
TurnoutSystem turnoutSys{nomTime};
|
|
|
|
|
|
BuffHandle result = port.weave (turnoutSys); // ◁————————— paramFun invoked here, then procFun
|
|
|
|
|
|
CHECK (LIFE_AND_UNIVERSE_4EVER == result.accessAs<uint>());// and procFun wrote param-value into result buffer
|
|
|
|
|
|
result.release();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-12-30 01:56:18 +01:00
|
|
|
|
|
|
|
|
|
|
/** @test create extended parameter data for use in recursive Node invocation.
|
|
|
|
|
|
* - demonstrate the mechanism of param-functor invocation,
|
|
|
|
|
|
* and how a Param-Spec is built to create and hold those functors
|
|
|
|
|
|
* - then instantiate an actual TurnoutSystem, as is done for a Node invocation,
|
|
|
|
|
|
* with an embedded »absolute nominal time« parameter value
|
|
|
|
|
|
* - can then invoke the param-functors and materialise results into a Param-Data-Block
|
|
|
|
|
|
* - which then can be linked internally to be reachable through the TurnoutSystem
|
|
|
|
|
|
* - other code further down the call-stack can thus access those parameter values.
|
|
|
|
|
|
* - The second part of the test uses the same scheme embedded into a Param(Agent)Node
|
|
|
|
|
|
*/
|
2024-12-26 21:42:32 +01:00
|
|
|
|
void
|
|
|
|
|
|
feedParamNode()
|
2024-12-07 18:15:44 +01:00
|
|
|
|
{
|
2024-12-29 03:23:59 +01:00
|
|
|
|
steam::asset::meta::TimeGrid::build("grid_sec", 1);
|
|
|
|
|
|
|
2024-12-30 01:56:18 +01:00
|
|
|
|
// Parameter-functor based on time-quantisation into a 1-seconds-grid
|
2024-12-29 03:23:59 +01:00
|
|
|
|
auto fun1 = [](TurnoutSystem& turSys)
|
|
|
|
|
|
{
|
2024-12-30 01:56:18 +01:00
|
|
|
|
return FrameNr::quant (turSys.getNomTime(), "grid_sec");
|
2024-12-29 03:23:59 +01:00
|
|
|
|
};
|
|
|
|
|
|
|
2024-12-30 01:56:18 +01:00
|
|
|
|
// The Param-Spec is used to coordinate type-safe access
|
|
|
|
|
|
// and also is used as a blueprint for building a Param(Agent)Node
|
2024-12-29 03:23:59 +01:00
|
|
|
|
auto spec = buildParamSpec()
|
|
|
|
|
|
.addValSlot (LIFE_AND_UNIVERSE_4EVER)
|
|
|
|
|
|
.addSlot (move (fun1))
|
|
|
|
|
|
;
|
|
|
|
|
|
|
2024-12-30 01:56:18 +01:00
|
|
|
|
// The implied type of the parameter-tuple to generate
|
|
|
|
|
|
using ParamTup = decltype(spec)::ParamTup;
|
|
|
|
|
|
CHECK (showType<ParamTup>() == "tuple<uint, long>"_expect);
|
|
|
|
|
|
|
|
|
|
|
|
// can now store accessor-functors for later use....
|
|
|
|
|
|
auto acc0 = spec.slot<0>().makeAccessor();
|
|
|
|
|
|
auto acc1 = spec.slot<1>().makeAccessor();
|
|
|
|
|
|
|
|
|
|
|
|
// drive test with a random »nominal Time« <10s with ms granularity
|
|
|
|
|
|
Time nomTime{rani(10'000),0};
|
|
|
|
|
|
TurnoutSystem turnoutSys{nomTime};
|
|
|
|
|
|
// can now immediately invoke the embedded parameter-functors
|
2024-12-29 03:23:59 +01:00
|
|
|
|
auto v0 = spec.slot<0>().invokeParamFun (turnoutSys);
|
|
|
|
|
|
auto v1 = spec.slot<1>().invokeParamFun (turnoutSys);
|
2024-12-30 01:56:18 +01:00
|
|
|
|
CHECK (v0 == LIFE_AND_UNIVERSE_4EVER); // ◁————————— the first paramFun yields the configured fixed value
|
|
|
|
|
|
CHECK (v1 == FrameNr::quant (nomTime, "grid_sec")); // ◁————————— the second paramFun accesses the time in TurnoutSystem
|
|
|
|
|
|
|
|
|
|
|
|
{ // Now build an actual storage block in local scope,
|
|
|
|
|
|
// thereby invoking the embedded parameter-functors...
|
|
|
|
|
|
auto paramBlock = spec.buildParamDataBlock (turnoutSys);
|
|
|
|
|
|
// Values are now materialised into paramBlock
|
|
|
|
|
|
CHECK (v0 == paramBlock.get<0>());
|
|
|
|
|
|
CHECK (v1 == paramBlock.get<1>());
|
|
|
|
|
|
|
|
|
|
|
|
// link this extension block into the parameter-chain in TurnoutSystem
|
|
|
|
|
|
turnoutSys.attachChainBlock(paramBlock);
|
|
|
|
|
|
|
|
|
|
|
|
// can now access the parameter values through the TurnoutSystem as front-End
|
|
|
|
|
|
CHECK (v0 == spec.slot<0>().getParamVal (turnoutSys));
|
|
|
|
|
|
CHECK (v1 == spec.slot<1>().getParamVal (turnoutSys));
|
|
|
|
|
|
// and can also use the accessor-functors stored above
|
|
|
|
|
|
CHECK (v0 == turnoutSys.get(acc0));
|
|
|
|
|
|
CHECK (v1 == turnoutSys.get(acc1));
|
|
|
|
|
|
|
|
|
|
|
|
// should detach extension block before leaving scope
|
|
|
|
|
|
turnoutSys.detachChainBlock(paramBlock);
|
|
|
|
|
|
}
|
2024-12-29 18:27:05 +01:00
|
|
|
|
|
2024-12-07 18:15:44 +01:00
|
|
|
|
TODO ("implement a simple Builder for ParamAgent-Node");
|
|
|
|
|
|
TODO ("then use both together to demonstrate a param data feed here");
|
|
|
|
|
|
}
|
2009-08-31 00:49:08 +02:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/** Register this test class... */
|
2024-12-06 23:43:18 +01:00
|
|
|
|
LAUNCHER (NodeFeed_test, "unit node");
|
2009-08-31 00:49:08 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2018-11-15 23:55:13 +01:00
|
|
|
|
}}} // namespace steam::engine::test
|