关于STM32 GPIO->BSRR GPIO->BRR的问题

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了关于STM32 GPIO->BSRR GPIO->BRR的问题相关的知识,希望对你有一定的参考价值。

#define LCD_EN(bit) if(bit) GPIOD->BSRR = LCD_CS; else GPIOD->BRR = LCD_CS;
请问这个程序是什么意思

if 后面的是置位,else 后面的是清零。

BSRR的高16位写1是对应口的清零,低16位写1是置位。
BRR只有低16位,写1是清零。
参考技术A 这个程序有问题吧。你的if后面的语句和else后面的语句是一样的,这个样的话不管怎么样,都会将LCD_CS上的数据给PD口。具体会怎么样看你的LCD_CS里面是什么了。 参考技术B 意思是如果bit=1就置位,如果bit=0就复位 参考技术C 使能某个位 ,

stm32寄存器版学习笔记01 GPIO口的配置(LED按键)

  STM32的I/O口可以由软件配置成如下8种模式:输入浮空、输入上拉、输入下拉、模拟输入、开漏输出、推挽输出、推挽式复用功能及开漏复用功能。每个I/O口由7个寄存器来控制:配置模式的端口配置寄存器CRL和CRH(模式、速度);数据寄存器IDR和ODR;置位/复位寄存器BSRR;复位寄存器BRR;锁存寄存器LCKR。

 

I/O口模式:

GPIO的8种模式
通用输出 推挽输出(Push-Pull) 可以输出高、低电平,连接数字器件  
开漏输出(Open-Drain) 开漏引脚不连接外部的上拉电阻时,只能输出低电平;如果需要同时具备输出高电平的功能,则需要接上拉电阻  
复用功能输出 复用功能推挽输出 片内外设功能(I2C的SCL,SDA) GPIO口被用作第二功能时的配置情况(即并非作为通用IO口使用)
复用功能开漏输出 片内外设功能(TX1,MOSI,MISO,SCK,SS)
输入 模拟输入 应用ADC模拟输入,或者低功耗下省电  
浮空输入 可以做KEY识别,外部按键输入 IO的电平状态是不确定,完全由外部输入决定
下拉输入 IO内部下拉电阻输入 不确定信号->低电平
上拉输入 IO内部上拉电阻输入 不确定信号->高电平

 

  

 



 

 

 

 

 

 

 

1.GPIO口配置步骤

①使能PORTx(x=A~G)

  APB2外设时钟使能寄存器(RCC_APB2ENR)

置1开启。清0关闭。

8-2位使能GPIO G-A

  Eg:RCC->APB2ENR| = 1 << 2;  //使能PORTA时钟

②配置IO口模式 低8位(CRL) 高8位(CRH)

  端口配置低寄存器(GPIOx_CRL) (x=A..E)

  Eg:GPIOA->CRL |= 0x00000003;  //PA0推挽输出

③配置端口输入和输出电平

  端口输入数据寄存器(GPIOx_IDR) (x=A..E)

 

  端口输出数据寄存器(GPIOx_ODR) (x=A..E)

Eg: GPIOA->ODR |= 1 <<8;  //PA8输出高

 

2.GPIO配置相关寄存器

  端口位设置/清除寄存器(GPIOx_BSRR) (x=A..E)

ODR寄存器只进行置1操作,不支持写0操作。用BSRR寄存器进行清除。

 

  端口位清除寄存器(GPIOx_BRR) (x=A..E)

 

  端口位清除寄存器(GPIOx_BRR) (x=A..E)  具体参看数据手册

  当执行正确的写序列设置了位16(LCKK)时,该寄存器用来锁定端口位的配置。位[15:0]用于锁定GPIO端口的配置。在规定的写入操作期间,不能改变LCKP[15:0]。当对相应的端口位执行了LOCK序列后,在下次系统复位之前将不能再更改端口位的配置。 每个锁定位锁定控制寄存器(CRL, CRH)中相应的4个位。

 

3.LED

//led.c                    
//LED IO口初始化
void LED_Init(void)
{
    RCC->APB2ENR|=1<<2;      //使能PORTA时钟        
    RCC->APB2ENR|=1<<5;      //使能PORTD时钟         
    GPIOA->CRH&=0XFFFFFFF0; 
    GPIOA->CRH|=0X00000003;//PA8推挽输出     
  GPIOA->ODR|=1<<8;                //PA8输出高
                                              
    GPIOD->CRL&=0XFFFFF0FF;
    GPIOD->CRL|=0X00000300;//PD2推挽输出
    GPIOD->ODR|=1<<2;         //PD2输出高
}

