2021-08-29 14:18:45 +02:00
|
|
|
#include <hex/data_processor/node.hpp>
|
|
|
|
|
2021-08-29 22:15:18 +02:00
|
|
|
#include <hex/helpers/fmt.hpp>
|
|
|
|
|
2022-02-01 18:09:40 +01:00
|
|
|
#include <hex/api/localization.hpp>
|
|
|
|
#include <hex/providers/provider.hpp>
|
|
|
|
|
2021-08-29 14:18:45 +02:00
|
|
|
namespace hex::dp {
|
|
|
|
|
2022-10-06 09:14:46 +02:00
|
|
|
int Node::s_idCounter = 1;
|
2022-02-01 18:09:40 +01:00
|
|
|
|
|
|
|
Node::Node(std::string unlocalizedTitle, std::vector<Attribute> attributes) : m_id(Node::s_idCounter++), m_unlocalizedTitle(std::move(unlocalizedTitle)), m_attributes(std::move(attributes)) {
|
2021-08-29 14:18:45 +02:00
|
|
|
for (auto &attr : this->m_attributes)
|
|
|
|
attr.setParentNode(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<u8> Node::getBufferOnInput(u32 index) {
|
|
|
|
auto attribute = this->getConnectedInputAttribute(index);
|
|
|
|
|
|
|
|
if (attribute == nullptr)
|
2022-01-15 14:14:53 +01:00
|
|
|
throwNodeError(hex::format("Nothing connected to input '{0}'", LangEntry(this->m_attributes[index].getUnlocalizedName())));
|
2021-08-29 14:18:45 +02:00
|
|
|
|
|
|
|
if (attribute->getType() != Attribute::Type::Buffer)
|
|
|
|
throwNodeError("Tried to read buffer from non-buffer attribute");
|
|
|
|
|
|
|
|
markInputProcessed(index);
|
|
|
|
attribute->getParentNode()->process();
|
|
|
|
|
|
|
|
auto &outputData = attribute->getOutputData();
|
|
|
|
|
|
|
|
if (!outputData.has_value())
|
|
|
|
throw std::runtime_error("No data available at connected attribute");
|
|
|
|
|
|
|
|
return outputData.value();
|
|
|
|
}
|
|
|
|
|
2022-03-27 00:01:28 +01:00
|
|
|
i64 Node::getIntegerOnInput(u32 index) {
|
2021-08-29 14:18:45 +02:00
|
|
|
auto attribute = this->getConnectedInputAttribute(index);
|
|
|
|
|
|
|
|
if (attribute == nullptr)
|
2022-01-15 14:14:53 +01:00
|
|
|
throwNodeError(hex::format("Nothing connected to input '{0}'", LangEntry(this->m_attributes[index].getUnlocalizedName())));
|
2021-08-29 14:18:45 +02:00
|
|
|
|
|
|
|
if (attribute->getType() != Attribute::Type::Integer)
|
|
|
|
throwNodeError("Tried to read integer from non-integer attribute");
|
|
|
|
|
|
|
|
markInputProcessed(index);
|
|
|
|
attribute->getParentNode()->process();
|
|
|
|
|
|
|
|
auto &outputData = attribute->getOutputData();
|
|
|
|
|
|
|
|
if (!outputData.has_value())
|
|
|
|
throw std::runtime_error("No data available at connected attribute");
|
|
|
|
|
|
|
|
if (outputData->size() < sizeof(u64))
|
|
|
|
throw std::runtime_error("Not enough data provided for integer");
|
|
|
|
|
2022-03-27 00:01:28 +01:00
|
|
|
return *reinterpret_cast<i64 *>(outputData->data());
|
2021-08-29 14:18:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
float Node::getFloatOnInput(u32 index) {
|
|
|
|
auto attribute = this->getConnectedInputAttribute(index);
|
|
|
|
|
|
|
|
if (attribute == nullptr)
|
2022-01-15 14:14:53 +01:00
|
|
|
throwNodeError(hex::format("Nothing connected to input '{0}'", LangEntry(this->m_attributes[index].getUnlocalizedName())));
|
2021-08-29 14:18:45 +02:00
|
|
|
|
|
|
|
if (attribute->getType() != Attribute::Type::Float)
|
|
|
|
throwNodeError("Tried to read float from non-float attribute");
|
|
|
|
|
|
|
|
markInputProcessed(index);
|
|
|
|
attribute->getParentNode()->process();
|
|
|
|
|
|
|
|
auto &outputData = attribute->getOutputData();
|
|
|
|
|
|
|
|
if (!outputData.has_value())
|
|
|
|
throw std::runtime_error("No data available at connected attribute");
|
|
|
|
|
|
|
|
if (outputData->size() < sizeof(float))
|
|
|
|
throw std::runtime_error("Not enough data provided for float");
|
|
|
|
|
2022-03-27 00:01:28 +01:00
|
|
|
float result = 0;
|
|
|
|
std::memcpy(&result, outputData->data(), sizeof(float));
|
|
|
|
return result;
|
2021-08-29 14:18:45 +02:00
|
|
|
}
|
|
|
|
|
2022-03-04 20:52:39 +01:00
|
|
|
void Node::setBufferOnOutput(u32 index, const std::vector<u8> &data) {
|
2021-08-29 14:18:45 +02:00
|
|
|
if (index >= this->getAttributes().size())
|
|
|
|
throw std::runtime_error("Attribute index out of bounds!");
|
|
|
|
|
|
|
|
auto &attribute = this->getAttributes()[index];
|
|
|
|
|
|
|
|
if (attribute.getIOType() != Attribute::IOType::Out)
|
|
|
|
throw std::runtime_error("Tried to set output data of an input attribute!");
|
|
|
|
|
|
|
|
attribute.getOutputData() = data;
|
|
|
|
}
|
|
|
|
|
2022-03-27 00:01:28 +01:00
|
|
|
void Node::setIntegerOnOutput(u32 index, i64 integer) {
|
2021-08-29 14:18:45 +02:00
|
|
|
if (index >= this->getAttributes().size())
|
|
|
|
throw std::runtime_error("Attribute index out of bounds!");
|
|
|
|
|
|
|
|
auto &attribute = this->getAttributes()[index];
|
|
|
|
|
|
|
|
if (attribute.getIOType() != Attribute::IOType::Out)
|
|
|
|
throw std::runtime_error("Tried to set output data of an input attribute!");
|
|
|
|
|
|
|
|
std::vector<u8> buffer(sizeof(u64), 0);
|
|
|
|
std::memcpy(buffer.data(), &integer, sizeof(u64));
|
|
|
|
|
|
|
|
attribute.getOutputData() = buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Node::setFloatOnOutput(u32 index, float floatingPoint) {
|
|
|
|
if (index >= this->getAttributes().size())
|
|
|
|
throw std::runtime_error("Attribute index out of bounds!");
|
|
|
|
|
|
|
|
auto &attribute = this->getAttributes()[index];
|
|
|
|
|
|
|
|
if (attribute.getIOType() != Attribute::IOType::Out)
|
|
|
|
throw std::runtime_error("Tried to set output data of an input attribute!");
|
|
|
|
|
|
|
|
std::vector<u8> buffer(sizeof(float), 0);
|
|
|
|
std::memcpy(buffer.data(), &floatingPoint, sizeof(float));
|
|
|
|
|
|
|
|
attribute.getOutputData() = buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Node::setOverlayData(u64 address, const std::vector<u8> &data) {
|
|
|
|
if (this->m_overlay == nullptr)
|
|
|
|
throw std::runtime_error("Tried setting overlay data on a node that's not the end of a chain!");
|
|
|
|
|
|
|
|
this->m_overlay->setAddress(address);
|
|
|
|
this->m_overlay->getData() = data;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|