⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 demo_11_4.i

📁 这是马潮老师最新力做<<AVR单片机潜入式系统原理与应用实践>>中有关带音乐报时的C程序,用CVAVR编译,对想熟悉和学习AVR单片机的朋友一定有用.注释详细,一看就懂.
💻 I
字号:
/*********************************************
File name	        : demo_11_4.c
Chip type           : ATmega16
Program type        : Application
Clock frequency     : 1.000000 MHz
Memory model        : Small
External SRAM size  : 0
Data Stack size     : 256
*********************************************/

// CodeVisionAVR C Compiler
// (C) 1998-2001 Pavel Haiduc, HP InfoTech S.R.L.

// I/O registers definitions for the ATmega16


#pragma used+
sfrb TWBR=0;
sfrb TWSR=1;
sfrb TWAR=2;
sfrb TWDR=3;
sfrb ADCL=4;
sfrb ADCH=5;
sfrw ADCW=4;      // 16 bit access
sfrb ADCSRA=6;
sfrb ADMUX=7;
sfrb ACSR=8;
sfrb UBRRL=9;
sfrb UCSRB=0xa;
sfrb UCSRA=0xb;
sfrb UDR=0xc;
sfrb SPCR=0xd;
sfrb SPSR=0xe;
sfrb SPDR=0xf;
sfrb PIND=0x10;
sfrb DDRD=0x11;
sfrb PORTD=0x12;
sfrb PINC=0x13;
sfrb DDRC=0x14;
sfrb PORTC=0x15;
sfrb PINB=0x16;
sfrb DDRB=0x17;
sfrb PORTB=0x18;
sfrb PINA=0x19;
sfrb DDRA=0x1a;
sfrb PORTA=0x1b;
sfrb EECR=0x1c;
sfrb EEDR=0x1d;
sfrb EEARL=0x1e;
sfrb EEARH=0x1f;
sfrw EEAR=0x1e;   // 16 bit access
sfrb UBRRH=0x20;
sfrb UCSRC=0X20;
sfrb WDTCR=0x21;
sfrb ASSR=0x22;
sfrb OCR2=0x23;
sfrb TCNT2=0x24;
sfrb TCCR2=0x25;
sfrb ICR1L=0x26;
sfrb ICR1H=0x27;
sfrb OCR1BL=0x28;
sfrb OCR1BH=0x29;
sfrw OCR1B=0x28;  // 16 bit access
sfrb OCR1AL=0x2a;
sfrb OCR1AH=0x2b;
sfrw OCR1A=0x2a;  // 16 bit access
sfrb TCNT1L=0x2c;
sfrb TCNT1H=0x2d;
sfrw TCNT1=0x2c;  // 16 bit access
sfrb TCCR1B=0x2e;
sfrb TCCR1A=0x2f;
sfrb SFIOR=0x30;
sfrb OSCCAL=0x31;
sfrb OCDR=0x31;
sfrb TCNT0=0x32;
sfrb TCCR0=0x33;
sfrb MCUCSR=0x34;
sfrb MCUCR=0x35;
sfrb TWCR=0x36;
sfrb SPMCR=0x37;
sfrb TIFR=0x38;
sfrb TIMSK=0x39;
sfrb GIFR=0x3a;
sfrb GICR=0x3b;
sfrb OCR0=0X3c;
sfrb SPL=0x3d;
sfrb SPH=0x3e;
sfrb SREG=0x3f;
#pragma used-

// Interrupt vectors definitions


// Needed by the power management functions (sleep.h)
#asm
	#ifndef __SLEEP_DEFINED__
	#define __SLEEP_DEFINED__
	.EQU __se_bit=0x40
	.EQU __sm_mask=0xB0
	.EQU __sm_powerdown=0x20
	.EQU __sm_powersave=0x30
	.EQU __sm_standby=0xA0
	.EQU __sm_ext_standby=0xB0
	.EQU __sm_adc_noise_red=0x10
	.SET power_ctrl_reg=mcucr
	#endif
#endasm



flash unsigned char led_7[10]={0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F}; 
flash unsigned char position[6]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf};

flash unsigned int t[9] = {0,956,865,759,716,638,568,506,470};
flash unsigned char d[9] = {0,105,116,132,140,157,176,198,209};
flash unsigned char music[32] = {5,2,8,2,5,2,4,2,3,2,2,2,1,4,1,2,1,2,2,2,3,2,3,2,1,2,3,2,4,2,5,8}; 

unsigned char note_n;
unsigned int int_n;
bit play_on;

unsigned char time[3],time_set[3];			// 时、分、秒计数和设置单元
unsigned char dis_buff[6];					// 显示缓冲区,存放要显示的6个字符的段码值
unsigned char time_counter,key_stime_counter;	// 时间计数单元,
unsigned char clock_state = 6,return_time;
bit	point_on,set_on,time_1s_ok,key_stime_ok;

void display(void)					// 6位LED数管动态扫描函数
{
    static unsigned char posit=0;
	PORTC = 0xff;
	PORTA = led_7[dis_buff[posit]];
	if (set_on && (posit==clock_state)) PORTA= 0x00;		// 校时闪烁
	if (point_on && (posit==2||posit==4)) PORTA |= 0x80;	// 秒闪烁
	PORTC = position[posit]; 
	if (++posit >=6 ) posit = 0;	// (3)
}

