#include<pic.h>
//內(nèi)部振蕩器4M,看門狗關,上電復位延時,開外部復位腳,數(shù)據(jù)保護關,代碼保護關,關掉電復位
__CONFIG(FOSC_INTRCIO & WDTE_OFF & PWRTE_ON & MCLRE_ON & CP_OFF & CPD_OFF & BOREN_OFF); //12F675
#define uchar unsigned char
#define uint unsigned int
#define TX GP0
#define RX GP1
#define KEY GP2
#define R_S GP4
#define LED GP5
uchar TX_REG,T_TMP;
bit TX_IF;
void delay_ms(uint wm) //進入退出一次8個指令周期
{
uchar i;
for(;wm!=0;wm--)
{
i=163;
while(i--);
NOP();
NOP();
}
}
/*
模擬串口,定時器中斷發(fā)送方式,波特率9600
*/
void UART(uchar U_D)
{
while(T0IE); //等待上一次數(shù)據(jù)發(fā)送完畢
TMR0=175; //預充值
TX=0; //發(fā)送起始位
T0IF=0; //清零定時器中斷
T0IE=1; //定時器0中斷使能
TX_REG=U_D; //轉(zhuǎn)移數(shù)據(jù)到發(fā)送緩存
T_TMP=8; //預置需要發(fā)送的位個數(shù)(8位數(shù)據(jù))
TX_IF=1; //數(shù)據(jù)發(fā)送完畢標志
}
void init()
{
CMCON = 0b00000111; //關閉內(nèi)部比較器
ANSEL = 0; //全部為數(shù)字IO
TRISIO = 0b00001110; //設置RA0為TX,RA1為RX,RA2為IRQ,RA3為復位,RA4為,RA5為
GPIO = 0b00000001; //
OPTION_REG=0b00001110;//GPIO上拉,TMR0時鐘為內(nèi)部
WPU = 0b00001110;
INTCON = 0b11000000;
}
void main(void)
{
init();
while(1)
{
if(KEY==0)
{
delay_ms(100);
if(KEY==0)
{
UART('O');
UART('K');
UART('\r');
UART('\n');
}
}
}
}
void interrupt INT()
{
if(T0IF) //檢測到定時器0中斷
{
TMR0=169; //預充值
if(T_TMP) //檢測數(shù)據(jù)是否發(fā)送完畢
{
TX=TX_REG; //輸出數(shù)據(jù)
TX_REG>>=1; //移位數(shù)據(jù)
--T_TMP; //
}
else //發(fā)送完畢后進行停止位發(fā)送
{
if(TX_IF)
{
TX=1;
TX_IF=0;
}
else
{
T0IE=0; //發(fā)送完畢后關閉定時器中斷
}
}
T0IF=0;
}
}
以上程序為定時器中斷方式IO模擬串口,
優(yōu)點就是占用CPU少(只使用一次子函數(shù)后一直等待定時器中斷才處理一次),
缺點是占用一個定時器
|