串口mcu:重新编写的ldisc

 * Modification:
 *  - Data    : 2015年9月12日
 *  - Revision: 初稿
 *  - Author  : chen_guangjian
 *  - Contents: 创建文件
 * 
 *-------------------------------------------------------------------------------------------------------------------*/
/*----------------------------------------------*
 * 包含头文件                                   *
 *----------------------------------------------*/
 /**内核依赖的头文件**/
  #include "drvbox_c.h"

#include <linux/module.h>
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/sched.h>
#include <linux/types.h>
#include <linux/fcntl.h>
#include <linux/interrupt.h>
#include <linux/ptrace.h>
#include <linux/poll.h>
#include <linux/in.h>
#include <linux/ioctl.h>
#include <linux/slab.h>
#include <linux/tty.h>
#include <linux/errno.h>
#include <linux/smp_lock.h>
#include <linux/string.h> 
#include <linux/signal.h>
#include <linux/if.h>
#include <linux/bitops.h>
#include <linux/syscalls.h>


#include <asm/system.h>
#include <asm/termios.h>
#include <asm/uaccess.h>
#include <asm/ioctls.h>
/*本地头文件*/
#include "mcu_common.h"
#include "mcuinfo_resource.h"
#include "mcu_if_uart.h"

 


//static int L_FD = -1;
static struct file   *gfilp=NULL;


/************************************************************************
 * 外部函数实现                         *
 ************************************************************************/
int32_t Mcu_IF_Uart_Open(McuInfoRes_t * mcuRes)
{
int err =-1;
int i_ld = N_UART;

//打开串口
#if 0  //sys_open's 方式,内核模块找不到符号表
struct ktermios option;
//L_FD = sys_open(mcuRes->UART_Res.Res.Name,0,0);
if (L_FD >= 0) {
//初始化uart配置信息
//底层驱动注册的时候已经有默认的设置tty_std_termios{}
err = sys_ioctl(L_FD,TCGETS,(long)&option);//先获取回来,
/*
option.c_cflag = B9600 | CS8 | CREAD | HUPCL | CLOCAL;
option.c_iflag = ICRNL | IXON;
option.c_oflag = OPOST | ONLCR;
option.c_lflag = ISIG | ICANON | ECHO | ECHOE | ECHOK |ECHOCTL | ECHOKE | IEXTEN;
option.c_cc = INIT_C_CC;
*/
//设置波特率
option.c_ispeed = option.c_ospeed = mcuRes->UART_Res.baudrate;

err = sys_ioctl(L_FD,TCSETSF,(long)&option);//--->set_termios  /*TCSETSF:TCSETSW*/

//设置线路规则
err = sys_ioctl(L_FD, TIOCSETD, (long)&i_ld);
}
else
{
OSA_ERROR("mcu_if_uart_open error");
}


OSA_INFO("mcu_if_uart_open end  err = %d \n",err);
#endif
#if 1//filp _open 的方法
struct termios settings;
mm_segment_t oldfs;

OSA_printf(KERN_INFO"type =%d  \n",mcuRes->UART_Res.Res);
OSA_printf(KERN_INFO"name= %s \n",mcuRes->UART_Res.Res.Name);

gfilp = filp_open(mcuRes->UART_Res.Res.Name,O_RDWR,0);


      if (IS_ERR(gfilp)) {
printk(" open dev error = %ld\n",PTR_ERR(gfilp));
}
else
{
//linux 3.0内核的tty_ops是unlocked_ioctl()
if(!(gfilp->f_op)||!(gfilp->f_op->unlocked_ioctl)||!(gfilp->f_op->write) )
{
            OSA_ERROR("file f_op %s  gfilp=%x cann't readable or writable?\n",mcuRes->UART_Res.Res.Name,gfilp);
  return -1;
      }
/*系统调用在内核空间会检查地址是否是__user,所以改变非用户空间地址的限制,
否则内核空间使用的系统调用的参数地址会出错*/
oldfs = get_fs();
set_fs(KERNEL_DS);


//初始化uart配置信息
//底层驱动注册的时候已经有默认的设置tty_std_termios{}
err = gfilp->f_op->unlocked_ioctl(gfilp, TCGETS, (unsigned long)&settings);
if(mcuRes->UART_Res.baudrate == 9600)
settings.c_cflag = B9600;
err = gfilp->f_op->unlocked_ioctl( gfilp, TCSETS, (unsigned long)&settings);
//设置线路规
err = gfilp->f_op->unlocked_ioctl(gfilp, TIOCSETD, (long)&i_ld);

set_fs(oldfs);
}


OSA_INFO("mcu_if_uart_open end  err = %d \n",err);
#endif
return err;
}


