• Linux内核中ideapad-laptop.c文件全解析4


    接上一篇文章《Linux内核中ideapad-laptop.c文件全解析3》,链接为:

    Linux内核中ideapad-laptop.c文件全解析3_蓝天居士的博客-CSDN博客

    上一回讲到了ideapad_sysfs_init函数由上到下的调用路线,本章我们来看具体内容。

    再la来回顾一下调用栈,重点关注传入参数:

    ideapad_sysfs_init

    ---> device_add_group(&priv->platform_device->dev, &ideapad_attribute_group)

    ---> sysfs_create_groups(&dev->kobj, groups);

    ---> internal_create_groups(kobj, 0, groups)

    ideapad_attribute_group的定义在同文件(drivers/platform/x86/ideapad-laptop.c)中,代码如下:

    1. static const struct attribute_group ideapad_attribute_group = {
    2. .is_visible = ideapad_is_visible,
    3. .attrs = ideapad_attributes
    4. };

    ideapad_attributes的定义在同文件中,代码如下:

    1. static struct attribute *ideapad_attributes[] = {
    2. &dev_attr_camera_power.attr,
    3. &dev_attr_conservation_mode.attr,
    4. &dev_attr_fan_mode.attr,
    5. &dev_attr_fn_lock.attr,
    6. &dev_attr_touchpad.attr,
    7. &dev_attr_usb_charging.attr,
    8. NULL
    9. };

     先来看一下ideapad_attributes指针数组中各个项的定义:

    1. static DEVICE_ATTR_RW(camera_power);
    2. static DEVICE_ATTR_RW(conservation_mode);
    3. static DEVICE_ATTR_RW(fan_mode);
    4. static DEVICE_ATTR_RW(fn_lock);
    5. static DEVICE_ATTR_RW(touchpad);
    6. static DEVICE_ATTR_RW(usb_charging);

    DEVICE_ATTR_RW的定义在include/linux/device.h中:

    1. #define DEVICE_ATTR_RW(_name) \
    2. struct device_attribute dev_attr_##_name = __ATTR_RW(_name)

    而__ATTR_RW的定义在include/linux/sysfs.h中:

    #define __ATTR_RW(_name) __ATTR(_name, 0644, _name##_show, _name##_store)

    __ATTR的定义在同文件中:

    1. #define __ATTR(_name, _mode, _show, _store) { \
    2. .attr = {.name = __stringify(_name), \
    3. .mode = VERIFY_OCTAL_PERMISSIONS(_mode) }, \
    4. .show = _show, \
    5. .store = _store, \
    6. }

    综上,像数学公式般展开来,最终得到以下结果(以camera_power为例):

    1. struct device_attribute dev_attr_camera_power = {
    2. .attr = {.name = "camera_power",
    3. .mode = 0644 },
    4. .show = camera_power_show,
    5. .store = camera_power_store,
    6. }

    其它几项也是同样的方法和形式,在这里就不一一展开了。
    看一下各个项的实际内容:

    • camera_power
    1. static ssize_t camera_power_show(struct device *dev,
    2. struct device_attribute *attr,
    3. char *buf)
    4. {
    5. struct ideapad_private *priv = dev_get_drvdata(dev);
    6. unsigned long result;
    7. int err;
    8. err = read_ec_data(priv->adev->handle, VPCCMD_R_CAMERA, &result);
    9. if (err)
    10. return err;
    11. return sysfs_emit(buf, "%d\n", !!result);
    12. }
    13. static ssize_t camera_power_store(struct device *dev,
    14. struct device_attribute *attr,
    15. const char *buf, size_t count)
    16. {
    17. struct ideapad_private *priv = dev_get_drvdata(dev);
    18. bool state;
    19. int err;
    20. err = kstrtobool(buf, &state);
    21. if (err)
    22. return err;
    23. err = write_ec_cmd(priv->adev->handle, VPCCMD_W_CAMERA, state);
    24. if (err)
    25. return err;
    26. return count;
    27. }
    28. static DEVICE_ATTR_RW(camera_power);
    • conservation_mode
    1. static ssize_t conservation_mode_show(struct device *dev,
    2. struct device_attribute *attr,
    3. char *buf)
    4. {
    5. struct ideapad_private *priv = dev_get_drvdata(dev);
    6. unsigned long result;
    7. int err;
    8. err = eval_gbmd(priv->adev->handle, &result);
    9. if (err)
    10. return err;
    11. return sysfs_emit(buf, "%d\n", !!test_bit(GBMD_CONSERVATION_STATE_BIT, &result));
    12. }
    13. static ssize_t conservation_mode_store(struct device *dev,
    14. struct device_attribute *attr,
    15. const char *buf, size_t count)
    16. {
    17. struct ideapad_private *priv = dev_get_drvdata(dev);
    18. bool state;
    19. int err;
    20. err = kstrtobool(buf, &state);
    21. if (err)
    22. return err;
    23. err = exec_sbmc(priv->adev->handle, state ? SBMC_CONSERVATION_ON : SBMC_CONSERVATION_OFF);
    24. if (err)
    25. return err;
    26. return count;
    27. }
    28. static DEVICE_ATTR_RW(conservation_mode);
    • fan_mode
    1. static ssize_t fan_mode_show(struct device *dev,
    2. struct device_attribute *attr,
    3. char *buf)
    4. {
    5. struct ideapad_private *priv = dev_get_drvdata(dev);
    6. unsigned long result;
    7. int err;
    8. err = read_ec_data(priv->adev->handle, VPCCMD_R_FAN, &result);
    9. if (err)
    10. return err;
    11. return sysfs_emit(buf, "%lu\n", result);
    12. }
    13. static ssize_t fan_mode_store(struct device *dev,
    14. struct device_attribute *attr,
    15. const char *buf, size_t count)
    16. {
    17. struct ideapad_private *priv = dev_get_drvdata(dev);
    18. unsigned int state;
    19. int err;
    20. err = kstrtouint(buf, 0, &state);
    21. if (err)
    22. return err;
    23. if (state > 4 || state == 3)
    24. return -EINVAL;
    25. err = write_ec_cmd(priv->adev->handle, VPCCMD_W_FAN, state);
    26. if (err)
    27. return err;
    28. return count;
    29. }
    30. static DEVICE_ATTR_RW(fan_mode);
    • fn_lock
    1. static ssize_t fn_lock_show(struct device *dev,
    2. struct device_attribute *attr,
    3. char *buf)
    4. {
    5. struct ideapad_private *priv = dev_get_drvdata(dev);
    6. unsigned long hals;
    7. int err;
    8. err = eval_hals(priv->adev->handle, &hals);
    9. if (err)
    10. return err;
    11. return sysfs_emit(buf, "%d\n", !!test_bit(HALS_FNLOCK_STATE_BIT, &hals));
    12. }
    13. static ssize_t fn_lock_store(struct device *dev,
    14. struct device_attribute *attr,
    15. const char *buf, size_t count)
    16. {
    17. struct ideapad_private *priv = dev_get_drvdata(dev);
    18. bool state;
    19. int err;
    20. err = kstrtobool(buf, &state);
    21. if (err)
    22. return err;
    23. err = exec_sals(priv->adev->handle, state ? SALS_FNLOCK_ON : SALS_FNLOCK_OFF);
    24. if (err)
    25. return err;
    26. return count;
    27. }
    28. static DEVICE_ATTR_RW(fn_lock);
    • touchpad
    1. static ssize_t touchpad_show(struct device *dev,
    2. struct device_attribute *attr,
    3. char *buf)
    4. {
    5. struct ideapad_private *priv = dev_get_drvdata(dev);
    6. unsigned long result;
    7. int err;
    8. err = read_ec_data(priv->adev->handle, VPCCMD_R_TOUCHPAD, &result);
    9. if (err)
    10. return err;
    11. return sysfs_emit(buf, "%d\n", !!result);
    12. }
    13. static ssize_t touchpad_store(struct device *dev,
    14. struct device_attribute *attr,
    15. const char *buf, size_t count)
    16. {
    17. struct ideapad_private *priv = dev_get_drvdata(dev);
    18. bool state;
    19. int err;
    20. err = kstrtobool(buf, &state);
    21. if (err)
    22. return err;
    23. err = write_ec_cmd(priv->adev->handle, VPCCMD_W_TOUCHPAD, state);
    24. if (err)
    25. return err;
    26. return count;
    27. }
    28. static DEVICE_ATTR_RW(touchpad);
    • usb_charging
    1. static ssize_t usb_charging_show(struct device *dev,
    2. struct device_attribute *attr,
    3. char *buf)
    4. {
    5. struct ideapad_private *priv = dev_get_drvdata(dev);
    6. unsigned long hals;
    7. int err;
    8. err = eval_hals(priv->adev->handle, &hals);
    9. if (err)
    10. return err;
    11. return sysfs_emit(buf, "%d\n", !!test_bit(HALS_USB_CHARGING_STATE_BIT, &hals));
    12. }
    13. static ssize_t usb_charging_store(struct device *dev,
    14. struct device_attribute *attr,
    15. const char *buf, size_t count)
    16. {
    17. struct ideapad_private *priv = dev_get_drvdata(dev);
    18. bool state;
    19. int err;
    20. err = kstrtobool(buf, &state);
    21. if (err)
    22. return err;
    23. err = exec_sals(priv->adev->handle, state ? SALS_USB_CHARGING_ON : SALS_USB_CHARGING_OFF);
    24. if (err)
    25. return err;
    26. return count;
    27. }
    28. static DEVICE_ATTR_RW(usb_charging);

    可以看到,show和store函数中很多地方都调用了read_ec_data、write_ec_data等函数,也就是与笔记本EC(Embedded Controller)打交道的函数,这些函数我们单独用1-2个章节介绍。

    至此,ideapad-laptop.c中sysfs部分的代码就已全部分析完了。

  • 相关阅读:
    【tg】calls_call 是整个app的发动机
    陈志泊主编《数据库原理及应用教程第4版微课版》的实验题目参考答案实验4
    C语言-数组指针与指针数组
    八、Linux中的用户与文件权限
    【无标题】
    熟悉使用“Linux中lvm”技术,实现容量管理自由
    Hololens 2 新建自定义按钮
    Elasticsearch系列-安装部署
    从零开始学习 Java:简单易懂的入门指南之Map集合(二十三)
    【华为机试真题 JAVA】数字涂色-100
  • 原文地址:https://blog.csdn.net/phmatthaus/article/details/128091531