解析linux或android添加文件系统的属性接口的方法

第一种:

1、添加关键头文件:

#include <linux/of_gpio.h>
#include <linux/gpio.h>
#include <linux/delay.h>

#include <linux/module.h>
#include <linux/types.h>
#include <linux/kobject.h>

2、在已经存在驱动文件中搜索"DEVICE_ATTR"关键字,如果存在,直接参考已经存在的方法添加一个即可,如下:

unsigned int Gpio134_OtgID = 134; //定义全局变量
static unsigned int otgid_status = 1;
…

3、定义文件系统的读写函数:

//add zhaojr gpio134 control OTG ID for host or device mode
static ssize_t setotgid_store(struct device *dev, struct device_attribute *attr,const char *buf, size_t count)
{
 unsigned int ret=0;
 pr_err("%s: \n", __func__);
 //ret = kstrtoint(buf, 10, &otgid_status);
 ret = kstrtouint(buf, 10, &otgid_status);
 //sscanf(buf, "%lu", &otgid_status);
 if (ret < 0){
  pr_err("%s::kstrtouint() failed \n", __func__);
 }
 //sscanf(buf, "%d", &otgid_status);
 pr_err("%s: otgid_status=%d \n", __func__,otgid_status);
 if(otgid_status > 0){
  gpio_set_value(Gpio134_OtgID, 1);
 }else{
  gpio_set_value(Gpio134_OtgID, 0);
 }
 return count;
}
static ssize_t setotgid_show(struct device *dev,struct device_attribute *attr, char *buf)
{
 pr_err("%s: \n", __func__);
 return sprintf(buf, "%d\n",otgid_status);
}
//static DEVICE_ATTR_RW(setotgid);
/*struct device_attribute dev_attr_setotgid = {
  .attr = {.name ="setotgid",
  .mode = 0664},
  .show = setotgid_show,
  .store = setotgid_store,
};*/
//setotgid的一致性,第一个参数setotgid和setotgid_show、setotgid_store前钻必须保持一致
static DEVICE_ATTR(setotgid, 0664, setotgid_show, setotgid_store);
//end zhaojr add
static struct device_attribute *android_usb_attributes[] = {
 &dev_attr_state,
 &dev_attr_setotgid, //setotgid跟DEVICE_ATTR定义的name必须保持一致
 NULL
};

4、在probe()函数中定义针对具体GPIO管脚的请求和初始化

static int mdss_mdp_probe(struct platform_device *pdev)
{
....................................................................................
//zhaojr add for gpio134 to usb host or device mode
 ret_status=gpio_request(Gpio134_OtgID, "Gpio134-OtgID");
 if(ret_status<0){
  pr_err("usb gadget configfs %s::Gpio134_OtgID gpio_request failed\n",__func__);
  }
 pr_err("android_device_create()::Gpio134_OtgID gpio_request OK\n");
 gpio_direction_output(Gpio134_OtgID,1);
 if(otgid_status > 0){ //有自定义初始化状态就添加上这个判断,没有就不需要添加if else操作
  pr_err("%s-Gpio134_OtgID pin set 1\n", __func__);
  gpio_set_value(Gpio134_OtgID, 1);
  //msleep(5);
 }else{
  pr_err("%s-Gpio134_OtgID pin set 0\n", __func__);
  gpio_set_value(Gpio134_OtgID, 0);
  //msleep(5);
 }
 //end zhaojr add
................................................................
}

5、在remove()函数中添加资源的释放

static int mdss_mdp_remove(struct platform_device *pdev)
{
 struct mdss_data_type *mdata = platform_get_drvdata(pdev);
 if (!mdata)
  return -ENODEV;
 pr_err("%s\n", __func__);
 gpio_free(Gpio134_OtgID); //zhaojr add free gpio otgid pin
 ........................................................
}

第二种方法:

在要添加驱动文件中没有搜索"DEVICE_ATTR"关键字的情况,如添加音频功放打开和关闭的控制接口:
1、添加关键头文件:

#include <linux/of_gpio.h>
#include <linux/gpio.h>
#include <linux/delay.h>

#include <linux/module.h>
#include <linux/types.h>
#include <linux/kobject.h>

2、定义全局变量和定义打开和关闭的接口并组织属性数组:

