标题:
PL3201芯片的串口2使用(修复BUG)
[打印本页]
作者:
xuwei
时间:
2015-6-13 15:56
标题:
PL3201芯片的串口2使用(修复BUG)
原本想着这个小事小弟完全可以搞定,可是小弟却令我有些失望,而且还是51的内核,只不过是他是在原来芯片上扩展的一个而已,操作起来也无非是:
第一歩:设置下定时器的工作模式
第二歩:设置下培增
第三部:定时器初始化
第四部:配置串口方式
第五步:启动定时器发生波特率
但是需要注意的是 他没有倍增器,第二定时器的初值,第三他的控制寄存器,
一下是代码,我测试的,很好用,!
#include<string.h>
#include <PL3201_Addr_Map.h>
#include "pl3201.h"
#include "usart.h"
#include "string.h"
extern void Protocol_analysis (unsigned char *buf,unsigned char len) ;
sbit PLC_TLED=P0^2;
sbit PLC_RLED=P0^4; //if it is 0,well be open ! IS A INDCATER!
sbit RUN=P0^3;
uchar data PLC_Rxd_pointer;
uchar data PLC_Rxd_tlen;
uchar data PLC_send_pointer;
uchar data PLC_timer;
uchar data PLC_total_len;
uchar data PLC_Rxd_timer;
uchar data WDI;
bit bdata PLC_Rxd_state;
bit bdata PLC_Rxd_OK;
bit bdata PLC_sending;
bit data PLC_TX_EN=0;
uchar xdata PLC_recv_buf[50];
uchar xdata PLC_Tx_Buf[50];
uchar put_buf=0;
uint8_t start_rec_data;
uint8_t rec;
unsigned char nk;
void Check_sys_plc(void)
{
WDI=0x9a;
if(!(PLC_Rxd_timer||PLC_sending)) // no sending and receiving
{
PLC_rset();
}
initsys_ref();
}
void Input_Rx_Process(void)
{
/*Into the upper layer protocol frame*/
Protocol_analysis (PLC_recv_buf,PLC_recv_buf[0]);
PLC_Rxd_state=0;
}
void timeint0 ( void ) interrupt 1
{
RUN=~RUN;
if ( PLC_Rxd_timer>0 )
{
PLC_Rxd_timer--;
}
else
{
PLC_Rxd_timer=0;
PLC_Rxd_state=0; // resume receive state
}
if ( PLC_timer>0 )
PLC_timer--;
nk++;
if(nk==20){nk=0;PLC_TX_EN=1;}
TH0=0x63; // 置定时器计数 100ms 中断一次 ( 4.8MHz )
TL0=0xcc;
if ( WDI == 0x9a )
{
WDT_RST=0xa1; // 清看门狗
WDI=0;
}
}
uint8_t n,len;
void PLC_int ( void ) interrupt 7
{
uchar data state;
uchar data SSC_adr_bak;
SSC_adr_bak=SSC_ADR;
SSC_ADR=0x00;
state= SSC_DAT;
if ( ( state&0x01 ) == 1 ) //载波发送状态
{ // 载波发送
if ( PLC_send_pointer !=PLC_total_len )
{
SSC_BUF=PLC_Tx_Buf[PLC_send_pointer]; //发送下一个字节
PLC_send_pointer++;
}
else
{ //发送完成
PLC_sending=0; // 清正在发送标志
// PLC_data=0;
// PLC_TLED=1; // 发送指示灯灭
}
}
else
{
PLC_Rxd_timer=20; // 最大字节接收延时时间为1S
if ( ( state&0x04 ) == 0x04 ) //如果接收的是祯头标志
{
//帧
if ( PLC_Rxd_state == 0 ) //如果处于等待接收状态
{
//接收第一个字节,并准备接收后面的字节
PLC_recv_buf[0]=SSC_BUF;
PLC_Rxd_pointer=0; //lxw2006-3-2 17:07
PLC_Rxd_state=1;
PLC_Rxd_tlen=PLC_recv_buf[0];
}
else // translate to rec
{
PLC_Rxd_state=0;
SSC_ADR=0X0;
SSC_DAT=0X0;
}
}
//数据流
else
{
if ( PLC_Rxd_state == 1 ) // 如果已经接收了祯头,继续接收后续字节
{
PLC_recv_buf[PLC_Rxd_pointer]=SSC_BUF;
PLC_Rxd_pointer++;
if ( PLC_Rxd_pointer==PLC_Rxd_tlen )
{
PLC_Rxd_OK=1;
PLC_Rxd_state=0;
SSC_ADR=0;
SSC_DAT=0;
}
}
else
{
PLC_Rxd_state=0;
SSC_ADR=0X0;
SSC_DAT=0X0;
}
}
}
SSC_ADR=SSC_adr_bak;
}
/*void set_PLC_3200 ( void )
{
SSC_ADR=0XFF;
SSC_DAT=0XFF; // 禁止写保护
SSC_ADR=0x01; // 选择3201载波通讯控制寄存器1
SSC_DAT=40; // ZZZ 2007-10-15 前导1bit为40个
SSC_ADR=0x03; // 选择3201载波通讯控制寄存器3
SSC_DAT=0xc0; // 开启载波通讯功能,打开中断,选择接收状态置为3200模式
SSC_ADR=0x04; // 设置捕获门限
SSC_DAT=LIMIT_3200;
SSC_ADR=0XFF;
SSC_DAT=0XAA; // 使能写保护
} */
void set_PLC_3105 ( void )
{
SSC_ADR=0XFF; // 2005.11.29 LXW
SSC_DAT=0XFF; // 禁止写保护
SSC_ADR=0x01; // 选择3105载波通讯控制寄存器1
SSC_DAT=64; // ZZZ 2007-10-15 前导1bit为64个
SSC_ADR=0x03; // 选择3201载波通讯控制寄存器3
SSC_DAT=0xe0; // 开启载波通讯功能,打开中断,选择接收状态置为3105模式
SSC_ADR=0x04; // 设置捕获门限
SSC_DAT=LIMIT_3105;
SSC_ADR=0XFF;
SSC_DAT=0XAA; // 使能写保护
}
void initsys ( void )
{ // 控制看门狗喂狗时间长度
CKCON=0xfF;
// 8位/16位运算模式选择位设置
ALU_MOD=0; // 设置成8位运算模式
//载波通讯配置
//************ 3201的载波寄存器设置 **********//
set_PLC_3105 ( ); //上电为3105模式
SSC_ADR=0xFF;
SSC_DAT=0xFF;
SSC_ADR=0x02;
SSC_DAT=0x1F;
SSC_ADR=0xFF;
SSC_DAT=0x00;
//************ 3201的载波寄存器设置 **********//
EIE=0X01; // ( A9H 使能INT2中断(EX2=1)
IT2=1; // ( C8H ) 为边沿触发方式(IT2=1)
//串口通讯配置
//TMOD1 = 0x10 ; // 设 T/C1 为1分频模式,ZZZ
TMOD = 0x21 ; // 设 T/C1 为定时方式2 设置为8位自装载计数器 T/C0 为定时方式 1
SCON=0x50; // 工作方式1
PCON=0x80; // 波特率加倍
TH1=BPS2400;
TL1=BPS2400; // 加倍设定波特率 9.6MHz ( 按1分频2400计算)
TR1=1; // 启动定时器1
TMOD1=0x32; //time2方式2
SCON1=0x50; //
TH2=0xc2;
TL2=0xc2;
TR2 =1;
// 定时器0中断配置
TH0=0x63;
TL0=0xcc; // 定时 50 ms 一个定时中断
TR0 = 1; // 启动 T0 定时器
// 中断级别设置
IP=0x42; // 定时器 T0 中断为高级中断
EIP=1; // 外部中断2 为高级中断
// 中断允许设置
IE=0x82;
// 允许ET0=1定时器 T0 中断; 开放中断
}
void initsys_ref( void )
{ // 载波设置重置
uchar data s1;
uchar data GOLD_KSAMI;
if ( EIE!=1 )
EIE=0X01; // ( A9H 使能INT2中断(EX2=1)
if ( IT2!=1 )
IT2=1; // ( C8H ) 为边沿触发方式(IT2=1)
SSC_ADR=0x02;
GOLD_KSAMI=SSC_DAT;
if ( GOLD_KSAMI!=0x1F )
{
SSC_ADR=0xFF;
SSC_DAT=0xFF;
SSC_ADR=0x02;
SSC_DAT=0x1F;
SSC_ADR=0xFF;
SSC_DAT=0x00;
}
// 串口设置重置
if ( TMOD1!=0x10 ) //2007-10-15 ZZZ
TMOD1 = 0x10 ; //1分频模式
if ( TMOD!=0x21 )
TMOD = 0x21 ; // 设 T/C1 为定时方式2 设置为8位自装载计数器 T/C0 为定时方式 14
if ( PCON!=0x80 )
PCON=0X80; // 波特率加倍
s1=SCON;
s1=s1&0xf0;
if ( s1!=0x50 )
initsys ();
if ( TH1!=BPS2400 )
TH1=BPS2400;
if ( TR1!=1 )
TR1=1; // 启动定时器1
// 定时器0重置
/*嘻嘻,*/
TMOD1=0x32; //time2方式2
SCON1=0x50; //
TH2=OMG;
TL2=OMG;
TR2 =1;
if ( TR0!=1 )
TR0 = 1; // 启动 T0 定时器
// 中断设置重置
if ( IE!=0x82 )
IE=0x82;
if ( IP!=2 )
IP=2;
if ( EIP!=1 )
EIP=1; // 外部中断2 为高级中断
}
void PLC_rset(void)
{
uchar s1;
/*if( function_now == 0 )
{
// MEAN 3200 FUNCTION
SSC_ADR=0x03;
s1=SSC_DAT;
if ( ( s1&0xf0 )!=0xc0 )
set_PLC_3200 ();
} */
//else if ( function_now == 0xff ) //3105
//{
SSC_ADR=0x03;
s1=SSC_DAT;
if ( ( s1&0xf0 )!=0xe0 )
set_PLC_3105 ();
// }
}
void init_PLC3201(void)
{
uchar data i;
uchar data j;
for ( i=0; i<5; i++ ) //wait for the stabilization of power
{
for ( j=0; j<0xff; j++ );
}
for ( i=0; i<5; i++ )
{
for ( j=0; j<0xff; j++ );
}
PLC_Rxd_OK=0;
PLC_sending=0;
// PLC_Rxd_state=0;
PLC_Rxd_timer=0x00;
}
uint8_t Send_PLC_Data( const uint8_t *Src,uint8_t len)
{
memcpy (PLC_Tx_Buf ,Src, len);
PLC_total_len=len ;
PLC_send_pointer=0;
PLC_sending=1;
SSC_ADR=0x03;
SSC_DAT= ( SSC_DAT|0x01 );
SSC_BUF= PLC_Tx_Buf[PLC_send_pointer];
PLC_timer=30; // 载波发送过程延时控制1.5秒
while (PLC_sending)
{
if( 0==PLC_timer) return 0;//超时
}
if ( PLC_timer == 0 ) // PLC sending finish
{
SSC_ADR=0x03;
SSC_DAT= (SSC_DAT&0xFE); // 设置3201为REC状态REDUNDANCY
PLC_sending=0;
}
return 1;//成功
}
复制代码
欢迎光临 (http://www.51hei.com/bbs/)
Powered by Discuz! X3.1