自己想要打印EMIO管脚的物理地址,在SDK提供的头文件中加入printf是无法打印的,基于此

我将需要打印地址的几个函数提取出来,放在main函数中,然后在里面加入printf打印这些用户管脚的地址:

Ps_SetDirectionPin   //设置方向,原函数XGpioPs_SetDirectionPin
Ps_SetOutputEnablePin//设置使能,原函数XGpioPs_SetOutputEnablePin
Ps_WritePin //写入值,原函数XGpioPs_WritePin

将其修改为自定义函数,目的就是利用串口打印物理地址

#include <stdio.h>
#include "xgpiops.h"
#include "sleep.h" /****************************************************************************/
/**
*
* Set the Direction of the specified pin.
*
* @param InstancePtr is a pointer to the XGpioPs instance.
* @param Pin is the pin number to which the Data is to be written.
* Valid values are 0-117 in Zynq and 0-173 in Zynq Ultrascale+ MP.
* @param Direction is the direction to be set for the specified pin.
* Valid values are 0 for Input Direction, 1 for Output Direction.
*
* @return None.
*
*****************************************************************************/
void Ps_SetDirectionPin(XGpioPs *InstancePtr, u32 Pin, u32 Direction)
{
u8 Bank;
u8 PinNumber;
u32 DirModeReg; Xil_AssertVoid(InstancePtr != NULL);
Xil_AssertVoid(InstancePtr->IsReady == XIL_COMPONENT_IS_READY);
Xil_AssertVoid(Pin < InstancePtr->MaxPinNum);
Xil_AssertVoid(Direction <= (u32)); /* Get the Bank number and Pin number within the bank. */
XGpioPs_GetBankPin((u8)Pin, &Bank, &PinNumber); DirModeReg = XGpioPs_ReadReg(InstancePtr->GpioConfig.BaseAddr,
((u32)(Bank) * XGPIOPS_REG_MASK_OFFSET) +
XGPIOPS_DIRM_OFFSET); if (Direction!=(u32)) { /* Output Direction */
DirModeReg |= ((u32) << (u32)PinNumber);
} else { /* Input Direction */
DirModeReg &= ~ ((u32) << (u32)PinNumber);
}
printf("Direction:%lx,%lx,%lx\n",InstancePtr->GpioConfig.BaseAddr,((u32)(Bank) * XGPIOPS_REG_MASK_OFFSET) +
XGPIOPS_DIRM_OFFSET,DirModeReg); XGpioPs_WriteReg(InstancePtr->GpioConfig.BaseAddr,
((u32)(Bank) * XGPIOPS_REG_MASK_OFFSET) +
XGPIOPS_DIRM_OFFSET, DirModeReg);
} /****************************************************************************/
/**
*
* Set the Output Enable of the specified pin.
*
* @param InstancePtr is a pointer to the XGpioPs instance.
* @param Pin is the pin number to which the Data is to be written.
* Valid values are 0-117 in Zynq and 0-173 in Zynq Ultrascale+ MP.
* @param OpEnable specifies whether the Output Enable for the specified
* pin should be enabled.
* Valid values are 0 for Disabling Output Enable,
* 1 for Enabling Output Enable.
*
* @return None.
*
* @note None.
*
*****************************************************************************/
void Ps_SetOutputEnablePin(XGpioPs *InstancePtr, u32 Pin, u32 OpEnable)
{
u8 Bank;
u8 PinNumber;
u32 OpEnableReg; Xil_AssertVoid(InstancePtr != NULL);
Xil_AssertVoid(InstancePtr->IsReady == XIL_COMPONENT_IS_READY);
Xil_AssertVoid(Pin < InstancePtr->MaxPinNum);
Xil_AssertVoid(OpEnable <= (u32)); /* Get the Bank number and Pin number within the bank. */
XGpioPs_GetBankPin((u8)Pin, &Bank, &PinNumber); OpEnableReg = XGpioPs_ReadReg(InstancePtr->GpioConfig.BaseAddr,
((u32)(Bank) * XGPIOPS_REG_MASK_OFFSET) +
XGPIOPS_OUTEN_OFFSET); if (OpEnable != (u32)) { /* Enable Output Enable */
OpEnableReg |= ((u32) << (u32)PinNumber);
} else { /* Disable Output Enable */
OpEnableReg &= ~ ((u32) << (u32)PinNumber);
}
printf("OutputEn:%lx,%lx,%lx\n",InstancePtr->GpioConfig.BaseAddr,((u32)(Bank) * XGPIOPS_REG_MASK_OFFSET) +
XGPIOPS_OUTEN_OFFSET,OpEnableReg); XGpioPs_WriteReg(InstancePtr->GpioConfig.BaseAddr,
((u32)(Bank) * XGPIOPS_REG_MASK_OFFSET) +
XGPIOPS_OUTEN_OFFSET, OpEnableReg);
} /****************************************************************************/
/**
*
* Write data to the specified pin.
*
* @param InstancePtr is a pointer to the XGpioPs instance.
* @param Pin is the pin number to which the Data is to be written.
* Valid values are 0-117 in Zynq and 0-173 in Zynq Ultrascale+ MP.
* @param Data is the data to be written to the specified pin (0 or 1).
*
* @return None.
*
* @note This function does a masked write to the specified pin of
* the specified GPIO bank. The previous state of other pins
* is maintained.
*
*****************************************************************************/
void Ps_WritePin(XGpioPs *InstancePtr, u32 Pin, u32 Data)
{
u32 RegOffset;
u32 Value;
u8 Bank;
u8 PinNumber;
u32 DataVar = Data; Xil_AssertVoid(InstancePtr != NULL);
Xil_AssertVoid(InstancePtr->IsReady == XIL_COMPONENT_IS_READY);
Xil_AssertVoid(Pin < InstancePtr->MaxPinNum); /* Get the Bank number and Pin number within the bank. */
XGpioPs_GetBankPin((u8)Pin, &Bank, &PinNumber);
printf("bank=%d,PinNumber=%d\n",Bank,PinNumber);
if (PinNumber > 15U) {
/* There are only 16 data bits in bit maskable register. */
PinNumber -= (u8);
RegOffset = XGPIOPS_DATA_MSW_OFFSET;
} else {
RegOffset = XGPIOPS_DATA_LSW_OFFSET;
} /*
* Get the 32 bit value to be written to the Mask/Data register where
* the upper 16 bits is the mask and lower 16 bits is the data.
*/
DataVar &= (u32)0x01;
Value = ~((u32) << (PinNumber + 16U)) & ((DataVar << PinNumber) | 0xFFFF0000U); printf("write:%lx,%lx,%lx\n",InstancePtr->GpioConfig.BaseAddr,((u32)(Bank) * XGPIOPS_DATA_MASK_OFFSET) +
RegOffset,Value); XGpioPs_WriteReg(InstancePtr->GpioConfig.BaseAddr,
((u32)(Bank) * XGPIOPS_DATA_MASK_OFFSET) +
RegOffset, Value);
} int main()
{ #if 1
static XGpioPs psGpioInstancePtr;
XGpioPs_Config* GpioConfigPtr; int iPinNumber= ; //想想为什么是54
u32 uPinDirection = 0x1; //1表示输出,0表示输入 int xStatus;
//--MIO的初始化
GpioConfigPtr = XGpioPs_LookupConfig(XPAR_PS7_GPIO_0_DEVICE_ID);
if(GpioConfigPtr == NULL)
return XST_FAILURE;
printf("baseAddr:0x%lx\n",GpioConfigPtr->BaseAddr);
xStatus = XGpioPs_CfgInitialize(&psGpioInstancePtr,GpioConfigPtr, GpioConfigPtr->BaseAddr);
if(XST_SUCCESS != xStatus)
print(" PS GPIO INIT FAILED \n\r"); //--MIO的输入输出操作
Ps_SetDirectionPin(&psGpioInstancePtr, iPinNumber,uPinDirection);//配置IO输出方向
Ps_SetOutputEnablePin(&psGpioInstancePtr, iPinNumber,);//配置IO的输出
//while(1)
//{
Ps_WritePin(&psGpioInstancePtr, iPinNumber, );//输出1
sleep();//延时
Ps_WritePin(&psGpioInstancePtr, iPinNumber, );//输出0
sleep();//延时
//} #endif return ; }

上述代码是利用SDK提供的API接口实现led的控制。

如果在做驱动开发时,需要知道led对应的(控制寄存器、数据寄存器)物理地址,因此根据上述代码实现了自己的裸机驱动代码

#include <stdio.h>
#include "xgpiops.h"
#include "sleep.h" #define MY_GPIO_BASE_ADDR 0xE000A000
#define XGPIOPS_DATA_MASK_OFFSET 0x00000008U /* Data/Mask Registers offset */
#define XGPIOPS_REG_MASK_OFFSET 0x00000040U /* Registers offset */
#define XGPIOPS_DIRM_OFFSET 0x00000204U /* Direction Mode Register, RW */
#define XGPIOPS_OUTEN_OFFSET 0x00000208U /* Output Enable Register, RW */
#define XGPIOPS_DATA_LSW_OFFSET 0x00000000U /* Mask and Data Register LSW, WO */
#define XGPIOPS_DATA_MSW_OFFSET 0x00000004U /* Mask and Data Register MSW, WO */ int main()
{ //引脚EMIO54等价于bank=2,PinNum=0
//引脚EMIO57等价于bank=2,PinNum=3
u8 Bank=;
u8 PinNumber=; /**************对应管脚功能配置**********************/
u32 Direction=0X01;//设置MIO_54为输出
u32 OpEnable=0X01;//使能
u32 DataVar = 0X01;//1亮 //方向寄存器地址(由bank决定)
u32 *Gpio_DIR =( u32 *)(MY_GPIO_BASE_ADDR+
((u32)(Bank) * XGPIOPS_REG_MASK_OFFSET) +
XGPIOPS_DIRM_OFFSET); //方向寄存器,1为输出
//使能寄存器地址(由bank决定)
u32 *Gpio_EN =( u32 *)(MY_GPIO_BASE_ADDR+
((u32)(Bank) * XGPIOPS_REG_MASK_OFFSET) +
XGPIOPS_OUTEN_OFFSET);//使能寄存器,1使能
//数据寄存器地址(由bank和PinNumuber决定)
u32 RegOffset; //16位宽,后面要注意
if (PinNumber > 15U) {
/* There are only 16 data bits in bit maskable register. */
PinNumber -= (u8);
RegOffset = XGPIOPS_DATA_MSW_OFFSET;
} else {
RegOffset = XGPIOPS_DATA_LSW_OFFSET;
}
u32 *Gpio_DATA =( u32 *)(MY_GPIO_BASE_ADDR+
((u32)(Bank) * XGPIOPS_DATA_MASK_OFFSET) +
RegOffset);//数据寄存器,1亮 //根据用户方向值,填充方向寄存器值
u32 DirModeReg=XGpioPs_ReadReg(MY_GPIO_BASE_ADDR,
((u32)(Bank) * XGPIOPS_REG_MASK_OFFSET) +
XGPIOPS_OUTEN_OFFSET);
if (Direction!=(u32)) { /* Output Direction */
DirModeReg |= ((u32) << (u32)PinNumber);
} else { /* Input Direction */
DirModeReg &= ~ ((u32) << (u32)PinNumber);
}
*Gpio_DIR = DirModeReg;
//根据用户使能值,填充使能寄存器值
u32 OpEnableReg=XGpioPs_ReadReg(MY_GPIO_BASE_ADDR,
((u32)(Bank) * XGPIOPS_REG_MASK_OFFSET) +
XGPIOPS_OUTEN_OFFSET);
if (OpEnable != (u32)) { /* Enable Output Enable */
OpEnableReg |= ((u32) << (u32)PinNumber);
} else { /* Disable Output Enable */
OpEnableReg &= ~ ((u32) << (u32)PinNumber);
}
*Gpio_EN =OpEnableReg; //根据用户DATA值,填充数据寄存器值
/*
* Get the 32 bit value to be written to the Mask/Data register where
* the upper 16 bits is the mask and lower 16 bits is the data.
*/
DataVar &= (u32)0x01;
u32 Value = ~((u32) << (PinNumber + 16U)) & ((DataVar << PinNumber) | 0xFFFF0000U);
*Gpio_DATA =Value;//1亮 (fffe0001亮,fffe0000灭) //测试物理地址
printf("Gpio_DIR:%lx,value:%lx\n",Gpio_DIR,DirModeReg);
printf("Gpio_EN:%lx,value:%lx\n",Gpio_EN,OpEnableReg);
printf("Gpio_DATA:%lx,value:%lx\n",Gpio_DATA,Value); return ; }

结合UG585手册能够清晰的确定三个主要寄存器(控制、使能、数据)的基地址,长度(32位),该填充的内容,这里没有自己研究,仅仅将代码测试出来,

后面二者结合能够更好的明白ZYNQ EMIO对应的寄存器配置。

该代码仅仅分析了EMIO54-57,分别对应bank2的0-3

用户仅仅修改,以下区域即可,其他代码均是由这些“用户配置”完成

    //引脚EMIO54等价于bank=2,PinNum=0
//引脚EMIO57等价于bank=2,PinNum=3
u8 Bank=;
u8 PinNumber=; /**************对应管脚功能配置**********************/
u32 Direction=0X01;//设置MIO_54为输出
u32 OpEnable=0X01;//使能
u32 DataVar = 0X01;//1亮
  方向寄存器(地址) 方向寄存器(输出) 使能寄存器(地址) 使能寄存器(使能) s数据寄存器(地址) 数据寄存器(亮) 数据寄存器(灭)
Y21    e000a284 f e000a288 f e000a010 fffe0001 fffe0000
G2   e000a284 f e000a288 f e000a010 fffd0002 fffd0000
W21   e000a284 1f e000a288 1f e000a010 fffb0004 fffb0000
A17   e000a284 1f e000a288 1f e000a010 fff70008 fff70000

注意:这里的地址都是绝对地址,也就是最终的物理地址,便于我们做后序驱动开发。

#include <stdio.h>
#include "xgpiops.h"
#include "sleep.h" #define MY_GPIO_BASE_ADDR 0xE000A000
#define XGPIOPS_DATA_MASK_OFFSET 0x00000008U /* Data/Mask Registers offset */
#define XGPIOPS_REG_MASK_OFFSET 0x00000040U /* Registers offset */
#define XGPIOPS_DIRM_OFFSET 0x00000204U /* Direction Mode Register, RW */
#define XGPIOPS_OUTEN_OFFSET 0x00000208U /* Output Enable Register, RW */
#define XGPIOPS_DATA_LSW_OFFSET 0x00000000U /* Mask and Data Register LSW, WO */
#define XGPIOPS_DATA_MSW_OFFSET 0x00000004U /* Mask and Data Register MSW, WO */ int main()
{ //引脚EMIO54等价于bank=2,PinNum=0
//引脚EMIO57等价于bank=2,PinNum=3
u8 Bank=;
u8 PinNumber=; /**************对应管脚功能配置**********************/
u32 Direction=0X01;//设置MIO_54为输出
u32 OpEnable=0X01;//使能
u32 DataVar = 0X00;//1亮 //方向寄存器地址(由bank决定)
u32 *Gpio_DIR =( u32 *)(MY_GPIO_BASE_ADDR+
((u32)(Bank) * XGPIOPS_REG_MASK_OFFSET) +
XGPIOPS_DIRM_OFFSET); //方向寄存器,1为输出
//使能寄存器地址(由bank决定)
u32 *Gpio_EN =( u32 *)(MY_GPIO_BASE_ADDR+
((u32)(Bank) * XGPIOPS_REG_MASK_OFFSET) +
XGPIOPS_OUTEN_OFFSET);//使能寄存器,1使能
//数据寄存器地址(由bank和PinNumuber决定)
u32 RegOffset; //16位宽,后面要注意
if (PinNumber > 15U) {
/* There are only 16 data bits in bit maskable register. */
PinNumber -= (u8);
RegOffset = XGPIOPS_DATA_MSW_OFFSET;
} else {
RegOffset = XGPIOPS_DATA_LSW_OFFSET;
}
u32 *Gpio_DATA =( u32 *)(MY_GPIO_BASE_ADDR+
((u32)(Bank) * XGPIOPS_DATA_MASK_OFFSET) +
RegOffset);//数据寄存器,1亮 //根据用户方向值,填充方向寄存器值
u32 DirModeReg=XGpioPs_ReadReg(MY_GPIO_BASE_ADDR,
((u32)(Bank) * XGPIOPS_REG_MASK_OFFSET) +
XGPIOPS_OUTEN_OFFSET);
if (Direction!=(u32)) { /* Output Direction */
DirModeReg |= ((u32) << (u32)PinNumber);
} else { /* Input Direction */
DirModeReg &= ~ ((u32) << (u32)PinNumber);
}
*Gpio_DIR = DirModeReg;
//根据用户使能值,填充使能寄存器值
u32 OpEnableReg=XGpioPs_ReadReg(MY_GPIO_BASE_ADDR,
((u32)(Bank) * XGPIOPS_REG_MASK_OFFSET) +
XGPIOPS_OUTEN_OFFSET);
if (OpEnable != (u32)) { /* Enable Output Enable */
OpEnableReg |= ((u32) << (u32)PinNumber);
} else { /* Disable Output Enable */
OpEnableReg &= ~ ((u32) << (u32)PinNumber);
}
*Gpio_EN =OpEnableReg; //根据用户DATA值,填充数据寄存器值
/*
* Get the 32 bit value to be written to the Mask/Data register where
* the upper 16 bits is the mask and lower 16 bits is the data.
*/ DataVar &= (u32)0x01;
u32 Value = ~((u32) << (PinNumber + 16U)) & ((DataVar << PinNumber) | 0xFFFF0000U);
*Gpio_DATA =Value;//1亮 (fffe0001亮,fffe0000灭) //测试物理地址
printf("Gpio_DIR:%lx,value:%lx\n",Gpio_DIR,DirModeReg);
printf("Gpio_EN:%lx,value:%lx\n",Gpio_EN,OpEnableReg);
printf("Gpio_DATA:%lx,value:%lx\n",Gpio_DATA,Value); return ; }

驱动学习3.1:获取zynqled的物理地址的更多相关文章

  1. BCM wifi驱动学习

    BCMwifi驱动学习 一.wifi详解1 1.代码路径:Z:\home\stonechen\svn\TD550_X\TD550\3rdparty\wifi\BCM43362\special\bcmd ...

  2. 【Linux驱动学习】SD卡规范学习

    摘要: 学习SD卡的相关规范,包括定义,硬件特性,数据传输,命令系统等.不涉及代码. 文章针对Linux驱动开发而写,以助于理解SD卡驱动,不会涉及过多硬件内容. 纲要: 1. SD卡介绍 2. SD ...

  3. linux驱动之中断方式获取键值

    linux驱动之中断方式获取键值 ----------------------------------------------------------------------------------- ...

  4. linux 驱动学习笔记01--Linux 内核的编译

    由于用的学习材料是<linux设备驱动开发详解(第二版)>,所以linux驱动学习笔记大部分文字描述来自于这本书,学习笔记系列用于自己学习理解的一种查阅和复习方式. #make confi ...

  5. Linux驱动学习步骤(转载)

    1. 学会写简单的makefile 2. 编一应用程序,可以用makefile跑起来 3. 学会写驱动的makefile 4. 写一简单char驱动,makefile编译通过,可以insmod, ls ...

  6. linux驱动学习(二) Makefile高级【转】

    转自:http://blog.csdn.net/ghostyu/article/details/6866863 版权声明:本文为博主原创文章,未经博主允许不得转载. 在我前一篇写的[ linux驱动学 ...

  7. linux驱动学习_1

    目前项目需要,需要做linux驱动了,记录一下 学习驱动,大家一定都会写一个hello world代码,网上也有很多范例,但是记录一下遇到的问题. 1.make之后,使用insmod加载,终端没有打印 ...

  8. Java学习-058-Jsoup爬虫获取中国所有的三级行政区划数据(三),处理二级编码缺失

    通过查看数据可知,直辖市或者某些三级行政区域没有对应的二级区域,为方便后续的地址使用,可自定义缺失的二级地址. 如下示例自定义的二级行政区域的名称为一级区域的名称,对应的源码如下所示: 将此段源码添加 ...

  9. IIC驱动学习笔记,简单的TSC2007的IIC驱动编写,测试

    IIC驱动学习笔记,简单的TSC2007的IIC驱动编写,测试 目的不是为了编写TSC2007驱动,是为了学习IIC驱动的编写,读一下TSC2007的ADC数据进行练习,, Linux主机驱动和外设驱 ...

随机推荐

  1. 如何理解IPD+CMMI+Scrum一体化研发管理解决方案之CMMI篇

    如何快速响应市场的变化,如何推出更有竞争力的产品,如何在竞争中脱颖而出,是国内研发企业普遍面临的核心问题,为了解决这些问题,越来越多的企业开始重视创新与研发管理,加强研发过程的规范化,集成产品开发(I ...

  2. Live Archive 训练题

    7091 Height Ordering Mrs. Chambers always has her class line up in height order (shortest at the fro ...

  3. 《C》数据类型

  4. 人生苦短,我用Python!

    一.程序分析 1.读取文件到缓冲区 def process_file(): # 读文件到缓冲区 try: # 打开文件 f = open("C:\\Users\\panbo\\Desktop ...

  5. 团队Alpha冲刺(一)

    目录 组员情况 组员1(组长):胡绪佩 组员2:胡青元 组员3:庄卉 组员4:家灿 组员5:凯琳 组员6:丹丹 组员7:家伟 组员8:政演 组员9:黄鸿杰 组员10:刘一好 组员11:何宇恒 展示组内 ...

  6. lintcode-81-数据流中位数

    81-数据流中位数 数字是不断进入数组的,在每次添加一个新的数进入数组的同时返回当前新数组的中位数. 说明 中位数的定义: 中位数是排序后数组的中间值,如果有数组中有n个数,则中位数为A[(n-1)/ ...

  7. C++ Primer Plus学习:第十一章

    运算符重载 使用方法: 在类的声明中定义重载运算符 datatype operator操作符(datatype); 定义:datatype classname:: operator操作符(dataty ...

  8. exce族函数详解

    exec函数族 函数族说明 fork() 函数用于创建一个新的子进程,该子进程几乎复制了父进程的全部内容,但是,这个新创建的子进程如何执行呢?exec 函数族就提供了一个在进程中启动另一个程序执行的方 ...

  9. Js 中实现重定向的几种方式

    之所以要总结这个,是因为在项目中使用了 AJAX,当请求成功时需要重定向到另一个页面. 1 . <script type="text/javascript"> wind ...

  10. utuntu下安装eclipse+jdk

    安装jdk: 1.下载一个可以用的jdk压缩包.下载地址:http://www.oracle.com/technetwork/java/javase/downloads/jdk7-downloads- ...