1、 簡(jiǎn)介
XADC是zynq芯片內(nèi)部進(jìn)行溫度和電壓檢測(cè)的模塊,通過(guò)(Xilinx Wiki - xadc.html)這篇wiki可以知道,XADC控制器有兩種表現(xiàn)形式,一種是位于PS內(nèi)部,即文檔中提到的the PS-XADC interface for the PS software to control the XADC,另一種是位于PL內(nèi)部,通過(guò)IP核的方式實(shí)現(xiàn)。目前常用的是第一種。
通過(guò)ug480_7Series_XADC這篇文檔得知,The ADCs can access up to 17 external analog input channels,也就是除了內(nèi)部原有的監(jiān)測(cè)項(xiàng)外,XADC最多支持17路擴(kuò)展通道,可以參看下圖。
	
對(duì)硬件有了基本的了解后,就可以進(jìn)行devicetree的設(shè)置了。
2、 devicetree配置
由于是在PS部分,XADC控制器設(shè)置就像其他控制器一樣即可,system.hdf中給出了地址:
	
	Devicetree設(shè)置如下:
	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>;
	};
	};
	};
這里只啟用了三個(gè)channel,如果啟用更多channel可以在devicetree中進(jìn)行添加。
	3、 kernel配置
	內(nèi)核中已經(jīng)包含了xadc的驅(qū)動(dòng)程序,位置是drivers\iio\adc,最后三個(gè)xilinx打頭的文件。將驅(qū)動(dòng)加入內(nèi)核配置過(guò)程如下:
	
	
	
	
如果為了省事,也可以直接在defconfig文件中加入配置信息,這樣默認(rèn)是選中的。
	
	4、 測(cè)試
	加載devicetree,uimage,uramdisk后能在sys目錄下找到檢測(cè)信息:
	
打開(kāi)in_temp0_raw能看到數(shù)字,注意,該數(shù)字是原始數(shù)字,需要轉(zhuǎn)換才能得到正確的攝氏度。
	
該數(shù)字不斷變化,表明溫度在變化。其他電壓信息類似。
	為了更直觀的展現(xiàn)監(jiān)測(cè)信息,可以做一個(gè)用戶app來(lái)打印,代碼如下:
	#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;
                        電子發(fā)燒友App
                    
                
                
          
        
        





           
            
            
                
            
評(píng)論