// add zhaojr gpio63 for close or speaker pa enable
struct kobject *spk_pa_kobj = NULL;
unsigned int gpio63_spk_pa_gpio; //for speaker pa ic enable
//extern unsigned int gpio63_spk_pa_gpio;
static unsigned int SpkPa_Gpio_Enable = 0;
static ssize_t spkpaon_store(struct device *dev, struct device_attribute *attr,const char *buf, size_t count)
{
 unsigned int ret=0;
 //ret = kstrtoint(buf, 10, &backlight_enable);
 ret = kstrtouint(buf, 10, &SpkPa_Gpio_Enable);
 if (ret < 0){
 pr_err("%s::kstrtouint() failed \n", __func__);
 }
 pr_err("%s: SpkPa_Gpio_Enable=%d \n", __func__,SpkPa_Gpio_Enable);
 if(SpkPa_Gpio_Enable > 0){
 //gpio_set_value(gpio63_spk_pa_gpio, 1);
 pr_err("%s: gpio_set_value gpio63 speaker pa enable \n", __func__);
 //功放打开的时序
 gpio_set_value(gpio63_spk_pa_gpio,0);
 udelay(8);
 gpio_set_value(gpio63_spk_pa_gpio,1);
 udelay(8);
 gpio_set_value(gpio63_spk_pa_gpio,0);
 udelay(8);
 gpio_set_value(gpio63_spk_pa_gpio,1);
 //sdm660_cdc->ext_spk_pa_set = true;
 }else{
 pr_err("%s: gpio_set_value gpio63 speaker pa disable \n", __func__);
 //功放关闭的时序
 gpio_set_value(gpio63_spk_pa_gpio,0);
 udelay(600);
 //sdm660_cdc->ext_spk_pa_set = false;
 }
 return count;
}
static ssize_t spkpaon_show(struct device *dev,struct device_attribute *attr, char *buf)
{
 return sprintf(buf, "%d\n",SpkPa_Gpio_Enable);
}
static DEVICE_ATTR(spkpaon, 0664, spkpaon_show, spkpaon_store);
static struct attribute *spkpa_attributes[] = {
 &dev_attr_spkpaon.attr,
 NULL
};
static const struct attribute_group apkpa_attr_group = {
 .attrs = spkpa_attributes,
 NULL
};
//end zhaojr add

3、在probe()函数中添加文件系统属性接口的注册:
在注册的时候并不需要对功放进行初始化,所以probe()函数中并没有对sdm660_cdc->spk_pa_gpio(GPIO63),只操作了请求。具体的请求操作请参考:msm8953 audio部分的EAR和Speaker输出的声音配置中的音频部分的

vendor/qcom/opensource/audio-kernel/asoc/codecs/sdm660_cdc/msm-analog-cdc.c文件操作

