有这个想法的初衷

喜欢电子和DIY硬件的朋友对稚晖君应该都不陌生,他定期都会分享一些自己做的好玩的硬件,他之前做了一个ElectronBot桌面机器人我就很感兴趣,所以就自己也做了一个。



起初我只是自己开发了一个叫电子脑壳的上位机软件,大家在之前的博客里应该也有见到,是个用WinUI(WASDK)开发的Windows应用软件。

ElectronBot机器人要一直和电脑连接通过应用进行控制,很多的网友都想ElectronBot脱离电脑使用,于是我也想着能有什么好办法,所以也就有了这个标题的想法。

项目代码地址

技术选型

大家想到树莓派肯定就会想起python,大学的时候玩树莓派确实用python比较多,但是作为一个.NET开发人员当然要尝试下.NET的跨平台能力到底如何了。

由于先有了之前写的上位机软件电子脑壳,也想着让之前封装的SDK能够直接用在树莓派上,所以选择.NET是顺理成章的事情了。

  • 应用开发选择.NET框架
  • 通讯方式选择GRPC
  • USB操作选择LibUsbDotNet调用libusb

整体方案如下图:

环境配置

1. 树莓派系统安装及网络配置

树莓派支持的系统有很多种,像官方的Debian,Ubuntu等等。

我平时使用Debian比较多,但是测试的时候好像usb操作有些问题,应该是我用的.net互操作库对Debian支持的不太好,所以我选择了ubuntu的系统。

安装烧录工具,下载系统,然后烧录系统。如下图所示:

目前树莓派Zero 2 W的Ubuntu只有Server版本和Core版本,需要使用命令行做操作。

网络配置可以参考下面的文章进行配置。

树莓派4B ubuntu server ssh服务器部署全过程详解(内网穿透)

能正常联网安装软件就可以了。

2. .NET6的运行环境

由于.NET6 Arm64官方没提供包管理安装,所以需要大家通过脚本,或者下载文件安装。

网络好的可以通过脚本安装,简单,本文以脚本安装示例。

参考文档如下:

使用安装脚本或通过提取二进制文件在 Linux 上安装 .NET

下载安装脚本dotnet-install.sh放到用户家目录

执行以下指令添加执行权限

  1. sudo chmod +x ./dotnet-install.sh

执行指令安装

  1. ./dotnet-install.sh -c Current

下载解压结束需要我们配置环境变量,就像windows安装软件一样,配置完环境变量,才可以直接通过dotnet直接使用了,不然是提示找不到指令的。

以下指令是编辑环境变量的,我选择了~/.bashrc,在文档最底部添加环境变量内容。

  1. sudo vim ~/.bashrc

内容

  1. export DOTNET_ROOT=$HOME/.dotnet
  2. export PATH=$PATH:$HOME/.dotnet:$HOME/.dotnet/tools

保存并退出,然后执行dotnet如果正常就可以进行下面的操作了。

实践过程记录

1. GRPC服务的开发与调试

首先用创建一个grpc服务

微软官方的文档协议文件编写介绍

协议文件编写和服务编写

以我的协议内容为例包含了float,bool,bytes类型

float 对应 c# float

bool 对应 c# bool

bytes 对应 c# byte[]

  1. syntax = "proto3";
  2. option csharp_namespace = "Verdure.ElectronBot.GrpcService";
  3. package electronbotactiongrpc;
  4. // The electronbotaction service definition.
  5. service ElectronBotActionGrpc {
  6. // Sends a greeting
  7. rpc PlayEmoticonAction (EmoticonActionFrameRequest) returns (EbHelloReply);
  8. }
  9. // The request message containing the user's name.
  10. message EmoticonActionFrameRequest {
  11. float J1 = 1;
  12. float J2 = 2;
  13. float J3 = 3;
  14. float J4 = 4;
  15. float J5 = 5;
  16. float J6 = 6;
  17. bool Enable = 7;
  18. bytes FrameBuffer = 8;
  19. }
  20. // The response message containing the greetings.
  21. message EbHelloReply {
  22. string message = 9;
  23. }

然后生成对应的服务代码

主要注意点是bytes类型的转换

服务端写好了以后,就可以写调用端代码了,为了图方便我就在电子脑壳代码里加了调用代码。

