欧美极品高清xxxxhd,国产日产欧美最新,无码AV国产东京热AV无码,国产精品人与动性XXX,国产传媒亚洲综合一区二区,四库影院永久国产精品,毛片免费免费高清视频,福利所导航夜趣136

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 4227|回復: 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 共享資料的黑幣獎勵!

查看全部評分

回復

使用道具 舉報

ID:142059 發表于 2019-3-12 00:33 | 顯示全部樓層
你這個不算簡潔,你看看我的,更貼近51
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表