本文檔介紹了在 iTOP-4418 開發板上用 PWM 控制蜂鳴器輸出的測試歷程,基于 QtE 系統。4418MCU 共提供了 5 路 PWM 輸出,其中一路未引出,所以共有 4 路可用的 PWM 輸出。 注意:本文檔中提供的例程,沒有注冊設備和驅動,只是在驅動入口和出口函數中進行了配置。如果用戶需要生成設備節點,則需要自行添加剩余部分,這部分可以參考 GPIO 操作的文檔。 1 配置 IO 打開底板電路圖,搜索“beep”,可以看到 beep 的網絡名為“MCU_ISO7816_CLK”,如下圖所示。 ![]() 在核心板原理圖,搜索該關鍵詞“MCU_ISO7816_CLK”,可見其對應 PWM2,如下圖所示。 ![]() 所以,接下來我們便對 PWM2 進行操作。在下面的操作之前,我們需要配置內核,取消內核中 buzzer 的驅動,解除該驅動對蜂鳴器的占用,其目錄如下圖所示。 ![]() 將該選項改為未選中狀態,如下圖所示。 ![]() 接下來,編譯燒寫該內核鏡像(boot.img)到開發板。再進行下面的操作即可。 2 編寫驅動程序 在 linux 內核中有一個規律,Linux 內核開發者把通用的東西都總結出來,個性化的東西就留出接口,和 GPIO 驅動類似,PWM 驅動在內核中也提供了對應的接口函數,內核提供的接口函數聲明在 include/linux/pwm.h 中。 //申請一個 PWM 資源 struct pwm_device *pwm_request(int pwm_id, const char *label); //釋放一個 PWM 資源 void pwm_free(struct pwm_device *pwm); //配置 PWM int pwm_config(struct pwm_device *pwm, int duty_ns, int period_ns); //使能 PWM,duty_ns 為高電平所用時間,period_ns 整個周期為所用時間,單位為納秒。 int pwm_enable(struct pwm_device *pwm); //不使能 PWM void pwm_disable(struct pwm_device *pwm); 根據以上這些,我們便可以編寫一個簡單的 pwm 輸出程序,來控制蜂鳴器的頻率。創建 名稱為 4418x_pwm.c 的文件,程序代碼如下所示。 #include #include #include #include #include #include #include #include #include /*pwm for this buzzer*/ struct pwm_device *pwm = NULL; static int __init buzzer_init(void) { int ret; printk(" check buzzer init.\n"); pwm = pwm_request(2, "buzzer"); if ( pwm == NULL ) { printk("buzzer open error.\n"); } //printk(KERN_EMERG "pwm_request %d ",pwm); ret=pwm_config(pwm,100000,200000);//設置了 1000Hz 頻率的聲音 printk("pwm_config %d ",ret); printk("pwm_config %d ",ret); ret=pwm_enable(pwm); printk("pwm_enable %d ",ret); printk(KERN_EMERG "done2. \n") ; return 0; } static void __exit buzzer_exit(void) { pwm_config(pwm,0,0); //關閉蜂鳴器輸出 pwm_disable(pwm); // 關閉 pwm pwm_free(pwm); // 釋放 pwm 資源 } module_init(buzzer_init); module_exit(buzzer_exit); MODULE_DESCRIPTION("pwm_buzzer driver"); MODULE_LICENSE("GPL"); 2.2 編寫 Makefile 接下來進行編寫 Makefile 文件。 export ARCH=arm obj-m += 4418x_pwm.o KDIR := /home/topeet/4418/4G/20170914/android/kernel PWD = $(shell pwd) all: make -C $(KDIR) M=$(PWD) modules clean: rm -rf *.o modules.order *.ko *mod.c Module.symvers 腳本中,export ARCH=arm 表示設置目標 CPU 類別為 arm,也就是編譯的依賴內核和驅動模塊目標 CPU 為 ARM。 obj-m += 4418x_pwm.o 表示編譯的源文件為 4418x_pwm.c, 如果源文件名有變化,則需要修改成對應的文件名。 KDIR 參數指向對應的內核源碼目錄。作者的內核源碼是在/home/topeet/4418/4G/20170914/android/kernel 目錄下,用戶要根據自己的具體情況來修改。 2.3 編譯運行 首先設置環境變量,使其在編譯時使用源碼中的編譯器。在源碼目錄中使用”cdkernel”進入 kernel 目錄。然后使用命令“make menuconfig”打開內核缺省配置界面,如下圖所示。 ![]() 進入圖中高亮的“General setup”,如下圖所示。 ![]() 可以看到圖中高亮的文本,描述了當前源碼使用的編譯器為“arm-eabi-”,我們回到源碼文件夾,使用命令“find ./ -name arm-eabi-*”,可以得到源碼中編譯器所在路徑,如下圖所示。 ![]() 這樣,源碼編譯器的絕對路徑為源碼所在路徑加上上圖中紅框的路徑,在本文中為“/home/topeet/4418/4G/20170914/android/prebuilts/gcc/linux-x86/arm/arm-eabi-4.7/bin/”,因為筆者是用 root 登錄的,所以打開文件“/root/.bashrc”在里面添加這樣一條內容,如下圖所示。 ![]() 接下來,我們便可以進行編譯了。 將 Makefile 與 C 程序放在 Ubuntu 系統的同一目錄。如下圖所示。 ![]() 在當前目錄輸入“make”開始編譯,生成內核模塊文件“ 4418x_pwm.ko”,如下圖所示。 ![]() 將該內核模塊文件拷貝到開發板,接下來在超級終端使用命令“insmod4418x_pwm.ko”加載該模塊,如下圖所示。 ![]() 模塊加載成功,同時蜂鳴器響起高頻聲音。然后使用命令“rmmod 4418x_pwm”卸載該驅動,如下圖所示。 ![]() 此時,蜂鳴器停止播放高頻聲音,PWM 蜂鳴器測試例程到此結束。 ![]() |