自己想要打印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. Python如何对折线进行平滑曲线处理?

    在用python绘图的时候,经常由于数据的原因导致画出来的图折线分界过于明显,因此需要对原数据绘制的折线进行平滑处理,本文介绍利用插值法进行平滑曲线处理: 实现所需的库 numpy.scipy.mat ...

  2. 根据Unicode码生成汉字

    最近需要一批汉字字符数据,类似数字字符与ASCII码之间的对应关系,汉字字符与Unicode码之间也存在对应关系. 所以可以遍历Unicode码批量生成汉字. 其中,汉字为宽字符,输出时候注意需要修改 ...

  3. iOS开发 常见错误

    一.NSAppTransportSecurity 错误提示:NSURLSession/NSURLConnection HTTP load failed (kCFStreamErrorDomainSSL ...

  4. timestamp 学习

    该答案摘抄自CSDN. 哇,奇迹,跨度三年了,不知道楼主是否已经解决了此问题. 路过,简单说一下,timestamp 主要是记录该行的最后修改时间戳, 注意,这个时间戳是不可以转换为时间的,只能标注该 ...

  5. texbbox,combobox设置属性

    --输入框 $("#xx").textbox('setValue','value');  //设置输入框的值 $('#xx').textbox('textbox').attr('r ...

  6. Java多线程下单例

    /* 多线程下的单例 */ //饿汉式 class Single { private static final Single s = new Single(); private Single(){} ...

  7. 【转载】mysql建表date类型不能设置默认值

    如题,mysql建表date类型的不能设置一个默认值,比如我这样: CREATE TABLE `new_table` ( `biryhday` datetime NULL DEFAULT '1996- ...

  8. LoadRunner录制用户操作

    先说明一点,使用录制的手段拿到的测试脚本和工程师自己编写的测试脚本其实是一样的,不要觉得录制的方式low,而自己编写脚本就显得高大上,这是不对的.除非工程师本身对开发们写的代码逻辑很熟,对业务上的各个 ...

  9. 集合里面的 E是泛型 暂且认为是object

    集合里面的 E是泛型 暂且认为是object

  10. poj3107 Godfather 求树的重心

    Description Last years Chicago was full of gangster fights and strange murders. The chief of the pol ...