最近由于在学习STM32看到别人用寄存器编程控制跑马灯,于是自己也想试一试。可是试了好久终究弄不出来。回头看了下库函数的调用关系才搞明白。首先通过查看GPIOA的设置函数发现设置如下:

void GPIO_Init(GPIO_TypeDef* GPIOx, GPIO_InitTypeDef* GPIO_InitStruct)
{
uint32_t pinpos = 0x00, pos = 0x00 , currentpin = 0x00; /* Check the parameters */
assert_param(IS_GPIO_ALL_PERIPH(GPIOx));
assert_param(IS_GPIO_PIN(GPIO_InitStruct->GPIO_Pin));
assert_param(IS_GPIO_MODE(GPIO_InitStruct->GPIO_Mode));
assert_param(IS_GPIO_PUPD(GPIO_InitStruct->GPIO_PuPd)); /* ------------------------- Configure the port pins ---------------- */
/*-- GPIO Mode Configuration --*/
for (pinpos = 0x00; pinpos < 0x10; pinpos++)
{
pos = ((uint32_t)0x01) << pinpos;
/* Get the port pins position */
currentpin = (GPIO_InitStruct->GPIO_Pin) & pos; if (currentpin == pos)
{
GPIOx->MODER &= ~(GPIO_MODER_MODER0 << (pinpos * ));
GPIOx->MODER |= (((uint32_t)GPIO_InitStruct->GPIO_Mode) << (pinpos * )); if ((GPIO_InitStruct->GPIO_Mode == GPIO_Mode_OUT) || (GPIO_InitStruct->GPIO_Mode == GPIO_Mode_AF))
{
/* Check Speed mode parameters */
assert_param(IS_GPIO_SPEED(GPIO_InitStruct->GPIO_Speed)); /* Speed mode configuration */
GPIOx->OSPEEDR &= ~(GPIO_OSPEEDER_OSPEEDR0 << (pinpos * ));
GPIOx->OSPEEDR |= ((uint32_t)(GPIO_InitStruct->GPIO_Speed) << (pinpos * )); /* Check Output mode parameters */
assert_param(IS_GPIO_OTYPE(GPIO_InitStruct->GPIO_OType)); /* Output mode configuration*/
GPIOx->OTYPER &= ~((GPIO_OTYPER_OT_0) << ((uint16_t)pinpos)) ;
GPIOx->OTYPER |= (uint16_t)(((uint16_t)GPIO_InitStruct->GPIO_OType) << ((uint16_t)pinpos));
} /* Pull-up Pull down resistor configuration*/
GPIOx->PUPDR &= ~(GPIO_PUPDR_PUPDR0 << ((uint16_t)pinpos * ));
GPIOx->PUPDR |= (((uint32_t)GPIO_InitStruct->GPIO_PuPd) << (pinpos * ));
}
}
}
GPIO_TypeDef端口设置结构体如下:
typedef struct
{
__IO uint32_t MODER; /*!< GPIO port mode register, Address offset: 0x00 */
__IO uint32_t OTYPER; /*!< GPIO port output type register, Address offset: 0x04 */
__IO uint32_t OSPEEDR; /*!< GPIO port output speed register, Address offset: 0x08 */
__IO uint32_t PUPDR; /*!< GPIO port pull-up/pull-down register, Address offset: 0x0C */
__IO uint32_t IDR; /*!< GPIO port input data register, Address offset: 0x10 */
__IO uint32_t ODR; /*!< GPIO port output data register, Address offset: 0x14 */
__IO uint16_t BSRRL; /*!< GPIO port bit set/reset low register, Address offset: 0x18 */
__IO uint16_t BSRRH; /*!< GPIO port bit set/reset high register, Address offset: 0x1A */
__IO uint32_t LCKR; /*!< GPIO port configuration lock register, Address offset: 0x1C */
__IO uint32_t AFR[]; /*!< GPIO alternate function registers, Address offset: 0x20-0x24 */
} GPIO_TypeDef;

改机构体根据结构体成员函数的大小自己形成了针对首地址的偏移量。从而与需要操作的寄存器地址偏移一一对应。通过查看设置函数可以看出,初始设置需要进行几步必要的设置,经过实验发现只需要设置

