嵌入式系统作业六

1、利用SysTick定时器编写倒计时程序,如初始设置为2分30秒,每秒在屏幕上输出一次时间,倒计时为0后,红灯亮,停止屏幕输出,并关闭SysTick定时器的中断。

话不多说,直接上代码

main.c:

//====================================================================
//文件名称:main.c(应用工程主函数)
//框架提供:SD-Arm(sumcu.suda.edu.cn)
//版本更新:2017.08, 2020.05
//功能描述:见本工程的<01_Doc>文件夹下Readme.txt文件
//====================================================================

#define GLOBLE_VAR
#include "includes.h"      //包含总头文件

//----------------------------------------------------------------------
//声明使用到的内部函数
//main.c使用的内部函数声明处

//----------------------------------------------------------------------
//主函数,一般情况下可以认为程序从此开始运行(实际上有启动过程见书稿)
int main(void)
{
    //(1)======启动部分(开头)==========================================
    //(1.1)声明main函数使用的局部变量
    uint8_t  mFlag;           //主循环使用的临时变量
    uint8_t  mSec;	        //记当前秒的值
    //(1.2)【不变】关总中断
    DISABLE_INTERRUPTS;
    wdog_stop();
    
    //(1.3)给主函数使用的局部变量赋初值
    mFlag='A';              //主循环使用的临时变量:蓝灯状态标志
    
    //(1.4)给全局变量赋初值
   	//"时分秒"缓存初始化(00:00:00)
   	gTime[0] = 0;       //时
   	gTime[1] = 0;	 	//分
   	gTime[2] = 0;	 	//秒
   	mSec = 0;	//记住当前秒的值
    //(1.5)用户外设模块初始化
    gpio_init(LIGHT_BLUE,GPIO_OUTPUT,LIGHT_OFF);    //初始化蓝灯
    gpio_init(LIGHT_RED,GPIO_OUTPUT,LIGHT_OFF);    //初始化红灯
    systick_init(10);      //设置systick为10ms中断
    //(1.6)使能模块中断
    //(1.7)【不变】开总中断
    ENABLE_INTERRUPTS;

    printf("------------------------------------------------------\n"); 
    printf("金葫芦提示:                                            \n");
    printf(" (1)蓝灯闪烁\n");
   	printf(" (2)每10ms中断触发SysTick定时器中断处理程序一次。 \n");
   	printf(" (3)进入SysTick定时器中断处理程序后,静态变量10ms单元+1, \n");
   	printf(" (4)达到一秒时,调用秒+1,程序,计算“时、分、秒”。 \n");
   	printf(" (5)使用全局变量字节型数组gTime[3],分别存储“时、分、秒”。 \n");
   	printf(" (6)可通过时间测试程序C#2019测试30秒的时间间隔来校准Systick \n");
   	printf(" (7)注意其中静态变量的使用 \n");
    printf("------------------------------------------------------\n"); 

    //for(;;) {  }     //在此打桩,理解蓝色发光二极管为何亮起来了?
    
    //(1)======启动部分(结尾)==========================================
    
    //(2)======主循环部分(开头)=========================================
    for(;;)     //for(;;)(开头)
    {
   		if (gTime[2] == mSec) continue;
   		mSec=gTime[2];
        //以下是1秒到的处理,灯的状态切换(这样灯每秒闪一次)
        //切换灯状态
      
        if (mFlag=='A')   //若灯状态标志为'A'
        {
          if(gTime[2] == 3 && gTime[1]==0) 
   		{
   		  gpio_set(LIGHT_RED,LIGHT_ON);   //设置灯“亮”
           printf("%d:%d:%d\n",gTime[0],gTime[1],gTime[2]);
           DISABLE_INTERRUPTS;
           
   		}else{
   		
   		   gpio_set(LIGHT_BLUE,LIGHT_ON);   //设置灯“亮”
            printf("%d:%d:%d\n",gTime[0],gTime[1],gTime[2]);
            mFlag='L';                        //改变状态标志
   		
   		}
            
        }
        else                   //否则,若灯状态标志不为'A'
        {
            gpio_set(LIGHT_BLUE,LIGHT_OFF);  //设置灯“暗”
            printf("%d:%d:%d\n",gTime[0],gTime[1],gTime[2]);
            mFlag='A';                        //改变状态标志
        }
         
        
    }     //for(;;)结尾
    //(2)======主循环部分(结尾)========================================
}

//======以下为主函数调用的子函数存放处=====================================

