?? 1.引言
隨著嵌入式技術(shù)的飛速發(fā)展,基于嵌入式系統(tǒng)的新一代工業(yè)控制器也日益增多。同以往的控制器不同,新的儀器大多以32位嵌入式處理器為核心,并且安裝有嵌入式操作系統(tǒng),從而大幅度提高了處理能力,方便了設(shè)計開發(fā)。在各種嵌入式操作系統(tǒng)中,嵌入式Linux是免費的自由軟件,其構(gòu)建的系統(tǒng)成本較低,而且Linux是單內(nèi)核的操作系統(tǒng),并可按要求進行任意剪裁,因此越來越多的研究人員開始在用Linux平臺來開發(fā)自己的產(chǎn)品[1]。
嵌入式開發(fā)過程中,經(jīng)常需要為特定設(shè)備開發(fā)驅(qū)動程序。這些驅(qū)動程序的編寫和編譯與PC上的Linux驅(qū)動開發(fā)相比存在明顯的差異,需要考慮的因素更多,實現(xiàn)過程更為復(fù)雜。本文以SAMSUNG公司S3C2410X CPU為例,探討如何為使用嵌入式Linux的工業(yè)控制器開發(fā)字符設(shè)備驅(qū)動程序來驅(qū)動步進電動機。
2.Linux驅(qū)動程序概述
在Linux中,幾乎所有的內(nèi)容都是文件,對設(shè)備驅(qū)動的訪問也是以文件操作的方式實現(xiàn)的。Linux系統(tǒng)支持3種類型的硬件設(shè)備:字符設(shè)備、塊設(shè)備和網(wǎng)絡(luò)設(shè)備,這些設(shè)備的驅(qū)動程序是系統(tǒng)內(nèi)核的重要組成部分。對用戶程序而言,操作系統(tǒng)隱藏了設(shè)備的具體細節(jié),把設(shè)備映射為一個設(shè)備文件,用戶程序可以對設(shè)備文件進行open、CLOSE、read、write等操作。這些操作和驅(qū)動程序是通過STruct file_operations這一數(shù)據(jù)結(jié)構(gòu)關(guān)聯(lián)起來的,編寫設(shè)備驅(qū)動程序的主要工作就是編寫子函數(shù)填充file_operations的各個字段[2]。
3.1 嵌入式Linux設(shè)備驅(qū)動程序的結(jié)構(gòu)
嵌入式Linux下的設(shè)備總體上可以分為兩部分:
其一,驅(qū)動與內(nèi)核接口層,它實現(xiàn)驅(qū)動模塊在Linux內(nèi)核的注冊加載與卸除工作。主要任務(wù)就是在模塊加載時向內(nèi)核注冊驅(qū)動,以及實現(xiàn)虛擬文件系統(tǒng)的設(shè)備操作接口。對于采用中斷的設(shè)備,此部分還包括中斷處理函數(shù)的注冊與注銷。
其二,硬件設(shè)備接口層,這部分主要描述驅(qū)動程序與設(shè)備的交互。它主要包括硬件探測和初始化以及設(shè)備的讀寫訪問和設(shè)備控制操作。硬件探測主要是在驅(qū)動注冊加載時監(jiān)測設(shè)備是否存在,設(shè)備初始化主要是檢測到設(shè)備后對它進行初始化操作。設(shè)備的讀寫操作主要完成從設(shè)備接受數(shù)據(jù)和將數(shù)據(jù)發(fā)送給設(shè)備的操作。硬件設(shè)備接口層還需要包括一些設(shè)備的控制操作,設(shè)定設(shè)備的工作參數(shù)。
對于驅(qū)動程序與內(nèi)核接口層,Linux提供了標準的入口點函數(shù)init_module();在通過模塊化的設(shè)計方法設(shè)計驅(qū)動程序時,使用insmod加載核心模塊時會調(diào)用本函數(shù),通知內(nèi)核對驅(qū)動程序進行注冊。模塊的卸除工作與加載工作類似,通過rmmod卸載模塊時,調(diào)用cleanup_module()取消驅(qū)動程序的注冊。
3.2 步進電機驅(qū)動程序需求分析
步進電機是將電脈沖信號轉(zhuǎn)變?yōu)榻俏灰苹蚓€位移的開環(huán)控制元件。在非超負載的情況下,電機的轉(zhuǎn)速、停止的位置只取決于脈沖信號的頻率和脈沖數(shù),而不受負載變化的影響。所以在驅(qū)動程序中間只需要考慮這兩個方面的影響。
本系統(tǒng)的步進電機的四相由硬件地址0x28000006的bit0~bit3控制,bit0對應(yīng)MOTOR_A,bit1對應(yīng)MOTOR_B,bit2對應(yīng)MOTOR_C,bit3對應(yīng)MOTOR_D。本文所描述的驅(qū)動是針對整步模式下的步進電機,整步模式下的步距角18°。在整步模式下的脈沖分配信號如表所示。
所以在程序中需要通過編制脈沖分配表控制步進電機,并且通過修改脈沖分配表可以實現(xiàn)步進電機方向的控制。
系統(tǒng)的步進電機僅僅是一個輸出的通道,只能順序的進行控制的操作,因此作為一個字符設(shè)備來進行驅(qū)動。對于字符設(shè)備的操作而言驅(qū)動程序需要提供相關(guān)的幾個操作分別為open,read,write,ioctl等相關(guān)的函數(shù)入口點。在驅(qū)動程序的實現(xiàn)過程中需要定義這些文件相關(guān)的操作,填充進入file_operations結(jié)構(gòu)中。
與普通文件相比,設(shè)備文件的操作要復(fù)雜得多,不可能簡單的通過read、write等操作來實現(xiàn)。并且由于對于步進電機驅(qū)動程序沒有相關(guān)的輸入與輸出,更關(guān)注的是對硬件的控制,因此在驅(qū)動程序?qū)τ趙rite操作和read操作僅需返回0,而對于硬件的控制只需要在驅(qū)動程序中實現(xiàn)ioctl函數(shù),并在其中添加相應(yīng)的case即可。通過cmd區(qū)分操作,通過arg傳遞參數(shù)和結(jié)果[3]。
3.3 步進電機驅(qū)動程序設(shè)計
因為步進電機用到了I/O端口,而在ARM9中操作端口要用虛擬地址而非實際的物理地址,所以要修改內(nèi)核代碼。
修改文件內(nèi)核源代碼中間的smdk.c,在結(jié)構(gòu)體
static struct map_desc smdk_io_desc] __initdata = {
{ vCS8900_BASE, pCS8900_BASE, 0x00100000, DOMAIN_IO, 0, 1, 0, 0 },
{ vCF_MEM_BASE, pCF_MEM_BASE, 0x01000000, DOMAIN_IO, 0, 1, 0, 0 },
{ vCF_IO_BASE, pCF_IO_BASE, 0x01000000, DOMAIN_IO, 0, 1, 0, 0 },
LAST_DESC
};
中添加一行數(shù)組元素{ 0xd3000000, 0x28000000, 0x01000000, DOMAIN_IO, 0, 1, 0, 0 },則步進電機的物理地址0x28000006對應(yīng)的虛擬地址為0xd3000006,在驅(qū)動程序中應(yīng)對這個地址進行操作。
定義全局變量num和status用來控制步進電機的速度和方向:
static int num=1;
static enum{off,clockwise,anticlockwise} status=off;
定義步進電機的整步模式正轉(zhuǎn)脈沖表:
unsigned char pulse_table[] =
{
0x05, 0x09, 0x0a, 0x06,
};
評論