嵌入式Linux:编译和使用Protobuf库

目录

1、开发环境和工具

2、安装和编译Protobuf、Protobuf-C库

3、编写和编译proto文件

4、修改makefile文件

5、测试示例

6、参考资料


Protobuf(Protocol Buffers)是由 Google 开发的一种轻量级、高效的结构化数据序列化方式,用于在不同应用之间进行数据交换和存储。它可以用于多种编程语言,并支持自动生成代码,使得数据结构定义和序列化/反序列化过程更加简洁和高效。

Protobuf-C 是 Protocol Buffers 的 C 语言实现,它专门针对 C 语言环境进行了优化,提供了类似于官方实现的功能,同时支持与其他语言生成的 Protobuf 数据进行交互。Protobuf-C 生成的库文件可以被 C 语言项目使用,使得在 C 语言环境中进行高效的数据序列化和反序列化成为可能。

Protobuf优点包括:

  • 高效性:protobuf 生成的数据格式通常比 XML 和 JSON 更加紧凑,序列化和反序列化速度更快。
  • 可扩展性:支持向已有消息类型添加新的字段或消息,而不破坏向后兼容性。
  • 语言无关性:protobuf 支持多种编程语言,包括 C++, Java, Python, Go, 和 C# 等。
  • 自动代码生成:通过 .proto 文件定义消息格式后,可以使用编译器自动生成目标语言的代码,简化开发工作。

Protobuf代码仓库:https://github.com/protocolbuffers/protobuf

Protobuf-C代码仓库:https://github.com/protobuf-c/protobuf-c

由于我需要在SoC开发板上使用C语言版的Protobuf库,所以需要使用到Protobuf和Protobuf-C

Protobuf 提供了 Protobuf 工具,用于将 .proto 文件转换为 C 源代码和头文件,而 Protobuf-c 生成了编译所需的动态库。

1、开发环境和工具

硬件环境

台湾联咏NT96570BG

软件环境

Ubuntu 18.04.6

SDK

na51055_linux_sdk-release.tar.gz

交叉编译工具链

nvt-96570-toolchain.tar.gz

Protobuf版本

V3.6.1(SoC需要和上位机通信,保持双方版本一致)

2、安装和编译Protobuf、Protobuf-C库

SoC编译和使用Protobuf库有2种方式:

  • 下载Protobuf、Protobuf-C源码,集成到SoC SDK包中,修改makefile文件和相关配置,每次编译SDK固件时,也会编译和生成Protobuf所需的库和文件。
  • Ubuntu系统下载和编译Protobuf、Protobuf-C源码,将编译好的库和文件拷贝到SoC APP应用工程中,修改makefile文件和相关配置,直接使用。

这里我们使用第二种方式。

1、安装依赖项,指令如下:

sudo apt-get install autoconf automake libtool curl make g++ unzip pkg-config

2、安装Protobuf

下载Protobuf V3.6.1,解压后进入文件夹,指令如下:

cd protobuf
./autogen.sh
./configure
make
sudo make install
sudo ldconfig

含义如下:

  • cd protobuf: 进入名为 protobuf 的目录。
  • ./autogen.sh: 运行 autogen.sh 脚本,用于生成 configure 配置脚本。
  • ./configure: 根据生成的配置脚本,配置编译环境。
  • make: 编译源代码。
  • sudo make install: 安装编译生成的文件到系统中。
  • sudo ldconfig: 更新动态链接库缓存,使得系统能够找到新安装的库文件。

如果不需要使用指定版本的Protobuf,可以使用git指令下载库:

git clone https://github.com/protocolbuffers/protobuf.git

3、安装protobuf-c

protobuf-c不需要指定版本,直接使用git指令下载仓库,指令如下:

git clone https://github.com/protobuf-c/protobuf-c.git
cd protobuf-c
./autogen.sh
./configure --host=arm-linux-gnueabihf CC=/opt/arm/arm-ca9-linux-gnueabihf-6.5/usr/bin/arm-ca9-linux-gnueabihf-gcc CXX=/opt/arm/arm-ca9-linux-gnueabihf-6.5/usr/bin/arm-ca9-linux-gnueabihf-g++ --disable-protoc --prefix=$PWD/tmp_out
make
sudo make install

