当前位置:   article > 正文

Zynq-Linux移植学习笔记之16-Zynq下linux XADC驱动_zynq linux xadc

zynq linux xadc

1、  简介

XADC是zynq芯片内部进行温度和电压检测的模块,通过(Xilinx Wiki - xadc.html)这篇wiki可以知道,XADC控制器有两种表现形式,一种是位于PS内部,即文档中提到的the PS-XADC interface for the PS software to control the XADC,另一种是位于PL内部,通过IP核的方式实现。目前常用的是第一种。

通过ug480_7Series_XADC这篇文档得知,The ADCs can access up to 17 external analog input channels,也就是除了内部原有的监测项外,XADC最多支持17路扩展通道,可以参看下图。


对硬件有了基本的了解后,就可以进行devicetree的设置了。

 

2、  devicetree配置

由于是在PS部分,XADC控制器设置就像其他控制器一样即可,system.hdf中给出了地址:


Devicetree设置如下:

  1. xadc@f8007100 {
  2. compatible= "xlnx,zynq-xadc-1.00.a";
  3. reg =<0xf8007100 0x20>;
  4. interrupts= <0 7 4>;
  5. interrupt-parent = <&gic>;
  6. clocks =<&pcap_clk>;
  7. xlnx,channels {
  8. #address-cells = <1>;
  9. #size-cells = <0>;
  10. channel@0 {
  11. reg= <0>;
  12. };
  13. channel@1 {
  14. reg= <1>;
  15. };
  16. channel@8 {
  17. reg= <8>;
  18. };
  19. };
  20. };


这里只启用了三个channel,如果启用更多channel可以在devicetree中进行添加。

 

3、  kernel配置

内核中已经包含了xadc的驱动程序,位置是drivers\iio\adc,最后三个xilinx打头的文件。将驱动加入内核配置过程如下:







如果为了省事,也可以直接在defconfig文件中加入配置信息,这样默认是选中的。


 

4、  测试

加载devicetree,uimage,uramdisk后能在sys目录下找到检测信息:


打开in_temp0_raw能看到数字,注意,该数字是原始数字,需要转换才能得到正确的摄氏度。


该数字不断变化,表明温度在变化。其他电压信息类似。

