|
引言 為了滿足實時性要求較高系統的設計需求,針對串聯多個器件在一線制總線上的結構導致的在查詢多點溫度時速度緩慢的問題,北京銘正同創科技有限公司提出了一種快速查詢多點溫度的解決方案。本方案以Dallas公司開發的一線制數字溫度傳感器DS18B20為核心,通過采用每個并行端口上連接一個DS18B20器件,實現同時對多個DS18B20進行同步操作的方法。本方案可廣泛應用于各種工業控制、儀器儀表產品中。 關鍵字快速多點溫度查詢 工業控制 儀器儀表 銘正同創
1 技術概述 Dallas公司開發的一線制數字溫度傳感器DS18B20是一款性能優異的數字式傳感器,廣泛應用于各種工業控制、儀器儀表產品當中。DS18B20與傳統的熱敏電阻溫度傳感器相比,能夠直接讀出被測溫度,并且根據實際要求通過簡單的編程可設置9~12位的分辨率,可以在750ms內將溫度轉化為12位的數字量,具有多種可選的封裝方式。因而使用DS18B20可使系統結構更加簡單,可靠性更高。DS18B20器件具體的封裝形式如下圖所示:

GND 接地 DQ 一線制總線(輸入/輸出) VD 供電電源而DS18B20的一線制總線獨特而經濟的特點,使用戶可輕松地組建傳感器網絡,為測量系統的

在一線制總線上串接多個DS18B20器件時,實現對其中一個DS18B20器件進行一次溫度轉換和讀取操作主要包括以下13個步驟(所有的操作都是通過DQ線進行信號傳輸的): 1 主機MCU發復位脈沖 2 DS18B20發應答脈沖(即MCU接收該應答信號,以確認器件在總線上) 3 主機發匹配ROM命令 4 主機發64位器件序列號(器件序列號與總線上的某個DS18B20器件一一對應) 5 主機發溫度轉換指令 6 總線保持高電平50ms 7 主機發復位命令 8 DS18B20發應答脈沖 9 主機發匹配ROM命令 10 主機發64位器件代碼 11 主機發讀數據寄存器指令 12 主機接收數據 13 主機發復位脈沖 參考DS18B20的數據手冊可知,當DS18B20的精度設置為12位精度表示時,依據上面的步驟完成對一個器件的測溫、讀取溫度值的過程,大概會消耗掉1秒鐘的時間。而如果總線上存在8個DS18B20器件的話,完成一次8個器件的查詢需要8秒的時間,這不還沒計算在系統初始化時,對總線上的器件序列號進行初始化過程所消耗的時間。 針對所述利用多個DS18B20器件串接在一線制總線上進行多點溫度查詢時速度慢的原因,做進一步分析如下。 DS18B20器件在進行一線制總線操作時,僅有一根DQ線用于雙向的數據傳輸,每一個操作最小的細分至每一個的讀寫過程,即一個位的讀寫操作為一線制總線操作的最小單位,可以參考DS18B20的手冊,了解到每一次最小單位的總線操作利用了規定時間內MCU驅動DQ線的高低電平來決定讀/寫的操作,然后在其后的規定時間內完成讀/寫一個位數據的操作。這樣,就決定了每一次操作的過程中,要傳輸的數據位數越多,每一次的操作耗時越長。而DS18B20的一線制總線的操作對時序的要求很嚴格,一般在設計MCU對其總線操作的程序時,都是利用延時去保證每個讀寫周期的時間準確性,即說明這些時間內CPU必然是閑置的。如下圖示意了一個寫0和1操作的時序:

另外,在多個器件串接在一線制總線上時,為了區分每次操作是針對總線上哪一個器件,DS18B20器件在內部提供了每個器件獨有的64位ROM序列號,也就是說每一次操作都要首先在對DS18B20器件的ROM序列號進行匹配后,方可對其中的某一個器件進行測溫/讀取溫度值的操作?梢怨浪愠,每一次序列號的匹配操作,差不多需要4ms的時間,完成一次完整的測溫/讀取溫度值操作,就需要進行兩次序列號匹配,即消耗掉大概8個ms的時間。 多個器件串接在總線上時,對所有的器件的查詢操作,需要一個一個來,完成一次全部器件的查詢需要成倍的操作時間,整個系統把大量時間消耗在時序所要求的延時上。 此外,當采用多個器件串接在一線制總線的系統時,還需要在系統的初始化其間花銷較長的時間來進行煩瑣的總線上器件的序列號查詢,并以此獲知總線上的每個器件的序列號。 如前所述,可以總結出,影響查詢多點DS18B20溫度速度的最主要因素有如下幾個: 1.每次操作都需要附加兩次對64位序列號的匹配過程; 2.多個器件串接,完成全部的查詢就需要與器件個數成倍增長的耗時。 這樣的應用在一些對實時性要求相對較高的系統當中,是非常占用資源的(雖然省掉了端口資源,但CPU不得不等待N長時間后方可獲取多點的溫度值),所以使用起來總會有些遺憾。下面,介紹一種快速查詢多點DS18B20溫度的方法,包括硬件的連線構成以及軟件的編程思路。 2 解決方案 由于一般都會將對DS18B20器件的溫度查詢放置在中斷當中實現或者是在程序的主循環當中采用定時查詢的方法實現,所以這就要求每次對DS18B20的操作都能快速的完成,盡快退出來進行其它的處理。所以為了解決串聯多個器件在一線制總線上的結構導致的在查詢多點溫度時速度緩慢的問題,本設計提出一種解決方案,具體說是通過修改硬件連接來實現方便快捷的查詢多點DS18B20器件溫度的方法。 2.1 快速查詢多點DS18B20溫度的方法簡述 當一線制總線上僅有一個DS18B20器件時,可以用skip ROM操作(即跳過ROM匹配)命令來代替64位序列號的匹配過程,這點也是使用單個DS18B20器件的系統常用的方法。所以,要想節省掉64位序列號匹配的時間開銷,就必需設計成一個一線制總線上僅有一個DS18B20器件的系統。 DS18B20的一線制總線在時序上的嚴格要求,也從另一方面意味著在一定的彈性范圍內,不同DS18B20器件的時序細節上的一致性應該是非常好,所以可以將系統設計成利用MCU的并行端口同時對多個DS18B20進行統一的操作,不過這時候并行端口上的每一個端口連接著一個DS18B20器件而已。 本文所述的解決方案正是以端口的消耗為代價,換取對多點DS18B20溫度查詢的速度,并在程序結構的設計上采用一些巧妙的處理方法,使得系統對DS18B20的操作上花更少的時間。此外,采用本設計實現的快速多點溫度查詢系統,可以省掉煩瑣的總線上器件序列號的查詢操作,并可節省大量的存儲空間(原用于存儲總線上器件的序列號所用的空間)。 從理論上分析,本設計方案的采用,查詢多個DS18B20器件操作所消耗的時間與查詢一個DS18B20器件操作所消耗的時間是等量的。 下面以查詢8個DS18B20器件為例詳細分析此方法的設計思想。 2.2 系統硬件連接 本系統方案8個DS18B20器件連接在MCU的一組端口的8個I/O口上,連線示意圖如下所示:

當然,上圖中的示意圖并沒有考慮諸如端口驅動能力、抗干擾處理等,僅表明一個邏輯的連接示意,具體在產品級的設計時會根據產品的應用做必要的處理,比如增加一些必要的電路等,此處不作為討論的重點。 從上圖可見,每個端口連接有一個DS18B20器件,也即一條一線制總線上僅有一個DS18B20器件,符合了前面所述的解決方法。實際在對DS18B20器件進行操作時,只需統一地對這一組并行端口進行操作(每個端口在同一時間輸出相同的電平狀態)即可。 一個端口對應一個DS18B20器件,也就表示每組端口的某一個位的讀回數據狀態也就是該端口所對應的器件的輸出狀態,所以,這樣的系統里面是不需要進行每個器件的序列號搜索、匹配的操作的。可知,在對DS18B20器件進行操作時,可以使用skip ROM命令來跳過ROM序列號匹配的操作,也即在所有的DS18B20器件的ROM操作時可以使用相同的端口輸出時序。 2.3 軟件設計思路 總結前面所介紹的電路示意圖,下面詳細介紹程序設計思想。 在接下來的軟件介紹中,會以C語言的例子介紹具體的編程思路,但這些代碼并非就是實際中所使用的代碼,僅作為邏輯性的參考,以便大家理解。這些代碼是從一個產品的應用當中摘出的,而程序設計的結構也是從具體的設計當中分解出來的,供大家參考。 軟件設計從最底層的與DS18B20時序相關的驅動,到與一線制總線器件處理過程控制/協議的接口函數,再上升到應用API接口函數的關系如下圖所示:

在對連在一組8位端口上的8個DS18B20操作時,是同時對該組端口進行操作,也即同時對8個DS18B20器件進行同步的操作。 范例程序是根據筆者的項目當中的功能需求而設計的,不一定會適合所有人的使用方法,但程序設計思想是可以參考的,這點請使用者在參考本文時對這里的范例進行一定的取舍。下面詳細介紹一個以MCS51系列單片機的應用為例的范例程序,其中約定與8個DS18B20器件進行連接的是P1端口。 2.3.1 底層時序驅動 底層時序驅動程序與DS18B20的一線制總線的協議保持一致,根據一線制總線時序的特點,設計了四個基本的函數: 總線寫1時序控制函數: void DS18B20_Write_1(void) { P1 = 0x00; //8個DQ 線全部設置為低電平 Delay_1us(10); //延時10us左右 P1 = 0xff; //8個DQ線全部輸出高電平 Delay_1us(30); //延時30us左右 } 總線寫0時序控制函數: void DS18B20_Write_0(void) { P1 = 0x00; //8個DQ 線全部設置為低電平 Delay_1us(40); //延時 P1 = 0xff; //端口恢復高電平 Delay_1us(1); } 總線讀取一個數據位時序控制函數: unsigned char DS18B20_ReadDQ(void) { unsigned char DQ_S=0; P1 = 0x00; //8個DQ 線全部設置為低電平 Delay_1us(10); P1 = 0xff; //端口置1,準備讀取 Delay_1us(1); //延時待總線準備好數據 DQ_S = P1; //一次性讀取8條DQ線的數據狀態 P1 = 0xff; //恢復端口電平 Delay_1us(30); //延時 return DQ_S; //返回讀取的值 } 在讀取一個總線狀態數據位的函數中,將會返回一個byte的數據,該數據的8個位正好與連接在P2端口上的8個I/O口對應,如下圖所示:

總線復位時序控制函數: void DS18B20_Reset(void) { unsigned char Error_Counter=0; P1 = 0x00; //8個DQ 線全部設置為低電平 Delay_1us(500); //保持總線低電平500us P1 = 0xff; Delay_1us(100); if(P1!=0x00) B20_Error = P1;//如檢測到DS18B20總線響應了回復信號,則讀取當前8條 //總線的狀態 Delay_1us(50); P1 = 0xff; for(Error_Counter=0;Error_Counter<200;Error_Counter++) { if((P1&(~B20_Error))==(~B20_Error)) break; //如檢測到總線的回復信號結 //束,則退出循環 Delay_1us(1); } P1 = 0xff; //恢復端口電平 Delay_1us(200); //延時 200us~~~ } 在復位時序控制的函數中,使用了B20_Error全局變量,它將會傳遞給上一層的數據處理函數作為判斷當前8個I/O口所接的DS18B20是否正常工作,或者是否在各自的總線上。 2.3.2 操作協議相關的函數 分析DS18B20的一線制總線控制命令,可以提煉出兩個最基本的操作函數,一個是寫一個byte數據至DS18B20器件,另一為讀取DS18B20器件的數據。而在本文的范例程序當中,僅僅為了提取DS18B20器件的轉換完后的溫度值,所以在讀取DS18B20的數據時,僅讀取存放在數據地址前兩個字節的溫度數據,而不讀取其它字節的數據,包括CRC校驗值也沒有進行讀取,供參考。 寫字節操作函數: void DS18B20_WriteByte(unsigned char Com) { unsigned char i; for(i=0;i<8;i++) { if(Com&0x01) DS18B20_Write_1(); else DS18B20_Write_0(); Com = Com>>1; } } 調用DS18B20_WriteByte函數,連在8個I/O口上的一線制總線上的8個DS18B20器件,將都會接收到同樣的一個字節的數據:Com。 讀數據操作函數: unsigned char Read_buf_8ch[16]; //buffer of Read DS18B20 void DS18B20_Read2Byte(void) { unsigned int i; for(i=0;i<16;i++) { Read_buf_8ch = DS18B20_ReadDQ(); } } 前面已經介紹過了,在本范例中,只讀取位到DS18B20內部數據區域的前兩節字的溫度值數據,所以數據讀取函數設計成讀取兩個字節的函數,即需要連續讀取16個位(對應于每一個DS18B20器件來說是連續的16個位)。而將讀回的數據保存于一個Read_buf_8ch(簡寫:Rb)的數組中,可以根據系統的接線圖對讀回的16個字節的數據進行分析,如下圖所示:

讀取DS18B20的數據時,先讀高位再讀低位;所以可以從上圖看到,以TM2的DS18B20的數據為例,TM2的兩個字節的數據由Read_buf_8ch數組的16個字節數據中的每個字節的bit2位組成。可知,完成一次數據讀取的操作后,可以同時讀回8個DS18B20器件的數據,在數據處理時,只需針對上圖的數據結構對Read_buf_8ch數組的數據進行處理即可得到每個DS18B20器件的測溫值。 2.3.3 API功能函數: 供上層應用程序直接調用的函數相對來說,是與系統的具體硬件接法沒有太多的關系,只需要依照DS18B20器件的操作流程進行操作即可。在此,提供兩個API的范例,分別是啟動溫度轉換控制函數和讀取溫度值函數。 啟動溫度轉換控制函數: void DS18B20_Conver(void) { DS18B20_Reset(); DS18B20_WriteByte(0xcc); //Skip ROM DS18B20_WriteByte(0x44); //啟動測溫 } 讀取溫度值函數: void DS18B20_ReadTemp(void) { DS18B20_Reset(); DS18B20_WriteByte(0xcc); //Skip ROM DS18B20_WriteByte(0xbe); //送入讀取數據命令 DS18B20_Read2Byte(); } 調用讀取溫度值函數后,8個DS18B20器件的測溫數據將保存在數組Read_buf_8ch的16個字節單元當中,還有待進行下一步的處理,方可得到對應每個DS18B20器件的測溫值。下面介紹簡單的處理代碼片斷: char i,j; unsigned int uiData[8]; unsigned char Mask; //OS the resoult of Temperature for(i=15;i>=0;i--) { Mask = 0x01; for(j=0;j<8;j++) { uiData[j] = uiData[j]<<1; if(Read_buf_8ch&Mask) uiData[j]++; Mask = Mask<<1; } } 經過上述簡單的處理,8個DS18B20器件的測溫數據將保存在數組uiData當中的8個單元里,就可以根據自身程序設計的需求來對這些數據進行具體的處理了。 3 結語 本文介紹的快速查詢多點DS18B20溫度的設計方案,解決了串聯多個器件在一線制總線上的結構導致的在查詢多點溫度時速度緩慢的問題,基本的設計思想是:將系統設計為在每個并行端口上連接一個DS18B20器件,利用MCU的并行端口同時對多個DS18B20進行統一的操作,實現操作多個DS18B20器件的時間等同于操作單個DS18B20器件的時間。本設計思想,可以大大減少在查詢多個DS18B20測溫值的時間開銷,滿足了實時性要求較高的系統的設計需求;同時,也省掉了煩瑣的總線上多個器件序列號搜索的代碼的步驟,并且節省了用于存儲這些器件的序列號的存儲單元,使得利用DS18B20進行多點測溫的操作變得更方便、容易。 雖然本文介紹的方法是以犧牲端口資源為代價,但具體在進行系統設計時,也可以通過一些擴展端口、串轉并端口、多路模擬開關等硬件電路設計來彌補這些端口資源的消耗,也可通過這些硬件電路來擴展更多的DS18B20器件(如果有必要的話)。 本文所介紹的方法已經在筆者參與設計的大型恒溫系統當中應用,目前系統運行穩定、可靠 |