找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 3337|回复: 1
打印 上一主题 下一主题
收起左侧

基于STM32的位带操作源程序(基础篇)

[复制链接]
跳转到指定楼层
楼主
ID:482205 发表于 2019-3-10 11:27 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
大家都知道51单片机上有一中叫位操作的,使得程序操作更简单,更通俗易懂,有人会问了?那STM32芯片可以跟51芯片一样用位带操作吗,答案是可以的,接下来给大家带来的使我自己手写的位带操作,通俗易懂版,还不懂可以询问我,欢迎讨教

单片机源程序如下:
  1. #include "stm32f10x.h"
  2. #include "bsp_led.h"  
  3. #include "bsp_key.h"

  4. //// 这里只定义了 GPIO ODR和IDR这两个寄存器的位带别名区地址,其他寄存器的没有定义

  5. ////SRAM 位带区:    0X2000 0000~0X2010 0000
  6. ////SRAM 位带别名区:0X2200 0000~0X23FF FFFF

  7. ////外设 位带区:    0X4000 0000~0X4010 0000
  8. ////外设 位带别名区:0X4200 0000~0X43FF FFFF

  9. //// 把“位带地址+位序号”转换成别名地址的宏
  10. //#define BITBAND(addr, bitnum) ((addr & 0xF0000000)+0x02000000+((addr & 0x00FFFFFF)<<5)+(bitnum<<2))
  11. ///*
  12. // *addr & 0xF0000000,取地址的高4位,看看是2还是4,用于区分SRAM和外设地址,
  13. // *如果是2,+0x02000000则=0X2200 0000,即是SRAM,如果是4,+0x02000000则=0X4200 0000,即是外设
  14. // *
  15. // *addr & 0x000FFFFFF,屏蔽掉高两位,相当于-0X2000 0000或者-0X4000 0000,结果表示偏移位带区多少个字节
  16. // *<<5  等于*8*4,因为位带区一个地址表示一个字节,一个字节有8个bit,一个bit可以膨胀成一个字,即4个字节
  17. // *<<2 等于*4,因为一个位可以膨胀成一个字,即4个字节
  18. // *
  19. // *分解成两条公式应该就是这样:
  20. // *SRAM位带别名地址
  21. // *AliasAddr= 0x22000000+((A-0x20000000)*8+n)*4 =0x22000000+ (A-0x20000000)*8*4 +n*4
  22. // *外设位带别名地址
  23. // *AliasAddr= 0x22000000+((A-0x20000000)*8+n)*4 =0x22000000+ (A-0x20000000)*8*4 +n*4
  24. // */


  25. //// 把一个地址转换成一个指针
  26. //#define MEM_ADDR(addr)  *((volatile unsigned long  *)(addr))

  27. //// 把位带别名区地址转换成指针
  28. //#define BIT_ADDR(addr, bitnum)   MEM_ADDR(BITBAND(addr, bitnum))   

  29. //// GPIO ODR 和 IDR 寄存器地址映射
  30. //#define GPIOA_ODR_Addr    (GPIOA_BASE+12) //0x4001080C   
  31. //#define GPIOB_ODR_Addr    (GPIOB_BASE+12) //0x40010C0C   
  32. //#define GPIOC_ODR_Addr    (GPIOC_BASE+12) //0x4001100C   
  33. //#define GPIOD_ODR_Addr    (GPIOD_BASE+12) //0x4001140C   
  34. //#define GPIOE_ODR_Addr    (GPIOE_BASE+12) //0x4001180C   
  35. //#define GPIOF_ODR_Addr    (GPIOF_BASE+12) //0x40011A0C      
  36. //#define GPIOG_ODR_Addr    (GPIOG_BASE+12) //0x40011E0C      
  37. //  
  38. //#define GPIOA_IDR_Addr    (GPIOA_BASE+8)  //0x40010808   
  39. //#define GPIOB_IDR_Addr    (GPIOB_BASE+8)  //0x40010C08   
  40. //#define GPIOC_IDR_Addr    (GPIOC_BASE+8)  //0x40011008   
  41. //#define GPIOD_IDR_Addr    (GPIOD_BASE+8)  //0x40011408   
  42. //#define GPIOE_IDR_Addr    (GPIOE_BASE+8)  //0x40011808   
  43. //#define GPIOF_IDR_Addr    (GPIOF_BASE+8)  //0x40011A08   
  44. //#define GPIOG_IDR_Addr    (GPIOG_BASE+8)  //0x40011E08


  45. //// 单独操作 GPIO的某一个IO口,n(0,1,2...16),n表示具体是哪一个IO口
  46. //#define PAout(n)   BIT_ADDR(GPIOA_ODR_Addr,n)  //输出   
  47. //#define PAin(n)    BIT_ADDR(GPIOA_IDR_Addr,n)  //输入   
  48. //  
  49. //#define PBout(n)   BIT_ADDR(GPIOB_ODR_Addr,n)  //输出   
  50. //#define PBin(n)    BIT_ADDR(GPIOB_IDR_Addr,n)  //输入   
  51. //  
  52. //#define PCout(n)   BIT_ADDR(GPIOC_ODR_Addr,n)  //输出   
  53. //#define PCin(n)    BIT_ADDR(GPIOC_IDR_Addr,n)  //输入   
  54. //  
  55. //#define PDout(n)   BIT_ADDR(GPIOD_ODR_Addr,n)  //输出   
  56. //#define PDin(n)    BIT_ADDR(GPIOD_IDR_Addr,n)  //输入   
  57. //  
  58. //#define PEout(n)   BIT_ADDR(GPIOE_ODR_Addr,n)  //输出   
  59. //#define PEin(n)    BIT_ADDR(GPIOE_IDR_Addr,n)  //输入  
  60. //  
  61. //#define PFout(n)   BIT_ADDR(GPIOF_ODR_Addr,n)  //输出   
  62. //#define PFin(n)    BIT_ADDR(GPIOF_IDR_Addr,n)  //输入  
  63. //  
  64. //#define PGout(n)   BIT_ADDR(GPIOG_ODR_Addr,n)  //输出   
  65. //#define PGin(n)    BIT_ADDR(GPIOG_IDR_Addr,n)  //输入  
  66. #define LED_GPIOB_ADDR   GPIOB_BASE+0x0c
  67. #define  PBout(n)    *(unsigned int *)((LED_GPIOB_ADDR&0xF0000000)+0x02000000+(LED_GPIOB_ADDR&0x00FFFFFF<<5)+(n<<2))
  68. void Delay(uint32_t count)
  69. {
  70.         for(;count!=0;count--);
  71. }
  72. /**
  73.   * @brief  主函数
  74.   * @param  无
  75.   * @retval 无
  76.   */
  77. int main(void)
  78. {       
  79.         /* LED端口初始化 */
  80.         LED_GPIO_Config();
  81.         LED1_OFF;

  82.         /* 按键端口初始化 */
  83.         Key_GPIO_Config();
  84.         #if 1
  85.         while(1)
  86.         {
  87.         //GPIO_SetBits(LED_G_GPIO_PORT, LED_G_GPIO_PIN);//关GPIO函数
  88. // LED_G(OFF);
  89.                  // PDout(2)=  1;
  90.                 Delay(0xFFFFFF);
  91.         //GPIO_ResetBits(LED_G_GPIO_PORT,LED_G_GPIO_PIN);//开GPIO函数
  92.                 //LED_G(ON);
  93.                   PBout(2)= 0;
  94.                 Delay(0xFFFFFF);
  95.         }
  96.         #else
  97.         /* 轮询按键状态,若按键按下则反转LED */
  98.         while(1)                           
  99.         {          
  100.                 //if( Key_Scan(KEY1_GPIO_PORT,KEY1_GPIO_PIN) == KEY_ON  )
  101.                
  102.                
  103.                         /*LED1反转*/
  104.                         LED1_TOGGLE;
  105.                 }

  106.         //        if( Key_Scan(KEY2_GPIO_PORT,KEY2_GPIO_PIN) == KEY_ON  )
  107.                 {
  108.                         /*LED2反转*/
  109.                         LED2_TOGGLE;
  110.                 }               
  111.                 #endif
  112.         }

  113. /*********************************************END OF FILE**********************/
复制代码

所有资料51hei提供下载:
GPIO位带操作.7z (178.89 KB, 下载次数: 19)


评分

参与人数 1黑币 +100 收起 理由
admin + 100 共享资料的黑币奖励!

查看全部评分

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏1 分享淘帖 顶 踩
回复

使用道具 举报

沙发
ID:142059 发表于 2019-3-12 00:33 | 只看该作者
你这个不算简洁,你看看我的,更贴近51
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表