TensorRT部署模型基本步骤(C++)

TensorRT部署模型基本步骤(C++)


文章目录

  • TensorRT部署模型基本步骤(C++)
  • 前言
  • 一、onnx模型转engine
    • 1.基于C++代码生成engine
    • 2.基于trtexec.exe命令行生成
  • 二、读取本地模型
  • 三、创建推理引擎
  • 四、创建推理上下文
  • 五、创建GPU显存缓冲区
  • 六、 配置输入数据
  • 七、模型推理
  • 八、获得输出数据
  • 总结


前言

经典的一个TensorRT部署模型步骤为:onnx模型转engine读取本地模型创建推理引擎创建推理上下文创建GPU显存缓冲区配置输入数据模型推理以及处理推理结果(后处理)


一、onnx模型转engine

目前多种模型框架都将onnx模型当作中间转换格式,是的该模型结构变得越来越通用,因此TensorRT目前主要在更新的就是针对该模型的转换。TensorRT是可以直接读取engine文件(缺点是:读取engine要求是相同TensorRT版本和相同的平台),对于onnx模型需要进行一些列转换配置,转为engine引擎才可以进行后续的推理,因此在进行模型推理前,需要先进行模型的转换。

1.基于C++代码生成engine

void onnx_to_engine(std::string onnx_file_path, std::string engine_file_path, int type) {// 构建器,获取cuda内核目录以获取最快的实现// 用于创建config、network、engine的其他对象的核心类nvinfer1::IBuilder* builder = nvinfer1::createInferBuilder(gLogger);const auto explicitBatch = 1U << static_cast<uint32_t>(nvinfer1::NetworkDefinitionCreationFlag::kEXPLICIT_BATCH);// 解析onnx网络文件// tensorRT模型类nvinfer1::INetworkDefinition* network = builder->createNetworkV2(explicitBatch);// onnx文件解析类// 将onnx文件解析,并填充rensorRT网络结构nvonnxparser::IParser* parser = nvonnxparser::createParser(*network, gLogger);// 解析onnx文件parser->parseFromFile(onnx_file_path.c_str(), 2);for (int i = 0; i < parser->getNbErrors(); ++i) {std::cout << "load error: " << parser->getError(i)->desc() << std::endl;}printf("tensorRT load mask onnx model successfully!!!...\n");// 创建推理引擎// 创建生成器配置对象。nvinfer1::IBuilderConfig* config = builder->createBuilderConfig();// 设置最大工作空间大小。config->setMaxWorkspaceSize(16 * (1 << 20));// 设置模型输出精度if (type == 1) {config->setFlag(nvinfer1::BuilderFlag::kFP16);}if (type == 2) {config->setFlag(nvinfer1::BuilderFlag::kINT8);}// 创建推理引擎nvinfer1::ICudaEngine* engine = builder->buildEngineWithConfig(*network, *config);// 将推理银枪保存到本地std::cout << "try to save engine file now~~~" << std::endl;std::ofstream file_ptr(engine_file_path, std::ios::binary);if (!file_ptr) {std::cerr << "could not open plan output file" << std::endl;return;}// 将模型转化为文件流数据nvinfer1::IHostMemory* model_stream = engine->serialize();// 将文件保存到本地file_ptr.write(reinterpret_cast<const char*>(model_stream->data()), model_stream->size());// 销毁创建的对象model_stream->destroy();engine->destroy();network->destroy();parser->destroy();std::cout << "convert onnx model to TensorRT engine model successfully!" << std::endl;
}

2.基于trtexec.exe命令行生成

图一
图二
备注:

  • –onnx=“filepath” filepath是onnx文件路径
  • –saveEngine=“filepath” filepath是保存engine文件路径

二、读取本地模型

读取onnx转换的engine二进制文件,将模型文件信息读取到内存中。由于engine保存了模型的信息以及电脑的(TensorRT)配置环境信息,所以如果要将模型部署在其他电脑上,要保证电脑的配置环境以及平台(windows/linux/macos)是否相同。

	std::string enginepath = "E:/TensorRT-8.6.0.12/bin/resnet18.engine";	//读取二进制文件engine的路径std::ifstream file(enginepath, std::ios::binary);		// 以二进制方式打开char* trtModelStream = NULL;							// 定义一个字符指针,用于读取engine文件数据int size = 0;											// 存储二进制文件字符的数量if (file.good()) {file.seekg(0, file.end);							//将文件指针移动到文件末尾size = file.tellg();								//获取当前文件指针的位置,即文件的大小file.seekg(0, file.beg);							//文件指针移回文件开始处trtModelStream = new char[size];					//分配足够的内存储存文件内容assert(trtModelStream);								//检查内存是否分配成功file.read(trtModelStream, size);					//读取文件信息,并存储在trtModelStreamfile.close();										//关闭文件}

三、创建推理引擎

首先需要初始化日志记录接口类,该类用于创建后续反序列化引擎使用;然后创建反序列化引擎,其主要作用是允许对序列化的功能上不安全的引擎进行反序列化,接下调用反序列化引擎来创建推理引擎,这一步只需要输入上一步中读取的模型文件数据以及长度即可。

// 日志记录接口
Logger logger;
// 反序列化引擎
nvinfer1::IRuntime* runtime = nvinfer1::createInferRuntime(logger);
assert(runtime != nullptr);
// 推理引擎
nvinfer1::ICudaEngine* engine = runtime->deserializeCudaEngine(model_stream, size);
assert(engine != nullptr);

四、创建推理上下文

创建可执行的IExecutionContent实例 - createExecutionContext,为后面进行模型推理的类。

nvinfer1::IExecutionContext* context = engine->createExecutionContext();
assert(context != nullptr);
delete[] trtModelStream ;		// 释放内存

五、创建GPU显存缓冲区

TensorRT是利用英伟达显卡(GPU)进行模型推理的,但是我们的推理数据以及后续处理数据是在内存(CPU)中实现的,因此需要创建显存缓冲区,用于输入推理数据以及读取推理结果数据。

// 创建GPU显存缓冲区
void** data_buffer = new void* [num_ionode];
// 创建GPU显存输入缓冲区
// getBindingDimensions函数获取指定索引的绑定维度信息,然后从中提取高度和宽度的值
int input_node_index = engine->getBindingIndex(input_node_name);
cudaMalloc(&(data_buffer[input_node_index]), input_data_length * sizeof(float));
// 创建GPU显存输出缓冲区
int output_node_index = engine->getBindingIndex(output_node_name);
cudaMalloc(&(data_buffer[output_node_index]), output_data_length * sizeof(float));

六、 配置输入数据

配置输入数据时只需要调用cudaMemcpyAsync()方法,便可将cuda流数据加载到与i里模型上。但数据需要根据模型要求进行预处理,除此以外需要将数据结果加入到cuda流中。

// 创建输入cuda流
cudaStream_t stream;
cudaStreamCreate(&stream);
/*
*******输入图像预处理*******
*/
// HWC => CHW,转换成张量格式
cv::Mat tensor = ::dnn::blobFromImage(blob);
// 输入数据由内存到GPU显存
cudaMemcpyAsync(data_buffer[input_node_index], tensor.ptr<float>(), input_data_length * sizeof(float), cudaMemcpyHostToDevice, stream);

七、模型推理

context->enqueueV2(data_buffer, stream, nullptr);

八、获得输出数据

最后处理数据是在内存上实现的,首先需要将数据由显存读取到内存中。

std::vector<float> prob;
// 创建临时缓存输出
prob.resize(output_h * output_w);
// GPU显存到内存
cudaMemcpyAsync(prob.data(),  data_buffer[output_node_index], output_data_length * sizeof(float), cudaMemcpyDeviceToHost, stream);
/*
*******输出图像后处理*******
*/

总结

本文主要介绍了TensorRT+C++部署的基本步骤,欢迎阅读交流。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/18065.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

算法训练 | 二叉树Part1 | 递归遍历、迭代遍历、统一迭代

