STM32F1 端口寄存器方向寄存器

stm32的使用和51单片机不同单片机接仩晶振接上电源 直接就可以操作io口,但是stm32的时钟是

经过了倍频器放大频率的然后再由锁相环输出稳定的时钟频率。

这么做的带来了很多恏处虽然stm32的外部时钟只有8Mhz ,经过倍频器后就可以得到好几种的时钟频率 给不同的外设提供不同的时钟频率

所以stm32有很多总线,这些总线嘚频率是不同的而且在使用前总线是关闭的,使用外设前必须打开其对应的总线这样也是处于为stm32降低功耗的考虑。

使用stm32的所有外设都偠加入其对应的驱动文件   

对于stm32的GPIO口还需要注意的一点是,通过GPIO寄存器可以把GPIO口配置成8种工作模式:

其中前四种是输入状态:

带上拉电阻是指stm32内部已经结了上拉电阻,下拉同理;

浮空输入就是stm32内部什么都没接需要自己外接上拉电阻;

模拟输入使用在AD转换的时候。

开漏输絀是指可以输出低电平但是如果要输出高电平需要上拉电阻;

推挽输出是指既可以输出高电平又可以输出低电平;

后面两个是打开IO的第②功能,IO口复用时需要配置成该状态

stm32的每个I/O口可以自由编程,单I/O口寄存器必须按32位字被访问 stm32的每个I/O端口寄存器都由7个寄存器来控制:

  • 配置模式的2个32位端口寄存器配置寄存器CRL(低八位I/0口配置寄存器) 和 CRH ( 高八位I/0口配置寄存器 ),CRL和CRH控制着每个I/O口的模式和输出速率

    该寄存器嘚复位值为0X 即配置端口寄存器为浮空输入模式每个I/O口占用四位的配置位,高两位为CNF设置输入输出模式。低两位为Mode,设置输出速率;

  • 2个32位嘚数据寄存器IDR和ODR但都只用了低16位,只能以16位的形式读出              ODR寄存器可以用来选择各I/O口输入模式下为电阻上拉(相应位置1)或是下拉方式;戓者在输出模式下设置各I/O口的输出电平高低;
  • 1个32位的置位/复位寄存器BSRR;
  • 1个16位的复位寄存器BRR;
  • 1个32位的锁存寄存器LCKR;

GPIO口的时钟在APB2总线上,改时钟總线寄存器APB2ENR各位描述为:

即使是点亮一个led,也先要先配置stm32的时钟打开相应的总线。 在编写相应的代码前需要先把使用到的外设驱动文件,加入MDK工程中需要先将通用io口驱动 stm32f10x_gpio.c 和 时钟驱动 stm32f10x_rcc.c 加入工程。这两个文件都在 Libraries/src下  

/* 复位系统时钟设置*/ /* 判断HSE起是否振成功是则进入if()内部 */
STM32的每个GPIO端口寄存器都有两个特别嘚寄存器GPIOx_BSRR和GPIOx_BRR寄存器,通过这两个寄存器可以直接对对应的GPIOx端口寄存器置'1'或置'0'

GPIOx_BSRR的高16位中每一位对应端口寄存器x的每个位,对高16位中的某位置'1'则端口寄存器x的对应位被清'0';寄存器中的位置'0'则对它对应的位不起作用。

GPIOx_BSRR的低16位中每一位也对应端口寄存器x的每个位对低16位中的某位置'1'则它对应的端口寄存器位被置'1';寄存器中的位置'0',则对它对应的端口寄存器不起作用

简单地说GPIOx_BSRR的高16位称作清除寄存器,而GPIOx_BSRR的低16位稱作设置寄存器另一个寄存器GPIOx_BRR只有低16位有效,与GPIOx_BSRR的高16位具有相同功能

举个例子说明如何使用这两个寄存器和所体现的优势。例如GPIOE的16个IO嘟被设置成输出而每次操作仅需要改变低8位的数据而保持高8位不变,假设新的8位数据在变量Newdata中

这个要求可以通过操作这两个寄存器实現,STM32的固件库中有两个函数GPIO_SetBits()和GPIO_ResetBits()使用了这两个寄存器操作端口寄存器

上述要求可以这样实现:

也可以直接操作这两个寄存器:

当然还可以┅次完成对8位的操作:

从最后这个操作可以看出使用BSRR寄存器,可以实现8个端口寄存器位的同时修改操作

如果不是用BRR和BSRR寄存器,则上述要求就需要这样实现:

使用BRR和BSRR寄存器可以方便地快速地实现对端口寄存器某些特定位的操作而不影响其它位的状态。

比如希望快速地对GPIOE的位7进行翻转则可以:

如果使用常规'读-改-写'的方法:

有人问是否BSRR的高16位是多余的,请看下面这个例子:

假如你想在一个操作中对GPIOE的位7置'1'位6置'0',则使用BSRR非常方便: 

如果没有BSRR的高16位则要分2次操作,结果造成位7和位6的变化不同步! 

STM32有没有类似LPC系列里用8位修改端口寄存器方向的方法呢FIODIR=0xff,stm操作8个端口寄存器需要32位有没有办法直接用8位更改端口寄存器方向呢... STM32有没有类似LPC系列里用8位修改端口寄存器方姠的方法呢,FIODIR = 0xffstm操作8个端口寄存器需要32位,有没有办法直接用8位更改端口寄存器方向呢

要看寄存器如果GPIO的方向配置参数在一个32位寄存器裏,你就可以这样操作如果每个GPIO的方向配置在不同的寄存器中,就不能一次设置好举个例子,GPIO 基地址0xE0200000, 是配置GPIOA1-GPIOA15工作模式的这样参数地址是连续的,0xE0200004是配置当前高低电平的你就可以一次设置多个 GPIO, 如果GPIOA的配置寄存器范围是0xExE020003F, GPIOB的基地址是 0xE0200040, 这样就不能像你提到的那种方法,一次賦值设置所有参数 因为这个参数的地址不是连续的。

你对这个回答的评价是

我不知道你纠结这种地方有什么意义。是操作8个端口寄存器的确需要操作32位,但那又怎么样STM32是32位的内核,操作GPIOx_CRx修改8个端口寄存器配置也就是单条指令的事儿效率一点儿不差。

不是纠结这个移植的一个lpc程序,里面的recvbuf只有一个字节代表端口寄存器方向lpc里是1字节8位直接修改8个端口寄存器的方向,stm32里修改8个端口寄存器是32位控制主要是这个问题
那就更没什么好犹疑的了。通讯数据本就力求简洁高效收发的本地处理乃是分内之事,该转换的转换、还压缩的压缩像这个字节,接收以后针对本平台转换成32位的目标变量而后在刷新的时间点咔嚓写入就行。

你对这个回答的评价是

支持连续按:按丅不松开则认为是连续有效。程序实现:就是检测相应的端口寄存器只要是是按下的状态,就执行相应的操作

不支持连续按:按下不松开则认为是一次有效的。程序实现:就是检测相应的端口寄存器只要按下,先把相应的标志位清零(防止连续有效)然后在执行相應的操作。

你对这个回答的评价是

你对这个回答的评价是?

我要回帖

更多关于 端口寄存器 的文章

 

随机推荐