第六章作业

内容:
 

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值