//====================================================================
/*

isr.c:

void SecAdd1(uint8_t *p);

//=====================================================================
//函数名称: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 >= 100)
	{
		SysTickCount = 0;
		SecAdd1(gTime);
		
	}
}

//===========================================================================
//函数名称:SecAdd1
//函数返回:无
//参数说明:*p:为指向一个时分秒数组p[3]
//功能概要:秒单元+1,并处理时分单元(00:00:00-23:59:59)
//===========================================================================
void SecAdd1(uint8_t *p)
{
	*(p+2)+=1;         //秒+1
	if(*(p+2)>=60)     //秒溢出
	{
		*(p+2)=0;       //清秒
		*(p+1)+=1;      //分+1
		if(*(p+1)>=60)  //分溢出
		{
			*(p+1)=0;    //清分
			*p+=1;       //时+1
			if(*p>=24)   //时溢出
			{
				*p=0;      //清时
			}
		}
	}
}

2、利用RTC显示日期(年月日、时分秒),每秒更新。并设置某个时间的闹钟。闹钟时间到时,屏幕上显示有你的姓名的文字,并点亮绿灯。

同样是在main.c和isr.c文件上进行更改

main.c:

//======================================================================
//文件名称:main.c(应用工程主函数)
//框架提供:SD-Arm(sumcu.suda.edu.cn)
//版本更新:20191108-20200419
//功能描述:见本工程的..\01_Doc\Readme.txt
//移植规则:【固定】
//======================================================================
#define GLOBLE_VAR
#include "includes.h"      //包含总头文件

//----------------------------------------------------------------------
//声明使用到的内部函数
//main.c使用的内部函数声明处

//----------------------------------------------------------------------
//主函数,一般情况下可以认为程序从此开始运行(实际上有启动过程,参见书稿)
int main(void)
{
//(1)======启动部分(开头)==========================================
//(1.1)声明main函数使用的局部变量
	uint32_t mMainLoopCount;  //主循环次数变量
	uint8_t  mFlag;           //灯的状态标志
	

 //(1.2)【不变】关总中断
	DISABLE_INTERRUPTS;

//(1.3)给主函数使用的局部变量赋初值
    mMainLoopCount=0;    //主循环次数变量
	mFlag='A';           //灯的状态标志
	
    
 //(1.4)给全局变量赋初值
    g_RTC_Flag=0;
//(1.5)用户外设模块初始化
	gpio_init(LIGHT_BLUE,GPIO_OUTPUT,LIGHT_ON);	//初始化蓝灯
	gpio_init(LIGHT_GREEN,GPIO_OUTPUT,LIGHT_OFF);	//初始化灯
    uart_init(UART_User,115200);
    RTC_Init();         //RTC初始化
	RTC_Set_Time(15,45,0);         //设置时间为0:0:0
    RTC_Set_Date(24,06,06,4);  //设置日期
//(1.6)使能模块中断
    RTC_PeriodWKUP_Enable_Int();                               //使能唤醒中断
    uart_enable_re_int(UART_User);
    RTC_Alarm_Enable_Int(0);
//(1.7)【不变】开总中断
	ENABLE_INTERRUPTS;
    RTC_Set_PeriodWakeUp(1);                            //配置WAKE UP中断,每秒中断一次
    RTC_Alarm_Enable_Int(0);
	RTC_Set_Alarm(0,4,15,45,10);
    
    printf("------------------------------------------------------\n"); 
    printf("金葫芦提示:                                           \n");
    printf(" (1)蓝灯闪烁\n");
    printf(" (2)设置日历基准时间为00/00/00 00:00:00 星期0\n");
   	printf(" (3)设置每秒唤醒中断,在中断输出MCU的相对时间\n");
   	printf(" (4)可通过User串口和RTC-测试程序C#2019改变基准时间\n");
    printf("------------------------------------------------------\n");
    
    
//(1)======启动部分(结尾)==========================================

//(2)======主循环部分(开头)========================================
	for(;;)   //for(;;)(开头)
	{
	   
//(2.1)主循环次数变量+1
        mMainLoopCount++;
//(2.2)未达到主循环次数设定值,继续循环
		if (mMainLoopCount<=12888999)  continue;
//(2.3)达到主循环次数设定值,执行下列语句,进行灯的亮暗处理
//(2.3.1)清除循环次数变量
		mMainLoopCount=0; 
	
		
		


		
	}  //for(;;)结尾
//(2)======主循环部分(结尾)========================================
}   //main函数(结尾)


//======以下为主函数调用的子函数===========================================


//========================================================================

isr.c:

//=====================================================================
//文件名称:isr.c(中断处理程序源文件)
//框架提供:SD-ARM(sumcu.suda.edu.cn)
//版本更新:20170801-20191020
//功能描述:提供中断处理程序编程框架
//移植规则:【固定】
//=====================================================================
#include "includes.h"
void User_SysFun(uint8_t ch);
uint8_t CreateFrame(uint8_t Data,uint8_t * buffer);		//组帧函数声明

//======================================================================
//程序名称:UART_User_Handler
//触发条件:UART_User串口收到一个字节触发
//备    注:进入本程序后,可使用uart_get_re_int函数可再进行中断标志判断
//          (1-有UART接收中断,0-没有UART接收中断)
//======================================================================
void UART_User_Handler(void)
{
	//(1)变量声明
    uint8_t flag,ch;
    DISABLE_INTERRUPTS;      //关总中断
    //(2)未触发串口接收中断,退出
    if(!uart_get_re_int(UART_User)) goto UART_User_Handler_EXIT;
    //(3)收到一个字节,读出该字节数据
    ch = uart_re1(UART_User,&flag);        //调用接收一个字节的函数
    if(!flag) goto UART_User_Handler_EXIT; //实际未收到数据,退出
  //(4)以下代码根据是否使用模板提供的User串口通信帧结构,及是否利用User串口
    //     进行带有设备序列号的进行程序更新而选择
    //(4.1)【自行组帧使用(开始)】
    if(CreateFrame(ch,gcRTCBuf))
	{   
        g_RTC_Flag=1;
	}
      
       //    【自行组帧使用(结束)】
    //(4.2)【使用模板提供的User串口通信帧结构(开始)】
    /*
       User_SysFun(ch);          //利用User串口进行程序更新
       if (gcRecvLen == 0) goto UART_User_Handler_EXIT; 
       //至此,不仅收到完整帧,且序号比较也一致,可以根据命令字节gcRecvBuf[16]进行编程
       switch(gcRecvBuf[16])  //帧标识
       {
           case 1:  //0之外的数据,自身命令
           break;
           default:
           break;
        }
        gcRecvLen = 0;   //帧已经使用完毕,下次若收到一个字节,可以继续组帧
        //【使用模板提供的User串口通信帧结构(结束)】
     */
    //(5)【公共退出区】