GPIOx->MODER和时钟就可以。同时需要对GPIO 端口置位/复位寄存器进行操作。由该寄存器可以知,低16位置位1,高16位复位0。由此得出寄存器操作代码如下:
//#define GPIPA  *(volatile unsigned long *)0x40020000;
#define GPIOA_MODER *(volatile unsigned long *)0x40020000
#define GPIOA_OSPEEDR *(volatile unsigned long *)0x40020008
#define GPIOA_OTYPER *(volatile unsigned long *)0x40020004
#define GPIOA_PUPDR *(volatile unsigned long *)0x4002000C
#define RCC_AHB1ENR *(volatile unsigned long *)0x40023830
#define GPIOA_BSRR *(volatile unsigned long *)0x40020018
//typedef struct //{
// __IO uint32_t MODER; /*!< GPIO port mode register, Address offset: 0x00 */
// __IO uint32_t OTYPER; /*!< GPIO port output type register, Address offset: 0x04 */
// __IO uint32_t OSPEEDR; /*!< GPIO port output speed register, Address offset: 0x08 */
// __IO uint32_t PUPDR; /*!< GPIO port pull-up/pull-down register, Address offset: 0x0C */
// __IO uint32_t IDR; /*!< GPIO port input data register, Address offset: 0x10 */
// __IO uint32_t ODR; /*!< GPIO port output data register, Address offset: 0x14 */
// __IO uint16_t BSRRL; /*!< GPIO port bit set/reset low register, Address offset: 0x18 */
// __IO uint16_t BSRRH; /*!< GPIO port bit set/reset high register, Address offset: 0x1A */
// __IO uint32_t LCKR; /*!< GPIO port configuration lock register, Address offset: 0x1C */
// __IO uint32_t AFR[2]; /*!< GPIO alternate function registers, Address offset: 0x20-0x24 */
//} GPIO_TypeDef;
void delay_ms(int time)
{
int i=*time;
while(i--);
} int main(void)
{
// delay_init(168);
int i = , j = , k = ; RCC_AHB1ENR |= 0x00000001;
GPIOA_MODER &= ~((<<(*))|(<<(*)));
GPIOA_MODER |= (<< (*))|(<<(*));
GPIOA_BSRR &= 0x0; //clear
/* Speed mode configuration */
// GPIOA_OSPEEDR &= ~((3<<(6*2))|(3<<(7*2)));
// GPIOA_OSPEEDR |= (3 << (6 * 2))|(3 << (7 * 2)); /* Output mode configuration*/
// GPIOA_OTYPER &= ~((3<<6)|(3<<7)) ;
// GPIOA_OTYPER |= (0 << 6)|(0 << 7); /* Pull-up Pull down resistor configuration*/
// GPIOA_PUPDR &= ~((3<<(6*2))|(3<<(7*2)));
// GPIOA_PUPDR |= (1<< (6*2))|(1<<(7*2));
// GPIOA_BSRRL = 0x0040|0x0080; while()
{ GPIOA_BSRR = (0x0040|0x0080)<<; //reset
delay_ms();
GPIOA_BSRR = 0x0040|0x0080; //set
delay_ms(); }
} void SystemInit()
{
;
}

以上代码便可以实现流水灯例子。

												

最新文章

  1. .Container与.container_fluid区别
  2. 安装TFS(2015)工作组模式代理服务器(Agent)
  3. 本周psp个人作业
  4. SVN
  5. Linux cat命令
  6. 21: Arithmetic Sequence--HZAU(dp)
  7. 微信公共平台开发2 .net
  8. 方法的可变长参数 传入参数个数不确定可用(Type ... values)
  9. [Hive - LanguageManual] Statistics in Hive
  10. WPF的重要新概念
  11. codeforces.com/contest/325/problem/B
  12. Asp.net MVC4.0(net4.5) 部署到window server 2003上的解决方案
  13. docker私服
  14. C++ 头文件系列(streambuf)
  15. Linux下定时备份文件
  16. 并发编程(七)——AbstractQueuedSynchronizer 之 CountDownLatch、CyclicBarrier、Semaphore 源码分析
  17. gson常用的方式
  18. gdb调试技巧 找到php执行进程当前执行的代码
  19. echarts给数据视图添加表格样式
  20. Mathematics | Mean, Variance and Standard Deviation

热门文章

  1. 笔记:Xen虚拟机如何迁移到KVM上?
  2. fiddler抓取APP请求
  3. 五大常用算法之四:回溯法[zz]
  4. 让UpdatePanel支持文件上传(2):服务器端组件 .
  5. os 模块 模块与包的初始
  6. 解决网卡无法自动获取ip的办法
  7. JavaScript的DOM_动态加载脚本和样式
  8. 2、Sublime Text 3 快捷键(实用,快速换行等)
  9. [转]CentOS 7忘记root密码解决办法
  10. 随手练——HDU 1284 动态规划入门