當(dāng)前位置:首頁 > 單片機 > 單片機
[導(dǎo)讀]/*************************************************************************用定時器 T0 或 T1 模擬串行口程序。 最高波特率(12 clock): 本程序收、發(fā)波特率相同。 11.059MHz -- 最高波特率 收: 9600, 最低波特

/*************************************************************************

用定時器 T0 或 T1 模擬串行口程序。
最高波特率(12 clock):
本程序收、發(fā)波特率相同。

11.059MHz -- 最高波特率 收: 9600, 最低波特率:300
30.000MHz -- 最高波特率 收: 28800 最低波特率:300
40.000MHz -- 最高波特率 收: 38400 最低波特率:300
...
使用說明:
1. 本程序使用一個定時器和任意 2 個 I/O 口模擬一個串行口。
2. 1位起始位,8位數(shù)據(jù)位,1位停止位。發(fā)數(shù)據(jù)位時先發(fā)低位。
3. 支持半雙工通訊。收、發(fā)波特率相同。
4. 應(yīng)把定時器中斷優(yōu)先級設(shè)置為最高級。
5. 本程序每接收一個字節(jié)后就把它放到一個隊列緩沖區(qū)中(也可使用環(huán)行緩沖區(qū)),
待緩沖區(qū)滿后,將緩沖區(qū)中的內(nèi)容原樣發(fā)回。這是為了測試多字節(jié)連續(xù)收發(fā)的
能力和簡化程序。實際應(yīng)用中應(yīng)防止緩沖區(qū)溢出。
6. 由接收轉(zhuǎn)換到發(fā)送時要先調(diào)用 soft_send_enable ();
由發(fā)送轉(zhuǎn)換到接收時要先調(diào)用 soft_receive_enable ()。
7. 發(fā)送最后一個字節(jié)后如果要立刻轉(zhuǎn)為接收,必須等待最后一個字節(jié)后發(fā)送完畢
while ( rs_f_TI == 0) ; // 等待最后一個字節(jié)發(fā)送完畢
**************************************************************************
編程說明:
----------------
發(fā)送:
由接收轉(zhuǎn)換到發(fā)送時要先調(diào)用 soft_send_enable (), 它為發(fā)送做初始化的工作。
以后就可以調(diào)用 rs_send_byte () 啟動發(fā)送一個字節(jié)的過程。
發(fā)送口平時為高電平,rs_send_byte ()函數(shù)使發(fā)送口變?yōu)榈碗娖介_始發(fā)送起始位;
同時設(shè)置和啟動定時器,為發(fā)送數(shù)據(jù)位在預(yù)定的時刻產(chǎn)生定時器中斷。發(fā)送數(shù)據(jù)位和
停止位都在定時器的中斷服務(wù)程序中進(jìn)行。
中斷服務(wù)程序中處理 4 種情況:發(fā)送數(shù)據(jù)位、發(fā)送停止位、發(fā)送完畢、處理錯誤。
----------------
接收:
由發(fā)送轉(zhuǎn)換到接收時要先調(diào)用 soft_receive_enable (), 它為接收做初始化的工
作。定時器以 3 到 4 倍波特率的頻率產(chǎn)生中斷(參見 rs_TEST0 的定義)檢測 PC
機發(fā)送的起始位。一旦檢測到起始位,立刻把定時器產(chǎn)生中斷的頻率調(diào)整到與波特率
相同,準(zhǔn)備在下一個定時器中斷中接收第 1 個數(shù)據(jù)位。
中斷服務(wù)程序中處理以下情況:
1. 收到的是 PC 機發(fā)送的起始位: 調(diào)整定時器產(chǎn)生中斷的頻率與波特率相同。
2. 收到第 8 位數(shù)據(jù)位: 存儲接收到的字節(jié)。
3. 收到第 1--7 位數(shù)據(jù)位: 存儲到收、發(fā)移位暫存器。
4. 收到停止位: 調(diào)用 soft_receive_enable(),檢測 PC 機發(fā)出的下一個起始位。
5. 處理出錯的情況。
**************************************************************************/