UART_User_Handler_EXIT:
    ENABLE_INTERRUPTS;//开总中断
}

//内部函数
void User_SysFun(uint8_t ch)
{
    //(1)收到的一个字节参与组帧
    if(gcRecvLen == 0)  gcRecvLen =useremuart_frame(ch,(uint8_t*)gcRecvBuf);
    //(2)字节进入组帧后,判断gcRecvLen=0?若为0,表示组帧尚未完成,
    //     下次收到一个字节,再继续组帧
    if(gcRecvLen == 0) goto User_SysFun_Exit;
    //(3)至此,gcRecvLen≠0,表示组帧完成,gcRecvLen为帧的长度,校验序列号后(与
    //     根据Flash中倒数一扇区开始的16字节进行比较)
    //     gcRecvBuf[16]进行跳转
    if(strncmp((char *)(gcRecvBuf),(char *)((MCU_SECTOR_NUM-1)*MCU_SECTORSIZE+
       MCU_FLASH_ADDR_START),16) != 0)
    {
        gcRecvLen = 0;         //恢复接收状态
        goto User_SysFun_Exit;
    }
    //(4)至此,不仅收到完整帧,且序号比较也一致, 根据命令字节gcRecvBuf[16]进行跳转
    //若为User串口程序更新命令,则进行程序更新
    switch(gcRecvBuf[16])  //帧标识
    {
        case 0:
            SYSTEM_FUNCTION((uint8_t *)(gcRecvBuf+17));
            gcRecvLen = 0;         //恢复接收状态
        break;
        default:
        break;
    }
User_SysFun_Exit:
    return;
}

//======================================================================
//程序名称:RTC_WKUP_IRQHandler
//函数参数:无
//中断类型:RTC闹钟唤醒中断处理函数
//======================================================================
 void RTC_WKUP_IRQHandler(void)
 {
 	uint8_t hour,min,sec;
 	uint8_t  year,month,date,week;
 	char *p;
	if(RTC_PeriodWKUP_Get_Int())         //唤醒中断的标志
	{
		RTC_PeriodWKUP_Clear();           //清除唤醒中断标志
		RTC_Get_Date(&year,&month,&date,&week); //获取RTC记录的日期
		RTC_Get_Time(&hour,&min,&sec);    //获取RTC记录的时间
		p=NumToStr("%02d/%02d/%02d %02d:%02d:%02d 星期%d\n",year,month,date,hour,min,sec,week);
		uart_send_string(UART_User,p);
		printf("%02d/%02d/%02d %02d:%02d:%02d 星期%d\n",year,month,date,hour,min,sec,week); 
		
		
		
	 }
 }
