NJUPT第一次积分赛

最近在忙第二次积分赛以及一些很复杂的队友关系(人际关系好复杂,好想电赛出个单机模式),但最后结果还是很满意的。

突然想起来第一次积分赛写的屎山,遂拿出来给大火闻闻

没啥很新颖的东西,都是找一堆开源然后缝合的,所以感觉开源也没啥关系,拿出来以便后人参考。

主控和遥控器部分采用的都是esp32,主控esp32跑的RTOS,一个核用来跑FOC算法,另一个用来跑lvgl以及其他的一些东西。代码如下:

#include <Arduino.h>
#include <SimpleFOC.h>
#include <Wire.h>
#include <TFT_eSPI.h>
#include <AiEsp32RotaryEncoder.h>
#include <lvgl.h>
#include <WiFi.h>
#include <esp_now.h>
#include <stdio.h>
#include <stdlib.h> #include "esp_freertos_hooks.h"
#include "esp_netif.h"
#include "esp_eth.h"
#include "esp_event.h"
#include "esp_log.h"
#include "esp_err.h"
#include "esp_system.h"
#include "esp_spi_flash.h" #include <soc/soc.h>
#include <soc/rtc_cntl_reg.h> #define ROTARY_ENCODER1_A_PIN 4
#define ROTARY_ENCODER1_B_PIN 16
#define ROTARY_ENCODER1_BUTTON_PIN 17
#define LV_TICK_PERIOD_MS 1
#define Press_KeyUP digitalRead(15)==LOW
#define Press_KeyDOWN digitalRead(19)==LOW
#define Press_KeyLEFT digitalRead(35)==LOW
#define Press_KeyRIGHT digitalRead(32)==LOW
#define Press_KeyCENTER digitalRead(34)==LOW typedef struct _MPU6050_
{
String boardName;
int touch;
int L_theta;
int R_theta;
int pitch;
int roll;
int yaw;
}_mpu; //跑双核,core0(Task1)用来跑FOC算法,core1(Task2)用来通信与控制,core1(Task3)用来显示gui
void Task1( void *pvParameters );
void Task2( void *pvParameters ); _mpu mpuData;
AiEsp32RotaryEncoder Encoder1 = AiEsp32RotaryEncoder(ROTARY_ENCODER1_A_PIN, ROTARY_ENCODER1_B_PIN, ROTARY_ENCODER1_BUTTON_PIN, -1);
volatile int16_t encoderButton = 0; //切换串口与旋钮控制
int16_t encoderValue = 0;
int16_t encoderAngle = 0;
int16_t encoderMulti = 0;
float target = 0;
volatile int flag_mode = 0; //切换模式
int wifiConnectionStatus = 0; //wifi连接状态
volatile int clientConnectionStatus = 0; //客户端连接状态
MagneticSensorI2C sensor = MagneticSensorI2C(AS5600_I2C); //编码器为AS5600,用iic通信
BLDCMotor motor = BLDCMotor(7);//电机极对数为7
BLDCDriver3PWM driver = BLDCDriver3PWM(33,25,26,27);//控制电机转动的三相pwm输出口分别为33,25,26,其中27是Enable口
Commander command = Commander(Serial); int shaft_angle; // current motor angle
int shaft_velocity; // current motor velocity
volatile int target_velocity = 10; // current target velocity
volatile float target_angle = 0; // current target angle
volatile float target_angle_MPU;
volatile int target_velocity_MPU; TFT_eSPI tft = TFT_eSPI(240,240); //TFT Object
static lv_disp_draw_buf_t draw_buf; //定义显示器变量
static lv_color_t buf[TFT_WIDTH * 10]; //定义刷新缓存
lv_obj_t * Data = NULL;
lv_timer_t * timer = NULL; static void lv_tick_task(void *arg) {(void) arg;lv_tick_inc(LV_TICK_PERIOD_MS);}
void doTarget(char* cmd){ command.scalar(&target, cmd);}
void onMotor(char*cmd){ command.motor(&motor,cmd); }
void tft_init();
int BLDC_init();
void Encoder_Init();
void lvgl_port_init();
void my_disp_flush(lv_disp_drv_t *disp, const lv_area_t *area, lv_color_t *color_p);
void timer_1cb(lv_timer_t * timer);
void WiFi_connect(const char *SSID, const char *Password);
void WiFi_Init();
void OnDataRecv(const uint8_t * mac, const uint8_t *incomingData, int len);
void Key_init(); void setup()
{
Serial.begin(115200); WRITE_PERI_REG(RTC_CNTL_BROWN_OUT_REG, 0);//关闭低电压检测,避免无限重启
xTaskCreatePinnedToCore(Task1, "Task1", 10000, NULL, 3, NULL, 0); //最后一个参数至关重要,决定这个任务创建在哪个核上.PRO_CPU 为 0, APP_CPU 为 1,或者 tskNO_AFFINITY 允许任务在两者上运行.
xTaskCreatePinnedToCore(Task2, "Task2", 10000, NULL, 1, NULL, 1);
}
void loop() {} /*--------------------------------------------------------------------------------------------------------------------------*/
/*--------------------------------------------------下面是任务循环-----------------------------------------------------------*/
/*--------------------------------------------------------------------------------------------------------------------------*/ void Task1(void *pvParameters) //跑FOC
{
BLDC_init();
vTaskDelay(1000);
while(1)
{
switch (flag_mode)
{
case 0:if(encoderButton) target = target_velocity;break;
case 1:if(encoderButton) target = target_angle;break;
case 2:target = target_angle;break;
case 3:target = target_velocity;break;
case 4:target = target_velocity;break;
case 5:target = target_velocity;break;
}
motor.loopFOC();
motor.move(target);
}
} /*--------------------------------------------------------------------------------------------------------------------------*/
/*--------------------------------------------------------------------------------------------------------------------------*/ void Task2(void *pvParameters) //干杂活
{
Encoder_Init();
Key_init();
WiFi_Init();
//上位机指令
command.add('T',doTarget,"target value");
command.add('M',onMotor,"my motor");
motor.useMonitoring(Serial); tft_init();
lvgl_port_init();
Data = lv_label_create(lv_scr_act());
timer = lv_timer_create(timer_1cb, 50, 0); const esp_timer_create_args_t periodic_timer_args = {
.callback = &lv_tick_task, // 超时回调函数
.name = "periodic_gui" // 定时器名称
};
esp_timer_handle_t periodic_timer; // 定义定时器
ESP_ERROR_CHECK(esp_timer_create(&periodic_timer_args, &periodic_timer)); // 创建定时器
ESP_ERROR_CHECK(esp_timer_start_periodic(periodic_timer, LV_TICK_PERIOD_MS * 1000)); // 开始定时器,每LV_TICK_PERIOD_MS * 1000微妙触发一次 if (esp_now_init() != 0)
{
Serial.println("Error initializing ESP-NOW");
}
esp_now_register_recv_cb(OnDataRecv);
vTaskDelay(100);
while(1)
{
command.run();
motor.monitor();
lv_timer_handler();
vTaskDelay(5);
switch(flag_mode)
{
case 0: motor.controller = MotionControlType::velocity;;break;
case 1: motor.controller = MotionControlType::angle;break;
case 2: motor.controller = MotionControlType::angle;break; //MPU--ANGLE
case 3: motor.controller = MotionControlType::velocity;break; //MPU--VELOCITY
case 4: motor.controller = MotionControlType::velocity;break;
case 5: motor.controller = MotionControlType::velocity;break;
}
encoderValue = Encoder1.readEncoder();
encoderAngle = encoderValue * 9 % 360;
if (Encoder1.isEncoderButtonClicked())
{
static unsigned long lastTimePressed = 0;
if (millis() - lastTimePressed < 500){}
else
{
encoderButton = ~encoderButton;
}
lastTimePressed = millis();
}
if(flag_mode == 0)
{
target_velocity = encoderValue;
}
if(flag_mode == 1)
{
target_angle = encoderValue * 9.0 / 360 * 6.28 ;
}
if(flag_mode == 2)
{
target_angle = -motor.shaftAngle() - (mpuData.yaw) / 360.0 * 6.28;
}
if(flag_mode == 3)
{
target_velocity = -mpuData.pitch / 3.0;
}
if(flag_mode == 4)
{
if(mpuData.L_theta)
target_velocity = 20;
else
target_velocity = 0;
if(mpuData.R_theta)
target_velocity = -20;
else
target_velocity = 0;
}
if(flag_mode == 5)
{
target_velocity = mpuData.roll / 3;
}
if(Press_KeyRIGHT)
{
vTaskDelay(180);
flag_mode++;
}
if(flag_mode > 5) flag_mode = 0;
if(flag_mode < 0) flag_mode = 5;
}
} /*--------------------------------------------------------------------------------------------------------------------------*/
/*-----------------------------------------------------下面是函数定义--------------------------------------------------------*/
/*--------------------------------------------------------------------------------------------------------------------------*/ void tft_init()
{
tft.init();
tft.fillScreen(TFT_BLACK);
tft.setRotation(1);
tft.setTextSize(2);
tft.setTextColor(TFT_WHITE);
tft.setCursor(0,0);
}
int BLDC_init()
{
Wire.setPins(21,22); //将GPIO21、GPIO22分别设置为SDA、SCL口
Wire.begin();
sensor.init(&Wire);
motor.linkSensor(&sensor);
driver.voltage_power_supply = 12; //12V供电
driver.init();
motor.linkDriver(&driver);
motor.foc_modulation = FOCModulationType::SpaceVectorPWM; //FOC调制类型为SVPWM
//PID参数
motor.PID_velocity.P = 0.2;
motor.PID_velocity.I = 1.5;
motor.PID_velocity.D = 0.005;
motor.P_angle.P = 5.2;
motor.P_angle.I = 0.026;
motor.P_angle.D = 0.0005; motor.voltage_limit = 6;
motor.LPF_velocity.Tf = 0.06; //对速度低通滤波
motor.velocity_limit = 40; //速度上限
motor.voltage_sensor_align = 1;
motor.init();
motor.initFOC();
return 1;
}
void Encoder_Init()
{
Encoder1.begin();
Encoder1.setup([]{Encoder1.readEncoder_ISR();});
Encoder1.disableAcceleration();
Encoder1.setBoundaries(-2000, 2000, false);//一圈最多39
}
void lvgl_port_init()
{
lv_init();
lv_disp_draw_buf_init(&draw_buf, buf, NULL, TFT_WIDTH * 10);
static lv_disp_drv_t disp_drv;
lv_disp_drv_init(&disp_drv);
disp_drv.hor_res = TFT_WIDTH;
disp_drv.ver_res = TFT_HEIGHT;
disp_drv.flush_cb = my_disp_flush;
disp_drv.draw_buf = &draw_buf;
lv_disp_drv_register(&disp_drv);
}
void my_disp_flush(lv_disp_drv_t *disp, const lv_area_t *area, lv_color_t *color_p)
{
uint32_t w = (area->x2 - area->x1 + 1);
uint32_t h = (area->y2 - area->y1 + 1); tft.startWrite(); //使能写功能
tft.setAddrWindow(area->x1, area->y1, w, h); //设置填充区域
tft.pushColors((uint16_t *)&color_p->full, w * h, true); //写入颜色缓存和缓存大小
tft.endWrite(); //关闭写功能
lv_disp_flush_ready(disp); //调用区域填充颜色函数
} void timer_1cb(lv_timer_t * timer)
{
shaft_angle = fabs((int)(motor.shaftAngle()*360/6.28) % 360);
shaft_velocity = (int)motor.shaftVelocity();
lv_label_set_text_fmt(Data,"EncoderButton: %d\nEncoderValue: %d\nRotaryAngle: %d\nMotorShaftAngle: %d\nMotorShaftVelocity: %d\nmode: %d\nClient Connection: %d\npitch: %d\nroll: %d\nyaw: %d\n JoyStick: %d\ntouch: %d\n",encoderButton,encoderValue,encoderAngle, shaft_angle, shaft_velocity, flag_mode%6, clientConnectionStatus, mpuData.pitch, mpuData.roll, mpuData.yaw,mpuData.L_theta / 6.28 * 360,mpuData.touch);
} void WiFi_connect(const char *SSID, const char *Password)
{
WiFi.begin(SSID,Password);
Serial.print("WIFI Connecting");
while (WiFi.status()!=WL_CONNECTED)
{
delay(1000);
Serial.print(".");
}
Serial.println();
Serial.println("WiFi Connection Established!");
wifiConnectionStatus = 1;
Serial.print("IP Address:"); Serial.println(WiFi.localIP());
}
void WiFi_Init()
{
WiFi.mode(WIFI_STA);
WiFi.setSleep(false);
}
void OnDataRecv(const uint8_t * mac, const uint8_t *incomingData, int len)
{
memcpy(&mpuData, incomingData, sizeof(mpuData));
clientConnectionStatus = 1;
Serial.print("Board name:"); Serial.println(mpuData.boardName);
}
void Key_init()
{
pinMode(15,INPUT_PULLUP);
pinMode(19,INPUT_PULLUP);
pinMode(32,INPUT_PULLUP);
pinMode(35,INPUT_PULLUP);
pinMode(34,INPUT_PULLUP);
}