含义如下:

  • cd protobuf-c: 进入名为 protobuf-c 的目录。
  • ./autogen.sh: 运行 autogen.sh 脚本,用于生成 configure 配置脚本。
  • ./configure --host=arm-linux-gnueabihf CC=/opt/arm/arm-ca9-linux-gnueabihf-6.5/usr/bin/arm-ca9-linux-gnueabihf-gcc CXX=/opt/arm/arm-ca9-linux-gnueabihf-6.5/usr/bin/arm-ca9-linux-gnueabihf-g++ --disable-protoc --prefix=$PWD/tmp_out: 配置编译环境,指定目标架构为 arm-linux-gnueabihf,并使用指定的交叉编译器进行编译。
  • make: 编译源代码。
  • sudo make install: 安装编译生成的文件到系统中。

重点说一下configure配置编译环境指令:

  • ./configure: 运行配置脚本。
  • --host=arm-linux-gnueabihf: 指定目标系统架构为 arm-linux-gnueabihf,表示编译生成的程序将在 ARM 架构上运行。
  • CC=/opt/arm/arm-ca9-linux-gnueabihf-6.5/usr/bin/arm-ca9-linux-gnueabihf-gcc: 指定 C 编译器为 /opt/arm/arm-ca9-linux-gnueabihf-6.5/usr/bin/arm-ca9-linux-gnueabihf-gcc,即指定了交叉编译器。
  • CXX=/opt/arm/arm-ca9-linux-gnueabihf-6.5/usr/bin/arm-ca9-linux-gnueabihf-g++: 指定 C++ 编译器为 /opt/arm/arm-ca9-linux-gnueabihf-6.5/usr/bin/arm-ca9-linux-gnueabihf-g++,即指定了交叉编译器。
  • --disable-protoc: 禁用 protoc 工具的构建,这表示只编译动态库,而不会生成 .proto 文件对应的 C 源码和头文件。
  • --prefix=$PWD/tmp_out: 指定安装路径为当前目录下的 tmp_out 目录。

如果不是ARM SoC使用,只是Ubuntu系统使用,配置编译环境就无需指定交叉编译工具链,指令如下:

./configure

Protobuf、Protobuf-C默认安装在/usr/local路径下:

使用指令可以查看Protobuf、Protobuf-C的版本,指令如下:

protoc-c --version

编译Protobuf-c代码时,指定了链接库输出在当前目录下的 tmp_out 目录。将编译输出物都拷贝到SoC APP应用工程中。

3、编写和编译proto文件

1、创建一个proto文件,文件命名为:LM_PCD_LD.proto,定义了一个消息类型:

syntax = "proto3";message Person {
  string name = 1;
  int32 id = 2;
  string email = 3;
}

2、使用 Protobuf 编译器(protoc)生成对应的C代码:

protoc --c_out=. LM_PCD_LD.proto.proto

编译生成:LM_PCD_LD.pb-c.c和LM_PCD_LD.pb-h文件。将文件拷贝到SoC APP应用工程中。

4、修改makefile文件

1、添加头文件路径:

2、添加动态链接库路径:

3、添加代码路径:

4、拷贝动态库到系统库文件下:

5、测试示例

#include <stdio.h>
#include "LM_PCD_LD.pb-c.h"int main() {
    // 创建并初始化 Person 消息对象
    Person person = PERSON__INIT;
    person.name = "John Doe";
    person.id = 1234;
    person.email = "johndoe@example.com";    // 序列化消息对象
    size_t packed_size = person__get_packed_size(&person);
    uint8_t buffer[packed_size];
    person__pack(&person, buffer);    // 反序列化消息对象
    Person *unpacked_person = person__unpack(NULL, packed_size, buffer);    // 打印反序列化后的消息内容
    printf("Name: %s\n", unpacked_person->name);
    printf("ID: %d\n", unpacked_person->id);
    printf("Email: %s\n", unpacked_person->email);    // 释放内存
    person__free_unpacked(unpacked_person, NULL);    return 0;
}

6、参考资料

https://blog.csdn.net/zhengnianli/article/details/110914259

https://blog.csdn.net/zhengnianli/article/details/109523798

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

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

相关文章

内联函数+auto关键字(C++11)+指针空指针nullptr(C++11)

内联函数auto关键字&#xff08;C11&#xff09;指针空指针nullptr&#xff08;C11&#xff09;详解 内联函数概念特性 auto关键字&#xff08;C11&#xff09;auto简介auto的使用细则auto不能推导的场景 基于范围的for循环(C11)范围for的语法范围for的使用条件 指针空指针null…

算法-卡尔曼滤波之卡尔曼滤波的第二个方程:预测方程(状态外推方程)

