版权声明:转载记得声明~~~ :) https://blog.csdn.net/ReCclay/article/details/87687121
工程可见Github<传送门>
一、主要代码
main.c
/*******************************************************************************
* 文件名:main.c
* 描 述:
* 作 者:CLAY
* 版本号:v1.0.0
* 日 期: 2019年2月19日
* 备 注:修改后的LCD例程
* 采用定时器实现的简单1s计时,实现LED1闪烁
*******************************************************************************
*/
#include "stm32f10x.h"
#include "lcd.h"
#include "e2prom.h"
#include "stdio.h"
#include "i2c.h"
#include "adc.h"
#include "rtc.h"
#include "usart2.h"
#include "pwm.h"
#include "pwm_oc.h"
#include "pwm_ic.h"
#include "timer.h"
#include "led.h"
u32 TimingDelay = 0;
u8 RxdCnt = 0;
u8 RxdOver = 0;
u8 RxdBuf[20];
void Delay_Ms(u32 nTime);
u8 RTC_Flag = 0;
//Main Body
int main(void)
{
STM3210B_LCD_Init();
LCD_Clear(Blue);
LCD_SetBackColor(Blue);
LCD_SetTextColor(White);
SysTick_Config(SystemCoreClock/1000);
//PWM_Init(500, 60);//500Hz 60%方波//PA1
PWM_OC_Init(500, 60);//500Hz 60%方波 PA1
PWM_IC_Init();//PA7
TIM4_Init(10000, 7200);//定时1s
LED_Init();//LED配置
while(1)
{
}
}
//
void Delay_Ms(u32 nTime)
{
TimingDelay = nTime;
while(TimingDelay != 0);
}
led.c
#include "stm32f10x.h"
void LED_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC|RCC_APB2Periph_GPIOD, ENABLE);//设能PC和PD口
//锁存器IO配置
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;//强推挽输出
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOD, &GPIO_InitStructure);
//8个LED的IO配置
GPIO_InitStructure.GPIO_Pin = 0xFF00;//PC8~PC15对应的LED位置
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;//强推挽输出
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOC, &GPIO_InitStructure);
GPIOC->ODR |= 0xFF00;//初始状态熄灭小灯
GPIOD->ODR |= (1<<2);//使能573
GPIOD->ODR &=~(1<<2);//失能573 注意是取非
}
led.h
#ifndef _LED_H
#define _LED_H
void LED_Init(void);
#endif
stm32f10x_it.c
...
void TIM4_IRQHandler(void)
{
if(TIM_GetITStatus(TIM4, TIM_IT_Update) == SET)
{
TIM_ClearITPendingBit(TIM4, TIM_IT_Update);
GPIOC->ODR ^= (1<<8);//PC8不断取反达到闪烁效果
GPIOD->ODR |= (1<<2);//使能573
GPIOD->ODR &= ~(1<<2);//失能573
}
}
...
二、注意事项
ODR寄存器
1、573在PD2 ; LED1在PC8
2、GPIO_InitStructure.GPIO_Pin = 0xFF00;
//PC8~PC15对应的LED位置
3、使能和失能573的操作
GPIOD->ODR |= (1<<2);//使能573
GPIOD->ODR &= ~(1<<2);//失能573,别忘了取非!!!
4、改变LED状态的操作技巧
GPIOC->ODR ^= (1<<8);//PC8不断取反达到闪烁效果
5、先开锁存器还是先给IO数据呢?
其实两个照现象来看的话都可以,但是实则不然,573在里面起到了关键作用,我们就来分析下573。
给锁存器的LE(N—LE,对应PD2)管脚一个上升沿脉冲即可把对应的电平锁存到锁存器的输出端(Q1~Q8),从而控制LED。
换句话说,只要有一个从低到高的脉冲,就会直接把输入端的数据给输出端,所以更准确来说应该在开锁存器前把数据准备好,然后再拉高LE,锁存完毕后再拉低LE为下次锁存做准备。
你可能会有疑问,连续拉高和拉低,之间的间隔够不够锁存完数据,这个口说无凭,需要参考手册咯。
最小24ns,而STM32使用8M晶振的话,一个机器周期大概是1/8M=125ns,所以完全够用,不必担心的!