[nnpackage_run] Introduce tensor_dumper (#6602)
author이상규/On-Device Lab(SR)/Principal Engineer/삼성전자 <sg5.lee@samsung.com>
Fri, 16 Aug 2019 04:39:51 +0000 (13:39 +0900)
committer이한종/On-Device Lab(SR)/Engineer/삼성전자 <hanjoung.lee@samsung.com>
Fri, 16 Aug 2019 04:39:51 +0000 (13:39 +0900)
- Introduce `dump` option in args.*
- Introduce tensor_dumper
- Use tensor_dumper in nnpackage_run

Signed-off-by: Sanggyu Lee <sg5.lee@samsung.com>
tests/tools/nnpackage_run/CMakeLists.txt
tests/tools/nnpackage_run/src/args.cc
tests/tools/nnpackage_run/src/args.h
tests/tools/nnpackage_run/src/nnpackage_run.cc
tests/tools/nnpackage_run/src/tensor_dumper.cc [new file with mode: 0644]
tests/tools/nnpackage_run/src/tensor_dumper.h [new file with mode: 0644]

index e1e50f7..6900b76 100644 (file)
@@ -8,6 +8,7 @@ endif(NOT BUILD_NEURUN)
 
 list(APPEND NNPACKAGE_RUN_SRCS "src/nnpackage_run.cc")
 list(APPEND NNPACKAGE_RUN_SRCS "src/args.cc")
+list(APPEND NNPACKAGE_RUN_SRCS "src/tensor_dumper.cc")
 
 nnfw_find_package(Boost REQUIRED)
 
index 30e276b..bcdbb91 100644 (file)
@@ -30,14 +30,14 @@ Args::Args(const int argc, char **argv) noexcept
 
 void Args::Initialize(void)
 {
-
   // General options
   po::options_description general("General options");
 
   // clang-format off
   general.add_options()
     ("help,h", "Display available options")
-    ("nnpackage", po::value<std::string>()->required());
+    ("nnpackage", po::value<std::string>()->required())
+    ("dump,d", po::value<std::string>()->default_value(""), "Output filename");
   // clang-format on
 
   _options.add(general);
@@ -71,6 +71,11 @@ void Args::Parse(const int argc, char **argv)
     exit(0);
   }
 
+  if (vm.count("dump"))
+  {
+    _dump_filename = vm["dump"].as<std::string>();
+  }
+
   if (vm.count("nnpackage"))
   {
     _package_filename = vm["nnpackage"].as<std::string>();
index e25d418..4268700 100644 (file)
@@ -32,6 +32,7 @@ public:
   void print(void);
 
   const std::string &getPackageFilename(void) const { return _package_filename; }
+  const std::string &getDumpFilename(void) const { return _dump_filename; }
 
 private:
   void Initialize();
@@ -42,6 +43,7 @@ private:
   po::options_description _options;
 
   std::string _package_filename;
+  std::string _dump_filename;
 };
 
 } // end of namespace NNPackageRun
index 9a6383a..aeb03b6 100644 (file)
@@ -16,6 +16,7 @@
 
 #include "args.h"
 #include "tflite/Diff.h"
+#include "tensor_dumper.h"
 #include "nnfw.h"
 
 #include <iostream>
@@ -120,6 +121,19 @@ int main(const int argc, char **argv)
   NNPR_ENSURE_STATUS(nnfw_run(session));
   run_ms = NowMicros() - run_ms;
 
+  // Dump output tensors
+  if (!args.getDumpFilename().empty())
+  {
+    NNPackageRun::TensorDumper dumper(args.getDumpFilename());
+    dumper.dumpInt32(num_outputs);
+    for (uint32_t i = 0; i < num_outputs; i++)
+    {
+      nnfw_tensorinfo ti;
+      NNPR_ENSURE_STATUS(nnfw_output_tensorinfo(session, i, &ti));
+      dumper.dumpTensor(ti, (void *)(outputs[i].data()), outputs[i].size() * sizeof(float));
+    }
+  }
+
   std::cout << "nnfw_prepare takes " << prepare_ms / 1e3 << " sec" << std::endl;
   std::cout << "nnfw_run     takes " << run_ms / 1e3 << " sec" << std::endl;
 
diff --git a/tests/tools/nnpackage_run/src/tensor_dumper.cc b/tests/tools/nnpackage_run/src/tensor_dumper.cc
new file mode 100644 (file)
index 0000000..ae4ed95
--- /dev/null
@@ -0,0 +1,54 @@
+/*
+ * 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 "tensor_dumper.h"
+
+#include <fstream>
+#include <iostream>
+#include <cstring>
+
+namespace NNPackageRun
+{
+TensorDumper::TensorDumper(const std::string &filename)
+{
+  // TODO Handle file open/write error
+  file_.open(filename, std::ios::out | std::ios::binary);
+  dumpInt32(version);
+}
+
+TensorDumper::~TensorDumper() { file_.close(); }
+
+void TensorDumper::dumpInt32(int32_t i)
+{
+  file_.write(reinterpret_cast<const char *>(&i), sizeof(i));
+}
+
+void TensorDumper::dumpSizeT(size_t i)
+{
+  file_.write(reinterpret_cast<const char *>(&i), sizeof(i));
+}
+
+void TensorDumper::dumpTensor(const nnfw_tensorinfo ti, void *buffer, size_t bytes)
+{
+  dumpInt32(ti.dtype);
+  dumpInt32(ti.rank);
+  for (uint i = 0; i < ti.rank; ++i)
+    dumpInt32(ti.dims[i]);
+  dumpSizeT(bytes);
+  file_.write(static_cast<char *>(buffer), bytes);
+}
+
+} // end of namespace NNPackageRun
\ No newline at end of file
diff --git a/tests/tools/nnpackage_run/src/tensor_dumper.h b/tests/tools/nnpackage_run/src/tensor_dumper.h
new file mode 100644 (file)
index 0000000..f274ad6
--- /dev/null
@@ -0,0 +1,31 @@
+#ifndef __NNPACKAGE_RUN_TENSOR_DUMPER_H__
+#define __NNPACKAGE_RUN_TENSOR_DUMPER_H__
+
+#include <memory>
+#include <string>
+#include <vector>
+#include <stddef.h>
+#include <fstream>
+
+#include "nnfw.h"
+
+namespace NNPackageRun
+{
+
+class TensorDumper
+{
+public:
+  TensorDumper(const std::string &filename);
+  void dumpTensor(const nnfw_tensorinfo ti, void *buffer, size_t bytes);
+  void dumpInt32(int32_t i);
+  void dumpSizeT(size_t i);
+  ~TensorDumper();
+
+private:
+  static constexpr int version = 1;
+  std::ofstream file_;
+};
+
+} // end of namespace NNPackageRun
+
+#endif // __NNPACKAGE_RUN_TENSOR_DUMPER_H__