树莓派之蓝牙编程

一、蓝牙配置及与手机通信

1、蓝牙介绍

(1)四种基本蓝牙profile
1. GAP Profile: Generic Access Profile,该Profile保证不同的Bluetooth产品可以互相发现对方并建立连接。 

2.SDAP Profile: Service Discovery Application Profile,通过该Profile,一个Bluetooth设备可以找到其它Bluetooth设备提供的服务,以及查询相关的信息。

3.SPP Profile: Serial Port Profile,模拟串口通讯。

4.GOEP Profile: Generic Object Exchange Profile,通用对象交换。这个Profile的名字有些费解,它定义的是数据的传输,包括同步,文件传输,或者推送其它的数据。你可以把它理解为内容无关的传输层协议,可以被任何应用用来传输自己定义的数据对象.

(2)九种应用协议(usage profile)
1.CTP Profile: Cordless Telephone Profile,无绳电话。 
2.IP Profile: Intercom Profile,这是在两个设备之间建立语音连接,换句话说,把两个昂贵的兰牙设备变成廉价的对讲机。
3. HS Profile: HeadSet Profile,用于连接耳机。  
4.DNP Profile: Dial-up Networking Profile,用于为PC提供拨号网络功能。 
5.FP Profile: Fax Profile,传真功能。 
6.LAP Profile: LAN Access Profile,使用PPP协议建立局域网。
7.OPP Profile: Object Push Profile,用于设备之间传输数据对象。 
8.FTP Profile: File Transfer Profile,用于文件传输。  
9.SP Profile: Synchronization Profile,用于不同的Bluetooth设备同步,保持数据的一致性

2、安装蓝牙相关软件包

sudo apt-get install pi-bluetooth bluez bluez-firmware blueman

3、添加pi用户到蓝牙组

sudo usermod -G bluetooth -a pi
sudo reboot
之后蓝牙可以使用

4、开启蓝牙设备

sudo vi /etc/systemd/system/dbus-org.bluez.service


修改文件中这两个语句(要根据自己的路径修改)

重启树莓派,输入hciconfig(类似ifconfig)命令查看蓝牙服务

5、查看适配器提供的各种功能

sudo sdptool browse local

6、等待连接

通过hciconfig看到我们的本地的设备是hci0,运行一下命令等待连接
sudo rfcomm watch hci0

让手机能够搜到我们的蓝牙
法1:点击Make Discoverable
这里写图片描述

法2:运行sudo hciconfig hci0 piscan

7、手机串口软件连接

这里写图片描述

手机连接上后ls /dev可以看到rfcomm0
这里写图片描述
只有在有蓝牙连接上去的时候才有这个设备

8、手机树莓派通信

  1. 树莓派上安装minicom–>设置串口为rfcomm0
  2. 手机上打开蓝牙串口,即可通信
    这里写图片描述
    这里写图片描述

参考网页:http://blog.csdn.net/wanyeye/article/details/52909869
https://www.raspberrypi.org/forums/viewtopic.php?p=919420
https://jasiek.me/2014/10/04/bluetooth-console-on-a-raspberry-pi-using-a-usb-adapter.html

9、绑定设备号 到某个MAC和channel上

创建设备节点:mknod /dev/rfcomm0 c 216 0
216是RFCOMM的设备号,可以参考…./bluez-utils-2.x/scripts/create_dev脚本
绑定:rfcomm bind /dev/rfcomm0 [MAC] [channel]
解除绑定:rfcomm release /dev/rfcomm0
查看绑定:rfcomm show /dev/rfcomm0

二、应用编程

1. 蓝牙串口操作所需要的头文件

 #include     <stdio.h>      /*标准输入输出定义*/  
 #include     <stdlib.h>     /*标准函数库定义*/  
 #include     <unistd.h>     /*Unix 标准函数定义*/  
 #include     <sys/types.h>    
 #include     <sys/stat.h>     
 #include     <fcntl.h>      /*文件控制定义*/  
 #include     <termios.h>    /*PPSIX 终端控制定义*/  
 #include     <errno.h>      /*错误号定义*/  

2.打开蓝牙串口

例如:fd = open( “/dev/rfcomm0 “, O_RDWR | O_NOCTTY | O_NDELAY);
标志O_NOCTTY可以告诉UNIX这个程序不会成为这个端口上的“控制终端”.如果不这样做的话,所有的输入,比如键盘上过来的Ctrl+C中止信号等等,会影响到你的进程.而有些程序比如getty(1M/8)则会在打开登录进程的时候使用这个特性,但是通常情况下,用户程序不会使用这个行为.
O_NDELAY标志则是告诉UNIX,这个程序并不关心DCD信号线的状态——也就是不关心端口另一端是否已经连接.如果不指定这个标志的话,除非DCD信号线上有space电压否则这个程序会一直睡眠.

3.设置蓝牙串口

波特率设置
struct termios options;
/*
 * Get the current options for the port...
 */
tcgetattr(fd, &options);
/*
 * Set the baud rates to 19200...
 */
cfsetispeed(&options, B19200);
cfsetospeed(&options, B19200);

/*
 * Enable the receiver and set local mode...
 */
options.c_cflag |= (CLOCAL | CREAD);

/*
 * Set the new options for the port...
 */
tcsetattr(fd, TCSANOW, &options);
设置字符大小

设置字符大小的时候,这里却没有像设置波特率那么方便的函数.所以,程序中需要一些位掩码运算来把事情搞定.字符大小以比特为单位指定:
options.c_flag &= ~CSIZE; / * Mask the character size bits */
options.c_flag |= CS8; /* Select 8 data bits */

设置奇偶校验和停止位
  • No parity (8N1)
    options.c_cflag &= ~PARENB
    options.c_cflag &= ~CSTOPB
    options.c_cflag &= ~CSIZE;
    options.c_cflag |= CS8;
  • Even parity (7E1)
    options.c_cflag |= PARENB
    options.c_cflag &= ~PARODD
    options.c_cflag &= ~CSTOPB
    options.c_cflag &= ~CSIZE;
    options.c_cflag |= CS7;
  • Odd parity (7O1)
    options.c_cflag |= PARENB
    options.c_cflag |= PARODD
    options.c_cflag &= ~CSTOPB
    options.c_cflag &= ~CSIZE;
    options.c_cflag |= CS7;
  • Space parity is setup the same as no parity (7S1)
    options.c_cflag &= ~PARENB
    options.c_cflag &= ~CSTOPB
    options.c_cflag &= ~CSIZE;
    options.c_cflag |= CS8;
设置输入输出

如果不是开发终端之类的,只是串口传输数据,而不需要串口来处理,那么使用原始模式(Raw Mode)方式来通讯,设置方式如下:
options.c_lflag &= ~(ICANON | ECHO | ECHOE | ISIG); /*Input*/
options.c_oflag &= ~OPOST; /*Output*/

经典输入是以面向行设计的.在经典输入模式中输入字符会被放入一个缓冲之中,这样可以以与用户交互的方式编辑缓冲的内容,直到收到CR(carriage return)或者LF(line feed)字符.
选择使用经典输入模式的时候,你通常需要选择ICANON,ECHO和ECHOE选项:
options.c_lflag |= (ICANON | ECHO | ECHOE);

通过在c_oflag成员变量中设置OPOST选项的方法程序可以选择加工过的输出.
options.c_oflag |= OPOST;
在所有选项当中,你可能只需要使用ONLCR选项来将行分隔符映射到CR-LF组合对上.其他选项主要是历史遗留,仅仅与行打印机和终端跟不上串行数据的年代有关.

设置硬件流控制

某些版本的UNIX系统支持通过CTS(Clear To Send)和RTS(Request To Send)信号线来设置硬件流控制.如果系统上定义了CNEW_RTSCTS和CRTSCTS常量,那么很可能它会支持硬件流控制.使用下面的方法将硬件流控制设置成有效:
options.c_cflag |= CNEW_RTSCTS;
将它设置成为无效的方法与此类似:
options.c_cflag &= ~CNEW_RTSCTS;

4.读写串口

发送数据
char  buffer[1024];    
int    Length;  
int    nByte;  
nByte = write(fd, buffer ,Length)     
读取数据
char  buff[1024];  
int    Len;  
int  readByte = read(fd,buff,Len);  

使用文件操作read函数读取,如果设置为原始数据模式(Raw Date Mode)传输数据,那么read函数返回的字符数是实际串口收到的字符数,也就是返回从串口输入缓冲区中实际得到的字符的个数.在不能得到数据的情况下,read(2)系统调用就会一直等着,只到有端口上新的字符可以读取或者发生超时或者错误的情况发生.
如果需要read(2)函数迅速返回的话,可以使用操作文件的函数来实现异步读取,如fcntl,或者select等来操作 。
fcntl(fd, F_SETFL, FNDELAY);
标志FNDELAY可以保证read(2)函数在端口上读不到字符的时候返回0.需要回到正常(阻塞)模式的时候,需要再次在不带FNDELAY标志的情况下调用fcntl(2)函数:
fcntl(fd, F_SETFL, 0);
当然,如果你最初就是以O_NDELAY标志打开串口的,你也可在之后使用这个方法改变读取的行为方式.

5.关闭串口

close(fd);

  • 7
    点赞
  • 81
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值