国产精品久久久久影院,成人午夜福利视频,国产精品久久久久高潮,国产精品 欧美 亚洲 制服,国产精品白浆无码流出

使用pinctrl和gpio子系統(tǒng)實現LED燈驅動

發(fā)布者:幸福自在最新更新時間:2024-06-24 來源: elecfans關鍵字:LED燈驅動 手機看文章 掃描二維碼
隨時隨地手機看文章

前邊已經學了兩種點燈,本質依然還是通過配置寄存器;在學習STM32的時候除了學習配置一下寄存器,基本都是使用庫來開發(fā),那么在i.MX6ULL還使用寄存器開發(fā)明顯是不太適合,那么i.MX6ULL有更方便的開發(fā)呢,這篇就來學習一下使用 pinctrl 和 gpio 子系統(tǒng)來完成 LED 燈驅動。


|修改設備樹文件


添加 pinctrl 節(jié)點



開發(fā)板上的 LED 燈使用了 GPIO1_IO04這個 PIN,打開 imx6ull-14x14-evk.dts,在 iomuxc 節(jié)點的 imx6ul-evk 子節(jié)點下創(chuàng)建一個名為“pinctrl_led”的子節(jié)點,節(jié)點內容如下所示:


/* 添加的 */

pinctrl_led: ledgrp { 

  fsl,pins = < 

    MX6UL_PAD_GPIO1_IO04__GPIO1_IO04 0x10B0 /* LED0 */

  >;

};

MX6UL_PAD_GPIO1_IO04__GPIO1_IO04 表示將該io復用為GPIO。


0x10b0 表示對PAD寄存器的配置值,具體含義為如下:

/*寄存器SW_PAD_SNVS_TAMPER3設置IO屬性

*bit 16:0 HYS關閉

*bit [15:14]: 00 默認下拉

*bit [13]: 0 kepper功能

*bit [12]: 1 pull/keeper使能

*bit [11]: 0 關閉開路輸出

*bit [7:6]: 10 速度100Mhz

*bit [5:3]: 110 R0/6驅動能力

*bit [0]: 0 低轉換率

*/


圖示:

9d077a5c-d006-11ed-bfe3-dac502259ad0.png

添加 LED 設備節(jié)點


在根節(jié)點“/”下創(chuàng)建 LED 燈節(jié)點,節(jié)點名為“gpioled”,節(jié)點內容如下:


gpioled { 

    #address-cells = <1>;

    #size-cells = <1>;

    compatible = 'atkalpha-gpioled';

    pinctrl-names = 'default';

    pinctrl-0 = <&pinctrl_led>;

    led-gpio = <&gpio1 4 GPIO_ACTIVE_LOW>;

status='okay';

};

pinctrl-0 屬性設置 LED 燈所使用的 PIN 對應的 pinctrl 節(jié)點。


led-gpio 屬性指定了 LED 燈所使用的 GPIO,在這里就是 GPIO1 的 IO04,低電平有效。稍后編寫驅動程序的時候會獲取 led-gpio 屬性的內容來得到 GPIO 編號,因為 gpio 子系統(tǒng)的 API 操作函數需要 GPIO 編號。


圖示:

9d615fcc-d006-11ed-bfe3-dac502259ad0.png

檢查 PIN 是否被其他外設使用

這一點非常重要?。?!

很多初次接觸設備樹的驅動開發(fā)人員很容易因為這個小問題栽了大跟頭!因為所使用的設備樹基本都是在半導體廠商提供的設備樹文件基礎上修改而來的,而半導體廠商提供的設備樹是根據自己官方開發(fā)板編寫的,很多 PIN 的配置和實際所使用的開發(fā)板不一樣。

比如 A 這個引腳在官方開發(fā)板接的是 I2C 的 SDA,而實際所使用的硬件可能將 A 這個引腳接到了其他的外設,比如 LED 燈上,接不同的外設,A 這個引腳的配置就不同。一個引腳一次只能實現一個功能,如果 A 引腳在設備樹中配置為了 I2C 的 SDA 信號,那么 A 引腳就不能再配置為 GPIO,否則的話驅動程序在申請 GPIO 的時候就會失敗。檢查 PIN 有沒有被其他外設使用包括兩個方面:

①、檢查 pinctrl 設置。

②、如果這個 PIN 配置為 GPIO 的話,檢查這個 GPIO 有沒有被別的外設使用。

因為本章實驗將 GPIO1_IO04這個 PIN 配置為了 GPIO,所以還需要查找一下有沒有其他的外設使用了 GPIO1_IO04,在 可能使用到的設備樹中搜索“gpio1 4”,看看是否被其他外設使用到:

簡單提前了解:使用pinctrl 和 gpio 子系統(tǒng)來完成 LED 燈驅動最明顯的變化就是不同操作寄存器,也就不用對物理地址映射成虛擬地址。完整的代碼如下:


#include 

#include 

#include 

#include 

#include 

#include 

#include 

#include 

#include 

#include 

#include 

#include 

#include 



/* 添加頭文件 */

#include 

#include 

#include 

#include 



#define CHRDEVBASE_CNT      1    /* 設備號個數 */

#define CHRDEVBASE_NAME   'chrdevbase'  /* 名字 */



#define LEDOFF 0 /* 關燈 */

#define LEDON 1 /* 開燈 */



/* chrdevbase 設備結構體 */

struct newchr_dev{

  dev_t devid;       /* 設備號 */

  struct cdev cdev;     /* cdev */

  struct class *class;   /* 類 */

  struct device *device;   /* 設備 */

  int major;         /* 主設備號 */

  int minor;         /* 次設備號 */

  struct device_node *nd; /* 設備節(jié)點 */

  int led_gpio; /* led 所使用的 GPIO 編號 */

};



struct newchr_dev chrdevbase;/* 自定義字符設備 */



/*

* @description : LED 硬件初始化

* @param : 無

* @return : 無

*/

static int led_hal_init(void)