#i nclude
sfr16DPTR = 0x82;

typedefunsigned charINT8U;
typedefunsigned intINT16U;

#defineYES 1
#defineNO 0

//定義使用哪個定時器, 只可定義一個
//#define TIMER_0
#define TIMER_1

//定義串口收、發(fā)送管腳。
sbit rs_TXD = P2^1;
sbit rs_RXD = P2^0;

//根據(jù)定時器確定參數(shù)
#ifdef TIMER_0
#define TMOD_AND_WORD 0xF0;
#define TMOD_TIME_MODE 0x01;
#define TMOD_COUNT_MODE 0x05; //設(shè)置計數(shù)模式位
sbit TCON_ENABLE_TIMER = TCON^4;
sbit TCON_TFx = TCON^5; //中斷標(biāo)志位
sbit IE_ETx = IE^1; //中斷允許位為 ET0
sbit IP_PTx = IP^1; //中斷優(yōu)先級

sfr rs_timerL = 0x8A; //TL0
sfr rs_timerH = 0x8C; //TH0
#endif

#ifdef TIMER_1
#define TMOD_AND_WORD 0x0F;
#define TMOD_TIME_MODE 0x10;
#define TMOD_COUNT_MODE 0x50; //設(shè)置計數(shù)模式位
sbit TCON_ENABLE_TIMER = TCON^6; //
sbit TCON_TFx = TCON^7; //中斷標(biāo)志位
sbit IE_ETx = IE^3; //中斷允許位為 ET1
sbit IP_PTx = IP^4; //中斷優(yōu)先級

sfr rs_timerL = 0x8B; //TL1
sfr rs_timerH = 0x8D; //TH1
#endif

INT8U bdata rs_BUF; //串行收、發(fā)時用的移位暫存器。
sbitrs_BUF_bit7 = rs_BUF^7; //移位暫存器的最高位。
INT8U rs_shift_count; //移位計數(shù)器。

INT8Ubdata rsFlags;
sbitrs_f_TI = rsFlags^0; //0:正在發(fā)送; 1: 一個字符完畢
sbitrs_f_RI_enable= rsFlags^1; //0:禁止接收; 1:允許接收
sbitrs_f_TI_enable= rsFlags^2; //0:禁止發(fā)送; 1:允許發(fā)送

//選擇以下一個晶體頻率
//#define Fosc 6000000 //6MHz
#define Fosc 11059200 //11.059MHz
//#define Fosc 12000000
//#define Fosc 18432000
//#define Fosc 20000000
//#define Fosc 24000000
//#define Fosc 30000000
//#define Fosc 40000000

//選擇以下一個波特率:
//#efine Baud 300 //11.059MHz時,baud 最低為 300
//#define Baud 1200
//#define Baud 2400
//#define Baud 4800
#define Baud 9600
//#define Baud 14400
//#define Baud 19200
//#define Baud 28800
//#define Baud 38400
//#define Baud 57600

//收、發(fā)一位所需定時器計數(shù)
#define rs_FULL_BIT0 ((Fosc/12) / Baud)
#define rs_FULL_BIT (65536 - rs_FULL_BIT0)
#define rs_FULL_BIT_H rs_FULL_BIT >> 8 //收、發(fā)一位所需定時器計數(shù)高位
#define rs_FULL_BIT_L (rs_FULL_BIT & 0x00FF) //收、發(fā)一位所需定時器計數(shù)低位

//檢測起始位的時間間隔所需定時器計數(shù)
#define rs_TEST0 rs_FULL_BIT0 / 4 //波特率較低時可以除以 3 或除以 2
#define rs_TEST ((~rs_TEST0))
#define rs_TEST_H rs_TEST >> 8 //高位
#define rs_TEST_L rs_TEST & 0x00FF //低位