// Timer 0 比较匹配中断服务,2ms定时
interrupt [20] void timer0_comp_isr(void)
{
	display();						// LED扫描显示
	if (++key_stime_counter >=5)
	{
		key_stime_counter = 0;
	    key_stime_ok = 1;							// 10ms到
		if (!(++time_counter % 25)) set_on = !set_on;	// 设置校时闪烁标志
		if (time_counter >= 100)
		{
			time_counter = 0;
			time_1s_ok = 1;        					// 1s到
		}
	}
}

// T/C1 比较匹配A中断服务
interrupt [7] void timer1_compa_isr(void)
{
    if (!play_on)
    {
        note_n = 0;
        int_n = 1;
        play_on = 1;
    }
    else
    {    
        if (--int_n == 0)
        {
            TCCR1B = 0x08;
            if (note_n < 32)
            {
                OCR1A = t[music[note_n]];
                int_n = d[music[note_n]];
                note_n++;
                int_n = int_n * music[note_n];
                note_n++;
                TCCR1B = 0x09;
            }
            else
                play_on = 0;
        }
    }
}

void time_to_disbuffer(unsigned char *time)				// 时钟时间送显示缓冲区函数
{
	unsigned char i,j=0;
	for (i=0;i<=2;i++)
	{
		dis_buff[j++] = time[i] % 10;
     	dis_buff[j++] = time[i] / 10;
	}
}


unsigned char read_key(void)
{
	static unsigned char key_state = 0,key_press;
	unsigned char key_return = 0;

	key_press = PINC			 & 0b11000000	;	// 读按键I/O电平
	switch (key_state)
	{
		case 0:				// 按键初始态
			if (key_press != 0b11000000	) key_state = 1;	
			break;				    	// 键被按下,状态转换到键确认态
		case 1:				// 按键确认态
			if (key_press == (PINC			 & 0b11000000	))
			{
				if (key_press == 0b01000000) key_return = 1;
				else if (key_press == 0b10000000) key_return = 2;
				key_state = 2;	// 状态转换到键释放态
			}
			else
				key_state = 0;	// 按键已抬起,转换到按键初始态
			break;
		case 2:
			if (key_press == 0b11000000	) key_state = 0;
			break;						//按键已释放,转换到按键初始态
	}
   	return key_return;
}

void main(void)
{
	unsigned char key_temp,i;

	DDRA=0xFF;          // LED段码输出
	PORTC=0xFF;
	DDRC=0x3F;          // LED位控输出
	DDRD=0x20;          // PD5音乐播放输出

	// T/C0初始化
	OCR0 = 0xF9;		// OCR0 = 0xF9(249),(249+1)/125=2ms
	TCCR0 = 0x0A;		// 内部时钟,8分频(1M/8=125KHz),CTC模式
	// T/C1初始化	
	TCCR1A=0x40;
    TCCR1B=0x08;
	TIMSK = 0x12;		// 允许T/C1比较匹配A中断,允许T/C0比较匹配中断

	time[2] = 23; time[1] = 58; time[0] = 55;	// 设时间初值23:58:55 

	#asm("sei")			// 开放全局中断

	while (1)
	{
		if (time_1s_ok)				// 1秒到
		{
			time_1s_ok = 0;
			point_on = ~point_on;		// 秒闪烁标志
			if (++time[0] >= 60)		// 秒加1,以下为时间调整
			{
				time[0] = 0;
				if (!play_on) TCCR1B = 0x09;    // 1分钟到,播放音乐
				if (++time[1] >= 60)
				{
					time[1] = 0;
					if (++time[2] >= 24) time[2] = 0;
				}
			}
			if ((++return_time >= 20) && (clock_state != 6)) clock_state = 6;
			if (clock_state == 6) time_to_disbuffer(time);
		}
		if (key_stime_ok)				// 10ms到,键处理
		{
			key_stime_ok = 0;
			key_temp = read_key();		// 调用按键接口程序
			if (key_temp)				// 确认有键按下
            {
				return_time = 0;
				if (key_temp == 1)		// K1键按下,状态转换
				{
					if (++clock_state >= 7) clock_state = 0;
					if (clock_state == 0)
					{
						for (i=0;i<=2;i++)	time_set[i] = 0;
						time_to_disbuffer(time_set);
					} 
 		    		if (clock_state == 6)
 		    		{
 		    			for (i=0;i<=2;i++)	time[i] = time_set[i];
 		    			time_to_disbuffer(time);	
 		    		}
				}
				if ((clock_state != 6) && (key_temp == 2))		// K2键按下
				{
					if (clock_state%2)	time_set[clock_state/2] += 10;
					else
					{
						if ((time_set[clock_state/2] % 10) == 9)
					 		time_set[clock_state/2] -= 9;
					 	else
					 		time_set[clock_state/2]+=1;
					}
					if (time_set[0] >= 60) time_set[0]-= 60;	// 以下设置时间调整
					if (time_set[1] >= 60) time_set[1]-= 60;
					if (time_set[2] >= 24) time_set[2]-= 10;
			 		time_to_disbuffer(time_set);				// 设置时间送显示缓存
				}
			}
		}
	}
}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -