迪文科技论坛

 找回密码
 立即注册
搜索
查看: 341|回复: 1

【2023.12.31获奖项目】智能培养室

[复制链接]

567

主题

167

回帖

1万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
11954
发表于 2024-1-12 10:44:11 | 显示全部楼层 |阅读模式
【开源】智能培养室

一、简介
本开源项目是一个智能培养室,可用于学校实验室,工程研究院,化学化工厂,材料研究院等。它是基于环境的湿量,温度,高度的等物理量来满足实验的基本环境,有加热模块和风机来控温和调温,有电源功率可调来驱动多路光照,培养实验所需的恒定光强,风量等。可以开启一键高效模式,让仪器按照用户设定的参数和系统标定的参数进行自动运行,无需过度的人为干预,当系统检测到危险时,仪器会记录下此故障号,还可以不使用模拟光照,而采用大自然的光,通过窗帘偏移位置和传感器来实时反馈调节的,为科创实验发挥一点点作用!

接下来,让我们直入主题吧,硬件上有rs485接口,配合modbus协议和各种从机配合实现整个功能。

二、GUI工程
图标素材




图片素材


DGUS Tool主界面



菜单界面




用户基础参数


维护标定参数

还有欢迎界面, 全键盘界面, 确认输入界面等请自行查看GUI工程。