为了更直观的展现监测信息,可以做一个用户app来打印,代码如下:

 

  1. #define MAX_PATH_SIZE 200
  2. #define MAX_NAME_SIZE 50
  3. #define MAX_VALUE_SIZE 100
  4. #define MAX_CMD_NAME_SIZE 100
  5. #define MAX_UNIT_NAME_SIZE 50
  6. #define SYS_PATH_IIO "/sys/bus/iio/devices/iio:device0"
  7. #define VCC_INT_CMD "xadc_get_value_vccint"
  8. #define VCC_AUX_CMD "xadc_get_value_vccaux"
  9. #define VCC_BRAM_CMD "xadc_get_value_vccbram"
  10. #define VCC_TEMP_CMD "xadc_get_value_temp"
  11. #define VCC_EXT_CH_CMD "xadc_get_value_ext_ch"
  12. static const int mV_mul = 1000;
  13. static const int multiplier = 1 << 12;
  14. static char gNodeName[MAX_NAME_SIZE];
  15. enum EConvType
  16. {
  17. EConvType_None,
  18. EConvType_Raw_to_Scale,
  19. EConvType_Scale_to_Raw,
  20. EConvType_Max
  21. };
  22. enum XADC_Param
  23. {
  24. EParamVccInt,
  25. EParamVccAux,
  26. EParamVccBRam,
  27. EParamTemp,
  28. EParamVAux0,
  29. EParamMax
  30. };
  31. struct command
  32. {
  33. const enum XADC_Param parameter_id;
  34. const char cmd_name[MAX_CMD_NAME_SIZE];
  35. const char unit[MAX_UNIT_NAME_SIZE];
  36. };
  37. struct command command_list[EParamMax] = {
  38. {EParamVccInt, VCC_INT_CMD, "mV"},
  39. {EParamVccAux, VCC_AUX_CMD, "mV"},
  40. {EParamVccBRam, VCC_BRAM_CMD, "mV"},
  41. {EParamTemp, VCC_TEMP_CMD, "Degree Celsius"},
  42. {EParamVAux0, VCC_EXT_CH_CMD, "mV"}
  43. };
  44. struct XadcParameter
  45. {
  46. const char name[MAX_NAME_SIZE];
  47. float value;
  48. float (* const conv_fn)(float,enum EConvType);
  49. };
  50. /*
  51. struct XadcParameter gXadcData[EParamMax] = {
  52. [EParamVccInt] = { "in_voltage0_vccint_raw", 0, conv_voltage},
  53. [EParamVccAux] = { "in_voltage4_vccpaux_raw", 0, conv_voltage},
  54. [EParamVccBRam]= { "in_voltage2_vccbram_raw", 0, conv_voltage},
  55. [EParamTemp] = { "in_temp0_raw", 0, conv_temperature},
  56. [EParamVAux0] = { "in_voltage8_raw", 0, conv_voltage_ext_ch}
  57. };
  58. */

  1. #include <stdio.h>
  2. #include <string.h>
  3. #include <errno.h>
  4. #include <unistd.h>
  5. #include <dirent.h>
  6. #include <stdlib.h>
  7. #include <sys/ioctl.h>
  8. #include <fcntl.h>
  9. #include <ctype.h>
  10. #include <pthread.h>
  11. #include <assert.h>
  12. #include "xadc_core.h"
  13. //utility functions
  14. float conv_voltage(float input, enum EConvType conv_direction)
  15. {
  16. float result=0;
  17. switch(conv_direction)
  18. {
  19. case EConvType_Raw_to_Scale:
  20. result = ((input * 3.0 * mV_mul)/multiplier);
  21. break;
  22. case EConvType_Scale_to_Raw:
  23. result = (input/(3.0 * mV_mul))*multiplier;
  24. break;
  25. default:
  26. printf("Convertion type incorrect... Doing no conversion\n");
  27. // intentional no break;
  28. case EConvType_None:
  29. result = input;
  30. break;
  31. }
  32. return result;
  33. }
  34. float conv_voltage_ext_ch(float input, enum EConvType conv_direction)
  35. {
  36. float result=0;
  37. switch(conv_direction)
  38. {
  39. case EConvType_Raw_to_Scale:
  40. result = ((input * mV_mul)/multiplier);
  41. break;
  42. case EConvType_Scale_to_Raw:
  43. result = (input/mV_mul)*multiplier;
  44. break;
  45. default:
  46. printf("Convertion type incorrect... Doing no conversion\n");
  47. // intentional no break;
  48. case EConvType_None:
  49. result = input;
  50. break;
  51. }
  52. return result;
  53. }
  54. float conv_temperature(float input, enum EConvType conv_direction)
  55. {
  56. float result=0;
  57. switch(conv_direction)
  58. {
  59. case EConvType_Raw_to_Scale:
  60. result = ((input * 503.975)/multiplier) - 273.15;
  61. break;
  62. case EConvType_Scale_to_Raw:
  63. result = (input + 273.15)*multiplier/503.975;
  64. break;
  65. default:
  66. printf("Conversion type incorrect... Doing no conversion\n");
  67. // intentional no break;
  68. case EConvType_None:
  69. result = input;
  70. break;
  71. }
  72. return result;
  73. }
  74. void get_iio_node()
  75. {
  76. struct dirent **namelist;
  77. int i,n;
  78. char value=0;
  79. int fd = -1;
  80. char upset[20];
  81. float raw_data=0;
  82. float true_data=0;
  83. int offset=0;
  84. int currpos;
  85. n = scandir(SYS_PATH_IIO, &namelist, 0, alphasort);
  86. for (i=0; i < n; i++)
  87. {
  88. sprintf(gNodeName,"%s/%s", SYS_PATH_IIO, namelist[i]->d_name);
  89. fd = open(gNodeName, O_RDWR );
  90. if(strstr(gNodeName,"temp"))
  91. {
  92. if(strstr(gNodeName,"raw"))
  93. {
  94. offset=0;
  95. while(offset<5)
  96. {
  97. lseek(fd,offset,SEEK_SET);
  98. read(fd,&value,sizeof(char));
  99. upset[offset]=value;
  100. offset++;
  101. }
  102. upset[offset]='\0';
  103. raw_data=atoi(upset);
  104. true_data=conv_temperature(raw_data, EConvType_Raw_to_Scale);
  105. printf("%s is %f cent\n",namelist[i]->d_name,true_data);
  106. }
  107. }
  108. else if(strstr(gNodeName,"voltage"))
  109. {
  110. if(strstr(gNodeName,"raw"))
  111. {
  112. offset=0;
  113. while(offset<5)
  114. {
  115. lseek(fd,offset,SEEK_SET);
  116. read(fd,&value,sizeof(char));
  117. upset[offset]=value;
  118. offset++;
  119. }
  120. upset[offset]='\0';
  121. raw_data=atoi(upset);
  122. true_data=conv_voltage(raw_data, EConvType_Raw_to_Scale);
  123. printf("%s is %f mv\n",namelist[i]->d_name,true_data);
  124. }
  125. }
  126. close(fd);
  127. }
  128. }
  129. int main(int argc, char *argv[])
  130. {
  131. get_iio_node();
  132. return 0;
  133. }


 

 

在编写这段代码时遇到好几个问题,首先用read函数取值取到的其实是文件中ASCII码字符,没有什么好办法只好改成一位一位读取存入字符串然后调用atoi函数转换为int数字。其次,用lseek函数获取文件大小时得到的值都为4096,但是这些文件并不是目录,不知道为何大小都显示的是linux最基本的文件块大小,最后只好投机取巧根据每个文件中的数字位数进行取值,例如temp和vcc这些文件只有4位,那么就从文件中读4位。最后输出结果如下:


上图中温度为39度,和实际情况差不多。至此XADC驱动告一段落。

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/很楠不爱3/article/detail/509922
推荐阅读
相关标签
  

闽ICP备14008679号