WinUI调用GRPC注意事项

直接将gprc相关的东西放到winui项目里好像有一些问题,所以我把它放到了一个库项目里进行操作了。

配置协议文件生成client代码

电子脑壳注入grpc-client

  1. services.AddGrpcClient<ElectronBotActionGrpc.ElectronBotActionGrpcClient>(o =>
  2. {
  3. o.Address = new Uri("http://192.168.3.236:5241");
  4. });

数据发送端也要注意bytes类型的转换

grpc服务的打包发布

打包的时候选择目标运行时为可移植(windows和linux都可以运行)

项目代码地址

2. 服务的部署与效果验证

通过ftp工具将grpc服务放到树莓派上,首先需要连接ElectronBot之后再运行服务

进入项目所在目录执行指令如下:

  1. dotnet Verdure.ElectronBot.GrpcService.dll --urls="http://*:5241"

如果通讯正常,那么应该会报错,提示libusb找不到,系统已经安装了libusb,只是.net调用的时候目录没有找到,需要我们创建软连接。

LibUsbDotNet README最底部有说

树莓派实际操作如下:

  1. sudo find / -name "libusb-1.0*.so*" //列出库的目录位置
  2. cd /usr/lib/aarch64-linux-gnu/ //进入库所在目录
  3. sudo ln -s libusb-1.0.so.0 libusb-1.0.so //创建链接

通过上面的操作 libusb的操作应该就没有问题了。

运行的效果如下图:

特殊问题与心得体会

最大的一个问题就是在sdk放到linux系统上测试写入数据异常。

  • 需要sdk单独处理linux环境
  • 需要sdk补一些内核驱动卸载逻辑
  • 特别注意otg数据线不要使用那种一拖几的,用一个口的就行不然数据发送不过去

由于使用的LibUsbDotNet上述逻辑没暴露到上层,我只好拉了一个分支改了发了一个包,问pr也没人理我。

sdk内部特别处理的代码如下:

执行了一个设备的SetAutoDetachKernelDriver

  1. if (wholeUsbDevice.DriverMode == UsbDevice.DriverModeType.MonoLibUsb)
  2. {
  3. _logger.LogInformation("MonoLibUsb DetachKernelDriver");
  4. var retDetach = wholeUsbDevice.SetAutoDetachKernelDriver(true);
  5. _logger.LogInformation(retDetach.ToString());
  6. }

通过这次的尝试,我算是明白了理想很丰满,现实很骨感的意思,本来觉得树莓派Zero 2 W小巧便携,而且我只是用它接收数据发送数据,结果数据发送的时候屏幕的刷新很慢,不知道是性能差劲,还是usb传输是瓶颈。下次换个4b进行测试一下。

至此整个过程就分享完了,这个项目基本上涵盖了.net技术内的很多东西,如果ElectronBot和树莓派结合独立以后能够很好的使用,我觉得用MAUI开发一个手机端的管理工具也是顺理成章了,可以期待一下,能看到这里的可以在评论区给点意见了。

电子脑壳 GRPC服务项目代码地址

LibUsbDotNet地址

