内容:
1、利用SysTick定时器编写倒计时程序,如初始设置为2分30秒,每秒在屏幕上输出一次时间,倒计时为0后,红灯亮,停止屏幕输出,并关闭SysTick定时器的中断。
2、利用RTC显示日期(年月日、时分秒),每秒更新。并设置某个时间的闹钟。闹钟时间到时,屏幕上显示有你的姓名的文字,并点亮绿灯。
3、利用PWM脉宽调制,交替显示红灯的5个短闪和5个长闪。
4、GEC39定义为输出引脚,GEC10定义为输入引脚,用杜邦线将两个引脚相连,验证捕捉实验程序Incapture-Outcmp-20211110,观察输出的时间间隔。
代码:
1.main.c
#define GLOBLE_VAR // 定义宏 GLOBLE_VAR
#include "includes.h" // 包含总头文件
//----------------------------------------------------------------------
//声明使用到的内部函数
//main.c使用的内部函数声明处
//----------------------------------------------------------------------
// 主函数,一般情况下可以认为程序从此开始运行(实际上有启动过程见书稿)
int main(void)
{
//(1)======启动部分(开头)==========================================
//(1.1)声明 main 函数使用的局部变量
uint8_t mSec; // 记录当前秒的值
//(1.2)【不变】关闭总中断
DISABLE_INTERRUPTS;
wdog_stop();
//(1.3)给主函数使用的局部变量赋初值
//(1.4)给全局变量赋初值
// "时分秒"缓存初始化为 00:02:30
gTime[0] = 0; // 时
gTime[1] = 2; // 分
gTime[2] = 30; // 秒
mSec = 0; // 记住当前秒的值
//(1.5)用户外设模块初始化
gpio_init(LIGHT_RED, GPIO_OUTPUT, LIGHT_OFF); // 初始化红灯
systick_init(10); // 设置 systick 为 10 毫秒中断
//(1.6)使能模块中断
//(1.7)【不变】开启总中断
ENABLE_INTERRUPTS;
// for(;;) { } // 在此打桩,理解红色发光二极管为何亮起来了?
//(1)======启动部分(结尾)==========================================
//(2)======主循环部分(开头)=========================================
for (;;)
{
if (gTime[2] == mSec) continue;
mSec = gTime[2];
// 以下是 1 秒到的处理,灯的状态切换(这样灯每秒闪一次)
// 切换灯状态
printf("%d:%d:%d\n", gTime[0], gTime[1], gTime[2]);
if (gTime[0] == 0 && gTime[1] == 0 && gTime[2] == 0)
{
gpio_set(LIGHT_RED, LIGHT_ON);
}
} // for(;;) 结尾
//(2)======主循环部分(结尾)========================================
}
2isr.c
#include "includes.h"
// 声明使用到的内部函数
// isr.c 使用的内部函数声明处
void SecMinus(uint8_t *p);
// SysTick 中断处理函数
void SysTick_Handler()
{
// printf("***\n");
static uint8_t SysTickCount = 0;
SysTickCount++; // Tick 单元加 1
wdog_feed(); // 看门狗“喂狗”
if (SysTickCount >= 10) // 这里可以调整机器 1 秒在实际中的时间,100 为 1 秒,调整为 10 为 0.1 秒
{
SysTickCount = 0;
SecMinus(gTime);
}
}
// 减少秒数函数
void SecMinus(uint8_t *p)
{
if (*(p + 2) > 0)
{
*(p + 2) -= 1; // 秒数减 1
}
else // 秒数不够
{
if (*(p + 1) >= 1)
{
*(p + 1) -= 1;
*(p + 2) = 59;
}
else
{
if (*p >= 1)
{
*p -= 1;
*(p + 1) = 59;
*(p + 2) = 59;
}
else
{
// 可以添加其他处理逻辑
}
}
}
}
2、
Mian.c
#define GLOBLE_VAR
#include "includes.h" //包含总头文件
//----------------------------------------------------------------------
//声明使用到的内部函数
//main.c使用的内部函数声明处
//----------------------------------------------------------------------
//主函数,一般情况下可以认为程序从此开始运行(实际上有启动过程,参见书稿)
int main(void)
{
//(1)======启动部分(开头)==========================================
//(1.1)声明main函数使用的局部变量
uint32_t mMainLoopCount; //主循环次数变量
//(1.2)【不变】关总中断
DISABLE_INTERRUPTS;
//(1.3)给主函数使用的局部变量赋初值
mMainLoopCount=0; //主循环次数变量
//(1.4)给全局变量赋初值
g_RTC_Flag=0;
//(1.5)用户外设模块初始化
gpio_init(LIGHT_GREEN,GPIO_OUTPUT,LIGHT_OFF);
uart_init(UART_User,115200);
RTC_Init(); //RTC初始化
RTC_Set_Time(11,50,0); //设置时间为0:0:0
RTC_Set_Date(24,5,16,4); //设置日期
//(1.6)使能模块中断
RTC_PeriodWKUP_Enable_Int(); //使能唤醒中断
uart_enable_re_int(UART_User);
RTC_Alarm_Enable_Int(0); //开启0号闹钟
RTC_Set_Alarm(0,4,11,50,10); //闹钟设置(闹钟号,周数,时,分,秒)
RTC_Set_PeriodWakeUp(1);
//(1.7)【不变】开总中断
ENABLE_INTERRUPTS;
//(1)======启动部分(结尾)==========================================
//(2)======主循环部分(开头)========================================
for(;;) //for(;;)(开头)
{
//(2.1)主循环次数变量+1
mMainLoopCount++;
//(2.2)未达到主循环次数设定值,继续循环
if (mMainLoopCount<=12888999) continue;
//(2.3)达到主循环次数设定值,执行下列语句,进行灯的亮暗处理
//(2.3.1)清除循环次数变量
mMainLoopCount=0;
if(g_RTC_Flag==1) //根据串口接收的数据设置基准时间
{
g_RTC_Flag=0;
gcRTC_Date_Time.Year=(uint8_t)((gcRTCBuf[1]-'0')*10+(gcRTCBuf[2]-'0'));
gcRTC_Date_Time.Month=(uint8_t)((gcRTCBuf[4]-'0')*10+(gcRTCBuf[5]-'0'));
gcRTC_Date_Time.Date=(uint8_t)((gcRTCBuf[7]-'0')*10+(gcRTCBuf[8]-'0'));
gcRTC_Date_Time.Hours=(uint8_t)((gcRTCBuf[10]-'0')*10+(gcRTCBuf[11]-'0'));
gcRTC_Date_Time.Minutes=(uint8_t)((gcRTCBuf[13]-'0')*10+(gcRTCBuf[14]-'0'));
gcRTC_Date_Time.Seconds=(uint8_t)((gcRTCBuf[16]-'0')*10+(gcRTCBuf[17]-'0'));
gcRTC_Date_Time.Weekday=(uint8_t)((gcRTCBuf[23]-'0'));
RTC_Set_Time(gcRTC_Date_Time.Hours,gcRTC_Date_Time.Minutes,gcRTC_Date_Time.Seconds); //设置时间
RTC_Set_Date(gcRTC_Date_Time.Year,gcRTC_Date_Time.Month,gcRTC_Date_Time.Date,gcRTC_Date_Time.Weekday); //设置日期
}
} //for(;;)结尾
//(2)======主循环部分(结尾)========================================
} //main函数(结尾)
Isr,c
#include "includes.h"
void RTC_Alarm_IRQHandler(void)
{
if(RTC_Alarm_Get_Int(A)) //闹钟A的中断标志位
{
RTC_Alarm_Clear(A); //清闹钟A的中断标志位
gpio_set(LIGHT_GREEN,LIGHT_ON); //灯“亮”
printf("周锦铭\n");
}
if(RTC_Alarm_Get_Int(B)) //闹钟A的中断标志位
{
RTC_Alarm_Clear(B); //清闹钟A的中断标志位
printf("This is ALARM_B!!!\n");
}
}
#define GLOBLE_VAR
#include "includes.h" //包含总头文件
void Delay_ms(uint16_t u16ms);
//----------------------------------------------------------------------
//声明使用到的内部函数
//main.c使用的内部函数声明处
//----------------------------------------------------------------------
//主函数,一般情况下可以认为程序从此开始运行(实际上有启动过程见书稿)
int main(void)
{
//(1)======启动部分(开头)==========================================
//(1.1)声明main函数使用的局部变量
uint8_t mFlag; //灯的状态标志
uint8_t Flag; //希望采集的电平高低标志
double m_duty; //占空比
double m_duty2; //占空比
uint32_t m_i; //控制在未知周期内不同占空比的波形只打印有限次
uint8_t m_K; //确保每次能正确打印输出PWM波形
uint8_t pd;
//(1.2)【不变】关总中断
DISABLE_INTERRUPTS;
//(1.3)给主函数使用的局部变量赋初值
Flag=1;
mFlag=0; //灯的状态标志
pd=1;
//(1.4)给全局变量赋初值
//(1.5)用户外设模块初始化
pwm_init(PWM_USER,1500,1000,m_duty,PWM_CENTER,PWM_MINUS); //PWM输出初始化
gpio_init(LIGHT_RED,1,1); //初始化蓝灯
//(1.6)使能模块中断
//(1.7)【不变】开总中断
ENABLE_INTERRUPTS;
printf("------------------------------------------------------\n");
printf("金葫芦提示: \n");
printf(" (1)蓝灯以不同亮暗程度交替闪烁\n");
printf(" (2)串口输出PWM的高低电平\n");
printf(" (3)可通过PWM-测试程序-C#2019观察波形变化\n");
printf("------------------------------------------------------\n");
//for(;;) //在此打桩,理解蓝色发光二极管为何亮起来了?
//(1)======启动部分(结尾)==========================================
//(2)======主循环部分(开头)=========================================
m_K=0;
m_duty2=10.0;
m_duty=90.0;
for(;;) //for(;;)(开头)
{ if(pd==0){
pwm_update(PWM_USER,m_duty); //调节占空比
}
else{
pwm_update(PWM_USER,m_duty2);
}
for (m_i=0;m_i<10;m_i++) //m_i<3为了控制未知周期内相同占空比的波形只打印三次
{
m_K=0; //保证每次输出打印完整的PWM波,再进入下一个循环
do
{
mFlag=gpio_get(PWM_USER);
if ((mFlag==1)&&(Flag==1))
{
printf("高电平:1\n");
Flag=0;
m_K++;
gpio_reverse(LIGHT_RED);//小灯反转
}
else if ((mFlag==0)&&(Flag==0))
{
printf("低电平:0\n");
Flag=1;
m_K++;
gpio_reverse(LIGHT_RED);
}
}
while (m_K<1);
}
if(pd==1){
pd=0;
}
else{pd=1;}
}
}
4mian.c
#define GLOBLE_VAR
#include "includes.h" //包含总头文件
void Delay_ms(uint16_t u16ms);
//----------------------------------------------------------------------
//声明使用到的内部函数
//main.c使用的内部函数声明处
//----------------------------------------------------------------------
//主函数,一般情况下可以认为程序从此开始运行(实际上有启动过程见书稿)
int main(void)
{
//(1)======启动部分(开头)==========================================
//(1.1)声明main函数使用的局部变量
uint8_t mFlag; //灯的状态标志
uint8_t flag; //标记高低电平
//(1.2)【不变】关总中断
DISABLE_INTERRUPTS;
//(1.3)给主函数使用的局部变量赋初值
mFlag='A'; //灯的状态标志
//(1.4)给全局变量赋初值
gTime[0] = 0; //分钟
gTime[1] = 0; //秒
gTime[2] = 0; //毫秒
period = 1000; //自动重装载寄存器初始值
//(1.5)用户外设模块初始化
gpio_init(LIGHT_BLUE,GPIO_OUTPUT,LIGHT_ON); //初始化蓝灯
outcmp_init(OUTCMP_USER,3000,200,50.0,CMP_REV); //输出比较初始化 占空比
incapture_init(INCAP_USER,375,1000,CAP_DOUBLE); //上升沿捕捉初始化
systick_init(1); //设置systick为1ms中断
//(1.6)使能模块中断
cap_enable_int(INCAP_USER); //使能输入捕捉中断
//(1.7)【不变】开总中断
ENABLE_INTERRUPTS;
printf("------------------------------------------------------\n");
printf("金葫芦提示: \n");
printf(" (1)蓝灯每秒闪烁一次,作为运行指示\n");
printf(" (2)设置GEC39为输出比较引脚,\n");
printf(" 设置GEC10为输入捕捉引脚,沿跳变捕捉\n");
printf(" (3)用导线将GEC39与GEC10连接 \n");
printf(" (4)程序使得输出比较引脚输出高低电平,输入捕捉引脚捕捉\n");
printf(" 后用printf输出,PC机程序据此显示波形引脚捕捉\n");
printf("------------------------------------------------------\n");
//for(;;) { } //在此打桩,理解蓝色发光二极管为何亮起来了?
//(1)======启动部分(结尾)==========================================
//(2)======主循环部分(开头)=========================================
for(;;) //for(;;)(开头)
{
flag = gpio_get(INCAP_USER);
//灯状态标志mFlag为'L',改变灯状态及标志
if (mFlag=='L' && flag == 1) //判断灯的状态标志
{
mFlag='A'; //灯的状态标志
gpio_set(LIGHT_BLUE,LIGHT_ON); //灯“亮”
}
//如灯状态标志mFlag为'A',改变灯状态及标志
else if(mFlag=='A' && flag == 0) //判断灯的状态标志
{
mFlag='L'; //灯的状态标志
gpio_set(LIGHT_BLUE,LIGHT_OFF); //灯“暗”
}
} //for(;;)结尾
//(2)======主循环部分(结尾)========================================
}
//======以下为主函数调用的子函数存放处=====================================
//======================================================================
//函数名称:Delay_ms
//函数返回:无
//参数说明:无
//功能概要:延时 - 毫秒级
//======================================================================
void Delay_ms(uint16_t u16ms)
{
uint32_t u32ctr;
for(u32ctr = 0; u32ctr < 8000*u16ms; u32ctr++)
{
__ASM("NOP");
}
}
Isr.c
#include "includes.h"
//声明使用到的内部函数
//isr.c使用的内部函数声明处
void SecAdd1(volatile uint16_t *p);
//======================================================================
//中断服务程序名称:UART_USER_Handler
//触发条件:UART_USE串口收到一个字节触发
//功 能:收到一个字节,直接返回该字节
//备 注:进入本程序后,可使用uart_get_re_int函数可再进行中断标志判断
// (1-有UART接收中断,0-没有UART接收中断)
//======================================================================
void UART_USER_Handler(void)
{
uint8_t ch;
uint8_t flag;
DISABLE_INTERRUPTS; //关总中断
//------------------------------------------------------------------
//接收一个字节
ch = uart_re1(UART_User, &flag); //调用接收一个字节的函数,清接收中断位
if(flag) //有数据
{
uart_send1(UART_User,ch);//回发接收到的字节
}
//------------------------------------------------------------------
ENABLE_INTERRUPTS; //开总中断
}
//=====================================================================
//函数名称:INCAP_USER_Handler(输入捕捉中断处理程序)
//参数说明:无
//函数返回:无
//功能概要:(1)每次捕捉到上升沿或者下降沿触发该程序;
// (2)每次触发都会上传当前捕捉到的上位机程序
//=====================================================================
void INCAP_USER_Handler(void)
{
//声明INCAP_USER_Handler中需要的变量
static uint8_t flag = 0;
DISABLE_INTERRUPTS; //关总中断
//------------------------------------------------------------------
//(在此处增加功能)
if(cap_get_flag(INCAP_USER))
{
//在捕获到上升沿之后,输出此刻捕获的是上升沿和时间
if(gpio_get(INCAP_USER)==1 && flag == 0){
printf("%d分钟:%d秒:%d毫秒此刻是上升沿\r\n",
gTime[0],gTime[1],gTime[2]);
flag = 1;
//修改自动重装载寄存器的值
period -= 100;//每次输出时钟间隔逐渐变短,如果时钟间隔大于一秒则重置到一秒
if (period < 100 || period > 1000)
period = 1000;
//通过修改自动重装载寄存器的值控制电平翻转时刻
outcmp_init(OUTCMP_USER,3000,period,50.0,CMP_REV); //修改输出比较电平周期,会触发中断,进入INCAP_USER_Handler(void)函数
}
//在捕获到下降沿之后,输出此刻捕获的是下降沿和时间
else if(gpio_get(INCAP_USER)==0 && flag == 1){
printf("%d分钟:%d秒:%d毫秒此刻是下降沿\r\n",
gTime[0],gTime[1],gTime[2]);
flag = 0;
//通过修改自动重装载寄存器的值控制电平翻转时刻
outcmp_init(OUTCMP_USER,3000,period,50.0,CMP_REV); //修改发输出比较电平周期,会触中断,进入INCAP_USER_Handler(void)函数
}
cap_clear_flag(INCAP_USER); //清中断
}
//------------------------------------------------------------------
ENABLE_INTERRUPTS; //关总中断
}
//=====================================================================
//函数名称:SYSTICK_USER_Handler(SysTick定时器中断处理程序)
//参数说明:无
//函数返回:无
//功能概要:(1)每10ms中断触发本程序一次;(2)达到一秒时,调用秒+1
// 程序,计算“时、分、秒”
//特别提示:(1)使用全局变量字节型数组gTime[3],分别存储“时、分、秒”
// (2)注意其中静态变量的使用
//=====================================================================
void SysTick_Handler()
{
//printf("***\n");
static uint8_t SysTickCount = 0;
SysTickCount++; //Tick单元+1
//wdog_feed(); //看门狗“喂狗”
if (SysTickCount >= 1)
{
SysTickCount = 0;
SecAdd1(gTime);
}
}
//===========================================================================
//函数名称:SecAdd1
//函数返回:无
//参数说明:*p:为指向一个时分秒数组p[3]
//功能概要:秒单元+1,并处理时分单元(00:00:00-23:59:59)
//===========================================================================
void SecAdd1(volatile uint16_t *p)
{
*(p+2)+=1; //秒+1
if(*(p+2)>=1000) //秒溢出
{
*(p+2)=0; //清秒
*(p+1)+=1; //分+1
if(*(p+1)>=60) //分溢出
{
*(p+1)=0; //清分
*p+=1; //时+1
if(*p>=60) //时溢出
{
*p=0; //清时
}
}
}
}
结果:
1、
2
3
4