[locomotiv] Support Forward node (#3436)
author박천교/On-Device Lab(SR)/Engineer/삼성전자 <ch.bahk@samsung.com>
Mon, 13 May 2019 03:46:54 +0000 (12:46 +0900)
committer박세희/On-Device Lab(SR)/Principal Engineer/삼성전자 <saehie.park@samsung.com>
Mon, 13 May 2019 03:46:54 +0000 (12:46 +0900)
This commit starts to support loco::Forward node. In terms of node
execution, Forward calculation is same with Push's.

Signed-off-by: Cheongyo Bahk <ch.bahk@samsung.com>
contrib/locomotiv/src/Node.lst
contrib/locomotiv/src/Node/Forward.cpp [new file with mode: 0644]
contrib/locomotiv/src/Node/Forward.test.cpp [new file with mode: 0644]

index 0754a22..a7c689c 100644 (file)
@@ -4,5 +4,6 @@
 
 // NODE(Name) : alphabetic order please
 
+NODE(Forward)
 NODE(Pull)
 NODE(Push)
diff --git a/contrib/locomotiv/src/Node/Forward.cpp b/contrib/locomotiv/src/Node/Forward.cpp
new file mode 100644 (file)
index 0000000..1c22aa0
--- /dev/null
@@ -0,0 +1,58 @@
+/*
+ * Copyright (c) 2019 Samsung Electronics Co., Ltd. All Rights Reserved
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *    http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+#include "NodeExecution.h"
+
+#include "NodeDataImpl.h"
+
+#include <stdexcept>
+
+namespace locomotiv
+{
+
+void NodeExecution::execute(loco::Forward *forward)
+{
+  auto input_data = annot_data(forward->input());
+
+  if (!input_data)
+  {
+    throw std::runtime_error("Ingredient not ready");
+  }
+
+  switch (input_data->dtype())
+  {
+  case loco::DataType::S32:
+  {
+    auto input_bufptr = input_data->as_s32_bufptr();
+    auto forward_data = make_data(*input_bufptr);
+    erase_annot_data(forward);
+    annot_data(forward, std::move(forward_data));
+    break;
+  }
+  case loco::DataType::FLOAT32:
+  {
+    auto input_bufptr = input_data->as_f32_bufptr();
+    auto forward_data = make_data(*input_bufptr);
+    erase_annot_data(forward);
+    annot_data(forward, std::move(forward_data));
+    break;
+  }
+  default:
+    throw std::runtime_error("NYI for this DataType");
+  }
+}
+
+} // namespace locomotiv
diff --git a/contrib/locomotiv/src/Node/Forward.test.cpp b/contrib/locomotiv/src/Node/Forward.test.cpp
new file mode 100644 (file)
index 0000000..0e1c989
--- /dev/null
@@ -0,0 +1,83 @@
+/*
+ * Copyright (c) 2019 Samsung Electronics Co., Ltd. All Rights Reserved
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *    http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+#include "NodeExecution.h"
+
+#include "locomotiv/NodeData.h"
+#include "NodeDataImpl.h"
+
+#include <nncc/core/ADT/tensor/Shape.h>
+#include <nncc/core/ADT/tensor/Buffer.h>
+#include <nncc/core/ADT/tensor/LexicalLayout.h>
+
+#include <gtest/gtest.h>
+
+using nncc::core::ADT::tensor::Index;
+using nncc::core::ADT::tensor::Shape;
+using nncc::core::ADT::tensor::LexicalLayout;
+using nncc::core::ADT::tensor::make_buffer;
+
+TEST(NodeExecution_Forward, s32)
+{
+  // Make pull-forward graph
+  auto g = loco::make_graph();
+  auto pull = g->nodes()->create<loco::Pull>();
+  pull->dtype(loco::DataType::S32);
+  pull->rank(1);
+  pull->dim(0) = loco::make_dimension(1);
+  auto forward = g->nodes()->create<loco::Forward>();
+  forward->input(pull);
+
+  // Make and assign data to pull node
+  auto pull_buf = make_buffer<int32_t, LexicalLayout>(Shape{1});
+  pull_buf.at(Index{0}) = 42;
+  auto pull_data = locomotiv::make_data(pull_buf);
+  locomotiv::annot_data(pull, std::move(pull_data));
+
+  locomotiv::NodeExecution::get().run(forward);
+
+  auto forward_data = locomotiv::annot_data(forward);
+  ASSERT_NE(forward_data, nullptr);
+  ASSERT_EQ(forward_data->dtype(), loco::DataType::S32);
+  ASSERT_EQ(*(forward_data->shape()), Shape{1});
+  ASSERT_EQ(forward_data->as_s32_bufptr()->at(Index{0}), pull_buf.at(Index{0}));
+}
+
+TEST(NodeExecution_Forward, f32)
+{
+  // Make pull-forward graph
+  auto g = loco::make_graph();
+  auto pull = g->nodes()->create<loco::Pull>();
+  pull->dtype(loco::DataType::FLOAT32);
+  pull->rank(1);
+  pull->dim(0) = loco::make_dimension(1);
+  auto forward = g->nodes()->create<loco::Forward>();
+  forward->input(pull);
+
+  // Make and assign data to pull node
+  auto pull_buf = make_buffer<float, LexicalLayout>(Shape{1});
+  pull_buf.at(Index{0}) = 3.14f;
+  auto pull_data = locomotiv::make_data(pull_buf);
+  locomotiv::annot_data(pull, std::move(pull_data));
+
+  locomotiv::NodeExecution::get().run(forward);
+
+  auto forward_data = locomotiv::annot_data(forward);
+  ASSERT_NE(forward_data, nullptr);
+  ASSERT_EQ(forward_data->dtype(), loco::DataType::FLOAT32);
+  ASSERT_EQ(*(forward_data->shape()), Shape{1});
+  ASSERT_FLOAT_EQ(forward_data->as_f32_bufptr()->at(Index{0}), pull_buf.at(Index{0}));
+}