树莓派Zero 2 W(ubuntu-22.04)通过.NET6和libusb操作USB读写的更多相关文章

  1. Qemu/Limbo/KVM镜像 Ubuntu 22.04 精简版,可运行Windows软件,内存占用不到200M

    镜像特征: Ubuntu 22.04系统 内置Wine 7.8,可运行大量Windows 软件 高度精简,内存占用仅200M不到. 自制UI,Windows3.1风格. 完全开源 镜像说明: 用户名为 ...

  2. Ubuntu 12.04下LVM2安装和操作实验

    实验环境: VirtualBox v4.3.20 Lubuntu 12.04LTS 前期准备: 1.添加虚拟盘:菜单"控制"->"设置"->&quo ...

  3. 三十项调整助力 Ubuntu 13.04 更上一层楼

    在Ubuntu 13.04 Raring Ringtail安装完成之后,我们还有三十项调整需要进行. 1.Ubuntu 13.04 Raring Ringtail安装完毕后,我又进行了一系列工作 大家 ...

  4. Ubuntu 20.04安装Docker

    Docker学习系列文章 入门必备:十本你不容错过的Docker入门到精通书籍推荐 day1.全面的Docker快速入门教程 day2.CentOS 8.4安装Docker day3.Windows1 ...

  5. 新版 Ubuntu 中 gnome-terminal 可恶的行间距问题逼我退回了 Ubuntu 20.04

    不知道从什么时候起(可能是 Ubuntu 21.04,也可能是 Ubuntu 21.10),Ubuntu 中的 gnome-terminal 的行间距就加大了,看起来极其不爽,特别是和 Powerli ...

  6. tips of my ubuntu 16.04 LTS

    update_0 : sudo .../idea.sh 才会把idea安装上,不加sudo也会启动,但是不会在开始菜单中找到程序. ---------------------------------- ...

  7. 树莓派初体验,安装Ubuntu 14.04 LTS

    转载自:http://www.polarxiong.com/archives/%E6%A0%91%E8%8E%93%E6%B4%BE%E5%88%9D%E4%BD%93%E9%AA%8C-%E5%AE ...

  8. 树莓派Ubuntu 16.04 MATA系统 修改用户文件夹名后,提示configure it with blueman-service

    自从修改了树莓派的Ubuntu 16.04 MATA 系统的 /home/ 下的用户文件夹名后,使用vncserver远程操作,看到桌面每次都提示 Configured directory for i ...

  9. 树莓派3b+ Ubuntu 16.04 MATA系统 ssh远程登陆后修改主机名、用户密码和用户名

    写在前面: 刚刚开始写博客,记录下自己的学习过程,备忘. 最近在使用树莓派做智能小车的开发,使用的是树莓派3b+,安装的是Ubuntu 16.04 MATA 系统,安装系统后需要修改主机名,登陆密码以 ...

随机推荐

  1. YII自定义小部件

    案例如下 common/widgets/TopMenu.php(地址可以自定义位置,命名空间一定要对应) <?php /** * Created by PhpStorm. * Date: 201 ...

  2. 【原创】Magisk+Shamiko过APP ROOT检测

    本文所有教程及源码.软件仅为技术研究.不涉及计算机信息系统功能的删除.修改.增加.干扰,更不会影响计算机信息系统的正常运行.不得将代码用于非法用途,如侵立删! Magisk+Shamiko过APP R ...

  3. React报错之Encountered two children with the same key

    正文从这开始~ 总览 当我们从map()方法返回的两个或两个以上的元素具有相同的key属性时,会产生"Encountered two children with the same key&q ...

  4. 解决使用 Eruda 绑定 dom 未在指定位置显示问题

    前言 开发项目中,使用到 Eruda 打印控制台信息显示 文档:https://github.com/liriliri/eruda 安装 Eruda npm install eruda --save ...

  5. java-servlet-EL表达式和java标签

    1      Servlet线程安全问题 1st.              为什么说servlet会有线程安全问题? 容器默认情况下,对于某个servlet,只会创建一个实例. 容器收到一个请求,就 ...

  6. 我就获取个时间,机器就down了

    本文主要讲解linux 时间管理系统中的一个问题 背景:linux 时间管理,包含clocksource,clockevent,timer,tick,timekeeper等等概念 , 这些概念有机地组 ...

  7. mustache.js常见用法

    一.mustache基于JS模板引擎,能较为快捷和简单得实现数据得渲染 用法: 1.CDN引入mustache.js,以下是4.0.1版本,有需要可以去github上查询其他版本的代码. (funct ...

  8. java方法---递归

    java方法递归 1.什么是递归 就是自己调用自己: 2.递归结构的两个部分 递归头:什么时候不调用自身方法.如果仅有头,将陷入死循环.递归体:什么时候需要调用自身方法.

  9. 如何不编写 YAML 管理 Kubernetes 应用?

    Kubernetes 将自身边界内的事物都抽象为资源.其中的主要部分,是以 Deployment.StatefulSet 为代表的 workload 工作负载控制器,其他各类资源都围绕这些主要的资源工 ...

  10. Linux云主机安全入侵排查步骤

    导语 经常有用户报障系统被植入恶意程序,如挖矿软件.ddos攻击病毒.syn映射攻击病毒等,可以按照以下流程为用户排查入侵病毒类型: 一.定位病毒进程 对于用户反馈云主机性能卡顿,CPU和内存占用较高 ...