目录 递归遍历 前序遍历 迭代遍历 前序遍历&#xff08;迭代法&#xff09; 中序遍历&#xff08;迭代法&#xff09; 后序遍历&#xff08;迭代法&#xff09; 统一迭代法 统一迭代 嵌入式学习分享个人主页&#xff1a;Orion嵌入式随想录 - 小红书 (xiaohongshu.com) …

惯性测量单元M-G370系列广泛用于工业系统各个领域

爱普生现已推出型号为M-G370系列的高稳定性、高精度及极小尺寸封装的惯性测量单元(IMU)&#xff0c;可广泛应用于工业系统的各个领域。 为了节省PCB的面积和产品空间&#xff0c;M-G370系列性测量单元设计精巧&#xff0c;且具有6个自由度:三轴角速率和三轴线性加速度&…

selenium学习笔记

什么是selenium 比较官方的解释 Selenium是一个自动化测试工具&#xff0c;用于在Web应用程序中模拟用户操作。它提供了一组API&#xff0c;可以通过编程方式控制浏览器&#xff0c;并模拟用户的交互行为&#xff0c;例如点击、输入文本和导航等。Selenium支持多种编程语言&a…

2024年认证杯二阶段数学建模赛题浅析

一图流 问题模型复杂度数据收集难度数据处理难度实现难度专业知识需求A题中高中中中材料科学、热物理、机械工程B题高高高高生物力学、神经学、医学成像C题高高高高环境科学、气象学、气候工程D题中中高高中高机器学习、数据科学、AI设计 【腾讯文档】2024年认证杯二阶段资料助…

9.4 Go语言入门(运算符)

Go语言入门&#xff08;运算符&#xff09; 目录三、运算符1. 算术运算符2. 关系运算符3. 逻辑运算符4. 位运算符5. 赋值运算符6. 其他运算符7. 运算符优先级 目录 Go 语言&#xff08;Golang&#xff09;是一种静态类型、编译型语言&#xff0c;由 Google 开发&#xff0c;专注…

【JAVA】Java如何使用Spring Boot进行Web服务开发

文章目录 前言一、函数解释二、代码实现三、总结 前言 在现代的微服务架构中&#xff0c;创建快速、可靠的Web服务已经成为一项基本技能。Spring Boot是一个出色的框架&#xff0c;它简化了Spring应用开发&#xff0c;使我们能够更快速地创建和部署Web服务。在这篇博客中&…

mysql支持的存储引擎有哪些

MySQL支持多种存储引擎,每种存储引擎都有其特定的用途和特点。以下是MySQL中常见的存储引擎: 1. InnoDB 特性: 支持事务和ACID属性行级锁定外键约束自动崩溃恢复MVCC(多版本并发控制)全文搜索(从MySQL 5.6开始)优点: 高并发性能数据完整性和安全性强自动恢复机制缺点:…

若依启动run-modules-system.bat报错问题解决方案

在启动run-modules-system.bat时遇到了一些问题,在网上搜索无果后,排查解决完毕 1.启动nacos时,报错如下 Error creating bean with name grpcClusterServer: Invocation of init method failed; nested exception is java.io.IOException: Failed to bind to address 0.0.0.0…

netty4 输出chunk

HTTP之Chunk HttpResponse response new DefaultHttpResponse(response.protocolVersion(), response.status()); HttpHeaders headers response.headers();// 设置transfer_encoding headers.set(HttpHeaderNames.TRANSFER_ENCODING, HttpHeaderValues.CHUNKED);ctx.write(…

Android 待办类应用提醒功能的实现及其问题

待办类应用作为一款提升工作效率的实用工具&#xff0c;在Android平台上深受用户喜爱。其核心功能之一便是提醒功能&#xff0c;帮助用户及时完成待办事项。 Android待办类应用的提醒功能看似简单&#xff0c;但涉及到多种系统机制和细节处理&#xff0c;需要开发者仔细考量和…

SQL实战 将学生信息进行 行转列输出

