⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 comm.c

📁 这是我在我们公司做的C51加油单片机程序
💻 C
📖 第 1 页 / 共 2 页
字号:
/* ---------------------------------------------------------- */
/*                                                            */
/*            IC卡加油机键盘通讯子模块                        */
/*            负责与PSAM卡、用户卡、加油机主板通讯            */
/*             最后修改日期:2002年4月21日                    */
/*                                                            */
/* ---------------------------------------------------------- */

#include "c:\comm_d\comm.h"



void send(unsigned char offset,unsigned char len)
{
	register unsigned char counter=0;
	bit tem;
        //ES=0;
	DIRECTION=1;			// 置数据传送方向为发送
	is_send=1;			// 置发送标志
	PT0=1;				// 置定时器0中断为高优先级
	ET0=1;				// 允许定时器0中断
	while(counter<len)		// 未发完len字节
	{
		TL0=0x84;		// 定时器0工作在模式2,晶振为14.31818M,定时溢出频率为9600(实际为9622)
		TH0=0x84;		// 用仿真器时用0x97
		TR0=1;			// 启动定时器0
		com_cur=0;		// 当前发送起始位
		com_tem=send_buf[offset+counter];	// 从外部读入要发送的字节
		com_tem1=0xe;
		ACC=com_tem;
		com_tem1|=P;		// 取校验位
		com_tem1=com_tem1<<1;
		tem=com_tem&0x80;	// 取d7
		com_tem1|=tem;		// 把d7放在com_tem1的D0位
		com_tem=com_tem<<1;	// com_tem存起始位(0)+d0~d6
		while(com_cur<13);	// 等待本字节发送完成
                //dog();
		counter++;
		TR0=0;			// 关定时器0
	}
	ET0=0;				// 禁止定时器0中断
        dog();
        //comm_status=0;
        //ES=1;
}

unsigned char recv(unsigned char offset,unsigned char len)
{
	register unsigned char timer=0;
        //ES=0;
	DIRECTION=0;			// 8888
	is_send=0;			// 置接收标志
	IE0=0;				// 清外部中断0触发标志
	IT0=1;				// 置为沿触发
	PX0=1;				// 置外部中断0为高优先级
	EX0=1;				// 允许外部中断0
	is_over=0;
	is_err=0;
	com_tem1=offset;		// 初始化接收指针
	com_limit=offset+len;
	while(!is_over && timer<100)	// 等待接收结束,2s超时
	{
		com_delay();
                dog();
		timer++;
	}
	EX0=0;				// 禁止外部中断0
	ET0=0;				// 禁止定时器0中断
	TR0=0;				// 关定时器0
        //comm_status=0;
        dog();
        //ES=1;
        if(is_err)return 1;		// 校验位错
	else if(is_over)return 0;	// 成功返回
	else return 2;			// 超时
}

void com_delay()
{
	register unsigned int i;

	for(i=0;i<3409;i++)
        {
                if(is_over)break;
                dog();
        }
}


/* 中断服务程序 */

void ic_timer0() interrupt 1		// 定时器0
{
	if(is_send)			// 发送
	{
		if(com_cur<8)		// 发送com_tem字节
		{
			com_sda=com_tem&0x1;	// 发送最低位
			com_tem=com_tem>>1;	// 右移1位
		}
		else			// 发送com_tem1
		{
			com_sda=com_tem1&0x1;
			com_tem1=com_tem1>>1;
		}
		com_cur++;
	}
	else				// 接收
	{
		if(com_cur<8)		// 收数据位
		{
			com_tem=com_tem>>1;
			if(com_sda)com_tem|=0x80;
		}
		else if(com_cur==8)	// 收校验位
		{
			ACC=com_tem;
			if(P!=com_sda)	// 校验错
			{
				is_err=1;
				is_over=1;
			}
		}
		else			// 停止位
		{
			rekk[com_tem1]=com_tem;
			com_tem1++;
			if(com_tem1<com_limit)	// 未收完
			{
				ET0=0;		// 禁止定时器0中断
				TR0=0;		// 关定时器0
				IE0=0;
				EX0=1;		// 允许外部中断0
			}
			else is_over=1;		// 收完
		}
		com_cur++;
	}
}

