随着大流行的持续,硬件不仅难以采购,而且难以安全地协同工作。在智能微型车辆的情况下,可用单元的数量本质上是有限的,我们部署它们的“环境”也是如此,例如赛道或障碍赛。我们如何解决硬件资源稀缺问题?我们将资源数字化!
> 本文最初发布于platis.solutions
SMCE是Smartcar 平台的数字双胞胎,以及一个酷炫且可定制的3D 世界,供您的漫游车漫游。核心概念是您编写可在两者上运行的 Arduino 代码真实的硬件和虚拟环境。通过这种方式,可以减轻或大大减少对汽车的专有和频繁访问的需求。理想情况下,模拟器允许限制与物理汽车的不可避免的接触,以至于开发团队可以在模拟器上开发他们的功能,并且只访问真实的硬件来验证它们。使用网络物理系统的虚拟表示不仅在大流行期间很有价值。我们这些使用嵌入式系统的人可能会痛苦地认识到由于缺乏硬件可用性而被阻止的不利影响。在这篇文章中,我将向您展示如何开始使用SMCE以及它的许多功能。如果你想要一些关于什么的灵感您可以使用该软件查看我们的学生如何利用SMCE 来模拟 Arduino 车辆。
在深入了解模拟器之前,让我们澄清一下 SMCE 不是什么:一个 3D 环境,用于您的超级自定义爱好项目,它使用各种奇异的传感器和第三方库。开箱即用的模拟器可支持Smartcar 库的用户并适应他们的典型用例和传感器设置。模拟器不仅适用于特定的库,但是,如果您使用它,您的生活会轻松得多。我并不是因为模拟器而这么说。它带有很多例子,它用途广泛,总体上是一个非常容易使用的库,您应该尝试一下!此外,我应该注意,模拟器不是由我制作或维护的,所以请不要联系我寻求支持。它是由我的两个非常有才华的前学生Ryan Janson和Ruthger Dijt 开发的。您可以通过在相关 GitHub 存储库之一上创建问题或开始讨论来联系他们。
SMCE 由两部分组成:“前端”(smce-gd)和“后端”(libSMCE)。libSMCE
是负责在您的计算机上编译和运行 Arduino 草图的库。
这个跨平台的 C++ 库为其消费者提供了在托管环境中编译和执行 Arduino 草图的能力,并绑定到其虚拟 I/O 端口以允许主机应用程序与其子草图进行交互。
这个跨平台的 C++ 库为其消费者提供了在托管环境中编译和执行 Arduino 草图的能力,并绑定到其虚拟 I/O 端口以允许主机应用程序与其子草图进行交互。
smce-gd
取决于在libSMCE
多彩的虚拟 3D 世界中可视化草图执行以及与周围环境的交互。除非您打算做一些“聪明”的事情,扩展或为项目做出贡献等,否则smce-gd
您应该主要关心的软件。对于本教程,我将使用1.3.1版本,smce-gd
在 Ubuntu 20.04 上运行。
为了在我的 Ubuntu 笔记本电脑上安装 SMCE,我按照Wiki上的这些说明进行操作。SMCE 的一个非常酷的地方是它可以在您的计算机上运行。SMCE 也可以安装在Windows或MacOS 上。如果 SMCE 安装正确,您将看到以下屏幕:smce-gd
选择Start Fresh选项,然后在下一个屏幕中单击+
标志。
然后是时候选择在 Arduino 上运行的代码了。我建议从Smartcar 库示例中的manualControl.ino草图开始。获取示例的一种简单方法是Smartcar shield
通过 Arduino IDE 的库管理器下载库,然后在您的光盘上找到它。让我们看一下草图:
#include
const int fSpeed = 70; // 70% of the full speed forward
const int bSpeed = -70; // 70% of the full speed backward
const int lDegrees = -75; // degrees to turn left
const int rDegrees = 75; // degrees to turn right
ArduinoRuntime arduinoRuntime;
BrushedMotor leftMotor(arduinoRuntime, smartcarlib::pins::v2::leftMotorPins);
BrushedMotor rightMotor(arduinoRuntime, smartcarlib::pins::v2::rightMotorPins);
DifferentialControl control(leftMotor, rightMotor);
SimpleCar car(control);
void setup()
{
Serial.begin(9600);
}
void loop()
{
handleInput();
}
void handleInput()
{ // handle serial input if there is any
if (Serial.available())
{
char input = Serial.read(); // read everything that has been received so far and log down
// the last entry
switch (input)
{
case 'l': // rotate counter-clockwise going forward
car.setSpeed(fSpeed);
car.setAngle(lDegrees);
break;
case 'r': // turn clock-wise
car.setSpeed(fSpeed);
car.setAngle(rDegrees);
break;
case 'f': // go ahead
car.setSpeed(fSpeed);
car.setAngle(0);
break;
case 'b': // go back
car.setSpeed(bSpeed);
car.setAngle(0);
break;
default: // if you receive something that you don't know, just stop
car.setSpeed(0);
car.setAngle(0);
}
}
}
上述示例指示流动站(即SimpleCar
实例)遵循从串行端口发送的简单命令。具体来说,当f
发送时,汽车将以预定义的速度前进,当b
发送时,它将向后行驶,r
它会向右和l
向左转弯。任何其他角色都会停下车。选择草图后,单击“编译”选项。
编译成功后,点击开始按钮。您会看到汽车在虚拟世界中弹出。酷吧?让我们开车吧!单击屏幕左下方的“串行”选项,键入并按键盘上的“Enter”。汽车将开始行驶。按下“跟随”选项,使相机随着汽车一起移动,并使用我们上面描述的简单命令进行操作。完成后,您可以单击停止。专业提示:您可能希望在函数结束时放置一个微小的延迟(例如),以避免占用您的 CPU 资源,因为仿真器将在您的草图中快速循环。f
loop()
delay(1)
现在您已经了解了环境物理原理以及如何通过串行端口向汽车发送命令,让我们使用一些传感器让汽车自动移动。默认情况下,汽车预装了一堆方便的传感器。默认配置包括车辆前部的超声波传感器 ( SR04) ,连接到引脚 6 和 7。如果距离小于 70 厘米的障碍物,让我们使用传感器停止汽车。
#include
ArduinoRuntime arduinoRuntime;
BrushedMotor leftMotor{arduinoRuntime, smartcarlib::pins::v2::leftMotorPins};
BrushedMotor rightMotor{arduinoRuntime, smartcarlib::pins::v2::rightMotorPins};
DifferentialControl control{leftMotor, rightMotor};
SimpleCar car(control);
const int triggerPin = 6; // D6
const int echoPin = 7; // D7
const unsigned int maxDistance = 100;
SR04 front{arduinoRuntime, triggerPin, echoPin, maxDistance};
void setup()
{
// Move the car with 50% of its full speed
car.setSpeed(50);
}
void loop()
{
const auto distance = front.getDistance();
// When distance is `0` it means there's no obstacle detected
if (distance > 0 && distance < 70) {
car.setSpeed(0);
}
#ifdef __SMCE__
// Avoid over-using the CPU if we are running in the emulator
delay(1);
#endif
}
加载草图并在模拟器中运行它。它会一直直行,直到遇到障碍物。您可以通过单击屏幕左侧的传感器读数来实时监控它们。在下面的屏幕截图中,我们可以看到,一旦遇到墙壁,电机就没有油门,并且检测到的距离在预期范围内。专业提示:如果按F3 ,您将获得传感器指向的位置及其范围的指示。这对于调试传感器输入特别有价值。
作为额外的真实感,传感器读数故意不完全准确并且包含噪声。默认设置中当前可用的传感器有:
通过串行端口发送命令并根据传感器输入使汽车自动行驶很有趣,但不可否认的是,您可以完成的任务是有限的。您经常需要您的车辆与“外部世界”进行通信,无论它可能是不同设备上的应用程序还是服务器。物理Smartcar平台围绕 ESP32 微控制器构建,因此它可以通过 WiFi 或蓝牙轻松连接到其他设备。SMCE 允许其用户通过WiFi 和 MQTT模拟连接。虽然 WiFi 库的模拟还没有完全实现,因为仍有工作要做,但应该不会超过一些#ifdef __SMCE__
编写一个可在真实硬件和仿真器上无缝运行的 Arduino 草图。让我们看看如何编写一个简单的草图,通过 MQTT 消息控制汽车并广播遥测数据,即前超声波传感器的距离测量值。注意:虽然下面的草图在 SMCE 上运行良好,但它需要一些添加/更改才能在实际的 ESP32 上运行,主要是关于 WiFi 连接。
#include <MQTT.h>
#include <WiFi.h>
#include <Smartcar.h>
#ifndef __SMCE__
WiFiClient net;
#endif
MQTTClient mqtt;
ArduinoRuntime arduinoRuntime;
BrushedMotor leftMotor(arduinoRuntime, smartcarlib::pins::v2::leftMotorPins);
BrushedMotor rightMotor(arduinoRuntime, smartcarlib::pins::v2::rightMotorPins);
DifferentialControl control(leftMotor, rightMotor);
SimpleCar car(control);
const auto oneSecond = 1000UL;
const auto triggerPin = 6;
const auto echoPin = 7;
const auto maxDistance = 400;
SR04 front(arduinoRuntime, triggerPin, echoPin, maxDistance);
void setup() {
Serial.begin(9600);
#ifdef __SMCE__
// ================= 1
// mqtt.begin("aerostun.dev", 1883, WiFi);
mqtt.begin(WiFi); // Will connect to localhost
#else
mqtt.begin(net);
#endif
// ================= 2
if (mqtt.connect("arduino", "public", "public")) {
mqtt.subscribe("/smartcar/control/#", 1);
mqtt.onMessage([](String topic, String message) {
if (topic == "/smartcar/control/throttle") {
car.setSpeed(message.toInt());
} else if (topic == "/smartcar/control/steering") {
car.setAngle(message.toInt());
} else {
Serial.println(topic + " " + message);
}
});
}
}
void loop() {
if (mqtt.connected()) {
mqtt.loop();
const auto currentTime = millis();
static auto previousTransmission = 0UL;
if (currentTime - previousTransmission >= oneSecond) {
previousTransmission = currentTime;
const auto distance = String(front.getDistance());
// ================= 3
mqtt.publish("/smartcar/ultrasound/front", distance);
}
}
#ifdef __SMCE__
// Avoid over-using the CPU if we are running in the emulator
delay(1);
#endif
}
我用内联注释突出显示了草图中的三个有趣点,然后是=================
:
begin
那么它将连接到localhost
. 您有责任在本地或远程主机上设置代理。
如果您使用的是 Ubuntu,Mosquitto是最容易设置的 MQTT 代理之一。我使用mosquitto_pub
和mosquitto_sub
实用程序订阅前超声波传感器测量并发送油门命令,如下面的屏幕截图所示。50
一旦在/smartcar/control/throttle
主题上得到适当的消息,汽车就开始加速行驶,并在/smartcar/ultrasound/front
.
当试图了解你的漫游者的环境时,传感器只能让你走这么远。SMCE 允许您获取虚拟世界的图像流。这样,您既可以在汽车的微控制器本身上进行一些非常简单的图像处理,也可以将图像流广播到功能更强大的设备并在那里进行图像处理。为简单起见,SMCE 仅支持Arduino_OV767X库,并且可以利用 MQTT 来广播流。让我们看一个草图,它为 MQTT 客户端侦听提供图像流/smartcar/camera
。和之前一样,只是增加了摄像头流:
#include
#include <MQTT.h>
#include <WiFi.h>
#include <OV767X.h>
#include <Smartcar.h>
#ifndef __SMCE__
WiFiClient net;
#endif
MQTTClient mqtt;
ArduinoRuntime arduinoRuntime;
BrushedMotor leftMotor(arduinoRuntime, smartcarlib::pins::v2::leftMotorPins);
BrushedMotor rightMotor(arduinoRuntime, smartcarlib::pins::v2::rightMotorPins);
DifferentialControl control(leftMotor, rightMotor);
SimpleCar car(control);
const auto oneSecond = 1000UL;
const auto triggerPin = 6;
const auto echoPin = 7;
const auto maxDistance = 400;
SR04 front(arduinoRuntime, triggerPin, echoPin, maxDistance);
std::vector<char> frameBuffer;
void setup() {
Serial.begin(9600);
Camera.begin(QVGA, RGB888, 15);
// ================= 1
frameBuffer.resize(Camera.width() * Camera.height() * Camera.bytesPerPixel());
#ifdef __SMCE__
// mqtt.begin("aerostun.dev", 1883, WiFi);
mqtt.begin(WiFi); // Will connect to localhost
#else
mqtt.begin(net);
#endif
if (mqtt.connect("arduino", "public", "public")) {
mqtt.subscribe("/smartcar/control/#", 1);
mqtt.onMessage([](String topic, String message) {
if (topic == "/smartcar/control/throttle") {
car.setSpeed(message.toInt());
} else if (topic == "/smartcar/control/steering") {
car.setAngle(message.toInt());
} else {
Serial.println(topic + " " + message);
}
});
}
}
void loop() {
if (mqtt.connected()) {
mqtt.loop();
const auto currentTime = millis();
static auto previousFrame = 0UL;
// ================= 2
if (currentTime - previousFrame >= 65) {
previousFrame = currentTime;
Camera.readFrame(frameBuffer.data());
mqtt.publish("/smartcar/camera", frameBuffer.data(), frameBuffer.size(),
false, 0);
}
static auto previousTransmission = 0UL;
if (currentTime - previousTransmission >= oneSecond) {
previousTransmission = currentTime;
const auto distance = String(front.getDistance());
mqtt.publish("/smartcar/ultrasound/front", distance);
}
}
#ifdef __SMCE__
// Avoid over-using the CPU if we are running in the emulator
delay(1);
#endif
}
65
毫秒,从相机中读取一帧并将其复制到frameBuffer
. 然后通过 MQTT 广播它。为了向您展示这在真实场景中会是什么样子,您可以使用参考 Android 应用程序通过 MQTT 消息控制汽车以及可视化图像流。它不是最漂亮的,但您将了解如何使用 JAVA 和 Android 完成事情的要点。
默认情况下,车辆预装了一组特定的传感器、执行器和可用引脚。这在 SMCE 的 wiki 中的车辆功能下进行了描述。要使用您自己的设置,您可以定义您的自定义配置,使用与您的草图json
位于同一目录中的文件。假设我们想要一个允许我们通过串行端口驾驶汽车的草图(如前所述),从车辆的所有四个侧面测量距离并将它们发送给我们。你最终会得到一个如下图所示的草图:
#include
const int fSpeed = 70; // 70% of the full speed forward
const int bSpeed = -70; // 70% of the full speed backward
const int lDegrees = -75; // degrees to turn left
const int rDegrees = 75; // degrees to turn right
const unsigned long transmissionInterval = 100; // In milliseconds
const int maxDistance = 300;
ArduinoRuntime arduinoRuntime;
BrushedMotor leftMotor(arduinoRuntime, smartcarlib::pins::v2::leftMotorPins);
BrushedMotor rightMotor(arduinoRuntime, smartcarlib::pins::v2::rightMotorPins);
DifferentialControl control(leftMotor, rightMotor);
SimpleCar car(control);
// ================= 1
SR04 left(arduinoRuntime, 2, 3, maxDistance); // trigger and echo pin respectively
SR04 right(arduinoRuntime, 4, 5, maxDistance);
SR04 front(arduinoRuntime, 6, 7, maxDistance);
SR04 back(arduinoRuntime, 16, 17, maxDistance);
void setup()
{
Serial.begin(9600);
}
void loop()
{
static auto previousTransmission = 0UL;
const auto currentTime = millis();
// ================= 2
if (currentTime > previousTransmission + transmissionInterval) {
previousTransmission = currentTime;
Serial.println("===");
Serial.println("Left: " + String(left.getDistance()));
Serial.println("Right: " + String(right.getDistance()));
Serial.println("Front: " + String(front.getDistance()));
Serial.println("Back: " + String(back.getDistance()));
}
handleInput();
#ifdef __SMCE__
// Avoid over-using the CPU if we are running in the emulator
delay(1);
#endif
}
void handleInput()
{
if (Serial.available())
{
char input = Serial.read(); // read everything that has been received so far and log down
// the last entry
switch (input)
{
case 'l': // rotate counter-clockwise going forward
car.setSpeed(fSpeed);
car.setAngle(lDegrees);
break;
case 'r': // turn clock-wise
car.setSpeed(fSpeed);
car.setAngle(rDegrees);
break;
case 'f': // go ahead
car.setSpeed(fSpeed);
car.setAngle(0);
break;
case 'b': // go back
car.setSpeed(bSpeed);
car.setAngle(0);
break;
default: // if you receive something that you don't know, just stop
car.setSpeed(0);
car.setAngle(0);
}
}
}
left
超声波 ( SR04
) 传感器连接到引脚2
和3
。前者是触发销,后者是回声销。transmissionInterval
。
默认设置不支持上面的草图,原因有两个:(a)默认汽车没有配备四个传感器,而SR04
只有一个,(b)在典型 ESP32 板上可用的引脚在默认情况下不可用仿真板。幸运的是,我们可以改变它!我们需要做的第一件事是使所有必要的引脚可用,并确保它们是正确的类型(即数字或模拟)。在这里我们应该注意,由于技术原因,传感器的回波引脚必须指定为。以下是您应该遵循的步骤:16
17
SR04
analog
board_config.json
gpio_drivers
属性来指定您的草图需要的所有引脚。注意:不要忘记您需要指定电机、里程表(如果您使用它们)、陀螺仪等所需的引脚。就我而言,我想移除一些我不使用的引脚(0
和),1
添加新的(16
和17
)并确保SR04
传感器使用的所有引脚都处于正确的模式。
{
"gpio_drivers": [
{ "pin": 2, "digital": true },
{ "pin": 3, "analog": true },
{ "pin": 4, "digital": true },
{ "pin": 5, "analog": true },
{ "pin": 6, "digital": true },
{ "pin": 7, "analog": true },
{ "pin": 16, "digital": true },
{ "pin": 17, "analog": true },
{ "pin": 12, "digital": true },
{ "pin": 13, "analog": true },
{ "pin": 14, "digital": true },
{ "pin": 18, "analog": true },
{ "pin": 25, "digital": true },
{ "pin": 26, "digital": true },
{ "pin": 27, "analog": true },
{ "pin": 34, "digital": true },
{ "pin": 35, "analog": true },
{ "pin": 85, "analog": true },
{ "pin": 135, "analog": true },
{ "pin": 39, "digital": true },
{ "pin": 36, "analog": true },
{ "pin": 86, "analog": true },
{ "pin": 136, "analog": true },
{ "pin": 250, "digital": true },
{ "pin": 205, "analog": true }
]
}
创建board_config.json
指定仿真 Arduino 板上可用引脚的文件后,是时候设置您的车辆了。同样,我们将使用默认设置来获取灵感并相应地对其进行修补。
vehicle_config.json
slots
您感兴趣的。在我的情况下,我想在和插槽上连接SR04
传感器Left
,使用与草图中相同的引脚,并确保这些引脚已经被.Right
Back
board_config.json
{
"slots": {
"Left": {
"class": "UltraSonic",
"name": "Left Ultrasound",
"props": {
"trigger_pin": 2,
"echo_pin": 3
}
},
"Right": {
"class": "UltraSonic",
"name": "Right Ultrasound",
"props": {
"trigger_pin": 4,
"echo_pin": 5
}
},
"FrontTop": {
"class": "UltraSonic",
"name": "Front Ultrasound",
"props": {
"trigger_pin": 6,
"echo_pin": 7
}
},
"Back": {
"class": "UltraSonic",
"name": "Back Ultrasound",
"props": {
"trigger_pin": 16,
"echo_pin": 17
}
}
}
}
专业提示:如果您想完全覆盖任何配置,可以将from_scratch
顶级元素添加为true
. 如果您想完全移除占用您根本不使用的插槽的摄像头或其他传感器,这将非常有用。引脚也是如此。例如,下面的配置将创建一辆汽车,其后侧仅配备一个超声波传感器和一对电机。没有别的,没有相机,没有里程表,没有陀螺仪。
{
"from_scratch": true,
"vehicle": "RayCar",
"slots": {
"Back": {
"class": "UltraSonic",
"name": "Back Ultrasound",
"props": {
"trigger_pin": 16,
"echo_pin": 17
}
}
},
"builtin": {
"Left BrushedMotor": {
"forward_pin": 12,
"backward_pin": 14,
"enable_pin": 13
},
"Right BrushedMotor": {
"forward_pin": 25,
"backward_pin": 26,
"enable_pin": 27
}
}
}
。
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
全部0条评论
快来发表一下你的评论吧 !