//發(fā)送起始位所需定時器總計數(shù)
#define rs_START_BIT 0xFFFF - (Fosc/12/Baud) + 0x28
#define rs_START_BIT_H rs_START_BIT >> 8 //發(fā)送起始位所需定時器計數(shù)高位
#define rs_START_BIT_L rs_START_BIT & 0x00FF //發(fā)送起始位所需定時器計數(shù)低位

#define rs_RECEIVE_MAX 128 //最大接收長度
INT8Uidata rs232buffer[rs_RECEIVE_MAX]; //收、發(fā)緩沖區(qū)
INT16UReceivePoint; //接收數(shù)據(jù)存儲指針

void soft_rs232_interrupt( void );

#ifdef TIMER_0
void timer0 (void) interrupt 1 using 3
{
if (rs_RXD == 0 | rs_shift_count > 0)
{ soft_rs232_interrupt(); }
else
{
rs_timerH = rs_TEST_H;
rs_timerL = rs_TEST_L;
}
}
#endif

#ifdef TIMER_1
void timer1 (void) interrupt 3 using 3
{
if (rs_RXD == 0 | rs_shift_count > 0)
{ soft_rs232_interrupt(); }
else
{
rs_timerH = rs_TEST_H;
rs_timerL = rs_TEST_L;
}
}
#endif
/***************************************/

void soft_rs232_init (void) //串口初始化
{
TCON_ENABLE_TIMER = 0; //停止定時器
TMOD &= TMOD_AND_WORD;
TMOD |= TMOD_TIME_MODE;
rs_RXD = 1; //接收腳置成高電平
rs_TXD = 1; //發(fā)射腳置成高電平
IP_PTx = 1; //置中斷優(yōu)先級為高
IE_ETx = 1; //允許定時器中斷
}

void soft_receive_init() //監(jiān)測起始位
{
TCON_ENABLE_TIMER = 0; //停止定時器
rs_timerH = rs_TEST_H;
rs_timerL = rs_TEST_L;
rs_shift_count = 0;
TCON_ENABLE_TIMER = 1; //啟動定時器
}


void soft_receive_enable() //允許接收
{
rs_f_RI_enable = 1; //允許接收
rs_f_TI_enable = 0; //禁止發(fā)送
soft_receive_init(); //監(jiān)測起始位, RXD 下降沿觸發(fā)接收字節(jié)過程.
}

void soft_send_enable (void) //允許發(fā)送
{
TCON_ENABLE_TIMER = 0; //停止定時器
rs_f_TI_enable = 1; //允許發(fā)送
rs_f_RI_enable = 0; //禁止接收

rs_shift_count = 0; //清移位計數(shù)器
rs_f_TI = 1; //發(fā)送一個字符完畢標(biāo)志
TCON_ENABLE_TIMER = 1; //啟動定時器
}