static int msm_anlg_cdc_probe(struct platform_device *pdev)
{
 int ret = 0;
 struct sdm660_cdc_priv *sdm660_cdc = NULL;
 struct sdm660_cdc_pdata *pdata;
 int adsp_state;
 ..................................
 dev_set_drvdata(&pdev->dev, sdm660_cdc);
 //kangting add
 sdm660_cdc->spk_pa_gpio = of_get_named_gpio(pdev->dev.of_node, "qcom,speaker-pa", 0);
 if (!gpio_is_valid(sdm660_cdc->spk_pa_gpio))
 pr_err("%s, sdm660_cdc->spk_pa_gpio not specified\n",__func__);
 else{
 pr_err("%s, sdm660_cdc->spk_pa_gpio is %d\n",__func__,sdm660_cdc->spk_pa_gpio);
 ret = gpio_request(sdm660_cdc->spk_pa_gpio, "spk_pa");
 if (ret) {
 pr_err("request spk_pa_gpio failed, ret=%d\n",ret);
 gpio_free(sdm660_cdc->spk_pa_gpio);
 }
 }
 //kangting end
 ret = snd_soc_register_codec(&pdev->dev,
   &soc_codec_dev_sdm660_cdc,
   msm_anlg_cdc_i2s_dai,
   ARRAY_SIZE(msm_anlg_cdc_i2s_dai));
 if (ret) {
 dev_err(&pdev->dev,
 "%s:snd_soc_register_codec failed with error %d\n",
 __func__, ret);
 goto err_supplies;
 }
 BLOCKING_INIT_NOTIFIER_HEAD(&sdm660_cdc->notifier);
 BLOCKING_INIT_NOTIFIER_HEAD(&sdm660_cdc->notifier_mbhc);
 //add by zhaojr
 gpio63_spk_pa_gpio = sdm660_cdc->spk_pa_gpio; //将设备树种定义的IO口号获取进来
 spk_pa_kobj = kobject_create_and_add("spk_pa", NULL); //创建/sys/spk_pa/目录
 ret = sysfs_create_group(spk_pa_kobj, &apkpa_attr_group); //创建/sys/class/spk_pa/spkpaon节点
 if (ret)
 dev_err(&pdev->dev,"%s:sysfs_create_group failed with error\n",__func__);
 //end zhaojr add
 ....................................

4、在remove函数中释放资源

static int msm_anlg_cdc_remove(struct platform_device *pdev)
{
 struct sdm660_cdc_priv *sdm660_cdc = dev_get_drvdata(&pdev->dev);
 struct sdm660_cdc_pdata *pdata = sdm660_cdc->dev->platform_data;
 int count;
 //add by zhaojr //释放资源
 gpio_free(sdm660_cdc->spk_pa_gpio);
 kobject_put(spk_pa_kobj); //关键函数
 sysfs_remove_group(spk_pa_kobj, &apkpa_attr_group); //关键函数
 //end zhaojr add
 for (count = 0; count < sdm660_cdc->child_count &&
 count < ANLG_CDC_CHILD_DEVICES_MAX; count++)
 platform_device_unregister(
 sdm660_cdc->pdev_child_devices[count]);
 snd_soc_unregister_codec(&pdev->dev);
 msm_anlg_cdc_disable_supplies(sdm660_cdc, pdata);
 wcd9xxx_spmi_irq_exit();
 devm_kfree(&pdev->dev, sdm660_cdc);
 return 0;
}

总结

到此这篇关于解析linux或android添加文件系统的属性接口的方法的文章就介绍到这了,更多相关linux 文件系统的属性接口内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • 详细分析Linux文件系统

    本片文章针对Linux文件系统从原理到运行机制做了非常详细的理论分析,有助于读者对此深入的理解,以下是具体内容: Linux上的文件系统一般来说就是EXT2或EXT3,但这篇文章并不准备一上来就直接讲它们,而希望结合Linux操作系统并从文件系统建立的基础--硬盘开始,一步步认识Linux的文件系统. 1.机械硬盘的物理存储机制 现代计算机大部分文件存储功能都是由机械硬盘这种设备提供的.(现在的SSD和闪存从概念和逻辑上都部分继承自机械硬盘,所以使用机械硬盘来进行理解也是没有问题的) 机械硬盘能

  • Linux中如何查看已挂载的文件系统类型详解

    前言 如你所知,Linux 支持非常多的文件系统,例如 ext4.ext3.ext2.sysfs.securityfs.FAT16.FAT32.NTFS 等等,当前被使用最多的文件系统是 ext4.你曾经疑惑过你的 Linux 系统使用的是什么类型的文件系统吗?没有疑惑过?不用担心!我们将帮助你.本指南将解释如何在类 Unix 的操作系统中查看已挂载的文件系统类型. 在 Linux 中查看已挂载的文件系统类型 有很多种方法可以在 Linux 中查看已挂载的文件系统类型,下面我将给出 8 种不同的

  • Linux查看分区文件系统类型的方法总结

    前言 在Linux 中如何查看分区的文件系统类型,下面总结几种查看分区文件系统类型的方法.下面话不多说了,来随着小编一起学习学习吧 1: df -T 命令查看 这个是最简单的命令,文件系统类型在Type列输出.只可以查看已经挂载的分区和文件系统类型.如下所示: [root@mylnx008 ~]# df -T /dev/sdb Filesystem Type 1K-blocks Used Available Use% Mounted on /dev/sdb xfs 315467264 43564

  • Linux中对lvm逻辑卷分区大小的调整教程(针对xfs与ext4不同文件系统)

    前言 当我们在安装系统的时候,由于没有合理分配分区空间,在后续维护过程中,发现有些分区空间不够使用,而有的分区空间却有很多剩余空间.如果这些分区在装系统的时候使用了lvm(前提是这些分区要是lvm逻辑卷分区),那么就可以轻松进行扩容或缩容!不同文件系统类型所对应的创建.检查.调整命令不同,下面就针对xfs和ext2/3/4文件系统的lvm分区空间的扩容和缩容的操作做一记录: -------------------------------------------------------------

  • 解析linux或android添加文件系统的属性接口的方法

    第一种: 1.添加关键头文件: #include <linux/of_gpio.h> #include <linux/gpio.h> #include <linux/delay.h> #include <linux/module.h> #include <linux/types.h> #include <linux/kobject.h> 2.在已经存在驱动文件中搜索"DEVICE_ATTR"关键字,如果存在,直接参

  • 利用jQuery来动态为属性添加或者删除属性的简单方法

    现在做的项目有这样一个需要: 先看图吧^^ 要求: 1.当点击导出Excel方式的时候,如果是"勾选导出"或"不分页导出"时,下面的文本框不能修改 2.当点击"分页导出"时,第一个文本框中的值可以被修改,但第二个文本框中的值不可以手动修改,但会随着第一个文本框中输入的值不断变化 实现: 这个页面我就不说怎么做的了,咱们看重点,如何来实现动态的增加属性值: function changeAttr(){ //onchange事件 添加和去除只读属性

  • Linux中Nginx添加自签证书TLS的方法

    创建自签证书TLS openssl req \ -newkey rsa:2048 \ -x509 \ -nodes \ -keyout test.com.key \ -new \ -out test.com.crt \ -subj /CN=test.com \ -reqexts SAN \ -extensions SAN \ -config <(cat /etc/pki/tls/openssl.cnf \ <(printf '[SAN]\nsubjectAltName=DNS:test.com

  • Android实现socket通信统一接口的方法

    目录 UDP通信与TCP通信的实现 UDP通信 TCP客户端通信 使用统一接口 定义接口 实现接口 实现应用层 总结 Android实现socket通信统一接口,统一接口之后可以在不需要大量修改应用层代码的情况下,使用与当前功能类似但是底层实现不同的功能,以实现的UDP与TCP两种通信方式的接口为例. UDP通信与TCP通信的实现 UDP通信 我们在使用UDP通信方式时,我们会这样实现 //设置socket val socket = DatagramSocket() val serverPort

  • Linux中把用户添加到组的4个方法总结

    前言 Linux 组是用于管理 Linux 中用户帐户的组织单位.对于 Linux 系统中的每一个用户和组,它都有惟一的数字标识号.它被称为 用户 ID(UID)和组 ID(GID).组的主要目的是为组的成员定义一组特权.它们都可以执行特定的操作,但不能执行其他操作. Linux 中有两种类型的默认组.每个用户应该只有一个 主要组 primary group  和任意数量的 次要组 secondary group  . • 主要组: 创建用户帐户时,已将主要组添加到用户.它通常是用户的名称.在执

  • Android 添加系统设置属性的实现及步骤

    Android 添加系统设置属性的实现及步骤 Android源码开发中,常常要用到一些全局标志或者说变量,这时候我们可以给android系统添加自己想要的属性. 1. 整个android系统都能访问的属性: 2.该属性值永久保存: 那么我们可以仿照Settings.System.SCRREN_OFF_TIMEOUT这个系统设置的方式来新建一个系统设置值: 操作步骤如下: 1. 设置该值的关键字: 涉及文件:frameworks/base/core/java/android/provider/Se

  • 解析Linux源码之epoll

    目录 一.前言 二.简单的epoll例子 2.1.epoll_create 2.2.struct eventpoll 2.3.epoll_ctl(add) 2.4.ep_insert 2.5.tfile->f_op->poll的实现 2.6.回调函数的安装 2.7.epoll_wait 2.8.ep_send_events 三.事件到来添加到epoll就绪队列(rdllist)的过程 3.1.可读事件到来 3.2.可写事件到来 四.关闭描述符(close fd) 五.总结 一.前言 在linu

  • Android给TextView添加点击事件的实现方法

    首先设定TextView的clickable属性为true. 可以在布局文件中进行设定,比如: <TextView android:id="@+id/phone" android:clickable="true" --------->设定此属性 android:layout_marginLeft="10dp" android:layout_below="@id/address" android:layout_toR

  • Android 添加TextView删除线(代码简单)

    在做商城的项目中,有这么个需求,就是一个产品下有两个价格,一个是市场价,一个是销售价,这时要把市场价添加个删除线: 刚开始遇到这个时,在网上找了半天的资料,看到最多的就是用SpannableString这个类中的方法,而且代码多,感觉有点繁琐,后来发现用Paint里的一些方法实现更好,只须一句代码 public class TestActivity extends Activity { private TextView tv; @Override public void onCreate(Bun

  • Android 自定义view和属性动画实现充电进度条效果

    近期项目中需要使用到一种类似手机电池充电进度的动画效果,以前没学属性动画的时候,是用图片+定时器的方式来完成的,最近一直在学习动画这一块,再加上复习一下自定义view的相关知识点,所以打算用属性动画和自定义view的方式来完成这个功能,将它开源出来,供有需要的人了解一下相关的内容. 本次实现的功能类似下面的效果: 接下来便详细解析一下如何完成这个功能,了解其中的原理,这样就能举一反三,实现其他类似的动画效果了. 详细代码请看大屏幕 https://github.com/crazyandcoder

随机推荐