int32_t Mcu_IF_Uart_Close(McuInfoRes_t * mcuRes)
{
int err =-1 ;

if(gfilp)
err = filp_close(gfilp,0);


gfilp  =  NULL;
return err;
}
/**
*读取一些杂项的数据,比如韦根应答包,升级应答包....
**/
int32_t Mcu_IF_ReadRaw(Char *pReadBuf,uint32_t size)
{
int ret = -1 ;

//if(gfilp)
//      ret = gfilp->f_op->unlocked_ioctl(gfilp, MCU_UART_RAW_READ, (unsigned long)pReadBuf);
if(gfilp)
{
gfilp->f_pos = 0;
ret = gfilp->f_op->read(gfilp, pReadBuf, size, &gfilp->f_pos);
}
OSA_INFO("Mcu_IF_ReadRaw end  ret = %d \n",ret);

return ret;
}


int32_t Mcu_IF_WriteRaw(const Char *pWriteBuf, Uint32 size)
{
int ret = -1 ;

if(gfilp)
{
gfilp->f_pos = 0;
ret = gfilp->f_op->write(gfilp, pWriteBuf, size, &gfilp->f_pos);
}


return ret;
}
/*
*读取按键信息的buf
*buf地址来自用户空间
*/
int32_t  Mcu_IF_GetKey(Uint8 *pBuf)
{
int ret = -1 ;


if(gfilp)
ret = gfilp->f_op->unlocked_ioctl(gfilp, MCU_UART_READKEY, (unsigned long)pBuf);

OSA_INFO("Mcu_IF_GetKey end  ret = %d \n",ret);

return ret;
}
/*
*读取iccard的信息
*/
int32_t Mcu_IF_ReadCard(Uint8 *pBuf)
{
int ret = -1 ;

if(gfilp)
ret = gfilp->f_op->unlocked_ioctl(gfilp, MCU_UART_READCARD, (unsigned long)pBuf);

OSA_INFO("Mcu_IF_GetKey end  ret = %d \n",ret);

return ret;
}


int32_t Mcu_IF_Uart_PollEvent(poll_table *wait)
{
int ret =-1;

if(gfilp)
ret  = gfilp->f_op->poll(gfilp, wait);

OSA_INFO("Mcu_IF_Uart_PollEvent end mask = 0x%x \n",ret);

return ret;
}/*******************************************************************************
*  描  述   : 开始升级,完成升级过程的握手,而updaet的数据,
通过RawWrite发送下来
*  输  入   : 无。
*  输  出   : 无。
*  返回值   : OSA_EFAIL: 失败
*             OSA_SOK  : 成功
*******************************************************************************/
 int32_t Mcu_IF_Uart_UpgradeStart(void)
 {
  int ret  = -1;
const Char update_buf_req[6] = {0xA6,0x00,0x07,0x60,0x00,0x0D};/*定义请求升级命令*/
        Char pReadBuf[13] ={0};
uint32_t resp= 0x00;

if(!gfilp)
return OSA_EFAIL;

////发送请求升级命令
ret = gfilp->f_op->write(gfilp, update_buf_req, sizeof(update_buf_req), &gfilp->f_pos);
OSA_INFO("write  update FW cmd ret = %x \n",ret);
if(ret < 0)
return OSA_EFAIL;
 
///等待升级命令的应答包
ret = gfilp->f_op->read(gfilp, pReadBuf, sizeof(pReadBuf), &gfilp->f_pos);
OSA_INFO("wait  respond = %x \n",ret);
  resp = pReadBuf[0] <<24 | pReadBuf[1]<<16 | pReadBuf[2]<<8 | pReadBuf[3];


  if(ret < 0 )
{
OSA_INFO("UpgradeStart not respond \n");
return OSA_EFAIL;
  }
//是否获取到正确的响应包,且允许升级pReadBuf[5] =1
if(( resp != MCU_UART_CMD_UPDATE_RESP )||(pReadBuf[5] !=1) )
{
return OSA_EFAIL;
}


// 获得单片机请求升级数据的第0包
ret = gfilp->f_op->read(gfilp, pReadBuf, sizeof(pReadBuf), &gfilp->f_pos);
OSA_INFO("wait  respond first request packet= %x \n",ret);
  if(ret < 0 )
{
OSA_INFO("UpgradeStart not respond \n");
return OSA_EFAIL;
  }
//是否获取到正确的响应包
if(resp != MCU_UART_CMD_UPDATE_REQUEST )
{
return OSA_EFAIL;
}
OSA_INFO("Mcu_IF_Uart_UpgradeStart end \n");

return OSA_SOK;
 }