void ic_int0() interrupt 0		// 外部中断0
{
	EX0=0;				// 禁止外部中断0
	TL0=0x74;			// 第一次定时为1.5个9600BAUD
	TH0=0x84;
	TF0=0;				// 清定时器0中断触发标志
	PT0=1;				// 置定时器0中断为高优先级
	ET0=1;				// 允许定时器0中断
	com_tem=0;
	com_cur=0;
	TR0=1;				// 启动定时器0
}
 void key_dlu()
{
        unsigned int k;
        for(k=0;k<2400;k++);
        {
                dog();
        }
}
unsigned int get_CRC(unsigned char xdata *buffer,unsigned int length)
{
	register unsigned int i,j;
	unsigned char data tem[2];
	bit lsb0,lsb1;
        //ES=0;
	tem[0]=0;				// 初始化CRC寄存器为0
	tem[1]=0;
	for(i=0;i<length;i++)
	{
		tem[1]^=buffer[i];		// 输入数据与CRC寄存器低字节异或
		for(j=0;j<8;j++)
		{
			lsb0=tem[0]&1;		// 取高字节的最低位
			tem[0]=tem[0]>>1;
			lsb1=tem[1]&1;		// 取低字节的最低位
			tem[1]=tem[1]>>1;
			if(lsb0)tem[1]|=0x80;
			if(lsb1)		// 最低位为1,CRC寄存器与0xa001异或
			{
				tem[0]^=0xa0;
				tem[1]^=0x01;
			}
		}
                dog();
	}
        //comm_status=0;
        //ES=1;
	return(*(unsigned int *)tem);
}
unsigned long bcdtohex(unsigned char xdata *bcd,unsigned char xdata bcd_num)
{

        unsigned long xdata tem;
	register unsigned char i;
	tem=0;
        //ZK_CS=1;

	for(i=bcd_num;i>0;i--)
        {
           //     tem=tem*100+(bcd[bcd_num-i]/16)*10+bcd[bcd_num-i]%16;
                  tem=tem*100;
                  tem=tem+(bcd[bcd_num-i]/16)*10;
                  tem=tem+bcd[bcd_num-i]%16;
                  dog();
        }

        return tem;

}
unsigned char * hextobcd(unsigned long xdata hexb)
{
        //unsigned long xdata hex_temp=0;
        unsigned char xdata char_temp[4]={0,0,0,0};
       // hex_temp=hexb;
       //ZK_CS=1;
       char_temp[1]=hexb/100000;
       char_temp[1]<<=4;
       hexb=hexb-hexb/100000*100000;
       char_temp[1]=char_temp[1]|hexb/10000;

       hexb=hexb-hexb/10000*10000;
       char_temp[2]=hexb/1000;
       char_temp[2]<<=4;
       hexb=hexb-hexb/1000*1000;
       char_temp[2]=char_temp[2]|hexb/100;

       hexb=hexb-hexb/100*100;
       char_temp[3]=hexb/10;
       char_temp[3]<<=4;
       hexb=hexb-hexb/10*10;
       char_temp[3]=char_temp[3]|hexb;

       return char_temp;
}
/*******************************************************************
                    向有子地址器件读取多字节数据函数
函数原型: bit  ISendStr(uchar sla,uchar suba,ucahr *s,uchar no);
功能:     从启动总线到发送地址,子地址,读数据,结束总线的全过程,从器件
          地址sla,子地址suba,读出的内容放入s指向的存储区,读no个字节。
           如果返回1表示操作成功,否则操作有误。
注意:    使用前必须已结束总线。
********************************************************************/
unsigned char IRcvStr(unsigned char sla,unsigned char suba,unsigned char no)
{
   unsigned char i;
   unsigned char *s;
   Start_I2c();               /*启动总线*/
   SendByte(sla);            /*发送器件地址*/
     if(ack==0)return(0);
   SendByte(suba);            /*发送器件子地址*/
     if(ack==0)return(0);

   Start_I2c();
   SendByte(sla+1);
      if(ack==0)return(0);

   for(i=0;i<no-1;i++)
    {
     kaohao[i]=RcvByte();               /*发送数据*/
      Ack_I2c(0);                /*发送就答位*/
     //s++;
    }
   kaohao[i]=RcvByte();
    Ack_I2c(1);                 /*发送非应位*/
 Stop_I2c();                    /*结束总线*/
  return(1);
}

/*******************************************************************
                    向有子地址器件发送多字节数据函数
函数原型: bit  ISendStr(uchar sla,uchar suba,ucahr *s,uchar no);
功能:     从启动总线到发送地址,子地址,数据,结束总线的全过程,从器件
          地址sla,子地址suba,发送内容是s指向的内容,发送no个字节。
           如果返回1表示操作成功,否则操作有误。
注意:    使用前必须已结束总线。
********************************************************************/
unsigned char ISendStr(unsigned char sla,unsigned char suba,unsigned char no)
{
   unsigned char i;
    //unsigned char *s;

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -