#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/ioctl.h>
#include<time.h>
#include <termios.h>
#include <sys/select.h>
#include <sys/types.h>
//#include "COMM.h"
#define ACK_SUCCESS 0x00 // 操作成功
#define ACK_FAIL 0x01 // 操作失敗
#define ACK_FULL 0x04 // Full
#define ACK_NOUSER 0x05 // 無此用戶
#define ACK_USER_EXIST 0x07 // 用戶已存在
#define ACK_TIMEOUT 0x08 // 採集超時
#define ACK_COUNT 0x3 //發生錯誤時候,重試次數
extern unsigned char TXBUF[9900];
extern unsigned char RXBUF[9900];
extern unsigned char rev_ok;
extern unsigned int TX_len;
extern unsigned int RX_len;
extern unsigned char one_onecontrast(unsigned char user_number_h,unsigned char user_number_l);
extern unsigned char one_morecontrast(void);
extern unsigned char Get_UserNumber_Right(void);
extern unsigned char set_addmode(unsigned char yn_repeat);
extern unsigned char add_Fingerprint(unsigned char time_number,unsigned char user_number_h,
unsigned char user_number_l,unsigned char user_right);
extern unsigned char del_alluser(void);
extern unsigned char del_oneuser(unsigned char user_number_h,unsigned char user_number_l);
extern unsigned char read_usernumber(void);
int fd;
unsigned char USER_NUMBER_H; // 用戶號高8位
unsigned char USER_NUMBER_L; // 用戶號低8位
unsigned char USER_RIGHT; // 用戶權限
unsigned char FEATURE_BUFFER[512]; // 要下傳的指紋特徵值數據
unsigned char FEATURE_LEN; // 要下傳的指紋特徵值的長度
unsigned char CharToHex(unsigned char ch);
unsigned char select_one_onecontrast(void);
void select_Get_Usernumber_right(void);
/*********************************************************************************************
* name: SERIAL_TX
* func: 發生數據到指紋模塊串口
* para: none
* ret: none
* modify:
* comment:
*********************************************************************************************/
int SERIAL_TX(void)
{
int ret;
ret = write(fd, TXBUF, TX_len); // 試圖從串口發送數 據
if(ret == -1) // 確實接收到了數據,並打印出來
{
// *(rcv_buf+ret)='\0';
printf(" Write device error!\n");
return -1;
// ret = 0;
}
return 0;
}
/*********************************************************************************************
* name: SERIAL_RX
* func: 從指紋模塊串口接收數據
* para: none
* ret: none
* modify:
* comment:
*********************************************************************************************/
void SERIAL_RX(void)
{
// read(fd, RXBUF , RX_len);
#if 1
int ret,n,pos,retval;
fd_set rfds;
struct timeval tv ;
pos = 0;//指向接收緩衝
tv.tv_sec = 2;
tv.tv_usec = 0;
for(n = 0; n < RX_len; n++)
{
RXBUF[n] = 0xFF;
}
//while(FD_ISSET(fd,&uart_r)||FD_ISSET(fd,&uart_w)); // 檢測串口是否有讀寫動作
while(1) // 檢測串口是否有讀寫動作
{
FD_ZERO(&rfds);// 清空串口接收端口集
FD_SET(fd,&rfds);// 設置串口接收端口集
retval = select(fd+1,&rfds,NULL,NULL,&tv);
if(retval == -1)
{
perror("select()");
break;
}
else if(retval)
{ //判斷是否還有數據
//sleep(2);
ret = read(fd, RXBUF, RX_len);
pos += ret;
//printf("ret = %d \n",ret);
if((RXBUF[pos-2] == '\r') & (RXBUF[pos-1] == '\n')) // 確實接收到了數據,並打印出來
{
FD_ZERO(&rfds);
FD_SET(fd,&rfds);
retval = select(fd+1,&rfds,NULL,NULL,&tv);
if(!retval)//no datas
{
break;
}
}
}
else
{
break;
}
}
}
void init_ttyS(int fd)
{
struct termios newtio;
bzero(&newtio, sizeof(newtio));
tcgetattr(fd, &newtio); // 得到當前串口的參數
cfsetispeed(&newtio, B19200); // 將輸入波特率設爲19200
cfsetospeed(&newtio, B19200); // 將輸出波特率設爲19200
newtio.c_cflag |= (CLOCAL | CREAD); // 使能接收並使能本地狀態
newtio.c_cflag &= ~PARENB; // 無校驗 8位數據位1位停止位
newtio.c_cflag &= ~CSTOPB;
newtio.c_cflag &= ~CSIZE;
newtio.c_cflag |= CS8;
newtio.c_lflag &= ~(ICANON | ECHO | ECHOE | ISIG); // 原始數據輸入
newtio.c_oflag &= ~(OPOST);
newtio.c_cc[VTIME] = 0; // 設置等待時間和最小接收字符數
newtio.c_cc[VMIN] = 0;
tcflush(fd, TCIFLUSH); // 處理未接收的字符
tcsetattr(fd,TCSANOW,&newtio); // 激活新配置
}
unsigned char set_addmode(unsigned char yn_repeat)
{
unsigned char check, i;
TXBUF[0] = 0xF5;
TXBUF[1] = 0x2D;
TXBUF[2] = 0x00;
TXBUF[3] = yn_repeat;
TXBUF[4] = 0x00;
TXBUF[5] = 0x00;
check = TXBUF[1];
for (i = 2; i < 6; i++)
{
check ^= TXBUF[i];
}
TXBUF[6] = check;
TXBUF[7] = 0xF5;
rev_ok = 1;
TX_len = 8;
RX_len = 8;
SERIAL_TX();
sleep(delaytime);
SERIAL_RX();
rev_ok = RXBUF[4];
return (rev_ok);
}
int main(int argc, char ** argv)
{
int t,ret;
char mode;
char *device = "/dev/tts/0"; // 設備路徑,初始使用UART0
for(t=1;t<argc;t++) // 獲取程序入口時輸入的參數
{
if(!strcmp(argv[t],"-d") && (argc > (t+1)))
{
device = argv[t+1];
}
}
if(!strcmp(device,"/dev/tts/1")) // 不允許使用UART1,因爲它已和PC相連。
{
printf("can not use /dev/tts/1\n");
return -1;
}
fd = open(device, O_RDWR); // 打開設備
if (fd < 0) // 設備打開失敗
{
printf("open device error\n");
return -1;
}
init_ttyS(fd); // 初始化設備
while(1)
{
set_addmode(1); // 設置指紋添加模式,禁止重複
}
close(fd); // 關閉打開的設備
return 0; // 正常返回
}
-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------2-----------------------------------------------------------------------------------------------------------------------------------
據通信的基本方式可分爲並行通信與串行通信兩種。
· 並行通信是指利用多條數據傳輸線將一個資料的各位同時傳送。它的特點是傳輸速度
快,適用於短距離通信,但要求傳輸速度較高的應用場合。
· 串行通信是指利用一條傳輸線將資料一位位地順序傳送。特點是通信線路簡單,利用
簡單的線纜就可實現通信,降低成本,適用於遠距離通信,但傳輸速度慢的應用場合。
串口設置詳解
本節主要講解設置串口的主要方法。
如前所述,設置串口中最基本的包括波特率設置,校驗位和停止位設置。串口的設置主
要是設置struct termios結構體的各成員值,如下所示:
#include
struct termio
{
unsigned short c_iflag; /* 輸入模式標誌 */
unsigned short c_oflag; /* 輸出模式標誌 */
unsigned short c_cflag; /* 控制模式標誌*/
unsigned short c_lflag; /*本地模式標誌 */
unsigned char c_line; /* line discipline */
unsigned char c_cc[NCC]; /* control characters */
};
在這個結構中最爲重要的是c_cflag,通過對它的賦值,用戶可以設置波特率、字符大小、
數據位、停止位、奇偶校驗位和硬件流控等。另外c_iflag 和c_cc 也是比較常用的標誌。在
此主要對這3 個成員進行詳細說明。
c_cflag支持的常量名稱
CBAUD 波特率的位掩碼
B0 0波特率(放棄DTR)
B1800 1800波特率
B2400 2400波特率
B4800 4800波特率
B9600 9600波特率
B19200 19200波特率
B38400 38400波特率
B57600 57600波特率
B115200 115200波特率
EXTA 外部時鐘率
EXTB 外部時鐘率
CSIZE 數據位的位掩碼
CS5 5個數據位
CS6 6個數據位
CS7 7個數據位
CS8 8個數據位
CSTOPB 2個停止位(不設則是1個停止位)
CREAD 接收使能
PARENB 校驗位使能
PARODD 使用奇校驗而不使用偶校驗
HUPCL 最後關閉時掛線(放棄DTR)
CLOCAL 本地連接(不改變端口所有者)
LOBLK 塊作業控制輸出
CNET_CTSRTS 硬件流控制使能
c_iflag支持的常量名稱
INPCK 奇偶校驗使能
IGNPAR 忽略奇偶校驗錯誤
PARMRK 奇偶校驗錯誤掩碼
ISTRIP 除去奇偶校驗位
IXON 啓動出口硬件流控
IXOFF 啓動入口軟件流控
IXANY 允許字符重新啓動流控
IGNBRK 忽略中斷情況
BRKINT 當發生中斷時發送SIGINT信號
INLCR 將NL映射到CR
IGNCR 忽略CR
ICRNL 將CR映射到NL
IUCLC 將高位情況映射到低位情況
IMAXBEL 當輸入太長時回覆ECHO
c_cc 支持的常量名稱
VINTR 中斷控制,對應鍵爲CTRL+C
VQUIT 退出操作,對應鍵爲CRTL+Z
VERASE 刪除操作,對應鍵爲Backspace(BS)
VKILL 刪除行,對應鍵爲CTRL+U
VEOF 位於文件結尾,對應鍵爲CTRL+D
VEOL 位於行尾,對應鍵爲Carriage return(CR)
VEOL2 位於第二行尾,對應鍵爲Line feed(LF)
VMIN 指定了最少讀取的字符數
VTIME 指定了讀取每個字符的等待時間
串口控制函數
Tcgetattr 取屬性(termios結構)
Tcsetattr 設置屬性(termios結構)
cfgetispeed 得到輸入速度
Cfgetospeed 得到輸出速度
Cfsetispeed 設置輸入速度
Cfsetospeed 設置輸出速度
Tcdrain 等待所有輸出都被傳輸
tcflow 掛起傳輸或接收
tcflush 刷清未決輸入和/或輸出
Tcsendbreak 送BREAK字符
tcgetpgrp 得到前臺進程組ID
tcsetpgrp 設置前臺進程組ID
[color=#ff0000]完整的串口配置模板,實用!把常用的選項在函數裏面列出,可大大方便用戶的調試使用[/color]
int set_opt(int fd,int nSpeed, int nBits, char nEvent, int nStop)
{
struct termios newtio,oldtio;
/*保存測試現有串口參數設置,在這裏如果串口號等出錯,會有相關的出錯信息*/
if ( tcgetattr( fd,&oldtio) != 0)
{
perror("SetupSerial 1");
return -1;
}
bzero( &newtio, sizeof( newtio ) );
/*步驟一,設置字符大小*/
newtio.c_cflag |= CLOCAL | CREAD;
newtio.c_cflag &= ~CSIZE;
/*設置停止位*/
switch( nBits )
{
case 7:
newtio.c_cflag |= CS7;
break;
case 8:
newtio.c_cflag |= CS8;
break;
}
/*設置奇偶校驗位*/
switch( nEvent )
{
case 'O': //奇數
newtio.c_cflag |= PARENB;
newtio.c_cflag |= PARODD;
newtio.c_iflag |= (INPCK | ISTRIP);
break;
case 'E': //偶數
newtio.c_iflag |= (INPCK | ISTRIP);
newtio.c_cflag |= PARENB;
newtio.c_cflag &= ~PARODD;
break;
case 'N': //無奇偶校驗位
newtio.c_cflag &= ~PARENB;
break;
}
/*設置波特率*/
switch( nSpeed )
{
case 2400:
cfsetispeed(&newtio, B2400);
cfsetospeed(&newtio, B2400);
break;
case 4800:
cfsetispeed(&newtio, B4800);
cfsetospeed(&newtio, B4800);
break;
case 9600:
cfsetispeed(&newtio, B9600);
cfsetospeed(&newtio, B9600);
break;
case 115200:
cfsetispeed(&newtio, B115200);
cfsetospeed(&newtio, B115200);
break;
case 460800:
cfsetispeed(&newtio, B460800);
cfsetospeed(&newtio, B460800);
break;
default:
cfsetispeed(&newtio, B9600);
cfsetospeed(&newtio, B9600);
break;
}
/*設置停止位*/
if( nStop == 1 )
newtio.c_cflag &= ~CSTOPB;
else if ( nStop == 2 )
newtio.c_cflag |= CSTOPB;
/*設置等待時間和最小接收字符*/
newtio.c_cc[VTIME] = 0;
newtio.c_cc[VMIN] = 0;
/*處理未接收字符*/
tcflush(fd,TCIFLUSH);
/*激活新配置*/
if((tcsetattr(fd,TCSANOW,&newtio))!=0)
{
perror("com set error");
return -1;
}
printf("set done!\n");
return 0;
}
[b]串口使用詳解[/b]
在配置完串口的相關屬性後,就可對串口進行打開,讀寫操作了。其使用方式與文件操作一樣,區別在於串口是一個終端設備。
[b]打開串口[/b]
fd = open( "/dev/ttyS0", O_RDWR|O_NOCTTY|O_NDELAY);
Open函數中除普通參數外,另有兩個參數O_NOCTTY和O_NDELAY。
O_NOCTTY: 通知linix系統,這個程序不會成爲這個端口的控制終端。
O_NDELAY: 通知linux系統不關心DCD信號線所處的狀態(端口的另一端是否激活或者停止)。
然後,恢復串口的狀態爲阻塞狀態,用於等待串口數據的讀入。用fcntl函數:
fcntl(fd, F_SETFL, 0);
接着,測試打開的文件描述府是否引用一個終端設備,以進一步確認串口是否正確打開。
isatty(STDIN_FILENO);
串口的讀寫與普通文件一樣,使用read,write函數。
read(fd,buff,8);
write(fd,buff,8);
實例
#i nclude stdio.h>
#i nclude string.h>
#i nclude sys/types.h>
#i nclude errno.h>
#i nclude sys/stat.h>
#i nclude fcntl.h>
#i nclude unistd.h>
#i nclude termios.h>
#i nclude stdlib.h>
int set_opt(int fd,int nSpeed, int nBits, char nEvent, int nStop)
{
struct termios newtio,oldtio;
if ( tcgetattr( fd,&oldtio) != 0) {
perror("SetupSerial 1");
return -1;
}
bzero( &newtio, sizeof( newtio ) );
newtio.c_cflag |= CLOCAL | CREAD;
newtio.c_cflag &= ~CSIZE;
switch( nBits )
{
case 7:
newtio.c_cflag |= CS7;
break;
case 8:
newtio.c_cflag |= CS8;
break;
}
switch( nEvent )
{
case 'O':
newtio.c_cflag |= PARENB;
newtio.c_cflag |= PARODD;
newtio.c_iflag |= (INPCK | ISTRIP);
break;
case 'E':
newtio.c_iflag |= (INPCK | ISTRIP);
newtio.c_cflag |= PARENB;
newtio.c_cflag &= ~PARODD;
break;
case 'N':
newtio.c_cflag &= ~PARENB;
break;
}
switch( nSpeed )
{
case 2400:
cfsetispeed(&newtio, B2400);
cfsetospeed(&newtio, B2400);
break;
case 4800:
cfsetispeed(&newtio, B4800);
cfsetospeed(&newtio, B4800);
break;
case 9600:
cfsetispeed(&newtio, B9600);
cfsetospeed(&newtio, B9600);
break;
case 115200:
cfsetispeed(&newtio, B115200);
cfsetospeed(&newtio, B115200);
break;
default:
cfsetispeed(&newtio, B9600);
cfsetospeed(&newtio, B9600);
break;
}
if( nStop == 1 )
newtio.c_cflag &= ~CSTOPB;
else if ( nStop == 2 )
newtio.c_cflag |= CSTOPB;
newtio.c_cc[VTIME] = 0;
newtio.c_cc[VMIN] = 0;
tcflush(fd,TCIFLUSH);
if((tcsetattr(fd,TCSANOW,&newtio))!=0)
{
perror("com set error");
return -1;
}
printf("set done!\n");
return 0;
}
int open_port(int fd,int comport)
{
char *dev[]={"/dev/ttyS0","/dev/ttyS1","/dev/ttyS2"};
long vdisable;
if (comport==1)
{ fd = open( "/dev/ttyS0", O_RDWR|O_NOCTTY|O_NDELAY);
if (-1 == fd){
perror("Can't Open Serial Port");
return(-1);
}
else
printf("open ttyS0 .....\n");
}
else if(comport==2)
{ fd = open( "/dev/ttyS1", O_RDWR|O_NOCTTY|O_NDELAY);
if (-1 == fd){
perror("Can't Open Serial Port");
return(-1);
}
else
printf("open ttyS1 .....\n");
}
else if (comport==3)
{
fd = open( "/dev/ttyS2", O_RDWR|O_NOCTTY|O_NDELAY);
if (-1 == fd){
perror("Can't Open Serial Port");
return(-1);
}
else
printf("open ttyS2 .....\n");
}
if(fcntl(fd, F_SETFL, 0)0)
printf("fcntl failed!\n");
else
printf("fcntl=%d\n",fcntl(fd, F_SETFL,0));
if(isatty(STDIN_FILENO)==0)
printf("standard input is not a terminal device\n");
else
printf("isatty success!\n");
printf("fd-open=%d\n",fd);
return fd;
}
int main(void)
{
int fd;
int nread,i;
char buff[]="Hello\n";
if((fd=open_port(fd,1))0){
perror("open_port error");
return;
}
if((i=set_opt(fd,115200,8,'N',1))0){
perror("set_opt error");
return;
}
printf("fd=%d\n",fd);
// fd=3;
nread=read(fd,buff,8);
printf("nread=%d,%s\n",nread,buff);
close(fd);
return;
}