Mcu_If_Driver_t Mcu_IF_Uart_Driver =
{
.pUsrArgs = NULL,
.OpOpen  = Mcu_IF_Uart_Open,
.OpClose  = Mcu_IF_Uart_Close,
.RawRead = Mcu_IF_ReadRaw,
.RawWrite = Mcu_IF_WriteRaw,
.UpgradeStart = Mcu_IF_Uart_UpgradeStart,
.UpgradeEnd = NULL,
.PollEvent = Mcu_IF_Uart_PollEvent,
.GetEvent = NULL,
.GetKey = Mcu_IF_GetKey,
.GetSoftwareVer = NULL,
.ReadCard = Mcu_IF_ReadCard,
.ResetMcu  =NULL,
};


/*******************************************************************
 * 内部函数实现***********************************************
 *******************************************************************/
 /* 调试log设置 */
#define DEBUG_LEVEL_INFO 1
static int debuglevel;
 /*
 * DH  uart ldisc的buffer管理函数
 */
static void Uart_Ldisc_buf_list_init(struct Uart_Ldisc_buf_list *list);
static void Uart_Ldisc_buf_put(struct Uart_Ldisc_buf_list *list,struct Uart_Ldisc_buf *buf);
static struct Uart_Ldisc_buf *Uart_Ldisc_buf_get(struct Uart_Ldisc_buf_list *list);
static struct Uart_Ldisc *Uart_Ldisc_alloc (void);


/* 收发数据的buf定义*/
#define DEFAULT_RX_BUF_COUNT 3
#define DEFAULT_TX_BUF_COUNT 3


static int max_rx_buf_frame = 15;
static int max_tx_buf_frame = 130;
#define RX_BUF_SIZE(sizeof(struct Uart_Ldisc_buf) + max_rx_buf_frame)
#define TX_BUF_SIZE(sizeof(struct Uart_Ldisc_buf) + max_tx_buf_frame)


/* TTY ldisc的回调函数 */
static ssize_t Uart_Ldisc_tty_read(struct tty_struct *tty, struct file *file,
  __u8 __user *buf, size_t nr);
static ssize_t Uart_Ldisc_tty_write(struct tty_struct *tty, struct file *file,
   const unsigned char *buf, size_t nr);
static int Uart_Ldisc_tty_ioctl(struct tty_struct *tty, struct file *file,
   unsigned int cmd, unsigned long arg);
static unsigned int Uart_Ldisc_tty_poll(struct tty_struct *tty, struct file *filp,
   poll_table *wait);
static int Uart_Ldisc_tty_open(struct tty_struct *tty);
static void Uart_Ldisc_tty_close(struct tty_struct *tty);
static void Uart_Ldisc_tty_receive(struct tty_struct *tty, const __u8 *cp,
      char *fp, int count);
static void Uart_Ldisc_tty_wakeup(struct tty_struct *tty);


/*定义tty  与ldisc之间的结构体获取*/
#define tty2Uart_Ldisc(tty) ((struct Uart_Ldisc *) ((tty)->disc_data))
#define Uart_Ldisc2tty(Uart_Ldisc) ((Uart_Ldisc)->tty)


.....


int  Uart_Ldisc_init(void)
{
int status=0;


debuglevel = 2;//调试等级设置

//注册ldisc,用于处理uart口收到的单片机数据。
status = tty_register_ldisc(N_UART, &Uart_Ldisc_ldisc);
if (!status)
OSA_INFO("Uart_Ldisc_init ok \n");
else
OSA_ERROR("Uart_Ldisc_init falied %d \n", status);


return status;

}


