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設置如下:
xadc@f8007100 {
compatible= "xlnx,zynq-xadc-1.00.a";
reg =<0xf8007100 0x20>;
interrupts= <0 7 4>;
interrupt-parent = <&gic>;
clocks =<&pcap_clk>;
xlnx,channels {
#address-cells = <1>;
#size-cells = <0>;
channel@0 {
reg= <0>;
};
channel@1 {
reg= <1>;
};
channel@8 {
reg= <8>;
};
};
};
這里只啟用了三個channel,如果啟用更多channel可以在devicetree中進行添加。
3、 kernel配置
內核中已經包含了xadc的驅動程序,位置是drivers\iio\adc,最后三個xilinx打頭的文件。將驅動加入內核配置過程如下:
如果為了省事,也可以直接在defconfig文件中加入配置信息,這樣默認是選中的。
4、 測試
加載devicetree,uimage,uramdisk后能在sys目錄下找到檢測信息:
打開in_temp0_raw能看到數字,注意,該數字是原始數字,需要轉換才能得到正確的攝氏度。
該數字不斷變化,表明溫度在變化。其他電壓信息類似。
為了更直觀的展現監測信息,可以做一個用戶app來打印,代碼如下:
#define MAX_PATH_SIZE 200
#define MAX_NAME_SIZE 50
#define MAX_VALUE_SIZE 100
#define MAX_CMD_NAME_SIZE 100
#define MAX_UNIT_NAME_SIZE 50
#define SYS_PATH_IIO "/sys/bus/iio/devices/iio:device0"
#define VCC_INT_CMD "xadc_get_value_vccint"
#define VCC_AUX_CMD "xadc_get_value_vccaux"
#define VCC_BRAM_CMD "xadc_get_value_vccbram"
#define VCC_TEMP_CMD "xadc_get_value_temp"
#define VCC_EXT_CH_CMD "xadc_get_value_ext_ch"
static const int mV_mul = 1000;
static const int multiplier = 1 << 12;
static char gNodeName[MAX_NAME_SIZE];
enum EConvType
{
EConvType_None,
EConvType_Raw_to_Scale,
EConvType_Scale_to_Raw,
EConvType_Max
};
enum XADC_Param
{
EParamVccInt,
EParamVccAux,
EParamVccBRam,
EParamTemp,
EParamVAux0,
EParamMax
};
struct command
{
const enum XADC_Param parameter_id;
const char cmd_name[MAX_CMD_NAME_SIZE];
const char unit[MAX_UNIT_NAME_SIZE];
};
struct command command_list[EParamMax] = {
{EParamVccInt, VCC_INT_CMD, "mV"},
{EParamVccAux, VCC_AUX_CMD, "mV"},
{EParamVccBRam, VCC_BRAM_CMD, "mV"},
{EParamTemp, VCC_TEMP_CMD, "Degree Celsius"},
{EParamVAux0, VCC_EXT_CH_CMD, "mV"}
};
struct XadcParameter
{
const char name[MAX_NAME_SIZE];
float value;
float (* const conv_fn)(float,enum EConvType);
};
/*
struct XadcParameter gXadcData[EParamMax] = {
[EParamVccInt] = { "in_voltage0_vccint_raw", 0, conv_voltage},
[EParamVccAux] = { "in_voltage4_vccpaux_raw", 0, conv_voltage},
[EParamVccBRam]= { "in_voltage2_vccbram_raw", 0, conv_voltage},
[EParamTemp] = { "in_temp0_raw", 0, conv_temperature},
[EParamVAux0] = { "in_voltage8_raw", 0, conv_voltage_ext_ch}
};
*/
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include "xadc_core.h"
//utility functions
float conv_voltage(float input, enum EConvType conv_direction)
{
float result=0;
評論
查看更多