#include<pic.h>
//內部振蕩器4M,看門狗關,上電復位延時,開外部復位腳,數據保護關,代碼保護關,關掉電復位
__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();
}
}
/*
模擬串口,定時器中斷發送方式,波特率9600
*/
void UART(uchar U_D)
{
while(T0IE); //等待上一次數據發送完畢
TMR0=175; //預充值
TX=0; //發送起始位
T0IF=0; //清零定時器中斷
T0IE=1; //定時器0中斷使能
TX_REG=U_D; //轉移數據到發送緩存
T_TMP=8; //預置需要發送的位個數(8位數據)
TX_IF=1; //數據發送完畢標志
}
void init()
{
CMCON = 0b00000111; //關閉內部比較器
ANSEL = 0; //全部為數字IO
TRISIO = 0b00001110; //設置RA0為TX,RA1為RX,RA2為IRQ,RA3為復位,RA4為,RA5為
GPIO = 0b00000001; //
OPTION_REG=0b00001110;//GPIO上拉,TMR0時鐘為內部
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) //檢測數據是否發送完畢
{
TX=TX_REG; //輸出數據
TX_REG>>=1; //移位數據
--T_TMP; //
}
else //發送完畢后進行停止位發送
{
if(TX_IF)
{
TX=1;
TX_IF=0;
}
else
{
T0IE=0; //發送完畢后關閉定時器中斷
}
}
T0IF=0;
}
}
以上程序為定時器中斷方式IO模擬串口,
優點就是占用CPU少(只使用一次子函數后一直等待定時器中斷才處理一次),
缺點是占用一個定時器
|