{

  int ret = 0;



  /* 設置 LED 所使用的 GPIO */

  /* 1、獲取設備節(jié)點:gpioled */

  chrdevbase.nd = of_find_node_by_path('/gpioled');

  if(chrdevbase.nd == NULL) {

    printk('chrdevbase node cant not found!

');

    return -EINVAL;

  } else {

    printk('chrdevbase node has been found!

');

  }



  /* 2、 獲取設備樹中的 gpio 屬性,得到 LED 所使用的 LED 編號 */

  chrdevbase.led_gpio = of_get_named_gpio(chrdevbase.nd, 'led-gpio', 0);

  if(chrdevbase.led_gpio < 0) {

    printk('can't get led-gpio');

    return -EINVAL;

  }

  printk('led-gpio num = %d

', chrdevbase.led_gpio);



  /* 3、設置 GPIO1_IO03 為輸出,并且輸出高電平,默認關閉 LED 燈 */

  ret = gpio_direction_output(chrdevbase.led_gpio, 1);

  if(ret < 0) {

    printk('can't set gpio!

');

  }



  return 0;

}



/*

 * @description    : 打開設備

 * @param - inode   : 傳遞給驅動的inode

 * @param - filp   : 設備文件,file結構體有個叫做private_data的成員變量

 *             一般在open的時候將private_data指向設備結構體。

 * @return       : 0 成功;其他 失敗

 */

static int chrdevbase_open(struct inode *inode, struct file *filp)

{

  printk('[BSP]chrdevbase open!

');

  filp->private_data = &chrdevbase; /* 設置私有數據 */

  return 0;

}



/*

 * @description    : 從設備讀取數據 

 * @param - filp   : 要打開的設備文件(文件描述符)

 * @param - buf   : 返回給用戶空間的數據緩沖區(qū)

 * @param - cnt   : 要讀取的數據長度

 * @param - offt   : 相對于文件首地址的偏移

 * @return       : 讀取的字節(jié)數,如果為負值,表示讀取失敗

 */

static ssize_t chrdevbase_read(struct file *filp, char __user *buf, size_t cnt, loff_t *offt)

{

  printk('chrdevbase read!

');

  return 0;

}



/*

 * @description    : 向設備寫數據 

 * @param - filp   : 設備文件,表示打開的文件描述符

 * @param - buf   : 要寫給設備寫入的數據

 * @param - cnt   : 要寫入的數據長度

 * @param - offt   : 相對于文件首地址的偏移

 * @return       : 寫入的字節(jié)數,如果為負值,表示寫入失敗

 */

static ssize_t chrdevbase_write(struct file *filp, const char __user *buf, size_t cnt, loff_t *offt)

{

  int retvalue = 0;

  char writebuf[1];

  struct newchr_dev *dev = filp->private_data;



  /* 接收用戶空間傳遞給內核的數據并且打印出來 */

  retvalue = copy_from_user(writebuf, buf, cnt);

  printk('[BSP]kernel recevdata data:%d!

',writebuf[0]);



  if(writebuf[0] == LEDON) { 

    gpio_set_value(dev->led_gpio, 0); /* 打開 LED 燈 */

  } else if(writebuf[0] == LEDOFF) {

    gpio_set_value(dev->led_gpio, 1); /* 關閉 LED 燈 */

  }



  // printk('chrdevbase write!

');

  return 0;

}



/*

 * @description    : 關閉/釋放設備

 * @param - filp   : 要關閉的設備文件(文件描述符)

 * @return       : 0 成功;其他 失敗

 */

static int chrdevbase_release(struct inode *inode, struct file *filp)

{

  printk('[BSP]release!

');

  return 0;

}



/*

 * 設備操作函數結構體

 */

static struct file_operations chrdevbase_fops = {

  .owner = THIS_MODULE,  

  .open = chrdevbase_open,

  .read = chrdevbase_read,

  .write = chrdevbase_write,

  .release = chrdevbase_release,

};



/*

 * @description  : 驅動入口函數 

 * @param     : 無

 * @return     : 0 成功;其他 失敗

 */

static int __init chrdevbase_init(void)

{

  /* 初始化硬件 */

  led_hal_init();



  /* 注冊字符設備驅動 */

  /* 1、創(chuàng)建設備號 */

  if (chrdevbase.major) { /* 定義了設備號 */

    chrdevbase.devid = MKDEV(chrdevbase.major, 0);

    register_chrdev_region(chrdevbase.devid, CHRDEVBASE_CNT, CHRDEVBASE_NAME);

  } else { /* 沒有定義設備號 */

    alloc_chrdev_region(&chrdevbase.devid, 0, CHRDEVBASE_CNT,CHRDEVBASE_NAME); /* 申請設備號 */

    chrdevbase.major = MAJOR(chrdevbase.devid); /* 獲取主設備號 */

    chrdevbase.minor = MINOR(chrdevbase.devid); /* 獲取次設備號 */

  }

  printk('newcheled major=%d,minor=%d

',chrdevbase.major,chrdevbase.minor);



  /* 2、初始化 cdev */

  chrdevbase.cdev.owner = THIS_MODULE;

  cdev_init(&chrdevbase.cdev, &chrdevbase_fops);



  /* 3、添加一個 cdev */

  cdev_add(&chrdevbase.cdev, chrdevbase.devid, CHRDEVBASE_CNT);



  /* 4、創(chuàng)建類 */

  chrdevbase.class = class_create(THIS_MODULE, CHRDEVBASE_NAME);

  if (IS_ERR(chrdevbase.class)) {

    return PTR_ERR(chrdevbase.class);

  }



  /* 5、創(chuàng)建設備 */

  chrdevbase.device = device_create(chrdevbase.class, NULL,chrdevbase.devid, NULL, CHRDEVBASE_NAME);

  if (IS_ERR(chrdevbase.device)) {

    return PTR_ERR(chrdevbase.device);

  }



  return 0;

}



/*

 * @description  : 驅動出口函數

 * @param     : 無

 * @return     : 無

 */

static void __exit chrdevbase_exit(void)

{

  /* 注銷字符設備 */

  cdev_del(&chrdevbase.cdev);/* 刪除 cdev */

  unregister_chrdev_region(chrdevbase.devid, CHRDEVBASE_CNT);/* 注銷設備號 */



  device_destroy(chrdevbase.class, chrdevbase.devid);/* 銷毀設備 */

  class_destroy(chrdevbase.class);/* 銷毀類 */



  printk('[BSP]chrdevbase exit!

');

}



/* 

 * 將上面兩個函數指定為驅動的入口和出口函數 

 */

module_init(chrdevbase_init);

module_exit(chrdevbase_exit);



/* 

 * LICENSE和作者信息

 */

MODULE_LICENSE('GPL');

MODULE_AUTHOR('zuozhongkai');

編譯設備樹和復制文件

編譯沒有問題:

9e031ab0-d006-11ed-bfe3-dac502259ad0.png

復制文件:

9e4144de-d006-11ed-bfe3-dac502259ad0.png

|編譯驅動程序

復制一份新字符驅動,對應改下名稱:

9e8b8206-d006-11ed-bfe3-dac502259ad0.png

編譯驅動:

9ebaa90a-d006-11ed-bfe3-dac502259ad0.png

復制驅動到對應位置:

9f0032ea-d006-11ed-bfe3-dac502259ad0.png

| 觀察效果

1、觀察設備樹節(jié)點

9f364ab0-d006-11ed-bfe3-dac502259ad0.png

2、加載驅動


depmod //第一次加載驅動的時候需要運行此命令
modprobe gpioled.ko //加載驅動


9f7d83e4-d006-11ed-bfe3-dac502259ad0.png

3、操作GPIO

9fab6228-d006-11ed-bfe3-dac502259ad0.png

使用pinctrl 和 gpio 子系統(tǒng)可以很方便對GPIO進行操作,可以不去查寄存器的地址也能實現,提高了程序員對底層驅動開發(fā)的效率。


關鍵字:LED燈驅動 引用地址:使用pinctrl和gpio子系統(tǒng)實現LED燈驅動

上一篇:CH9434嵌入式Linux與安卓系統(tǒng)驅動移植和使用教程
下一篇:一種可用于單片機的0-10V模擬量采集電路(一)

推薦閱讀最新更新時間:2025-06-12 21:33

采用非隔離驅動器的LED燈泡可減小體積與成本
LED燈泡 的價格在不斷下跌。一年前,大概要花50美元才能買到一只用于替換老式燈泡的飛利浦60W調光 LED燈泡 ,而今天,在百思買只需花17美元就可以買到其家用品牌的8W、800流明Insignia燈泡,可替換60W白熾燈泡。在 LED燈泡 的設計中,哪些變化導致了這種成本的下降?對LED燈泡的拆解可以讓我們看清楚LED照明的一些設計趨勢,例如燈泡中如何放置LED,以及采用了何種 驅動器 架構等。 1. Insignia燈泡的外形類似于我們熟悉的傳統(tǒng)白熾燈,不同的是它有三個金屬散熱鰭片,以及塑料(而不是玻璃)的燈頭。 2. 用一支Dremel工具可以去掉塑料燈泡的外殼,露出六只Cree公司的白光LED,它們照亮了燈泡
[電源管理]
采用非隔離<font color='red'>驅動</font>器的<font color='red'>LED燈</font>泡可減小體積與成本
無閃爍LED燈驅動方案設計
傳統(tǒng)鹵素杯燈使用電子式變壓器并采用交流電輸入,因此目前市面上的LED燈具產品,其內部加裝整流電路可直接替代傳統(tǒng)鹵素杯燈。傳統(tǒng)電子式變壓器雖然價格便宜,但相對質量良莠不齊,由于電子式變壓器事實上是以半橋式整流控制,隨負載阻抗大小而變動輸出,因此電路穩(wěn)定度較差,當電路匹配有問題時,LED-MR16燈具就容易產生閃爍或不亮的情形發(fā)生。   現行3-5WLED-MR16燈的參考設計中,多采用3顆1W的LED串聯(lián)輸出,輸入電壓為交流12V±10%。若是LED驅動電路架構采用直流降壓模式時,在正常供電情況下可以有效輸出,但由于電子式變壓器穩(wěn)定度原因,使得LED驅動電路的輸入電壓可能高于或低于輸出電壓,造成直流降壓驅動電路無法工作,LED的
[電源管理]
無閃爍<font color='red'>LED燈</font>具<font color='red'>驅動</font>方案設計
離線式LED燈對LED驅動器IC要求
  過去幾年,用作高清電視機(HDTV)顯示器背光照明的LED一直是LED市場增長的主要驅動力。不過,隨著LED普通照明應用在商用和住宅環(huán)境中引起越來越大的關注,LED的增長將顯著加速。   LED照明高增長率背后的主要驅動力是,與傳統(tǒng)照明方法相比,LED照明的功耗大幅降低。與白熾燈照明相比,要提供同樣的光輸出 (以流明為單位),LED需要的電功率不到白熾燈的25%。LED照明還有其他很多優(yōu)勢,但是也有一些與LED照明有關的挑戰(zhàn)。LED照明的優(yōu)勢包括工作壽命比白熾燈長數個量級,這極大地降低了更換成本。能利用以前安裝的TRIAC調光器給LED調光,也是一個主要的成本優(yōu)勢,尤其是在住宅照明領域。LED能即時接通,不像CFL那樣需要
[電源管理]
離線式<font color='red'>LED燈</font>對LED<font color='red'>驅動</font>器IC要求
小廣播
設計資源 培訓 開發(fā)板 精華推薦

最新單片機文章

 
EEWorld訂閱號

 
EEWorld服務號

 
汽車開發(fā)圈

 
機器人開發(fā)圈

電子工程世界版權所有 京ICP證060456號 京ICP備10001474號-1 電信業(yè)務審批[2006]字第258號函 京公網安備 11010802033920號 Copyright ? 2005-2025 EEWORLD.com.cn, Inc. All rights reserved