void  Uart_Ldisc_exit(void)
{
int status = tty_unregister_ldisc(N_UART);


if (status)
OSA_ERROR("Uart_Ldisc_exit failed %d \n", status);
else
OSA_INFO("Uart_Ldisc_exit ok !");
}

 头文件

*     - 使用说明:  mcu_if_uart.h 作为uart接口的mcu与主机直接协议解析的驱动
 *
 *     - 局限性说明:
 *
 *     - 其他说明:
 *
 *
 * Modification:
 *  - Data    : 2015年9月12日
 *  - Revision: 初稿
 *  - Author  : chen_guangjian
 *  - Contents: 创建文件
 * 
 *-------------------------------------------------------------------------------------------------------------------*/
/*----------------------------------------------*
 * 包含头文件                                   *
 *----------------------------------------------*/
#include "mcu_common.h"
#include <linux/tty.h>


/*----------------------------------------------*
 * 宏定义                                                            *
 *----------------------------------------------*/
#define MCU_UART_IOC_MAGIC   'M'
/* 按键状态 */
#define MCU_UART_READKEY        _IOWR(MCU_IOC_MAGIC, 0x1, int)
/* 裸读MCU */
#define MCU_UART_RAW_READ         _IOWR(MCU_IOC_MAGIC, 0x2, int)
/* 读取卡信息*/
#define MCU_UART_READCARD             _IOWR(MCU_IOC_MAGIC, 0x3, int)

 


/**参考内核头文件tty.h**/
#define N_UART 19
#define LDISC_MAGIC 0xFFFF
/*----------------------------------------------*
 * 数据结构                                                            
 *----------------------------------------------*/
 
 /*
 *这里定义的是单片机与主机之间协议解析的头部命令
 *
*/
#define     MCU_UART_CMD_HEAD_LEN 4//前4个字节
#define MCU_UART_CMD_CARD  0xA6000710
#define MCU_UART_CMD_KEY0xA6000720
#define MCU_UART_CMD_KEEPALIVE0xA6000730
#define MCU_UART_CMD_KEEPALIVE_RESP0xA6000731
#define MCU_UART_CMD_SETLED 0xA6000740
#define MCU_UART_CMD_SETWEIGEN0xA6000750
#define MCU_UART_CMD_WEIGEN_RESP0xA6000751
#define MCU_UART_CMD_SETID0xA6000752
#define MCU_UART_CMD_UPDATE_START0xA6000760
#define MCU_UART_CMD_UPDATE_RESP0xA6000761
#define MCU_UART_CMD_UPDATE_REQUEST0xA6000762
#define MCU_UART_CMD_UPDATE_DATA0xA6000763
#define MCU_UART_CMD_UPDATE_END0xA6000764
#define MCU_UART_CMD_UPDATE_END_RESP0xA6000765
#define MCU_UART_CMD_UNKNONW0xFFFFFFFF

 

 


struct Uart_Ldisc_buf {
struct Uart_Ldisc_buf *link;
int  count;
char  buf[1];
};


struct Uart_Ldisc_buf_list {
struct Uart_Ldisc_buf *head;
struct Uart_Ldisc_buf *tail;
int  count;
spinlock_t  spinlock;
};


struct Uart_Ldisc {
int magic;
__u32 flags;
struct tty_struct*tty;
//传输控制标志
int tbusy;
int woke_up;

struct Uart_Ldisc_buf*tbuf; //当前正在发送的buf项
struct Uart_Ldisc_buf_listtx_buf_list; //用于发送的buf链表
struct Uart_Ldisc_buf_listtx_free_buf_list; //用于缓存应用发下来的buf链表
struct Uart_Ldisc_buf_listrx_card_buf_list; //用于发送给用户的card buf 链表
struct Uart_Ldisc_buf_listrx_free_card_buf_list; //用于缓存ldisc收到的card buf链表
struct Uart_Ldisc_buf_listrx_key_buf_list; //用于发送给用户的key buf 链表
struct Uart_Ldisc_buf_listrx_free_key_buf_list; //用于缓存ldisc收到的key buf链?struct Uart_Ldisc_buf_list rx_misc_buf_list;//用于发送给用户的杂项 buf 链表
struct Uart_Ldisc_buf_listrx_free_misc_buf_list; //用于缓存ldisc收到的杂项buf链?};


extern int  Uart_Ldisc_init(void);
extern void  Uart_Ldisc_exit(void);

 


 

 

你可能感兴趣的:(UART,ldisc)