下面是遥控部分。两块esp32采用esp_now通信,摇杆部分有点bug,不会搞,但最后还是蒙混过去了。

#include <Arduino.h>
#include <WiFi.h>
#include <Wire.h>
#include <MPU6050_tockn.h>
#include <esp_now.h> #define tm T6 typedef struct _MPU6050_
{
String boardname;
int touch;
int L_theta;
int R_theta;
int pitch;
int roll;
int yaw;
}_mpu; const int threshold=48;
uint8_t broadcastAddress[] = {0x08, 0xB6, 0x1F, 0x32, 0xFE, 0xE8}; //MAC地址
String sendJSONData;
_mpu mpuData;
MPU6050 mpu6050(Wire);
int JoyStickL_X;
int JoyStickL_Y;
int JoyStickR_X;
int JoyStickR_Y;
int tc; long timer = 0; void WiFi_connect(const char *SSID, const char *Password);
void WiFi_Init();
void OnDataSent(const uint8_t *mac_addr, esp_now_send_status_t status); // 数据发送回调函数 void setup()
{
Serial.begin(115200);
WiFi_Init();
if (esp_now_init() != ESP_OK)
{
Serial.println("Error initializing ESP-NOW");
}
mpuData.touch = 0;
esp_now_register_send_cb(OnDataSent); //设置发送数据回调函数
esp_now_peer_info_t peerInfo = {};
memcpy(peerInfo.peer_addr, broadcastAddress, 6);
peerInfo.channel = 0;
peerInfo.encrypt = false;
if (esp_now_add_peer(&peerInfo) != ESP_OK)
{
Serial.println("Failed to add peer");
} pinMode(2,OUTPUT);
digitalWrite(2,LOW);
Wire.begin(21,22);
mpu6050.begin();
mpu6050.calcGyroOffsets(true); } void loop()
{
mpu6050.update();
mpuData.boardname = "MyESP_1";
mpuData.pitch = (int)mpu6050.getAngleY();
mpuData.roll = (int)mpu6050.getAngleX();
mpuData.yaw = (int)mpu6050.getAngleZ();
JoyStickL_X = analogRead(35);
JoyStickL_Y = (int)analogRead(34);
JoyStickR_X = analogRead(33);
JoyStickR_Y = analogRead(32);
tc=touchRead(tm);
if(tc < threshold)
{
mpuData.touch = 0;
}
else mpuData.touch = 1;
if(JoyStickL_Y > 4070)
mpuData.L_theta = 1;
else if(JoyStickL_Y < 20)
mpuData.L_theta = -1;
else
mpuData.L_theta = 0; if(JoyStickR_Y < 20)
mpuData.R_theta = 1;
else if(JoyStickR_Y > 4090)
mpuData.R_theta = -1;
else
mpuData.R_theta = 0;
esp_err_t result = esp_now_send(broadcastAddress, (uint8_t *) &mpuData, sizeof(mpuData));
if (result == ESP_OK)
{
Serial.println("Sent with success");
}
else
{
Serial.println("Error sending the data");
}
Serial.print(JoyStickR_Y); Serial.print(" ");Serial.println(JoyStickL_Y);
Serial.println(mpuData.touch);
delay(10);
} void WiFi_connect(const char *SSID, const char *Password)
{
WiFi.begin(SSID,Password);
Serial.print("WIFI Connecting");
while (WiFi.status()!=WL_CONNECTED)
{
delay(1000);
Serial.print(".");
}
Serial.println();
Serial.println("WiFi Connection Established!");
Serial.print("IP Address:"); Serial.println(WiFi.localIP());
}
void WiFi_Init()
{
WiFi.mode(WIFI_STA);
WiFi.setSleep(false);
}
void OnDataSent(const uint8_t *mac_addr, esp_now_send_status_t status)
{
char macStr[18];
Serial.print("Packet to: ");
snprintf(macStr, sizeof(macStr), "%02x:%02x:%02x:%02x:%02x:%02x",
mac_addr[0], mac_addr[1], mac_addr[2], mac_addr[3], mac_addr[4], mac_addr[5]);
Serial.println(macStr);
Serial.print("Send status: ");
Serial.println(status == ESP_NOW_SEND_SUCCESS ? "Delivery Success" : "Delivery Fail");
Serial.println();
}