void soft_rs232_interrupt( void )
{
/************************ 接收 ****************************/
if (rs_f_RI_enable == 1)
{
if (rs_shift_count == 0) //移位計數(shù)器==0, 表示檢測到起始位的起點
{
if ( rs_RXD == 1 )
{
soft_receive_enable (); //起始位錯, 從新開始
}
else
{
//下次中斷在數(shù)據(jù)位或停止位中的某時刻發(fā)生
rs_timerL += rs_FULL_BIT_L + 0x10;
rs_timerH = rs_FULL_BIT_H;
rs_shift_count++;
rs_BUF = 0; //清移位緩沖變量
}
}
else
{
rs_timerL += rs_FULL_BIT_L; //下次中斷在數(shù)據(jù)位或停止位中發(fā)生
rs_timerH = rs_FULL_BIT_H;

rs_shift_count++; //2--9:數(shù)據(jù)位 10:停止位

if ( rs_shift_count == 9)
{
rs_BUF = rs_BUF >> 1; //接收第8位
rs_BUF_bit7 = rs_RXD;
if( ReceivePoint < rs_RECEIVE_MAX)
{ //保存收到的字節(jié)
rs232buffer[ReceivePoint++] = rs_BUF;
}
else
{
rs_f_RI_enable = 0; //緩沖區(qū)滿, 禁止接收
}
}
else
{
if (rs_shift_count < 9 ) //收到的是數(shù)據(jù)位 1 -- 7
{
rs_BUF = rs_BUF >> 1;
rs_BUF_bit7 = rs_RXD;
}
else
{ //收到停止位,繼續(xù)檢測 PC 機發(fā)出的下一個起始位
soft_receive_init();
}
}
}
TCON_TFx = 0; //清定時器中斷標(biāo)志
}
else
{
/************************ 發(fā)送 ****************************/
if (rs_f_TI_enable == 1)
{
rs_timerL += rs_FULL_BIT_L;//下次中斷在數(shù)據(jù)位的末尾時刻
rs_timerH = rs_FULL_BIT_H;

rs_shift_count--; //0:停止位末尾時刻到
//1:發(fā)送停止位
//2--9:發(fā)送數(shù)據(jù)位
if (rs_shift_count > 9) //錯誤狀態(tài)
{
rs_shift_count = 9;
rs_BUF = 0xFF;
}

if (rs_shift_count > 1) //2--9:發(fā)送數(shù)據(jù)位
{
ACC = rs_BUF;
ACC = ACC >> 1;
rs_TXD = CY;
rs_BUF = ACC;
}
else
{
if (rs_shift_count == 0) //0:停止位末尾時刻到
{
rs_TXD = 1;
rs_f_TI = 1; //已發(fā)送完畢一個字節(jié)
}
else
{
rs_TXD = 1; //1:發(fā)送停止位
}
}
}
}
}

//由收轉(zhuǎn)到發(fā)時,要先調(diào)用 soft_send_enable ()
void rs_send_byte(INT8U SendByte) //發(fā)送一個字節(jié)
{
while ( rs_f_TI == 0); //等待發(fā)送完畢前一個字節(jié)
rs_TXD = 1;
rs_timerL = rs_START_BIT_L; //下次中斷在起始位的末尾時刻
rs_timerH = rs_START_BIT_H;
rs_BUF = SendByte;
rs_shift_count = 10;
rs_TXD = 0; //發(fā)送起始位
rs_f_TI = 0; //清已發(fā)送完畢一個字節(jié)的標(biāo)志
}

void initiate_MCU (void) //系統(tǒng)初始化
{
soft_rs232_init(); //串口初始化
EA = 1; //開中斷
}

void main (void)
{
//首先發(fā)送 128 個字節(jié) 00H--7FH, 然后等待 PC 機發(fā)送的數(shù)據(jù)。當(dāng)收到 128
//個字節(jié)后,立刻將收到的 128 個數(shù)據(jù)回發(fā)送給 PC 機,然后繼續(xù)等待下一個
//數(shù)據(jù)塊。

INT8U i;
initiate_MCU(); //系統(tǒng)初始化

soft_send_enable (); //允許發(fā)送,禁止接收
for (i=0; i < rs_RECEIVE_MAX; i++ )
{
rs_send_byte(i);
}
while ( rs_f_TI == 0) ; // 等待最后一個字節(jié)發(fā)送完畢

while(1)
{
soft_receive_enable (); //啟動并開始接收,禁止發(fā)送
while (ReceivePoint < rs_RECEIVE_MAX); // 等待接收緩沖區(qū)滿

soft_send_enable (); //允許發(fā)送,禁止接收
for (i=0; i < rs_RECEIVE_MAX; i++ )
{
rs_send_byte(rs232buffer[i]);
}
while ( rs_f_TI == 0) ; //等待最后一個字節(jié)發(fā)送完畢
ReceivePoint = 0;
}
}

本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫毥谦F公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險,如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機 衛(wèi)星通信

要點: 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運營商 數(shù)字經(jīng)濟

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團(tuán))股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