xref: /aosp_15_r20/external/armnn/tests/MnistDatabase.cpp (revision 89c4ff92f2867872bb9e2354d150bf0c8c502810)
1*89c4ff92SAndroid Build Coastguard Worker //
2*89c4ff92SAndroid Build Coastguard Worker // Copyright © 2017 Arm Ltd. All rights reserved.
3*89c4ff92SAndroid Build Coastguard Worker // SPDX-License-Identifier: MIT
4*89c4ff92SAndroid Build Coastguard Worker //
5*89c4ff92SAndroid Build Coastguard Worker #include "MnistDatabase.hpp"
6*89c4ff92SAndroid Build Coastguard Worker 
7*89c4ff92SAndroid Build Coastguard Worker #include <armnn/Logging.hpp>
8*89c4ff92SAndroid Build Coastguard Worker 
9*89c4ff92SAndroid Build Coastguard Worker #include <fstream>
10*89c4ff92SAndroid Build Coastguard Worker #include <vector>
11*89c4ff92SAndroid Build Coastguard Worker 
12*89c4ff92SAndroid Build Coastguard Worker constexpr int g_kMnistImageByteSize = 28 * 28;
13*89c4ff92SAndroid Build Coastguard Worker 
EndianSwap(unsigned int & x)14*89c4ff92SAndroid Build Coastguard Worker void EndianSwap(unsigned int &x)
15*89c4ff92SAndroid Build Coastguard Worker {
16*89c4ff92SAndroid Build Coastguard Worker     x = (x >> 24) | ((x << 8) & 0x00FF0000) | ((x >> 8) & 0x0000FF00) | (x << 24);
17*89c4ff92SAndroid Build Coastguard Worker }
18*89c4ff92SAndroid Build Coastguard Worker 
MnistDatabase(const std::string & binaryFileDirectory,bool scaleValues)19*89c4ff92SAndroid Build Coastguard Worker MnistDatabase::MnistDatabase(const std::string& binaryFileDirectory, bool scaleValues)
20*89c4ff92SAndroid Build Coastguard Worker     : m_BinaryDirectory(binaryFileDirectory)
21*89c4ff92SAndroid Build Coastguard Worker     , m_ScaleValues(scaleValues)
22*89c4ff92SAndroid Build Coastguard Worker {
23*89c4ff92SAndroid Build Coastguard Worker }
24*89c4ff92SAndroid Build Coastguard Worker 
GetTestCaseData(unsigned int testCaseId)25*89c4ff92SAndroid Build Coastguard Worker std::unique_ptr<MnistDatabase::TTestCaseData> MnistDatabase::GetTestCaseData(unsigned int testCaseId)
26*89c4ff92SAndroid Build Coastguard Worker {
27*89c4ff92SAndroid Build Coastguard Worker     std::vector<unsigned char> I(g_kMnistImageByteSize);
28*89c4ff92SAndroid Build Coastguard Worker     unsigned int label = 0;
29*89c4ff92SAndroid Build Coastguard Worker 
30*89c4ff92SAndroid Build Coastguard Worker     std::string imagePath = m_BinaryDirectory + std::string("t10k-images.idx3-ubyte");
31*89c4ff92SAndroid Build Coastguard Worker     std::string labelPath = m_BinaryDirectory + std::string("t10k-labels.idx1-ubyte");
32*89c4ff92SAndroid Build Coastguard Worker 
33*89c4ff92SAndroid Build Coastguard Worker     std::ifstream imageStream(imagePath, std::ios::binary);
34*89c4ff92SAndroid Build Coastguard Worker     std::ifstream labelStream(labelPath, std::ios::binary);
35*89c4ff92SAndroid Build Coastguard Worker 
36*89c4ff92SAndroid Build Coastguard Worker     if (!imageStream.is_open())
37*89c4ff92SAndroid Build Coastguard Worker     {
38*89c4ff92SAndroid Build Coastguard Worker         ARMNN_LOG(fatal) << "Failed to load " << imagePath;
39*89c4ff92SAndroid Build Coastguard Worker         return nullptr;
40*89c4ff92SAndroid Build Coastguard Worker     }
41*89c4ff92SAndroid Build Coastguard Worker     if (!labelStream.is_open())
42*89c4ff92SAndroid Build Coastguard Worker     {
43*89c4ff92SAndroid Build Coastguard Worker         ARMNN_LOG(fatal) << "Failed to load " << imagePath;
44*89c4ff92SAndroid Build Coastguard Worker         return nullptr;
45*89c4ff92SAndroid Build Coastguard Worker     }
46*89c4ff92SAndroid Build Coastguard Worker 
47*89c4ff92SAndroid Build Coastguard Worker     unsigned int magic, num, row, col;
48*89c4ff92SAndroid Build Coastguard Worker 
49*89c4ff92SAndroid Build Coastguard Worker     // Checks the files have the correct header.
50*89c4ff92SAndroid Build Coastguard Worker     imageStream.read(reinterpret_cast<char*>(&magic), sizeof(magic));
51*89c4ff92SAndroid Build Coastguard Worker     if (magic != 0x03080000)
52*89c4ff92SAndroid Build Coastguard Worker     {
53*89c4ff92SAndroid Build Coastguard Worker         ARMNN_LOG(fatal) << "Failed to read " << imagePath;
54*89c4ff92SAndroid Build Coastguard Worker         return nullptr;
55*89c4ff92SAndroid Build Coastguard Worker     }
56*89c4ff92SAndroid Build Coastguard Worker     labelStream.read(reinterpret_cast<char*>(&magic), sizeof(magic));
57*89c4ff92SAndroid Build Coastguard Worker     if (magic != 0x01080000)
58*89c4ff92SAndroid Build Coastguard Worker     {
59*89c4ff92SAndroid Build Coastguard Worker         ARMNN_LOG(fatal) << "Failed to read " << labelPath;
60*89c4ff92SAndroid Build Coastguard Worker         return nullptr;
61*89c4ff92SAndroid Build Coastguard Worker     }
62*89c4ff92SAndroid Build Coastguard Worker 
63*89c4ff92SAndroid Build Coastguard Worker     // Endian swaps the image and label file - all the integers in the files are stored in MSB first(high endian)
64*89c4ff92SAndroid Build Coastguard Worker     // format, hence it needs to flip the bytes of the header if using it on Intel processors or low-endian machines
65*89c4ff92SAndroid Build Coastguard Worker     labelStream.read(reinterpret_cast<char*>(&num), sizeof(num));
66*89c4ff92SAndroid Build Coastguard Worker     imageStream.read(reinterpret_cast<char*>(&num), sizeof(num));
67*89c4ff92SAndroid Build Coastguard Worker     EndianSwap(num);
68*89c4ff92SAndroid Build Coastguard Worker     imageStream.read(reinterpret_cast<char*>(&row), sizeof(row));
69*89c4ff92SAndroid Build Coastguard Worker     EndianSwap(row);
70*89c4ff92SAndroid Build Coastguard Worker     imageStream.read(reinterpret_cast<char*>(&col), sizeof(col));
71*89c4ff92SAndroid Build Coastguard Worker     EndianSwap(col);
72*89c4ff92SAndroid Build Coastguard Worker 
73*89c4ff92SAndroid Build Coastguard Worker     // Reads image and label into memory.
74*89c4ff92SAndroid Build Coastguard Worker     imageStream.seekg(testCaseId * g_kMnistImageByteSize, std::ios_base::cur);
75*89c4ff92SAndroid Build Coastguard Worker     imageStream.read(reinterpret_cast<char*>(&I[0]), g_kMnistImageByteSize);
76*89c4ff92SAndroid Build Coastguard Worker     labelStream.seekg(testCaseId, std::ios_base::cur);
77*89c4ff92SAndroid Build Coastguard Worker     labelStream.read(reinterpret_cast<char*>(&label), 1);
78*89c4ff92SAndroid Build Coastguard Worker 
79*89c4ff92SAndroid Build Coastguard Worker     if (!imageStream.good())
80*89c4ff92SAndroid Build Coastguard Worker     {
81*89c4ff92SAndroid Build Coastguard Worker         ARMNN_LOG(fatal) << "Failed to read " << imagePath;
82*89c4ff92SAndroid Build Coastguard Worker         return nullptr;
83*89c4ff92SAndroid Build Coastguard Worker     }
84*89c4ff92SAndroid Build Coastguard Worker     if (!labelStream.good())
85*89c4ff92SAndroid Build Coastguard Worker     {
86*89c4ff92SAndroid Build Coastguard Worker         ARMNN_LOG(fatal) << "Failed to read " << labelPath;
87*89c4ff92SAndroid Build Coastguard Worker         return nullptr;
88*89c4ff92SAndroid Build Coastguard Worker     }
89*89c4ff92SAndroid Build Coastguard Worker 
90*89c4ff92SAndroid Build Coastguard Worker     std::vector<float> inputImageData;
91*89c4ff92SAndroid Build Coastguard Worker     inputImageData.resize(g_kMnistImageByteSize);
92*89c4ff92SAndroid Build Coastguard Worker 
93*89c4ff92SAndroid Build Coastguard Worker     for (unsigned int i = 0; i < col * row; ++i)
94*89c4ff92SAndroid Build Coastguard Worker     {
95*89c4ff92SAndroid Build Coastguard Worker         // Static_cast of unsigned char is safe with float
96*89c4ff92SAndroid Build Coastguard Worker         inputImageData[i] = static_cast<float>(I[i]);
97*89c4ff92SAndroid Build Coastguard Worker 
98*89c4ff92SAndroid Build Coastguard Worker         if(m_ScaleValues)
99*89c4ff92SAndroid Build Coastguard Worker         {
100*89c4ff92SAndroid Build Coastguard Worker             inputImageData[i] /= 255.0f;
101*89c4ff92SAndroid Build Coastguard Worker         }
102*89c4ff92SAndroid Build Coastguard Worker     }
103*89c4ff92SAndroid Build Coastguard Worker 
104*89c4ff92SAndroid Build Coastguard Worker     return std::make_unique<TTestCaseData>(label, std::move(inputImageData));
105*89c4ff92SAndroid Build Coastguard Worker }
106