時代不同了,連led都成子系統了,針對核心提供的通用模型,分析一下,好久沒寫文章了也!
代碼位於drivers/leds下,看一下Makefile 模型檔案主要是:
# LED Core
obj-$(CONFIG_NEW_LEDS) += led-core.o
obj-$(CONFIG_LEDS_CLASS) += led-class.o
obj-$(CONFIG_LEDS_TRIGGERS) += led-triggers.o
直接看led-core檔案吧,這個檔案無比個性,主體內容四行
DECLARE_RWSEM(leds_list_lock);
EXPORT_SYMBOL_GPL(leds_list_lock);
LIST_HEAD(leds_list); //連結所有led的全域鏈表
EXPORT_SYMBOL_GPL(leds_list);
再來看下led-class.c函數,這裡先來介紹一下描述led的核心結構體
struct led_classdev {
const char *name; //名字
int brightness; //亮度值,也可以用來表示開關特性
int max_brightness; //允許的最大亮度值
int flags; //標誌
/* Lower 16 bits reflect status */
#define LED_SUSPENDED (1 << 0)
/* Upper 16 bits reflect control information */
#define LED_CORE_SUSPENDRESUME (1 << 16)
/* Set LED brightness level */
/* Must not sleep, use a workqueue if needed */
void (*brightness_set)(struct led_classdev *led_cdev, //核心回呼函數,當設定/sys/class/leds/下的led介面裡的brightness屬性檔案時,會回調該函數
enum led_brightness brightness);
/* Get LED brightness level */
enum led_brightness (*brightness_get)(struct led_classdev *led_cdev); //核心回呼函數,當獲得led當前值時會調用
/* Activate hardware accelerated blink, delays are in
* miliseconds and if none is provided then a sensible default
* should be chosen. The call can adjust the timings if it cant
* match the values specified exactly. */
int (*blink_set)(struct led_classdev *led_cdev,
unsigned long *delay_on,
unsigned long *delay_off);
struct device *dev; //嵌入的標準裝置模型
struct list_head node; /* LED Device list */ //上面提到的全域led裝置的掛接點
const char *default_trigger; /* Trigger to use */
#ifdef CONFIG_LEDS_TRIGGERS
/* Protects the trigger data below */
struct rw_semaphore trigger_lock;
struct led_trigger *trigger;
struct list_head trig_list;
void *trigger_data;
#endif
};
好了,下面說下初始化函數:
static int __init leds_init(void)
{
leds_class = class_create(THIS_MODULE, "leds");
if (IS_ERR(leds_class))
return PTR_ERR(leds_class);
leds_class->suspend = led_suspend;
leds_class->resume = led_resume;
leds_class->dev_attrs = led_class_attrs; //屬性檔案,sys下的介面,重點看一下
return 0;
}
函數在/sys/class目錄下產生了leds這個子目錄,按照led模型註冊的裝置都會出現在該目錄下
static struct device_attribute led_class_attrs[] = {
__ATTR(brightness, 0644, led_brightness_show, led_brightness_store),
__ATTR(max_brightness, 0644, led_max_brightness_show,
led_max_brightness_store),
#ifdef CONFIG_LEDS_TRIGGERS
__ATTR(trigger, 0644, led_trigger_show, led_trigger_store),
#endif
__ATTR_NULL,
};
可見,屬性函數的通用設定函數為led_brightness_store,獲得函數位led_brightness_show,大同小異:
static ssize_t led_brightness_store(struct device *dev,
struct device_attribute *attr, const char *buf, size_t size)
{
struct led_classdev *led_cdev = dev_get_drvdata(dev); //獲得led裝置資訊
ssize_t ret = -EINVAL;
char *after;
unsigned long state = simple_strtoul(buf, &after, 10);
size_t count = after - buf;
if (isspace(*after))
count++;
if (count == size) {
ret = count;
if (state == LED_OFF)
led_trigger_remove(led_cdev);
led_set_brightness(led_cdev, state);
}
return ret;
}
跟進一下led_set_brightness函數:
static inline void led_set_brightness(struct led_classdev *led_cdev,
enum led_brightness value)
{
if (value > led_cdev->max_brightness)
value = led_cdev->max_brightness;
led_cdev->brightness = value;
if (!(led_cdev->flags & LED_SUSPENDED)) {
#ifdef CONFIG_HAS_EARLYSUSPEND
if (queue_brightness_change(led_cdev, value) != 0)
#endif
led_cdev->brightness_set(led_cdev, value);
}
}
可見最終確實是回調的 led_cdev->brightness_set來完成的該屬性的設定。
下面看一下給我們留的介面,註冊函數:
int led_classdev_register(struct device *parent, struct led_classdev *led_cdev)
{
led_cdev->dev = device_create(leds_class, parent, 0, led_cdev,
"%s", led_cdev->name); //在leds目錄下產生該裝置的目錄項
if (IS_ERR(led_cdev->dev))
return PTR_ERR(led_cdev->dev);
#ifdef CONFIG_LEDS_TRIGGERS
init_rwsem(&led_cdev->trigger_lock);
#endif
/* add to the list of leds */
down_write(&leds_list_lock);
list_add_tail(&led_cdev->node, &leds_list); //把該裝置加入全域鏈表
up_write(&leds_list_lock);
if (!led_cdev->max_brightness)
led_cdev->max_brightness = LED_FULL; //最大值如果沒設定,則設定成255
led_update_brightness(led_cdev);
#ifdef CONFIG_LEDS_TRIGGERS
led_trigger_set_default(led_cdev);
#endif
printk(KERN_DEBUG "Registered led device: %s",
led_cdev->name);
return 0;
}
總結:基本上簡要分析了一下led子系統,代碼不多,也很容易看,主要自己端的probe函數要配置好io口的功能,填充好led_cdev這個東東,然後註冊就可以了 ^.^~