在上一节中&#xff0c;使用了静态模型&#xff0c;我们推导出了卡尔曼滤波的状态更新方程&#xff0c;但是在实际情况下&#xff0c;系统都是动态&#xff0c;预测阶段&#xff0c;前后时刻的状态是改变的&#xff0c;此时我们引入预测方程&#xff0c;也叫状态外推方程&#…

[动画详解]LeetCode151.翻转字符串里的单词

&#x1f496;&#x1f496;&#x1f496;欢迎来到我的博客&#xff0c;我是anmory&#x1f496;&#x1f496;&#x1f496; 又和大家见面了 欢迎来到动画详解LeetCode算法系列 用通俗易懂的动画让算法题不再神秘 先来自我推荐一波 个人网站欢迎访问以及捐款 推荐阅读 如何低成…

go语言基础1

1.token token是构成源程序的基本不可在分割单元。编译器编译源程序的第一步就是将源程序分割为一个个独立的token&#xff0c;这个过程就是词法分析。Go语言的token可以分为关键字、标识符、操作符、分隔符和字面常量等&#xff0c;如图所示&#xff1a; Go token分隔符有两类…

c#多态性的应用

设计一个电脑游戏&#xff0c;游戏中有猪、牛、兔子、青蛙、鸭子等动物&#xff0c;这些动 物都继承于Vertebrata 类&#xff08;脊椎动物类&#xff09;&#xff0c;Vertebrata类有一个抽象方法Display()&#xff0c;每个动 物都从Vertebrata 类那里继承并重写了Display()方法…

数据驱动测试在接口测试和网站测试中的应用

什么是数据驱动测试 据驱动测试是一种测试方法&#xff0c;其中测试数据和测试逻辑是分开的&#xff0c;测试数据被存储在外部源中&#xff08;如Excel表格、JSON文件、数据库等&#xff09;&#xff0c;测试逻辑则独立于测试数据。在测试过程中&#xff0c;测试数据被读取并传…

ROS2入门21讲__第03讲__ROS2安装方法

目录 前言 Linux系统简介 Ubuntu系统简介 Ubuntu虚拟机安装 1. 下载系统镜像 2. 在虚拟机中创建系统 3. 设置虚拟机硬盘大小 4. 设置Ubuntu镜像路径 5. 启动虚拟机 6. 设置用户名和密码 7. 等待系统安装 8. 完成安装 ROS2系统安装 1. 设置编码 2. 添加源 3. 安装…

2024年5月面试准备

2024年5月面试准备 资料来源Java基础泛型注解异常反射SPI机制Java集合CollectionMap 并发基础线程并发关键字并发集合Lock核心类并发集合核心类原子类核心类线程池核心类ScheduledThreadPoolExecutorForkJoinPoolFokJoinTask JUC原子类: CAS, Unsafe和原子类详解JUC 工具类 Jav…

鸿蒙内核源码分析(内核态锁篇) | 如何实现快锁Futex(下)

本篇为快锁下篇&#xff0c;说清楚快锁在内核态的实现&#xff0c;解答以下问题&#xff0c;它们在上篇的末尾被提出来。 鸿蒙内核进程池默认上限是64个&#xff0c;除去两个内核进程外&#xff0c;剩下的都归属用户进程&#xff0c;理论上用户进程可以创建很多快锁&#xff0…

Android开发,日志级别

5个日志级别 Verbose (VERBOSE): 这是最低的日志级别&#xff0c;用于输出最为详尽的信息&#xff0c;包括开发和调试过程中的各种细节。在Log类中对应的方法是Log.v()。Debug (DEBUG): 此级别用于输出调试信息&#xff0c;帮助开发者理解程序运行流程或状态。通过Log.d()方法…

保研机试之【文件描述符】

A选项&#xff1a; 一个文件描述符对应着系统级文件表中的一项 B选项 C选项 D选项 E选项 F选项 综上&#xff0c;我认为这道题选择B、C、E、F~

九、e2studio VS STM32CubeIDE之const修饰BSP函数的形参

目录 一、概述/目的 二、通过串口发送函数对比 2.1 stm32 hal库 VS renesas FSP 2.2 const修改函数形参的作用 2.2.1 值传递-副本 2.2.2 指针传递&#xff08;就近原则&#xff09; 2.2.2.1 const修饰&#xff1a;*P 2.2.2.2 const修饰&#xff1a;指针变量P 2.2.2.3 …