就这么多吧,等后面忙完了把第二次积分赛的图像部分放出来。

NJUPT第一次积分赛的更多相关文章

  1. 【系列】关于NJUPT电赛自控方向第一次积分赛的总结

    本人是NJUPT电子科学与技术专业大一摸鱼狗一枚.本博客旨在总结与分享个人准备电赛所学知识,同时也是为了防止遗忘,锻炼写文章的能力.目前电赛方向为自控方向.主要研究方向为单片机.图像处理.自动控制相关 ...

  2. ACM集训第一次积分赛赛前复习+day4

    不知不觉4天过去了,我们迎来了我们第一次积分赛,赛前的四天我们学了以下知识点吧: day 1.排序 之前一直想用qsort,但是总是写不明白,STL的sort()可以说是很方便了. 先写一个最基础的数 ...

  3. 我的“第一次”,就这样没了:DDD(领域驱动设计)理论结合实践

    写在前面 插一句:本人超爱落网-<平凡的世界>这一期,分享给大家. 阅读目录: 关于DDD 前期分析 框架搭建 代码实现 开源-发布 后记 第一次听你,清风吹送,田野短笛:第一次看你,半弯 ...

  4. IIS初始化(预加载),解决第一次访问慢,程序池被回收问题

    你以为你可以慢,那是不可能的!你以为你可以不动,那也是不可能的! 河南是守株待兔故事情节的发源地,讲的是懒惰的农夫坐在树桩旁等待可爱的小毛兔撞树的故事,那么这种事情怎么可能天天出现呢!你以为的事并一定 ...

  5. 简历生成平台项目开发-STEP3第一次项目例会探讨

    时间:2016.7.13周三7点半 地点:图书馆 讨论主题:项目需求和功能分析.第一次任务分配 内容:按照之前的讨论,我们认为简历生成功能,不仅要适应学生求职的需求,更多的是要在格式和内容上满足HR的 ...

  6. android应用程序第一次启动时显示引导界面

    市面上好多优秀的应用(举例新浪微博.UC浏览器)都采用了欢迎页面与使用向导的方式给用户带来了良好的用户体验. 一般来说用户第一次安装应用或者安装了新版本后第一次进入应用都会显示成 欢迎页面-使用向导- ...

  7. 《Entity Framework 6 Recipes》中文翻译系列 (40) ------ 第七章 使用对象服务之从跟踪器中获取实体与从命令行生成模型(想解决EF第一次查询慢的,请阅读)

    翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 7-5  从跟踪器中获取实体 问题 你想创建一个扩展方法,从跟踪器中获取实体,用于数 ...

  8. Android笔记——判断程序是否第一次启动

    public class Welcome extends Activity { private final long SPLASH_LENGTH = 2000; Handler handler = n ...

  9. Coding道场:第一次

    10/23日,我在部门内部进行了一次内部学习,使用目前流行的Coding Dojo(道场)方式,进行了TDD开发的演练.演练的题目如下:     有关Coding道场的介绍,请自行百度一下,我就不再多 ...

  10. 第一次写博客Poj1044

    Date bugs Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 3005   Accepted: 889 Descript ...

随机推荐

  1. Python 中isinstance的用法

    isinstance()函数用来判断一个对象是否是一个已知的类型.isinstance(object, classinfo) 类似 type().isinstance() 与 type() 区别: t ...

  2. 使用V2R做反向代理内网穿透

    环境 内网服务器Prob1位于内网LAN1, 内网服务器Prob2位于内网LAN2, 外网服务器Serv1位于IP 123.123.123.123 内网节点配置 内网节点没有inbound,只需要配置 ...

  3. Widget模式

    Widget模式 Widget模式是指借用Web Widget思想将页面分解成组件,针对部件开发,最终组合成完整的页面,Web Widget指的是一块可以在任意页面中执行的代码块,Widget模式不属 ...

  4. Vue+SpringBoot+ElementUI实战学生管理系统-4.后端API编写

    1.章节介绍 前一篇介绍了项目的表结构设计,这一篇编写后端API,需要的朋友可以拿去自己定制.:) 2.获取源码 源码是捐赠方式获取,详细请QQ联系我 :)! 3.项目截图 登录页 列表操作 动态图 ...

  5. Vue+SpringBoot+ElementUI实战学生管理系统-3.表结构设计

    1.章节介绍 前一篇介绍了如何搭建前端工程,这一篇讲一下表结构设计,需要的朋友可以拿去自己定制.:) 2.获取源码 源码是捐赠方式获取,详细请QQ联系我 :)! 3.项目截图 登录页 列表操作 动态图 ...

  6. centos7安装postgresql9.6

    1.安装yum源 yum install -y https://download.postgresql.org/pub/repos/yum/9.6/redhat/rhel-7-x86_64/pgdg- ...

  7. 如何申请免费的SSL证书和通配符证书

    一,目前免费证书普遍存在的痛点 证书有效期普遍只有3个月 2023年12月22日阿里云官方发布公告,后续免费证书的有效期统一调整为3个月,我相信其它厂商很快也会做出调整. 调整为3个月有效期后,意味着 ...

  8. 负载均衡load balancing和算法分类概要介绍

    一.负载均衡介绍 1.1 什么是负载均衡 负载均衡(load balancing) 它是计算机的一种技术,用来在计算机集群.网络连接.CPU.磁盘驱动器或其他资源中分配负载,以达到优化资源使用.最大化 ...

  9. JAVA对象生命周期(三)-对象的销毁

    目录 从引用说起 指针直接引用 句柄引用 优缺点 如何判断对象死亡 引用计数法 可达性分析法 垃圾收集算法 标记-清除算法 复制算法 复制算法--优化 有关年轻代的JVM参数 标记-整理算法 分代收集 ...

  10. 【Azure Developer】Java代码访问Key Vault Secret时候的认证问题,使用 DefaultAzureCredentialBuilder 或者 ClientSecretCredentialBuilder

    问题描述 使用Java SDK获取Key Vault Secret机密信息时,需要获取授权.通常是使用AAD的注册应用(Client ID, Tenant ID, Client Secret)来获取  ...