//======================================================================
//程序名称:RTC_Alarm_IRQHandler
//中断类型:RTC闹钟中断处理函数
//======================================================================
void RTC_Alarm_IRQHandler(void)
{

	if(RTC_Alarm_Get_Int(A))            //闹钟A的中断标志位
	{
		RTC_Alarm_Clear(A);       //清闹钟A的中断标志位
		printf("时间到!\n");
		printf("杨俊锋");
		gpio_set(LIGHT_BLUE,LIGHT_OFF); //灯“暗”
		gpio_set(LIGHT_GREEN,LIGHT_ON);  //灯“亮”	
		DISABLE_INTERRUPTS;
	}
	if(RTC_Alarm_Get_Int(B))            //闹钟A的中断标志位
	{
		RTC_Alarm_Clear(B);       //清闹钟A的中断标志位
		printf("还没到!?\n");
	}
	
 }

//内部调用函数
//===========================================================================
//函数名称:CreateFrame
//功能概要:组建数据帧,将待组帧数据加入到数据帧中
//参数说明:Data:待组帧数据
//       buffer:数据帧变量
//函数返回:组帧状态    0-组帧未成功,1-组帧成功
//===========================================================================
uint8_t CreateFrame(uint8_t Data,uint8_t * buffer)
{
    static uint8_t frameLen=0;    //帧的计数器
    uint8_t frameFlag;            //组帧状态

    frameFlag=0;            //组帧状态初始化
    //根据静态变量frameLen组帧
    switch(frameLen)
    {
        case 0:    //第一个数据
        {
            if (Data=='?')    //收到数据是帧头FrameHead
            {
                buffer[0]=Data;
                frameLen++;
                frameFlag=0;        //组帧开始
            }
            break;
        }
        default:    //其他情况
        {
        	
            //如果接收到的不是帧尾
            if(frameLen>=1 && Data!='!')
            {
                buffer[frameLen]=Data;
                frameLen++;
                break;
            }

            //若是末尾数据则组帧成功
            if(Data=='!')
            {
                buffer[frameLen]=Data;
            	frameFlag=1;    //组帧成功
                frameLen=0;     //计数清0,准备重新组帧
                break;
            }
        }
    }
    return frameFlag;                 //返回组帧状态
}

3、利用PWM脉宽调制,交替显示红灯的5个短闪和5个长闪。

//====================================================================
//文件名称:main.c(应用工程主函数)
//框架提供:SD-Arm(sumcu.suda.edu.cn)
//版本更新:2017.08, 2020.05
//功能描述:见本工程的<01_Doc>文件夹下Readme.txt文件
//====================================================================

#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;          //占空比
    uint32_t m_i;           //控制在未知周期内不同占空比的波形只打印有限次
    uint8_t m_K;           //确保每次能正确打印输出PWM波形
     uint8_t f;             //希望采集的电平高低标志
    
    //(1.2)【不变】关总中断
    DISABLE_INTERRUPTS;
    f=1;
    //(1.3)给主函数使用的局部变量赋初值
    Flag=1;
    mFlag=0;     //灯的状态标志
    //(1.4)给全局变量赋初值
    
    //(1.5)用户外设模块初始化
    gpio_init(LIGHT_RED,GPIO_OUTPUT,LIGHT_OFF);    //初始化灯
    //pwm_init(PWM_USER,1500,1500,50.0,PWM_CENTER,PWM_MINUS);   //PWM输出初始化
    
    //(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;
    pwm_init(PWM_USER,375,750,50.0,PWM_CENTER,PWM_MINUS);   //PWM输出初始化
    //m_duty=50.0;
    for(;;)     //for(;;)(开头)
    {
    
   // pwm_update(PWM_USER,m_duty);         //调节占空比
        //m_duty=m_duty+5.0;                   //占空比每次递增5
        //if (m_duty>=90.0) m_duty=1.0;       //当占空比大于90后置为1,重新进行循环。
        
        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))
                {
                    if(f)
                    {printf("长闪高电平:1\n");
                    Delay_ms(1000);
                   
                    }
                    else
                    {
                    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);
        }
         printf("高\n");
         if(f)
         f=0;
         else f=1;
    }  //for(;;)结尾
    //(2)======主循环部分(结尾)========================================
}

//======以下为主函数调用的子函数存放处=====================================
//======================================================================
//函数名称:Delay_ms
//函数返回:无
//参数说明:无
//功能概要:延时 - 毫秒级
//======================================================================
void Delay_ms(uint16_t u16ms)
{
    uint32_t u32ctr;
    for(u32ctr = 0; u32ctr < 8000*u16ms; u32ctr++)
    {
        __ASM("NOP");
    }
}
//===================================================================

长闪:                                                                   短闪:

4、GEC39定义为输出引脚,GEC10定义为输入引脚,用杜邦线将两个引脚相连,验证捕捉实验程序Incapture-Outcmp-20211110,观察输出的时间间隔。

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");
    }
}

  • 9
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值