三、C51工程
C51的代码量很大,这里就以核心代码为举例,那就是主界面的代码,还有modbus的主处理主界面的:

  1. <font size="3" face="宋体">#include "main_win.h"
  2. #include "modbus.h"
  3. #include "sys_params.h"
  4. #include "func_handler.h"
  5. #include "uart2.h"
  6. #include <stdio.h>
  7. #include <string.h>

  8. #define TEMP_HUM_SLAVE_ADDR                        2               
  9. #define TEMP_HUM_VAL_MAX_NUM                2        

  10. #define ALERT_BIT_MAX_NUM                                30
  11. #define ALERT_BYTE_NUM                                        (ALERT_BIT_MAX_NUM/8+((ALERT_BIT_MAX_NUM%8)!=0))
  12. #define GET_ALERT_BIT(val, pos)        ((val[pos/8]>>(pos%8))&0x01)

  13. typedef struct{
  14.         char date[17];
  15.         u8 desc;
  16. }ALERT;
  17. #define ALERT_TABLE_LEN                                        20

  18. static u8 btn_sta[MAIN_WIN_BTN_MAX_NUM] = {0};
  19. static u8 btn_addr[MAIN_WIN_BTN_MAX_NUM] = {50, 51, 52, 69, 53, 54, 55, 70, 56, 57, 58, 59};
  20. u16 main_win_val[MAIN_WIN_VAL_MAX_NUM];
  21. u16 temp_hum_val[TEMP_HUM_VAL_MAX_NUM] = {0};
  22. u16 date_val[MAIN_WIN_DATE_MAX_NUM] = {0};
  23. u8 alert_val[ALERT_BYTE_NUM] = {0};
  24. u8 old_alert_val[ALERT_BYTE_NUM] = {0};
  25. ALERT alert_table[ALERT_TABLE_LEN];
  26. u16 alert_num = 0;
  27. bit is_main_win = 0;

  28. void main_win_update()
  29. {

  30. }


  31. void main_win_disp_date()
  32. {
  33.         u8 len;
  34.         
  35.         len = sprintf(common_buf, "%u:%u", (u16)date_val[3], (u16)date_val[4]);
  36.         common_buf[len+1] = 0;
  37.         sys_write_vp(MAIN_WIN_DATE_VP, common_buf, len/2+2);
  38. }
  39.         

  40. void main_win_process_alert()
  41. {
  42.         u8 i;
  43.         
  44.         for(i=0;i<ALERT_BIT_MAX_NUM;i++)
  45.         {
  46.                 if(GET_ALERT_BIT(old_alert_val, i))
  47.                         continue;
  48.                
  49.                 if(GET_ALERT_BIT(alert_val, i))
  50.                 {
  51.                         if(alert_num>=ALERT_TABLE_LEN)
  52.                                 alert_num = ALERT_TABLE_LEN-1;
  53.                         alert_table[alert_num].desc = i+1;
  54.                         sprintf(alert_table[alert_num].date, "%u/%u/%u %u:%u",
  55.                                 date_val[0], date_val[1], date_val[2], date_val[3], date_val[4]
  56.                         );
  57.                         alert_num++;
  58.                 }
  59.         }
  60.         memcpy(old_alert_val, alert_val, sizeof(alert_val));
  61. }


  62. void main_win_disp_alert()
  63. {
  64.         u16 i;
  65.         u16 val;
  66.         u16 len = 0;
  67.         
  68.         common_buf[0] = 0;
  69.         for(i=0;i<ALERT_TABLE_LEN;i++)
  70.         {
  71.                 val = 0;
  72.                 if(i<alert_num)
  73.                 {
  74.                         val = alert_table.desc;
  75.                         len += sprintf(common_buf+len, "%s\r\n", alert_table.date);
  76.                 }
  77.                 sys_write_vp(ALERT_WIN_DESC_START_VP+i, (u8*)&val, 1);
  78.         }
  79.         common_buf[len+1] = 0;
  80.         sys_write_vp(ALERT_WIN_DATE_VP, common_buf, len/2+2);
  81.         
  82. }


  83. void main_win_init()
  84. {
  85.         float fixed_val;
  86.         u8 i;
  87.         is_main_win = 1;

  88.         
  89.         main_win_val[5] = (u16)(temp_hum_val[0]/10.0+0.5f);
  90.         main_win_val[6] = (u16)(temp_hum_val[1]/10.0+0.5f);
  91.         for(i=0;i<MAIN_WIN_VAL_MAX_NUM;i++)
  92.         {
  93.                 if(i==0)
  94.                         continue;
  95.                 sys_write_vp(MAIN_WIN_WIND_SPEED_VP+MAIN_WIN_VAL_OFFSET*i, (u8*)&main_win_val, 1);
  96.         }
  97.         fixed_val = main_win_val[0]/WIND_SPEED_SCALE+FLOAT_FIX_VAL;
  98.         sys_write_vp(MAIN_WIN_WIND_SPEED_VP, (u8*)&fixed_val, 2);
  99. }


  100. void main_win_click_handler(u16 btn_val)
  101. {
  102.         u8 index;
  103.         
  104.         if(btn_val==0x0B)
  105.         {
  106.                 main_win_disp_alert();
  107.                 return;
  108.         }
  109.         
  110.         index = btn_val-1;
  111.         btn_sta[index] = !btn_sta[index];
  112.         if((index==3)||(index==7))
  113.                 btn_sta[index] = 1;
  114.         modbus_write_bit(btn_addr[index], btn_sta[index]?0xFF00:0x0000);
  115.         btn_val = btn_sta[index];
  116.         sys_write_vp(MAIN_WIN_BTN_STA_START_VP+MAIN_WIN_BTN_STA_OFFSET*index, (u8*)&btn_val, 1);
  117.         
  118.         if(index==9)
  119.                 is_main_win = 0;
  120.         else if((index==3)||(index==7))
  121.         {
  122.                 while(sys_get_touch_sta());
  123.                 modbus_write_bit(btn_addr[index], 0x0000);
  124.         }
  125. }

  126. void main_win_msg_handler(u8 *msg,u16 msg_len)
  127. {
  128.         u8 f_code = msg[MODBUS_RESPOND_POS_FUNC_CODE];
  129.         u8 data_len = msg[MODBUS_RESPOND_POS_DATA_LEN];
  130.         u8 i;
  131.         u8 offset;
  132.         msg_len = msg_len;
  133.         
  134.         if(!is_main_win)
  135.                 return;
  136.         if((f_code==MODBUS_FUNC_CODE_03)&&(data_len==MAIN_WIN_VAL_MAX_NUM*2))
  137.         {
  138.                 offset = MODBUS_RESPOND_POS_DATA;
  139.                 for(i=0;i<MAIN_WIN_VAL_MAX_NUM;i++)
  140.                 {
  141.                         main_win_val = SYS_GET_U16(msg[offset], msg[offset+1]);
  142.                         offset += 2;
  143.                 }
  144.                 main_win_update();
  145.         }else if((f_code==MODBUS_FUNC_CODE_01)&&(data_len==ALERT_BYTE_NUM))
  146.         {
  147.                 offset = MODBUS_RESPOND_POS_DATA;
  148.                 for(i=0;i<ALERT_BYTE_NUM;i++)
  149.                 {
  150.                         alert_val = msg[offset];
  151.                         offset++;
  152.                 }
  153.                 main_win_process_alert();
  154.         }else if((f_code==MODBUS_FUNC_CODE_03)&&(data_len==TEMP_HUM_VAL_MAX_NUM*2))
  155.         {
  156.                 offset = MODBUS_RESPOND_POS_DATA;
  157.                 for(i=0;i<TEMP_HUM_VAL_MAX_NUM;i++)
  158.                 {
  159.                         temp_hum_val = SYS_GET_U16(msg[offset], msg[offset+1]);
  160.                         offset += 2;
  161.                         modbus_write_word(5+i, temp_hum_val);
  162.                 }
  163.                 main_win_update();
  164.         }else if((f_code==MODBUS_FUNC_CODE_03)&&(data_len==MAIN_WIN_DATE_MAX_NUM*2))
  165.         {
  166.                 offset = MODBUS_RESPOND_POS_DATA;
  167.                 for(i=0;i<MAIN_WIN_DATE_MAX_NUM;i++)
  168.                 {
  169.                         date_val = SYS_GET_U16(msg[offset], msg[offset+1]);
  170.                         offset += 2;
  171.                 }
  172.                 main_win_disp_date();
  173.         }
  174. }


  175. void main_win_read_temp_hum()
  176. {
  177.         u8 old_slave_addr = SLAVE_ADDR;
  178.         
  179.         sys_params.user_config[5] = TEMP_HUM_SLAVE_ADDR;
  180.         modbus_read_words(0, TEMP_HUM_VAL_MAX_NUM);
  181.         sys_params.user_config[5] = old_slave_addr;//还原
  182. }


  183. void main_win_handler()
  184. {
  185.         static u8 flag = 0;
  186.         
  187.         if(is_main_win)
  188.         {
  189.                 if(alert_read_period==ALERT_READ_PERIOD)
  190.                 {
  191.                         alert_read_period = 0;
  192.                         modbus_read_bits(510, ALERT_BIT_MAX_NUM);
  193.                         return;
  194.                 }
  195.                 if(date_update_period==DATE_UPDATE_PERIOD)
  196.                 {
  197.                         date_update_period = 0;
  198.                         modbus_read_words(180, MAIN_WIN_DATE_MAX_NUM);
  199.                         return;
  200.                 }
  201.                
  202.                 flag = !flag;
  203.                 if(flag)
  204.                         modbus_read_words(0, MAIN_WIN_VAL_MAX_NUM);
  205.                 else
  206.                         main_win_read_temp_hum();
  207.         }
  208. }


  209. modbus的:
  210. #include "modbus.h"
  211. #include "crc16.h"
  212. #include "sys_params.h"

  213. #define UART_INCLUDE                        "uart2.h"
  214. #define UART_INIT                                        uart2_init
  215. #define UART_SEND_BYTES                uart2_send_bytes
  216. #define UART_BAUD                                        9600

  217. #define MODBUS_RECV_TIMEOUT                        (u8)(35000.0f/UART_BAUD+2)
  218. #define MODBUS_SEND_INTERVAL                150               



  219. #include UART_INCLUDE

  220. static bit is_modbus_recv_complete = 0;
  221. static u8 modbus_recv_buff[270];
  222. static u16 modbus_recv_len = 0;//接受的总字节长度
  223. static u8 modbus_recv_timeout = 0;//接受溢出时间
  224. static volatile u16 modbus_send_interval = 0;
  225. MODBUS_PACKET packet;


  226. void modbus_init()
  227. {
  228.         UART_INIT(UART_BAUD);

  229. }


  230. void modbus_send_bytes(u8 *bytes,u16 len)
  231. {
  232.         UART_SEND_BYTES(bytes,len);
  233. }



  234. void modbus_recv_byte(u8 byte)
  235. {
  236.         if(is_modbus_recv_complete)
  237.                 return;
  238.         if(modbus_recv_len<sizeof(modbus_recv_buff))
  239.                 modbus_recv_buff[modbus_recv_len++] = byte;
  240. }


  241. void modbus_check_recv_timeout()
  242. {
  243.         if(modbus_recv_timeout)
  244.         {
  245.                 modbus_recv_timeout--;
  246.                 if(modbus_recv_timeout==0)
  247.                 {
  248.                         is_modbus_recv_complete = 1;
  249.                 }
  250.         }
  251. }


  252. u8 modbus_send_packet(u8 *packet)
  253. {
  254.         u16 len;
  255.         u16 crc;
  256.         u8 func_code = packet[1];
  257.         
  258.         while(modbus_send_interval);
  259.         if(func_code==MODBUS_FUNC_CODE_10)
  260.         {
  261.                 ((MODBUS_10_PACKET*)packet)->byte_num = ((MODBUS_10_PACKET*)packet)->word_num*2;
  262.                 len = 9+((MODBUS_10_PACKET*)packet)->byte_num;
  263.         }else if(func_code==MODBUS_FUNC_CODE_0F)
  264.         {
  265.                 len = ((MODBUS_0F_PACKET*)packet)->bit_num;
  266.                 ((MODBUS_0F_PACKET*)packet)->byte_num = len/8+(len%8?1:0);
  267.                 len = 9+((MODBUS_0F_PACKET*)packet)->byte_num;
  268.         }else
  269.         {
  270.                 len = sizeof(MODBUS_PACKET);
  271.         }

  272.         crc = crc16(packet,len-2);
  273.         packet[len-2] = (u8)(crc>>8);
  274.         packet[len-1] = (u8)crc;
  275.         modbus_send_bytes(packet,len);
  276.         modbus_send_interval = MODBUS_SEND_INTERVAL;
  277.         
  278.         return 0;//成功
  279. }



  280. extern void modbus_msg_handler(u8 *msg,u16 msg_len);
  281. void modbus_handler()
  282. {
  283.         u16 crc;
  284.         
  285.         if(!is_modbus_recv_complete)
  286.                 return;
  287.         //校验crc值
  288.         crc = ((u16)modbus_recv_buff[modbus_recv_len-2]<<8)+modbus_recv_buff[modbus_recv_len-1];
  289.         if(crc16(modbus_recv_buff,modbus_recv_len-2)==crc)
  290.         {
  291.                 modbus_msg_handler(modbus_recv_buff,modbus_recv_len);
  292.         }
  293.         modbus_recv_len = 0;
  294.         is_modbus_recv_complete = 0;        
  295. }


  296. u8 modbus_send_fcode(u8 fcode, u16 addr, u16 len)
  297. {
  298.         packet.slave_addr = SLAVE_ADDR;
  299.         packet.func_code = fcode;//功能码
  300.         packet.start_addr = addr;//地址
  301.         packet.data_len = len;//写入的值
  302.         len = modbus_send_packet((u8*)&packet);
  303.         
  304.         return len;
  305. }</font>
复制代码

四、视频演示效果
视频入口





备注说明:如需源码,请参考此链接:http://inforum.dwin.com.cn:20080/forum.php?mod=viewthread&tid=9586&_dsign=2417fdbe






本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

1

主题

8

回帖

130

积分

注册会员

Rank: 2

积分
130
发表于 2024-3-3 14:32:39 | 显示全部楼层
醉得太好了,十分羡慕。加油。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|Archiver|手机版|小黑屋|迪文科技论坛 ( 京ICP备05033781号-1 )

GMT+8, 2024-11-21 22:53 , Processed in 0.054358 second(s), 22 queries .

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表