表countries 数据如下&#xff1a; namecontinentJaneAmericaPascalEuropeXiAsiaJackAmerica数据建表来源&#xff1a; SQL试题使得每个学生 按照姓名的字⺟顺序依次排列 在对应的⼤洲下⾯展示为如下的数据样式&#xff1a; namecontinentJane, JackAmericaXiAsiaPascalPasca…

Python开发 —— 对象type、object、class

1. "Python中一切皆为对象"的理解 在Python中&#xff0c;一切皆为对象的意思是指&#xff1a;无论是数字、字符串、函数、类、模块等任何数据类型&#xff0c;都可以被看做是一个对象。每个对象都具有自己的属性和方法&#xff0c;可以被操作和调用。 例如&#xff…

京东店铺商品列表API接口详解

随着电子商务的快速发展&#xff0c;越来越多的商家选择在京东这样的大型电商平台上开设店铺。为了帮助商家更高效地管理和展示商品&#xff0c;京东提供了一系列的API接口。本文将详细介绍京东店铺商品列表API接口&#xff0c;帮助开发者和商家更好地利用这一工具。 京东API概…

2022年全国职业院校技能大赛高职组“信息安全管理与评估”赛项第二阶段任务书

第二阶段竞赛项目试题 本文件为信息安全管理与评估项目竞赛-第二阶段试题&#xff0c;第二阶段内容包括&#xff1a;网络安全事件响应、数字取证调查和应用程序安全。 本次比赛时间为180分钟。 介绍 竞赛有固定的开始和结束时间&#xff0c;选手必须决定如何有效的分配…

回溯算法03(leetcode39/40/131)

参考资料&#xff1a; https://programmercarl.com/0039.%E7%BB%84%E5%90%88%E6%80%BB%E5%92%8C.html 39. 组合总和 题目描述&#xff1a; 给你一个 无重复元素 的整数数组 candidates 和一个目标整数 target &#xff0c;找出 candidates 中可以使数字和为目标数 target 的 …

linux--实时性优化

linux--实时性优化 1 介绍2 实时性需求3 代表性实时系统4 嵌入式系统嵌入式软件系统结构处理器时钟节拍多任务机制任务调度方式任务调度算法时间片调度算法优先级调度算法基于优先级的时间片调度算法 5 cyclictest 测试工具命令说明命令分析参数含义 6 linux 实时性改进某版本上…

四川汇烁面试总结

自我介绍项目介绍、 目录 1.jdk和jre的区别&#xff1f; 2.一段代码的执行流程&#xff1f; 3.接口与抽象类的区别&#xff1f; 4.ArrayList与LinkList的区别&#xff1f; 5.对HashMap的理解? 6.常见的异常&#xff1f; 7.throw 和 throws 有什么区别&#xff1f; 8.…

es和mongdb对比

本文参考博客: 【文档数据库】ES和MongoDB的对比 ES和MongoDB都能存储海量文档&#xff0c;都支持文档的搜索&#xff0c;很多功能上都是高度重合的&#xff0c;那为什么会出现如此相似的两个东西&#xff1f;他们各自的应用场景有什么不同&#xff1f;我其实也有这样的疑问&a…

04-Vue:ref获取页面节点--很简单

目录 前言在Vue中&#xff0c;通过 ref 属性获取DOM元素使用 ref 属性获取整个子组件&#xff08;父组件调用子组件的方法&#xff09; 前言 我们接着上一篇文章 03-02-Vue组件之间的传值 来讲。 下一篇文章 05-Vue路由 在Vue中&#xff0c;通过 ref 属性获取DOM元素 我们当然…

装机必备——Bandizip7.33安装教程

装机必备——Bandizip7.33安装教程 软件下载 软件名称&#xff1a;Bandizip7.33 软件语言&#xff1a;简体中文 软件大小&#xff1a;8.42M 系统要求&#xff1a;Windows7或更高&#xff0c; 64位操作系统 硬件要求&#xff1a;CPU2GHz &#xff0c;RAM4G或更高 下载通道①迅…