2019-11-25 19:34:34 +00:00
|
|
|
#include "SqueezeNetValidator.h"
|
|
|
|
|
#include "protobufHelpers.h"
|
|
|
|
|
#include "fileHelpers.h"
|
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
|
#include <winrt/Windows.Media.h>
|
|
|
|
|
#include <winrt/Windows.Graphics.Imaging.h>
|
|
|
|
|
#include <winrt/Windows.Storage.h>
|
|
|
|
|
#include <winrt/Windows.Storage.Streams.h>
|
|
|
|
|
|
|
|
|
|
#include "WinMLProfiler.h"
|
|
|
|
|
|
|
|
|
|
// using namespace winrt::Windows::Foundation;
|
|
|
|
|
using namespace winrt::Windows::AI::MachineLearning;
|
|
|
|
|
using namespace winrt::Windows::Foundation::Collections;
|
|
|
|
|
using namespace winrt::Windows::Graphics::Imaging;
|
|
|
|
|
using namespace winrt::Windows::Media;
|
|
|
|
|
using namespace winrt::Windows::Storage;
|
|
|
|
|
using namespace winrt::Windows::Storage::Streams;
|
|
|
|
|
|
|
|
|
|
namespace WinML::Engine::Test{
|
|
|
|
|
|
|
|
|
|
enum WINML_RUNTIME_TEST_PERF
|
|
|
|
|
{
|
|
|
|
|
PREP_TEST = 0,
|
|
|
|
|
CREATE_RUNTIME,
|
|
|
|
|
LOAD_MODEL,
|
|
|
|
|
CREATE_EVAL_CONTEXT,
|
|
|
|
|
RUN_TEST,
|
|
|
|
|
BIND_VALUE,
|
|
|
|
|
EVAL_MODEL,
|
|
|
|
|
EVAL_MODEL_FIRST_RUN,
|
|
|
|
|
kCount
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static std::vector<std::string> WINML_RUNTIME_TEST_PERF_NAMES =
|
|
|
|
|
{
|
|
|
|
|
"PREP TEST ",
|
|
|
|
|
" CREATE RUNTIME ",
|
|
|
|
|
" LOAD MODEL ",
|
|
|
|
|
" CREATE EVAL CONTEXT",
|
|
|
|
|
"RUN TEST ",
|
|
|
|
|
" BIND VALUE ",
|
|
|
|
|
" EVAL MODEL ",
|
|
|
|
|
" EVAL MODEL 1st Run "
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
#define MAX_PROFILING_LOOP 100
|
|
|
|
|
Profiler<WINML_RUNTIME_TEST_PERF> g_RuntimeProfiler;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void BindImage(
|
|
|
|
|
LearningModelBinding binding,
|
|
|
|
|
const wchar_t* name,
|
|
|
|
|
const wchar_t* fullImagePath,
|
|
|
|
|
bool bindAsInspectable = false)
|
|
|
|
|
{
|
|
|
|
|
auto imagefile = StorageFile::GetFileFromPathAsync(fullImagePath).get();
|
|
|
|
|
auto stream = imagefile.OpenAsync(FileAccessMode::Read).get();
|
|
|
|
|
auto decoder = BitmapDecoder::CreateAsync(stream).get();
|
|
|
|
|
auto softwareBitmap = decoder.GetSoftwareBitmapAsync().get();
|
|
|
|
|
auto frame = VideoFrame::CreateWithSoftwareBitmap(softwareBitmap);
|
|
|
|
|
|
|
|
|
|
if (bindAsInspectable)
|
|
|
|
|
{
|
|
|
|
|
EXPECT_NO_THROW(binding.Bind(name, frame));
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
auto imagetensor = ImageFeatureValue::CreateFromVideoFrame(frame);
|
|
|
|
|
EXPECT_NO_THROW(binding.Bind(name, imagetensor));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void BindTensor(
|
|
|
|
|
LearningModelBinding binding,
|
|
|
|
|
const wchar_t* name,
|
|
|
|
|
ITensor inputTensor,
|
|
|
|
|
bool bindAsInspectable = false)
|
|
|
|
|
{
|
|
|
|
|
EXPECT_TRUE(inputTensor != nullptr);
|
|
|
|
|
|
|
|
|
|
if (bindAsInspectable)
|
|
|
|
|
{
|
|
|
|
|
EXPECT_NO_THROW(binding.Bind(name, inputTensor.as<TensorFloat>().GetAsVectorView()));
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
EXPECT_NO_THROW(binding.Bind(name, inputTensor));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
ITensor BindOutput(
|
|
|
|
|
OutputBindingStrategy strategy,
|
|
|
|
|
LearningModelBinding binding,
|
|
|
|
|
const wchar_t* name,
|
|
|
|
|
const IVectorView<int64_t> shape = nullptr
|
|
|
|
|
)
|
|
|
|
|
{
|
|
|
|
|
ITensor outputTensor = nullptr;
|
|
|
|
|
switch (strategy)
|
|
|
|
|
{
|
|
|
|
|
case OutputBindingStrategy::Bound:
|
|
|
|
|
outputTensor = T::Create(shape);
|
|
|
|
|
EXPECT_NO_THROW(binding.Bind(name, outputTensor));
|
|
|
|
|
break;
|
|
|
|
|
case OutputBindingStrategy::Empty:
|
|
|
|
|
outputTensor = T::Create();
|
|
|
|
|
EXPECT_NO_THROW(binding.Bind(name, outputTensor));
|
|
|
|
|
break;
|
|
|
|
|
case OutputBindingStrategy::Unbound:
|
|
|
|
|
__fallthrough;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return outputTensor;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ImageFeatureValue BindImageOutput(
|
|
|
|
|
OutputBindingStrategy strategy,
|
|
|
|
|
LearningModelBinding binding,
|
|
|
|
|
const wchar_t* name
|
|
|
|
|
)
|
|
|
|
|
{
|
|
|
|
|
ImageFeatureValue outputTensor = nullptr;
|
|
|
|
|
switch (strategy)
|
|
|
|
|
{
|
|
|
|
|
case OutputBindingStrategy::Bound:
|
|
|
|
|
{
|
|
|
|
|
SoftwareBitmap bitmap(BitmapPixelFormat::Bgra8, 720, 720);
|
|
|
|
|
VideoFrame frame = VideoFrame::CreateWithSoftwareBitmap(bitmap);
|
|
|
|
|
outputTensor = ImageFeatureValue::CreateFromVideoFrame(frame);
|
|
|
|
|
EXPECT_NO_THROW(binding.Bind(name, outputTensor));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case OutputBindingStrategy::Unbound:
|
|
|
|
|
__fallthrough;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return outputTensor;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ModelValidator::FnsCandy16(
|
|
|
|
|
std::string instance,
|
|
|
|
|
LearningModelDeviceKind deviceKind,
|
|
|
|
|
OutputBindingStrategy outputBindingStrategy,
|
|
|
|
|
bool bindInputsAsIInspectable,
|
|
|
|
|
float dataTolerance)
|
|
|
|
|
{
|
2019-11-27 23:04:47 +00:00
|
|
|
ORT_UNUSED_PARAMETER(dataTolerance);
|
2019-11-25 19:34:34 +00:00
|
|
|
WINML_PROFILING_START(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::PREP_TEST);
|
|
|
|
|
// file name strings
|
|
|
|
|
static wchar_t* modelFileName = L"winmlperf_coreml_FNS-Candy_prerelease_fp16.onnx";
|
|
|
|
|
static wchar_t* inputDataImageFileName = L"fish_720.png";
|
|
|
|
|
static wchar_t* outputDataFileName = L"output.png";
|
|
|
|
|
static wchar_t* inputBindingName = L"inputImage";
|
|
|
|
|
static const wchar_t* outputDataBindingName = L"outputImage";
|
|
|
|
|
|
|
|
|
|
auto modulePath = FileHelpers::GetModulePath();
|
|
|
|
|
auto fullModelPath = modulePath + modelFileName;
|
|
|
|
|
auto outputFileName = modulePath + outputDataFileName;
|
|
|
|
|
WINML_PROFILING_STOP(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::PREP_TEST);
|
|
|
|
|
|
|
|
|
|
WINML_PROFILING_START(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::LOAD_MODEL);
|
|
|
|
|
// WinML model creation
|
|
|
|
|
LearningModel model = nullptr;
|
|
|
|
|
EXPECT_NO_THROW(model = LearningModel::LoadFromFilePath(fullModelPath));
|
|
|
|
|
WINML_PROFILING_STOP(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::LOAD_MODEL);
|
|
|
|
|
|
|
|
|
|
WINML_PROFILING_START(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::RUN_TEST);
|
|
|
|
|
WINML_PROFILING_START(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::CREATE_EVAL_CONTEXT);
|
|
|
|
|
LearningModelSession modelSession = nullptr;
|
|
|
|
|
EXPECT_NO_THROW(modelSession = LearningModelSession(model, LearningModelDevice(deviceKind)));
|
|
|
|
|
WINML_PROFILING_STOP(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::CREATE_EVAL_CONTEXT);
|
|
|
|
|
|
|
|
|
|
WINML_PROFILING_START(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::BIND_VALUE);
|
|
|
|
|
LearningModelBinding modelBinding(modelSession);
|
|
|
|
|
auto fullImagePath = modulePath + inputDataImageFileName;
|
|
|
|
|
BindImage(modelBinding, inputBindingName, fullImagePath.c_str(), bindInputsAsIInspectable);
|
|
|
|
|
|
|
|
|
|
// create the tensor for the actual output
|
|
|
|
|
auto output = model.OutputFeatures().First().Current();
|
|
|
|
|
EXPECT_TRUE(output.Kind() == LearningModelFeatureKind::Tensor);
|
|
|
|
|
|
|
|
|
|
auto shape = winrt::single_threaded_vector(std::vector<int64_t> {1, 1});
|
|
|
|
|
auto outputTensor = BindImageOutput(outputBindingStrategy, modelBinding, outputDataBindingName);
|
|
|
|
|
WINML_PROFILING_STOP(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::BIND_VALUE);
|
|
|
|
|
|
|
|
|
|
// Evaluate the model
|
|
|
|
|
WINML_PROFILING_START(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::EVAL_MODEL_FIRST_RUN);
|
|
|
|
|
std::cout << "Calling EvaluateSync on instance" << instance << "\n";
|
|
|
|
|
LearningModelEvaluationResult result = nullptr;
|
|
|
|
|
EXPECT_NO_THROW(result = modelSession.Evaluate(modelBinding, {}));
|
|
|
|
|
WINML_PROFILING_STOP(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::EVAL_MODEL_FIRST_RUN);
|
|
|
|
|
|
|
|
|
|
// Get results
|
|
|
|
|
if (outputBindingStrategy == OutputBindingStrategy::Unbound)
|
|
|
|
|
{
|
|
|
|
|
// When output binding strategy is unbound, the output tensor was not set on bind.
|
|
|
|
|
// Therefore, we need to retrieve it from the LearnignModelEvaluationResult
|
|
|
|
|
// TODO: is this right? outputTensorT is unused...
|
|
|
|
|
/*auto outputTensorT = */result.Outputs().Lookup(outputDataBindingName).as<TensorFloat16Bit>();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
EXPECT_EQ(result.Outputs().Lookup(outputDataBindingName), outputTensor);
|
|
|
|
|
|
|
|
|
|
auto softwareBitmap = outputTensor.VideoFrame().SoftwareBitmap();
|
|
|
|
|
|
|
|
|
|
auto folder = StorageFolder::GetFolderFromPathAsync(modulePath.c_str()).get();
|
|
|
|
|
auto imagefile = folder.CreateFileAsync(outputDataFileName, CreationCollisionOption::ReplaceExisting).get();
|
|
|
|
|
auto stream = imagefile.OpenAsync(FileAccessMode::ReadWrite).get();
|
|
|
|
|
auto encoder = BitmapEncoder::CreateAsync(BitmapEncoder::JpegEncoderId(), stream).get();
|
|
|
|
|
encoder.SetSoftwareBitmap(softwareBitmap);
|
|
|
|
|
encoder.FlushAsync();
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
WINML_PROFILING_STOP(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::RUN_TEST);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ModelValidator::SqueezeNet(
|
|
|
|
|
std::string instance,
|
|
|
|
|
LearningModelDeviceKind deviceKind,
|
|
|
|
|
float dataTolerance,
|
|
|
|
|
bool bindAsImage,
|
|
|
|
|
OutputBindingStrategy outputBindingStrategy,
|
|
|
|
|
bool bindInputsAsIInspectable)
|
|
|
|
|
{
|
|
|
|
|
g_RuntimeProfiler.Enable(ProfilerType::CPU);
|
|
|
|
|
g_RuntimeProfiler.Enable(ProfilerType::GPU);
|
|
|
|
|
g_RuntimeProfiler.Reset(ProfilerType::CPU);
|
|
|
|
|
g_RuntimeProfiler.Reset(ProfilerType::GPU);
|
|
|
|
|
|
|
|
|
|
WINML_PROFILING_START(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::PREP_TEST);
|
|
|
|
|
// file name strings
|
|
|
|
|
static wchar_t* modelFileName = L"model.onnx";
|
|
|
|
|
static wchar_t* inputDataFileName = L"test_data_0_input.pb";
|
|
|
|
|
static wchar_t* outputDataFileName = L"test_data_0_output.pb";
|
|
|
|
|
static wchar_t* inputBindingName = L"data_0";
|
|
|
|
|
static wchar_t* inputDataImageFileName = L"kitten_224.png";
|
|
|
|
|
static const wchar_t* outputDataBindingName = L"softmaxout_1";
|
|
|
|
|
|
|
|
|
|
auto modulePath = FileHelpers::GetModulePath();
|
|
|
|
|
auto fullModelPath = modulePath + modelFileName;
|
|
|
|
|
auto outputFileName = modulePath + outputDataFileName;
|
|
|
|
|
WINML_PROFILING_STOP(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::PREP_TEST);
|
|
|
|
|
|
|
|
|
|
WINML_PROFILING_START(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::LOAD_MODEL);
|
|
|
|
|
// WinML model creation
|
|
|
|
|
LearningModel model = nullptr;
|
|
|
|
|
EXPECT_NO_THROW(model = LearningModel::LoadFromFilePath(fullModelPath));
|
|
|
|
|
WINML_PROFILING_STOP(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::LOAD_MODEL);
|
|
|
|
|
|
|
|
|
|
WINML_PROFILING_START(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::RUN_TEST);
|
|
|
|
|
WINML_PROFILING_START(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::CREATE_EVAL_CONTEXT);
|
|
|
|
|
LearningModelSession modelSession = nullptr;
|
|
|
|
|
EXPECT_NO_THROW(modelSession = LearningModelSession(model, LearningModelDevice(deviceKind)));
|
|
|
|
|
WINML_PROFILING_STOP(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::CREATE_EVAL_CONTEXT);
|
|
|
|
|
|
|
|
|
|
WINML_PROFILING_START(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::BIND_VALUE);
|
|
|
|
|
LearningModelBinding modelBinding(modelSession);
|
|
|
|
|
|
|
|
|
|
if (bindAsImage)
|
|
|
|
|
{
|
|
|
|
|
std::wstring fullImagePath = modulePath + inputDataImageFileName;
|
|
|
|
|
BindImage(modelBinding, inputBindingName, fullImagePath.c_str(), bindInputsAsIInspectable);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
auto inputDataPath = modulePath + inputDataFileName;
|
|
|
|
|
auto inputTensor = ProtobufHelpers::LoadTensorFromProtobufFile(inputDataPath, false);
|
|
|
|
|
BindTensor(modelBinding, inputBindingName, inputTensor, bindInputsAsIInspectable);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// load up the expected output
|
|
|
|
|
auto expectedResultsTensor = ProtobufHelpers::LoadTensorFromProtobufFile(outputFileName, false);
|
|
|
|
|
EXPECT_TRUE(expectedResultsTensor != nullptr);
|
|
|
|
|
|
|
|
|
|
// create the tensor for the actual output
|
|
|
|
|
auto output = model.OutputFeatures().First().Current();
|
|
|
|
|
EXPECT_TRUE(output.Kind() == LearningModelFeatureKind::Tensor);
|
|
|
|
|
|
|
|
|
|
auto outputTensor = BindOutput<TensorFloat>(
|
|
|
|
|
outputBindingStrategy, modelBinding, outputDataBindingName, expectedResultsTensor.Shape());
|
|
|
|
|
WINML_PROFILING_STOP(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::BIND_VALUE);
|
|
|
|
|
|
|
|
|
|
// Evaluate the model
|
|
|
|
|
WINML_PROFILING_START(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::EVAL_MODEL_FIRST_RUN);
|
|
|
|
|
std::cout << "Calling EvaluateSync on instance" << instance << "\n";
|
|
|
|
|
LearningModelEvaluationResult result = nullptr;
|
|
|
|
|
EXPECT_NO_THROW(result = modelSession.Evaluate(modelBinding, {}));
|
|
|
|
|
WINML_PROFILING_STOP(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::EVAL_MODEL_FIRST_RUN);
|
|
|
|
|
|
|
|
|
|
// Get results
|
|
|
|
|
if (outputBindingStrategy == OutputBindingStrategy::Unbound)
|
|
|
|
|
{
|
|
|
|
|
// When output binding strategy is unbound, the output tensor was not set on bind.
|
|
|
|
|
// Therefore, we need to retrieve it from the LearnignModelEvaluationResult
|
|
|
|
|
outputTensor = result.Outputs().Lookup(outputDataBindingName).as<ITensor>();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
EXPECT_EQ(result.Outputs().Lookup(outputDataBindingName), outputTensor);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
auto outDataExpected = expectedResultsTensor.as<TensorFloat>().GetAsVectorView();
|
|
|
|
|
auto outDataActual = outputTensor.as<TensorFloat>().GetAsVectorView();
|
|
|
|
|
|
|
|
|
|
EXPECT_TRUE(outDataActual.Size() == outDataExpected.Size());
|
|
|
|
|
for (uint32_t i = 0; i < outDataActual.Size(); i++)
|
|
|
|
|
{
|
|
|
|
|
float delta = std::abs(outDataActual.GetAt(i) - outDataExpected.GetAt(i));
|
|
|
|
|
if (delta > dataTolerance)
|
|
|
|
|
{
|
|
|
|
|
ADD_FAILURE() << "EXPECTED: " << outDataExpected.GetAt(i) << " , ACTUAL: " << outDataActual.GetAt(i)
|
|
|
|
|
<< "instance " << instance << ", element " << i;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
WINML_PROFILING_STOP(g_RuntimeProfiler, WINML_RUNTIME_TEST_PERF::RUN_TEST);
|
|
|
|
|
|
|
|
|
|
std::cout << "Profiling data:\n";
|
|
|
|
|
for (int i = 0; i < WINML_RUNTIME_TEST_PERF::kCount; ++i)
|
|
|
|
|
{
|
|
|
|
|
std::cout << WINML_RUNTIME_TEST_PERF_NAMES[i]
|
|
|
|
|
<< ": Time=" << g_RuntimeProfiler[i].GetAverage(CounterType::TIMER)
|
|
|
|
|
<< "\tCPUUse(%%)=" << g_RuntimeProfiler[i].GetAverage(CounterType::CPU_USAGE)
|
|
|
|
|
<< "\tAvgWorkingSetDelta(MB)=" << g_RuntimeProfiler[i].GetAverage(CounterType::WORKING_SET_USAGE)
|
|
|
|
|
<< "\tMaxWorkingSetDelta(MB)=" << g_RuntimeProfiler[i].GetMax(CounterType::WORKING_SET_USAGE)
|
|
|
|
|
<< "\tGPUUse(%%)=" << g_RuntimeProfiler[i].GetAverage(CounterType::GPU_USAGE)
|
|
|
|
|
<< "\tGPUDedicatedMem(MB)=" << g_RuntimeProfiler[i].GetAverage(CounterType::GPU_DEDICATED_MEM_USAGE);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|