Zynq-Linux移植学习笔记之16-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设置如下:

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 <stdio.h>
#include <string.h>
#include <errno.h>
#include <unistd.h>
#include <dirent.h>
#include <stdlib.h>
#include <sys/ioctl.h>
#include <fcntl.h>
#include <ctype.h>
#include <pthread.h>
#include <assert.h>

#include "xadc_core.h"

//utility functions
float conv_voltage(float input, enum EConvType conv_direction)
{
	float result=0;

	switch(conv_direction)
	{
	case EConvType_Raw_to_Scale:
		result = ((input * 3.0 * mV_mul)/multiplier);
		break;
	case EConvType_Scale_to_Raw:
		result = (input/(3.0 * mV_mul))*multiplier;
		break;
	default:
		printf("Convertion type incorrect... Doing no conversion\n");
		//  intentional no break;
	case EConvType_None:
		result = input;
		break;
	}

	return result;
}

float conv_voltage_ext_ch(float input, enum EConvType conv_direction)
{
	float result=0;

	switch(conv_direction)
	{
	case EConvType_Raw_to_Scale:
		result = ((input * mV_mul)/multiplier);
		break;
	case EConvType_Scale_to_Raw:
		result = (input/mV_mul)*multiplier;
		break;
	default:
		printf("Convertion type incorrect... Doing no conversion\n");
		//  intentional no break;
	case EConvType_None:
		result = input;
		break;
	}

	return result;
}

float conv_temperature(float input, enum EConvType conv_direction)
{
	float result=0;

	switch(conv_direction)
	{
	case EConvType_Raw_to_Scale:
		result = ((input * 503.975)/multiplier) - 273.15;
		break;
	case EConvType_Scale_to_Raw:
		result = (input + 273.15)*multiplier/503.975;
		break;
	default:
		printf("Conversion type incorrect... Doing no conversion\n");
		//  intentional no break;
	case EConvType_None:
		result = input;
		break;
	}

	return result;
}


void get_iio_node()
{
	struct dirent **namelist;
	int i,n;
	char value=0;
	int fd = -1;
	char upset[20];
	float raw_data=0;
	float true_data=0;
	int offset=0;
	int currpos;

	n = scandir(SYS_PATH_IIO, &namelist, 0, alphasort);

	for (i=0; i < n; i++)
	{
		sprintf(gNodeName,"%s/%s", SYS_PATH_IIO, namelist[i]->d_name);

		fd = open(gNodeName, O_RDWR );

		if(strstr(gNodeName,"temp"))
		{
			if(strstr(gNodeName,"raw"))
			{
				offset=0;
				while(offset<5)
				{	
					lseek(fd,offset,SEEK_SET);
					read(fd,&value,sizeof(char));	
					upset[offset]=value;
					offset++;
				}	
				upset[offset]='\0';
				raw_data=atoi(upset);
				true_data=conv_temperature(raw_data, EConvType_Raw_to_Scale);
				printf("%s is %f cent\n",namelist[i]->d_name,true_data);
			}
		}
		else if(strstr(gNodeName,"voltage"))
		{
			if(strstr(gNodeName,"raw"))
			{
				offset=0;
				while(offset<5)
				{
					lseek(fd,offset,SEEK_SET);
					read(fd,&value,sizeof(char));	
					upset[offset]=value;
					offset++;
				}	
				upset[offset]='\0';
				raw_data=atoi(upset);
				true_data=conv_voltage(raw_data, EConvType_Raw_to_Scale);
				printf("%s is %f mv\n",namelist[i]->d_name,true_data);
			}
		}
		close(fd);

	}
}

int main(int argc, char *argv[])
{
	get_iio_node();
	return 0;
}



 

 

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


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

  • 5
    点赞
  • 54
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
### 回答1: "Zynq-Linux移植学习笔记"是一篇关于在Zynq SoC平台上移植Linux操作系统的学习笔记。该笔记主要介绍了如何在Zynq平台上搭建交叉编译环境、编译内核、制作根文件系统以及启动Linux系统等方面的知识。通过学习这篇笔记,读者可以了解到如何将Linux操作系统移植Zynq平台上,并且可以通过实践来深入理解相关的知识和技 ### 回答2: Zynq Linux移植是搭建Zynq硬件平台和在该平台上运行Linux系统的过程。它包括了硬件的设计和软件的开发,这有助于实现在Zynq平台上开发Linux应用程序的目标。 首先,进行Zynq Linux移植前需要研究设备的结构和硬件构造。zynq硬件平台包含两个主要组成部分:PS和PL。PS负责处理器系统和外设的管理和控制,PL则是可编程逻辑,支持FPGA逻辑的定制化,同时也支持外设的实现。在移植时,需要设计PS的硬件架构和软件驱动程序,同时也需要配置PL。一般情况下,需要进行如(Xilinx SDK)和(Petalinux)等的软件开发环境安装和配置。 接下来,进行Linux系统的移植。这部分工作需要了解Linux内核的结构、功能和特性,然后根据硬件架构,对Linux系统进行调整和定制,构建出适合硬件平台的Linux系统。这个过程需要进行的工作包括:解编译适合SOC的内核、制定设备树、调整内核参数等。 最后,这些工作完成后,就可以在Zynq平台上编译和运行Linux应用程序了。开发者可以尝试通过开发板上的GPIO、I2C、SPI或UART等接口,学习Linux的设备驱动程序、网络编程、文件系统管理等知识点。可逐步学习如何掌握Linux的shell、打包工具、交叉编译工具等。 总结来说,Zynq Linux移植工作是基础的硬件构造、嵌入式软件和Linux知识的综合应用,需要开发者有至少三方面的技能。需要熟练编写硬件设计,熟练掌握Linux内核编程以及Linux系统软件的维护和管理。这些技能的应用能力对于移植Zynq平台Linux系统和应用程序的开发工程师来说是非常必要的。 ### 回答3: 随着嵌入式系统的广泛应用,zynq-linux移植成为了嵌入式开发的一个重要研究课题。zynq-linux移植涉及到许多方面的知识,在学习过程中需要注意以下几点: 一、zynq-linux移植前需要了解的基础知识 在开始进行zynq-linux移植之前,需要对Linux系统、ARM技术、FPGA开发、C语言等基础知识有一定的了解。同时需要熟悉zynq系列的基本架构和应用场景。 二、zynq-linux移植必要的步骤 zynq-linux移植的过程主要分为以下几个步骤:首先是确定硬件平台和环境搭建;其次是进行内核编译和配置;然后是uboot编译和烧录;最后是Linux文件系统的制作和烧录。在整个移植过程中,需要注意各个步骤的顺序和详细操作,确保每一步都正确完成。 三、zynq-linux移植中可能会遇到的问题 在zynq-linux移植过程中,可能会遇到各种问题,例如硬件平台的兼容性、内核配置的错误、uboot烧录问题、文件系统制作出错等。在遇到这些问题时,需要耐心地进行排查和解决,同时也可以借助搜索引擎和社区的技术支持。 四、zynq-linux移植之后的应用与拓展 zynq-linux移植成功之后,可以将其应用于各种嵌入式系统中,例如网络设备、智能家居、工业控制等领域。同时,还可以进行拓展和优化,例如添加各种驱动程序、优化系统性能等。 总之,在进行zynq-linux移植学习和实践中,需要认真学习基础知识,仔细操作每个步骤,及时排查并解决问题,并在成功移植之后持续进行应用与拓展。
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值