[应用方案] free同学的开发日记-微信蓝牙协议栈移植与应用

[复制链接]
2511|14
 楼主| gaoyang9992006 发表于 2016-2-24 15:04 | 显示全部楼层 |阅读模式
本帖最后由 gaoyang9992006 于 2016-2-24 15:12 编辑

1、微信的基本架构(节选自微信官方协议文档) 001.png

厂商服务器和外设,由厂商开发完成。
微信会提供服务器的接口以对接厂商的服务器,会提供手机的接口(如本文规定的蓝牙协议)以对接厂商的外设。
主要功能该协议打通了设备和厂商服务器之间的数据链路,也就是支持将设备上的数据发送到厂商的服务器上,也支持将厂商的数据发送到设备。
厂商的数据对于微信来说,是黑盒,微信不对设备数据做分析。

该协议也打通了设备和微信服务器之间的数据链路。
设备和微信服务器之间的数据格式由微信规定,例如登录,新消息通知等。

2、微信蓝牙一般是基于串口的,所以只跟串口底层相关,也就是只需要将串口发送和接收两个函数填充到协议栈对外接口函数就行了。
先把微信SDK包加载在我们自己的项目中,如下我的项目添加到了interactive目录,4个c文件和4个h文件。
整个移植过程我们只需要改动proto.c文件里边5个接口函数。
002.jpg
3、微信传输的数据包结构
003.jpg
4、先把串口准备好,我的都在这个是uart.c文件中,以前的代码写的狗屎一般,看思路就行了。
  1. #include "uart.h"
  2. #include "uart_config.h"

  3. enum BLE_TRAN_FSM
  4. {
  5.     bMagicNumber = 0,
  6.     bVer,
  7.     nLenghth_hi,
  8.     nLenghth_low,
  9.     bdata,
  10. };  

  11. unsigned char recv_from_ble_buf[60]= {0};
  12. unsigned char m_frame_buffer[256]={0};   

  13. unsigned char rxbuf[256]={0};
  14. unsigned char rx_idx=0;
  15. unsigned short uart0_rx_len;
  16. unsigned short uart0_rx_cnt=0;
  17. unsigned short uart0_buf_num=0;
  18. static unsigned char uart0_tran_state = 0;
  19. unsigned char uart0_tran_complt = 0;

  20. void UART0_Init(void)
  21. {   
  22. }

  23. void uart0_putchar(unsigned char dat)
  24. {
  25. }

  26. unsigned char uart0_getchar(void)
  27. {
  28.     return 0;
  29. }

  30. unsigned char UART0_Put_N_Byte (unsigned char* data_buf,unsigned char data_len)
  31. {
  32.     unsigned char i;     
  33.     //BT_EN_OFF;   
  34.     for (i = 0; i < data_len; i++)
  35.     {
  36.         uart0_putchar(data_buf[i]);
  37.     }   
  38.     //BT_EN_ON;
  39.     return data_len;
  40. }

  41. //拷贝串口缓冲区数据
  42. unsigned char  UART0_Get_Buf(unsigned char* data_buf)
  43. {
  44.     unsigned char i;
  45.     if(uart0_tran_complt)      
  46.     {        
  47.         uart0_buf_num = uart0_rx_len;
  48.         for(i=0;i<uart0_rx_len;i++)
  49.         {
  50.             *data_buf ++ = rxbuf[i];
  51.         }
  52.         uart0_tran_complt = 0;
  53.         uart0_rx_len = 0;               
  54.     }
  55.     else
  56.     {
  57.         uart0_buf_num = 0;
  58.     }
  59.     return uart0_buf_num;
  60. }

  61. void UART0_ISR(void)
  62. {        
  63.     rxbuf[rx_idx]=uart0_getchar(); //串口接收是从数据寄存器你读取数据,这个为了脱离平台限制,
  64.                                    //添加了一个uart0_getchar()的空函数      
  65.     rx_idx+=1;   
  66.     switch(uart0_tran_state)
  67.     {
  68.         case bMagicNumber:
  69.         if(MAGIC_NUMBER_ID == rxbuf[0])
  70.         {            
  71.             uart0_tran_state = bVer;
  72.         }
  73.         else
  74.         {
  75.             uart0_tran_state = bMagicNumber; rx_idx=0;
  76.         }
  77.         break;
  78.         case bVer:
  79.         if(DEVEICE_VERSION_ID == rxbuf[1])
  80.         {
  81.             uart0_tran_state = nLenghth_hi;
  82.             //uart0_putstring("uart0_tran_state : bVer \r\n");
  83.             //预留用一个定时器来做超时检测
  84.             //uart0_timeout = 20;                     
  85.         }
  86.         else
  87.         {
  88.             uart0_tran_state = bMagicNumber; rx_idx=0;
  89.         }
  90.         break;
  91.         
  92.         case nLenghth_hi:                  
  93.         uart0_tran_state = nLenghth_low;        
  94.         break;  
  95.         
  96.         case nLenghth_low:      
  97.         uart0_rx_len = (unsigned short) ((rxbuf[2]<<8)|rxbuf[3]);
  98.         uart0_tran_state = bdata;
  99.         break;
  100.         
  101.         case bdata:      
  102.         if(rx_idx == uart0_rx_len)
  103.         {              
  104.             uart0_tran_complt = 1;
  105.             uart0_tran_state = bMagicNumber;
  106.             rx_idx=0;            
  107.         }
  108.         
  109.         //串口超时之后再添加  
  110.         //if(uart0_timeout)
  111.         //{
  112.         //    for(iRev=0;iRev<20;iRev++) {rxbuf[]=0;rx_idx=0;}         
  113.         //}               
  114.         break;
  115.         
  116.         default:
  117.         uart0_tran_state = bMagicNumber; rx_idx=0;        
  118.         break;
  119.     }           
  120. }