//led.h
#ifndef __LED_H
#define __LED_H     
#include "sys.h"

//LED端口定义
#define LED0 PAout(8)    // PA8
#define LED1 PDout(2)    // PD2    

void LED_Init(void);    //初始化                         
#endif 

//main.c
#include "sys.h"
#include "usart.h"        
#include "delay.h"    
#include "led.h" 
int main(void)
{         
    Stm32_Clock_Init(9); //系统时钟设置
    delay_init(72);             //延时初始化
    LED_Init();        //初始化与LED连接的硬件接口
    while(1)
    {
        LED0=0;
        LED1=1;
        delay_ms(300);
        LED0=1;
        LED1=0;
        delay_ms(300);
    }     
}

 

4.按键KEY输入

//key.c
#include "key.h"
#include "delay.h"
//按键初始化函数
//PA0 PA15 PC5设置成输入
void KEY_Init(void)
{
    RCC->APB2ENR|=1<<2;     //使能PORTA时钟
    RCC->APB2ENR|=1<<4;     //使能PORTC时钟
    JTAG_Set(SWD_ENABLE);    //关闭JTAG 开启SWD PA15占用了JTAG一个口
    GPIOA->CRL&=0XFFFFFFF0;    //PA0设置成输入  
    GPIOA->CRL|=0X00000008;   
    GPIOA->CRH&=0X0FFFFFFF;    //PA15设置成输入
    GPIOA->CRH|=0X80000000;              
    GPIOA->ODR|=1<<15;           //PA15上拉 PA0默认下拉
    GPIOC->CRL&=0XFF0FFFFF;    //PC5设置成输入  
    GPIOC->CRL|=0X00800000;   
    GPIOC->ODR|=1<<5;           //PC5上拉
} 
//按键处理函数
//mode:0不支持连续按;1,支持连续按
//响应优先级KEY0>KEY1>WK_UP
u8 KEY_Scan(u8 mode)
{     
    static u8 key_up=1;//按键按松开标志
    if(mode)key_up=1;  //支持连按      
    if(key_up&&(KEY0==0||KEY1==0||WK_UP==1))
    {
        delay_ms(10);//去抖动
        key_up=0;
        if(KEY0==0)
                    return KEY0_PRES;
        else if(KEY1==0)
                    return KEY1_PRES;
        else if(WK_UP==1)
                    return WKUP_PRES; 
    }else if(KEY0==1&&KEY1==1&&WK_UP==0)
            key_up=1;          
    return 0;// 无按键按下
}

//key.h
#ifndef __KEY_H
#define __KEY_H     
#include "sys.h"

#define KEY0_PRES    1        //KEY0按下
#define KEY1_PRES    2        //KEY1按下
#define WKUP_PRES    3        //WK_UP按下

#define KEY0  PCin(5)          //PC5
#define KEY1  PAin(15)         //PA15 
#define WK_UP PAin(0)          //PA0  WK_UP
     
void KEY_Init(void);        
u8 KEY_Scan(u8 mode);        
#endif

//main.c
#include "sys.h"
#include "usart.h"        
#include "delay.h"    
#include "led.h" 
#include "key.h" 

int main(void)
{         
    u8 t;      
    Stm32_Clock_Init(9);     
    delay_init(72);            
    LED_Init();                   
    KEY_Init();          
    LED0 = 0;                    
    while(1)
    {
        t=KEY_Scan(0);        
        switch(t)
        {                 
            case KEY0_PRES:
                LED0=!LED0;
                break;
            case KEY1_PRES:
                LED1=!LED1;
                break;
            case WKUP_PRES:                
                LED0=!LED0;
                LED1=!LED1;
                break;
            default:
                delay_ms(10);    
        } 
    }         
}

                    

 

以上是关于关于STM32 GPIO->BSRR GPIO->BRR的问题的主要内容,如果未能解决你的问题,请参考以下文章

stm32中GPIOx_BSRR和 GPIO_xBRR寄存允许对任何GPIOx进行读\更改的独立访问;

STM32中GPIOD->BSRR = addr(0x0000ffff<<16); <<16是啥意思

GPIO 配置之ODR, BSRR, BRR 详解

关于STM32的简单问题,GPIOA->BSRR=0x080004;这样写对吗?

Stm32基础

STM32F103X datasheet学习笔记---GPIOs and AFIOs