Skip to content
GitLab
Explore
Sign in
Primary navigation
Search or go to…
Project
Gaudi
Manage
Activity
Members
Labels
Plan
Issues
Issue boards
Milestones
Requirements
Jira
Code
Merge requests
Repository
Branches
Commits
Tags
Repository graph
Compare revisions
Locked files
Build
Pipelines
Jobs
Pipeline schedules
Test cases
Artifacts
Deploy
Releases
Model registry
Operate
Environments
Monitor
Incidents
Analyze
Value stream analytics
Contributor analytics
CI/CD analytics
Repository analytics
Code review analytics
Issue analytics
Insights
Model experiments
Help
Help
Support
GitLab documentation
Compare GitLab plans
Community forum
Contribute to GitLab
Provide feedback
Keyboard shortcuts
?
Snippets
Groups
Projects
Show more breadcrumbs
Hadrien Benjamin Grasland
Gaudi
Merge requests
!1
WIP: Data handle rework
Code
Review changes
Check out branch
Download
Patches
Plain diff
Closed
WIP: Data handle rework
new-data-handles
into
opensuse-hacks
Overview
31
Commits
41
Pipelines
0
Changes
16
Closed
Hadrien Benjamin Grasland
requested to merge
new-data-handles
into
opensuse-hacks
7 years ago
Overview
25
Commits
41
Pipelines
0
Changes
16
Expand
Moved to the Gaudi MR
gaudi/Gaudi!462 (merged)
.
Edited
7 years ago
by
Hadrien Benjamin Grasland
0
0
Merge request reports
Compare
opensuse-hacks
version 15
3b39fab9
7 years ago
version 14
c8845687
7 years ago
version 13
7be3f045
7 years ago
version 12
d9e38509
7 years ago
version 11
d9e38509
7 years ago
version 10
b3c1b232
7 years ago
version 9
e35ed148
7 years ago
version 8
660399ee
7 years ago
version 7
ce6790ab
7 years ago
version 6
cb9929ea
7 years ago
version 5
04a25136
7 years ago
version 4
832c7458
7 years ago
version 3
b1484213
7 years ago
version 2
bb503513
7 years ago
version 1
e2a8cd4f
7 years ago
opensuse-hacks (base)
and
latest version
latest version
446d5257
41 commits,
7 years ago
version 15
3b39fab9
40 commits,
7 years ago
version 14
c8845687
39 commits,
7 years ago
version 13
7be3f045
38 commits,
7 years ago
version 12
d9e38509
37 commits,
7 years ago
version 11
d9e38509
81 commits,
7 years ago
version 10
b3c1b232
81 commits,
7 years ago
version 9
e35ed148
37 commits,
7 years ago
version 8
660399ee
24 commits,
7 years ago
version 7
ce6790ab
21 commits,
7 years ago
version 6
cb9929ea
20 commits,
7 years ago
version 5
04a25136
18 commits,
7 years ago
version 4
832c7458
17 commits,
7 years ago
version 3
b1484213
11 commits,
7 years ago
version 2
bb503513
10 commits,
7 years ago
version 1
e2a8cd4f
6 commits,
7 years ago
16 files
+
1125
−
8
Inline
Compare changes
Side-by-side
Inline
Show whitespace changes
Show one file at a time
Files
16
Search (e.g. *.vue) (Ctrl+P)
GaudiExamples/src/DataHandles/DataFlowTestAlgs.cpp
0 → 100644
+
173
−
0
Options
#include
<array>
#include
<stdexcept>
#include
<string>
#include
<utility>
#include
"GaudiKernel/DataObjID.h"
#include
"GaudiKernel/Algorithm.h"
#include
"GaudiKernel/EventDataHandle.h"
// ============================================================================
/** @file
* Test harness checking that the new DataHandles declare data flow correctly
*
* The following data flow graph is being tested. It happens to be one of
* the simplest graphs that checks all useful combinations of 0, 1, and 2
* inputs and outputs (ie anything but no inputs and no outputs).
*
* In the schematic below, named boxes are algorithms and numbers are the
* values that get passed around between them, which double as graph edge
* identifiers. This simplifies implementation, at the cost of breaking
* reconfiguration of the data flow graph from the Python test script. For
* testing code, I think this is a fine trade-off.
*
* +---+ 1 +---+ 2 +---+ 3 +---+
* | A |-->| |--------->| |-->| C | A: 0 inputs and 1 output
* +---+ | | | | +---+ B: 0 inputs and 2 outputs
* | H | | E | C: 1 input and 0 outputs
* +---+ 4 | | 5 +---+ | | 6 +---+ D: 1 input and 1 output
* | |-->| |-->| | | |-->| | E: 1 input and 2 outputs
* | | +---+ | | +---+ | | F: 2 inputs and 0 outputs
* | B | | G | | F | G: 2 inputs and 1 output
* | | 7 +---+ 8 | | 9 | | H: 2 inputs and 2 outputs
* | |-->| D |-->| |--------->| |
* +---+ +---+ +---+ +---+
*
* @author Hadrien GRASLAND
* @date 2017-10-05
*/
// ============================================================================
// You may notice that in the graph above, all algorithms will be implemented
// in a similar way. All they do is to make sure that inputs and outputs are
// declared properly, and send/check dummy data around.
//
// So ideally, we would like to write a single algorithm template that we can
// instantiate at will for each of our eight test algorithms, like this:
//
// using AlgE = DataflowTestAlg<{2}, {3, 6}>;
//
// However, this is not valid C++ today, and we will need to go through some
// ugly metaprogramming tricks to get something similar.
//
// Let's start with what C++ does provides today, namely std::integer_sequence.
// That name is too long for humans to type repeatedly, so let's shorten it.
//
template
<
int
...
ids
>
using
IDs
=
std
::
integer_sequence
<
int
,
ids
...
>
;
//
// Can we now make sure that algorithms get instantiated using syntax like
// "using AlgE = DataflowTestAlg<IDs<2>, IDs<3, 6>>"? Yes, we can. But due to
// more unfortunate C++ limitations, we will also need to use specialization
// for this purpose instead of a pure template declaration.
//
template
<
typename
Inputs
,
typename
Outputs
>
class
DataHandleFlowAlg
;
//
template
<
int
...
inputIDs
,
int
...
outputIDs
>
class
DataHandleFlowAlg
<
IDs
<
inputIDs
...
>
,
IDs
<
outputIDs
...
>>
:
public
Gaudi
::
experimental
::
Algorithm
{
public:
using
Gaudi
::
experimental
::
Algorithm
::
Algorithm
;
/// Run the algorithm, checking the data flow graph and I/O values
StatusCode
execute
()
override
{
// TODO: Once reentrant Algorithms land in Gaudi, use their explicit context
auto
eventContext
=
getContext
();
// Check that inputs were declared correctly
info
()
<<
"Checking input keys"
<<
endmsg
;
DataObjIDColl
expectedInputKeys
;
for
(
const
auto
&
inputHandle
:
m_readers
)
{
expectedInputKeys
.
emplace
(
inputHandle
.
id
());
}
if
(
eventInputKeys
()
!=
expectedInputKeys
)
{
throw
std
::
runtime_error
(
"Unexpected input keys"
);
}
// Check that outputs were declared correctly
info
()
<<
"Checking output keys"
<<
endmsg
;
DataObjIDColl
expectedOutputKeys
;
for
(
const
auto
&
outputHandle
:
m_writers
)
{
expectedOutputKeys
.
emplace
(
outputHandle
.
id
());
}
if
(
eventOutputKeys
()
!=
expectedOutputKeys
)
{
throw
std
::
runtime_error
(
"Unexpected output keys"
);
}
// Check that we receive the expected input values
info
()
<<
"Checking input values"
<<
endmsg
;
for
(
size_t
i
=
0
;
i
<
N_inputs
;
++
i
)
{
if
(
m_readers
[
i
].
get
(
eventContext
)
!=
m_inputValues
[
i
])
{
throw
std
::
runtime_error
(
"Unexpected input #"
+
std
::
to_string
(
i
));
}
}
// Send our output values
info
()
<<
"Sending output values"
<<
endmsg
;
for
(
size_t
i
=
0
;
i
<
N_outputs
;
++
i
)
{
const
auto
&
output
=
m_outputValues
[
i
];
if
(
m_writers
[
i
].
put
(
eventContext
,
output
)
!=
output
)
{
throw
std
::
runtime_error
(
"Unexpected output #"
+
std
::
to_string
(
i
));
}
}
return
StatusCode
::
SUCCESS
;
}
private
:
// Count our inputs and outputs
static
constexpr
size_t
N_inputs
=
sizeof
...(
inputIDs
);
static
constexpr
size_t
N_outputs
=
sizeof
...(
outputIDs
);
// Record our input and output IDs for future use as input and output values
std
::
array
<
int
,
N_inputs
>
m_inputValues
{
inputIDs
...};
std
::
array
<
int
,
N_outputs
>
m_outputValues
{
outputIDs
...};
// Declare our inputs using ReadHandles
using
IntReadHandle
=
Gaudi
::
experimental
::
EventReadHandle
<
int
>
;
std
::
array
<
IntReadHandle
,
N_inputs
>
m_readers
{
IntReadHandle
{
this
,
"Input"
+
std
::
to_string
(
inputIDs
),
DataObjID
(
"/Event/Int"
+
std
::
to_string
(
inputIDs
)
)
}...
};
// Declare our outputs using WriteHandles
using
IntWriteHandle
=
Gaudi
::
experimental
::
EventWriteHandle
<
int
>
;
std
::
array
<
IntWriteHandle
,
N_outputs
>
m_writers
{
IntWriteHandle
{
this
,
"Output"
+
std
::
to_string
(
outputIDs
),
DataObjID
(
"/Event/Int"
+
std
::
to_string
(
outputIDs
)
)
}...
};
};
// clang-format off
// Generate the algorithms used by the test | Inputs | Outputs |
using
DataHandleFlowAlgA
=
DataHandleFlowAlg
<
IDs
<>
,
IDs
<
1
>
>
;
using
DataHandleFlowAlgB
=
DataHandleFlowAlg
<
IDs
<>
,
IDs
<
4
,
7
>
>
;
using
DataHandleFlowAlgC
=
DataHandleFlowAlg
<
IDs
<
3
>
,
IDs
<>
>
;
using
DataHandleFlowAlgD
=
DataHandleFlowAlg
<
IDs
<
7
>
,
IDs
<
8
>
>
;
using
DataHandleFlowAlgE
=
DataHandleFlowAlg
<
IDs
<
2
>
,
IDs
<
3
,
6
>
>
;
using
DataHandleFlowAlgF
=
DataHandleFlowAlg
<
IDs
<
6
,
9
>
,
IDs
<>
>
;
using
DataHandleFlowAlgG
=
DataHandleFlowAlg
<
IDs
<
5
,
8
>
,
IDs
<
9
>
>
;
using
DataHandleFlowAlgH
=
DataHandleFlowAlg
<
IDs
<
1
,
4
>
,
IDs
<
2
,
5
>
>
;
// clang-format on
// Now, make the algorithms accessible via Python
# define DECLARE_COMPONENT_ALIAS( Alg ) DECLARE_COMPONENT_WITH_ID( Alg, #Alg )
DECLARE_COMPONENT_ALIAS
(
DataHandleFlowAlgA
)
DECLARE_COMPONENT_ALIAS
(
DataHandleFlowAlgB
)
DECLARE_COMPONENT_ALIAS
(
DataHandleFlowAlgC
)
DECLARE_COMPONENT_ALIAS
(
DataHandleFlowAlgD
)
DECLARE_COMPONENT_ALIAS
(
DataHandleFlowAlgE
)
DECLARE_COMPONENT_ALIAS
(
DataHandleFlowAlgF
)
DECLARE_COMPONENT_ALIAS
(
DataHandleFlowAlgG
)
DECLARE_COMPONENT_ALIAS
(
DataHandleFlowAlgH
)
Loading