Newer
Older
Martin Christoph Hierholzer
committed
/*
* ScalarAccessor.h
*
* Created on: Jun 07, 2016
* Author: Martin Hierholzer
*/
Martin Christoph Hierholzer
committed
#ifndef CHIMERATK_SCALAR_ACCESSOR_H
#define CHIMERATK_SCALAR_ACCESSOR_H
#include <string>
#include <boost/smart_ptr/shared_ptr.hpp>
Martin Christoph Hierholzer
committed
#include <boost/thread.hpp>
Martin Christoph Hierholzer
committed
Martin Christoph Hierholzer
committed
#include <mtca4u/ScalarRegisterAccessor.h>
#include "Module.h"
#include "VariableNetworkNode.h"
Martin Christoph Hierholzer
committed
namespace ChimeraTK {
Martin Christoph Hierholzer
committed
/** Accessor for scalar variables (i.e. single values). Note for users: Use the convenience classes
* ScalarPollInput, ScalarPushInput, ScalarOutput instead of this class directly. */
Martin Christoph Hierholzer
committed
template< typename UserType >
Martin Christoph Hierholzer
committed
class ScalarAccessor : public mtca4u::ScalarRegisterAccessor<UserType> {
Martin Christoph Hierholzer
committed
public:
Martin Christoph Hierholzer
committed
Martin Christoph Hierholzer
committed
/** Change meta data (name, unit, description and optionally tags). This function may only be used on
* Application-type nodes. If the optional argument tags is omitted, the tags will not be changed. To clear the
* tags, an empty set can be passed. */
Martin Christoph Hierholzer
committed
void setMetaData(const std::string &name, const std::string &unit, const std::string &description) {
node.setMetaData(name,unit,description);
}
Martin Christoph Hierholzer
committed
void setMetaData(const std::string &name, const std::string &unit, const std::string &description,
const std::unordered_set<std::string> &tags) {
node.setMetaData(name,unit,description,tags);
}
Martin Christoph Hierholzer
committed
Martin Christoph Hierholzer
committed
/** Convert into VariableNetworkNode */
operator VariableNetworkNode() {
return node;
Martin Christoph Hierholzer
committed
}
Martin Christoph Hierholzer
committed
Martin Christoph Hierholzer
committed
/** Connect with other node */
VariableNetworkNode operator>>(const VariableNetworkNode &otherNode) {
return node >> otherNode;
Martin Christoph Hierholzer
committed
}
Martin Christoph Hierholzer
committed
/** Replace with other ScalarRegisterAccessor */
void replace(const ScalarAccessor<UserType> &newAccessor) {
mtca4u::NDRegisterAccessorBridge<UserType>::replace(newAccessor);
node = VariableNetworkNode(this, newAccessor.node.getName(), newAccessor.node.getDirection(), newAccessor.node.getUnit(),
newAccessor.node.getNumberOfElements(), newAccessor.node.getMode(), newAccessor.node.getDescription(),
&newAccessor.node.getValueType());
if(_owner != newAccessor._owner) {
if(_owner != nullptr) _owner->unregisterAccessor(*this);
Martin Christoph Hierholzer
committed
_owner = newAccessor._owner;
Martin Christoph Hierholzer
committed
}
void replace(const mtca4u::NDRegisterAccessorBridge<UserType> &newAccessor) = delete;
Martin Christoph Hierholzer
committed
Martin Christoph Hierholzer
committed
using mtca4u::ScalarRegisterAccessor<UserType>::operator=;
Martin Christoph Hierholzer
committed
Martin Christoph Hierholzer
committed
/** Move-assignment operator as an alternative for replace where applicable. This is needed to allow late
* initialisation of ApplicationModules using ScalarAccessors */
ScalarAccessor<UserType>& operator=(ScalarAccessor<UserType> &&rhs) {
mtca4u::NDRegisterAccessorBridge<UserType>::replace(rhs);
node.pdata = rhs.node.pdata;
node.pdata->appNode = this;
return *this;
}
Martin Christoph Hierholzer
committed
~ScalarAccessor() {
if(_owner != nullptr) _owner->unregisterAccessor(*this);
Martin Christoph Hierholzer
committed
}
Martin Christoph Hierholzer
committed
Martin Christoph Hierholzer
committed
/** Add a tag. Valid names for tags only contain alpha-numeric characters (i.e. no spaces and no special
* characters). */
void addTag(const std::string &tag) {
node.addTag(tag);
}
Martin Christoph Hierholzer
committed
protected:
Martin Christoph Hierholzer
committed
ScalarAccessor(Module *owner, const std::string &name, VariableDirection direction, std::string unit,
Martin Christoph Hierholzer
committed
UpdateMode mode, const std::string &description, const std::unordered_set<std::string> &tags={})
: node(this, name, direction, unit, 1, mode, description, &typeid(UserType), tags), _owner(owner)
Martin Christoph Hierholzer
committed
{
owner->registerAccessor(*this);
}
/** Default constructor creates a dysfunctional accessor (to be assigned with a real accessor later) */
ScalarAccessor() {}
Martin Christoph Hierholzer
committed
VariableNetworkNode node;
Martin Christoph Hierholzer
committed
Martin Christoph Hierholzer
committed
Module *_owner{nullptr};
Martin Christoph Hierholzer
committed
};
Martin Christoph Hierholzer
committed
/** Convenience class for input scalar accessors with UpdateMode::push */
template< typename UserType >
struct ScalarPushInput : public ScalarAccessor<UserType> {
Martin Christoph Hierholzer
committed
ScalarPushInput(Module *owner, const std::string &name, std::string unit,
const std::string &description, const std::unordered_set<std::string> &tags={})
: ScalarAccessor<UserType>(owner, name, VariableDirection::consuming, unit, UpdateMode::push,
description, tags)
Martin Christoph Hierholzer
committed
{}
ScalarPushInput() : ScalarAccessor<UserType>() {}
using ScalarAccessor<UserType>::operator=;
};
/** Convenience class for input scalar accessors with UpdateMode::poll */
template< typename UserType >
struct ScalarPollInput : public ScalarAccessor<UserType> {
Martin Christoph Hierholzer
committed
ScalarPollInput(Module *owner, const std::string &name, std::string unit,
const std::string &description, const std::unordered_set<std::string> &tags={})
: ScalarAccessor<UserType>(owner, name, VariableDirection::consuming, unit, UpdateMode::poll,
description, tags)
Martin Christoph Hierholzer
committed
{}
ScalarPollInput() : ScalarAccessor<UserType>() {}
Martin Christoph Hierholzer
committed
void doReadTransfer() override { this->doReadTransferLatest(); }
void read() { this->readLatest(); }
Martin Christoph Hierholzer
committed
using ScalarAccessor<UserType>::operator=;
};
/** Convenience class for output scalar accessors (always UpdateMode::push) */
template< typename UserType >
struct ScalarOutput : public ScalarAccessor<UserType> {
Martin Christoph Hierholzer
committed
ScalarOutput(Module *owner, const std::string &name, std::string unit,
const std::string &description, const std::unordered_set<std::string> &tags={})
: ScalarAccessor<UserType>(owner, name, VariableDirection::feeding, unit, UpdateMode::push,
description, tags)
Martin Christoph Hierholzer
committed
{}
ScalarOutput() : ScalarAccessor<UserType>() {}
using ScalarAccessor<UserType>::operator=;
};
Martin Christoph Hierholzer
committed
} /* namespace ChimeraTK */
#endif /* CHIMERATK_SCALAR_ACCESSOR_H */