5、将proto.c文件里边
    void Auth(unsigned char *pBuf , unsigned short  int Length);
        void Unpack_Auth_Response(unsigned char *buf , unsigned short  int Length );
        void Unpack_Init_Response(unsigned char *buf , unsigned short  int Length );
        void SendDataToManuf(unsigned char *buf , unsigned short  int Length);
        void Unpack_SendDataToManufacturerSvrResponse(unsigned char *buf , unsigned short  int Length );
5个函数中的串口发送函数换成我们自己的。
004.jpg
6、我们接收厂商服务器的数据都在
void Unpack_ManufacturerSvrSendDataPush(unsigned char *buf , unsigned short  int Length )
函数你解析.
005.jpg


7、连接过程
006.jpg


 楼主| gaoyang9992006 发表于 2016-2-24 15:07 | 显示全部楼层
free同学最近在做开发笔记,写的很不错,我征求了意见,将抽空转载过来。以便大家跟free学习他的成功经验。
打赏我的话,我会把钱钱分给他的。:lol
yiyigirl2014 发表于 2016-2-24 15:10 | 显示全部楼层
楼主这个是不是不分什么单片机都是可以用的?
 楼主| gaoyang9992006 发表于 2016-2-24 15:11 | 显示全部楼层
yiyigirl2014 发表于 2016-2-24 15:10
楼主这个是不是不分什么单片机都是可以用的?

是的,都是可以的。不谈什么单片机。
xyz549040622 发表于 2016-2-24 15:26 | 显示全部楼层
讲的有点简单。灰哥貌似发过一个详细的帖子的。
643757107 发表于 2016-2-24 19:44 | 显示全部楼层
该协议也打通了设备和微信服务器之间的数据链路。
神奇啊。虽然不长不过我都看懂了。
598330983 发表于 2016-2-24 21:27 | 显示全部楼层
微信蓝牙一般是基于串口的,所以只跟串口底层相关,也就是只需要将串口发送和接收两个函数填充到协议栈对外接口函数就行了。
嗯,我看好多的WIFI都是用的串口的模块。
dentsgot 发表于 2016-2-24 22:27 | 显示全部楼层
说的是微信可以在用户之间通过蓝牙传输?
Harvard 发表于 2016-2-25 11:27 | 显示全部楼层
mark一下 看看
yiyigirl2014 发表于 2016-2-25 20:50 | 显示全部楼层
原来可以通过本地的蓝牙方式进行直接通信,以前总觉得微信只能通过公共网络传输。
dongnanxibei 发表于 2016-2-25 22:08 | 显示全部楼层
厂商和微信提供了设备和接口,那么我们只需要对这些接口操作就行了,避免了复杂的协议,已经存储交换。
Jessicakjdsl 发表于 2016-2-26 08:59 | 显示全部楼层
感觉现在微信做的越来越强大了啊,以后估计能接入硬件了
wahahaheihei 发表于 2016-2-26 20:47 | 显示全部楼层
现在好多的物联网都接入了微信。另外我发现微信有的功能QQ有,而QQ功能更强大,为何大家都喜欢微信呢,我就好奇怪。
Bermanrep 发表于 2016-2-26 21:31 | 显示全部楼层
这个外围的硬件用的是什么啊
大苏牙 发表于 2016-2-28 22:53 | 显示全部楼层
楼主能否具体讲解一下微信蓝牙的定义呢
您需要登录后才可以回帖 登录 | 注册

本版积分规则

个人签名:如果你觉得我的分享或者答复还可以,请给我点赞,谢谢。

2052

主题

16403

帖子

222

粉丝
快速回复 在线客服 返回列表 返回顶部