Crnn.cpp 6.17 KB
Newer Older
Your Name's avatar
Your Name committed
1
2
3
4
5
6
7
8
9
10
#include <Crnn.h>
#include <migraphx/onnx.hpp>
#include <migraphx/gpu/target.hpp>
#include <Filesystem.h>
#include <SimpleLog.h>


namespace migraphxSamples
{

liucong's avatar
liucong committed
11
Crnn::Crnn()
Your Name's avatar
Your Name committed
12
13
14
15
16
17
18
19
20
21
22
{

}

Crnn::~Crnn()
{
    configurationFile.release();
}

ErrorCode Crnn::Initialize(InitializationParameterOfOcr initializationParameterOfOcr, bool dynamic)
{
liucong's avatar
liucong committed
23
24
    // 读取配置文件
    std::string configFilePath=initializationParameterOfOcr.configFilePath;
liucong's avatar
liucong committed
25
    if(!Exists(configFilePath))
Your Name's avatar
Your Name committed
26
    {
liucong's avatar
liucong committed
27
28
        LOG_ERROR(stdout, "no configuration file!\n");
        return CONFIG_FILE_NOT_EXIST;
Your Name's avatar
Your Name committed
29
    }
liucong's avatar
liucong committed
30
31
32
33
34
35
    if(!configurationFile.open(configFilePath, cv::FileStorage::READ))
    {
       LOG_ERROR(stdout, "fail to open configuration file\n");
       return FAIL_TO_OPEN_CONFIG_FILE;
    }
    LOG_INFO(stdout, "succeed to open configuration file\n");
Your Name's avatar
Your Name committed
36
37

    // 获取配置文件参数
liucong's avatar
liucong committed
38
39
    cv::FileNode netNode = configurationFile["CrnnDynamic"];
    std::string modelPath=(std::string)netNode["ModelPath"];
Your Name's avatar
Your Name committed
40
41

    // 加载模型
liucong's avatar
liucong committed
42
    if(!Exists(modelPath))
Your Name's avatar
Your Name committed
43
    {
liucong's avatar
liucong committed
44
        LOG_ERROR(stdout,"%s not exist!\n",modelPath.c_str());
Your Name's avatar
Your Name committed
45
46
47
48
        return MODEL_NOT_EXIST;
    }

    if(dynamic)
shizhm's avatar
shizhm committed
49
    {
Your Name's avatar
Your Name committed
50
51
52
        migraphx::onnx_options onnx_options;
        onnx_options.map_input_dims["input"]={1,1,32,512};

shizhm's avatar
shizhm committed
53
        net = migraphx::parse_onnx(modelPath, onnx_options);
liucong's avatar
liucong committed
54
        LOG_INFO(stdout,"succeed to load model: %s\n",GetFileName(modelPath).c_str());
Your Name's avatar
Your Name committed
55

liucong's avatar
liucong committed
56
57
58
59
60
        // 获取模型输入/输出节点信息
        std::unordered_map<std::string, migraphx::shape> inputs=net.get_inputs();
        std::unordered_map<std::string, migraphx::shape> outputs=net.get_outputs();
        inputName=inputs.begin()->first;
        inputShape=inputs.begin()->second;
liucong's avatar
liucong committed
61
62
63
64
65
        int N=inputShape.lens()[0];
        int C=inputShape.lens()[1];
        int H=inputShape.lens()[2];
        int W=inputShape.lens()[3];
        inputSize=cv::Size(W,H);
Your Name's avatar
Your Name committed
66
67

        // log输出日志信息
liucong's avatar
liucong committed
68
69
        LOG_INFO(stdout,"InputMaxSize:%dx%d\n",inputSize.width,inputSize.height);
        LOG_INFO(stdout,"InputName:%s\n",inputName.c_str());
Your Name's avatar
Your Name committed
70
71
72
73
74
75
    }
    else
    {
        migraphx::onnx_options onnx_options;
        onnx_options.map_input_dims["input"]={1,1,32,100};

shizhm's avatar
shizhm committed
76
        net = migraphx::parse_onnx(modelPath, onnx_options);
liucong's avatar
liucong committed
77
        LOG_INFO(stdout,"succeed to load model: %s\n",GetFileName(modelPath).c_str());
Your Name's avatar
Your Name committed
78

liucong's avatar
liucong committed
79
80
81
82
83
        // 获取模型输入/输出节点信息
        std::unordered_map<std::string, migraphx::shape> inputs=net.get_inputs();
        std::unordered_map<std::string, migraphx::shape> outputs=net.get_outputs();
        inputName=inputs.begin()->first;
        inputShape=inputs.begin()->second;
liucong's avatar
liucong committed
84
85
86
87
88
        int N=inputShape.lens()[0];
        int C=inputShape.lens()[1];
        int H=inputShape.lens()[2];
        int W=inputShape.lens()[3];
        inputSize=cv::Size(W,H);
Your Name's avatar
Your Name committed
89
90

        // log输出日志信息
liucong's avatar
liucong committed
91
92
        LOG_INFO(stdout,"InputSize:%dx%d\n",inputSize.width,inputSize.height);
        LOG_INFO(stdout,"InputName:%s\n",inputName.c_str());
Your Name's avatar
Your Name committed
93
94
95
96
97
98
99
100
    }

    // 设置模型为GPU模式
    migraphx::target gpuTarget = migraphx::gpu::target{};

    // 编译模型
    migraphx::compile_options options;
    options.device_id=0;                          // 设置GPU设备,默认为0号设备
shizhm's avatar
shizhm committed
101
102
    options.offload_copy=true;
    net.compile(gpuTarget,options);
liucong's avatar
liucong committed
103
    LOG_INFO(stdout,"succeed to compile model: %s\n",GetFileName(modelPath).c_str());
Your Name's avatar
Your Name committed
104

liucong's avatar
liucong committed
105
106
107
    // warm up
    std::unordered_map<std::string, migraphx::argument> inputData;
    inputData[inputName]=migraphx::argument{inputShape};
shizhm's avatar
shizhm committed
108
    net.eval(inputData);
Your Name's avatar
Your Name committed
109
110
111
112
113
114
115
116

    return SUCCESS;
}

ErrorCode Crnn::Infer(const cv::Mat &srcImage, std::vector<char> &resultsChar, bool raw, bool dynamic)
{
    if(srcImage.empty() || srcImage.type()!=CV_8UC3)
    {
liucong's avatar
liucong committed
117
        LOG_ERROR(stdout, "image error!\n");
Your Name's avatar
Your Name committed
118
119
120
121
122
        return IMAGE_ERROR;
    }

    cv::Mat inputImage, inputBlob;
    cv::cvtColor(srcImage, inputImage, CV_BGR2GRAY);
shizhm's avatar
shizhm committed
123

Your Name's avatar
Your Name committed
124
125
126
127
128
129
130
131
132
133
134
135
136
    int height, width, widthRaw;
    widthRaw = inputImage.cols;
    if(dynamic)
    {
        cv::resize(inputImage, inputImage, cv::Size(widthRaw, 32));
        height = inputImage.rows, width = inputImage.cols;
    }
    else
    {
        cv::resize(inputImage, inputImage, cv::Size(100, 32));
        height = inputImage.rows, width = inputImage.cols;
    }
    inputBlob = cv::dnn::blobFromImage(inputImage);
shizhm's avatar
shizhm committed
137

Your Name's avatar
Your Name committed
138
139
140
141
    for(int i=0; i<width * height; i++)
    {
        *((float*)inputBlob.data+i) = ((*((float*)inputBlob.data+i))/255.f - 0.5)/0.5;
    }
shizhm's avatar
shizhm committed
142

liucong's avatar
liucong committed
143
144
    // 创建输入数据
    std::unordered_map<std::string, migraphx::argument> inputData;
Your Name's avatar
Your Name committed
145
146
147
    if(dynamic)
    {
        std::vector<std::size_t> dynamicShape = {1, 1, 32, width};
liucong's avatar
liucong committed
148
        inputData[inputName] = migraphx::argument{migraphx::shape(inputShape.type(),dynamicShape), (float*)inputBlob.data};
Your Name's avatar
Your Name committed
149
150
151
    }
    else
    {
liucong's avatar
liucong committed
152
        inputData[inputName] = migraphx::argument{inputShape, (float*)inputBlob.data};
Your Name's avatar
Your Name committed
153
    }
shizhm's avatar
shizhm committed
154

Your Name's avatar
Your Name committed
155
156
    // 推理
    std::vector<migraphx::argument> inferenceResults = net.eval(inputData);
shizhm's avatar
shizhm committed
157

Your Name's avatar
Your Name committed
158
159
    // 获取推理结果
    std::vector<cv::Mat> outs;
shizhm's avatar
shizhm committed
160
    migraphx::argument result = inferenceResults[0];
Your Name's avatar
Your Name committed
161
162
163
164
165
166
167
168

    // 转换为cv::Mat
    migraphx::shape outputShape = result.get_shape();
    int shape[]={outputShape.lens()[0],outputShape.lens()[1],outputShape.lens()[2]};
    cv::Mat out(3,shape,CV_32F);
    memcpy(out.data,result.data(),sizeof(float)*outputShape.elements());
    outs.push_back(out);

shizhm's avatar
shizhm committed
169
    std::vector<int> predChars;
Your Name's avatar
Your Name committed
170
171
172
173
174
    const std::string alphabet = "-0123456789abcdefghijklmnopqrstuvwxyz";

    //获取字符索引序列
    for(uint i = 0; i < outs[0].size[0]; i++)
    {
liucong's avatar
liucong committed
175
        cv::Mat scores = cv::Mat(1,outs[0].size[2],CV_32F,outs[0].ptr<float>(i));
Your Name's avatar
Your Name committed
176
177
178
179
180
181
        cv::Point charIdPoint;
        double maxCharScore;
        cv::minMaxLoc(scores, 0, &maxCharScore, 0, &charIdPoint);
        int maxIdx = charIdPoint.x;
        predChars.push_back(maxIdx);
    }
shizhm's avatar
shizhm committed
182

Your Name's avatar
Your Name committed
183
    //字符转录处理
liucong's avatar
liucong committed
184
    for(uint i=0; i < predChars.size(); i++)
Your Name's avatar
Your Name committed
185
186
187
188
189
190
191
192
193
    {
        if(raw)
        {
            resultsChar.push_back(alphabet[predChars[i]]);
        }
        else
        {
            if(predChars[i] != 0)
            {
liucong's avatar
liucong committed
194
                if(!(i > 0 && predChars[i-1] == predChars[i]))
Your Name's avatar
Your Name committed
195
196
197
198
199
200
201
202
203
204
                {
                    resultsChar.push_back(alphabet[predChars[i]]);
                }
            }
        }
    }

    return SUCCESS;
}
}