U BOARD ir.

ورود به حساب کاربری
مرا بخاطر بسپار
گذرواژه را فراموش کرده اید؟
هنوز ثبت نام نکرده اید؟ تنها سه گزینه برای ثبت نام ثبت نام

U BOARD ir.

ثبت نام
ثبت نام
لطفا تمامی موارد خواسته شده را تکمیل نمایید لطفا ایمیل معتبر وارد نمایید لطفا گذرواژه را بیشتر از 6 کاراکتر وارد نمایید لطفا گذرواژه را مجددا بیشتر از 6 کاراکتر وارد نمایید
روش راه اندازی ADC و LCD گرافیکی KS108 با میکروکنترلرهای AVR

نمونه کد راه اندازی ADC در میکروکنترلر ATmega128A و نمایش در LCD گرافیکی KS108 در نرم افزار Atmel Studio، کتابخانۀ GPIO برای استفادۀ توابع LCD گرافیکی، مشاهدۀ نتیجۀ برنامۀ نوشته شده برای راه اندازی ADC و نمایش مقدار دیجیتال در LCD گرافیکی KS108 در Proteus و در عمل


رضا اسدی ۹۸/۰۹/۱۹ زمان موردنیاز برای مطالعه ۳۰ دقیقه

ADC در AVR مجموعه ای از یک بلوک سخت افزاری و نرم افزارهایی است که با استفاده از آنها یک تبدیل ADC انجام می شود. با برنامه نویسی مبدل آنالوگ به دیجیتال، ADC در AVR فعال می شود. ADC در AVR روش های تحریک متفاوت، فرکانس های کاری مختلف و تعداد کانال های متعدد (تکی و تفاضلی) دارد.

این که بخواهیم به طور کامل بدانیم ADC چیست، بحث مفصلی است و توضیح ADC در این نوشته نمی گنجد. در  نوشتۀ «بررسی بلوک ADC در میکروکنترلرهای AVR» به بحث ADC در AVR پرداخته می شود و توضیحات جامع تری دربارۀ ADC و راه اندازی آن بیان می گردد. ADC مخفف عبارت Analog to Digital Converter به معنای مبدل آنالوگ به دیجیتال است. وسایل و المان های دیجیتال برای پردازش سیگنال های آنالوگ نیاز دارند تا آنها را به مقادیر دیجیتال تبدیل کنند. این کار با ADC انجام می شود. کاربرد تبدیل سیگنال های آنالوگ به دیجیتال (ADC) در اندازه گیری خروجی سنسورهای آنالوگ ، اندازه گیری مقدار ولتاژ و اندازه گیری و پردازش هر سیگنال آنالوگ توسط سیستم های دیجیتال است. ADC در AVR یک بلوک سخت افزاری است. برای راه اندازی ADC در AVR باید برای این بلوک برنامه نوشت.

در این نوشته می خواهیم به بررسی یک نمونه کد برای راه اندازی ADC در AVR (در میکروکنترلر ATmega64A) به همراه راه اندازی LCD گرافیکی KS108 بپردازیم. برای راه اندازی KS108 از یک کتابخانه استفاده می کنیم که در ادامه توضیح داده خواهد شد. در این نمونه کد سعی شده از نکات و روش های گفته شده در دورۀ آموزشی «زبان C و C++ ویژۀ میکروکنترلرها» برای نوشتن کدها استفاده کنیم. با استفاده از نکات رعایت شده، برای راه اندازی ADC در AVR، نمونه کدی نوشتیم که مستقل از میکروکنترلر است و می توان آن را برای میکروکنترلرهای دیگر نیز به کار برد. این نمونه کد در نرم افزار اتمل استودیو نوشته شده است.

راه اندازی ADC در کدویزارد به راحتی و با چند تیک زدن انجام می شود. در کدویزارد رجیسترها توسط نرم افزار مقداردهی می شوند. اگر در کدویزارد بخواهیم از چند کانال ADC استفاده کنیم، کدی تولید می شود که کار با ADC را خیلی راحت می کند. این کار با استفاده از وقفۀ ADC امکان پذیر شده است.

در ادامه به بررسی برنامۀ نوشته شده در نرم افزار اتمل استودیو می پردازیم. برای مطالعۀ رجیسترهای مربوط به بلوک ADC در AVR می توانید به دیتاشیت ATmega64A، قسمت ADC آن مراجعه کنید. در دیتاشیت ATmega64A می بینیم که کانال های ورودی ADC، پایه های پورت F هستند.

 

بررسی کدهای نوشته شده

در این مرحله به شرح کدهای نوشته شده برای ADC و KS108 می پردازیم. برای درک بهتر عملکرد کدهای ADC در AVR و همچنین راه اندازی KS108،می توانید ویدئوی «روش راه اندازی ADC در AVR و LCD گرافیکی KS108» را مشاهده کنید.

main.cpp های درون فایل define بررسی

در ابتدا فرکانس کاری میکروکنترلر و نوع میکروکنترلر را define کرده ایم:

#ifndef F_CPU
#define F_CPU 8000000UL // You can enter the frequency you want for the processor here
#endif

#define AVR	   1
#define LPC	   2
#define McuFamily  AVR

 

فرکانس کاری میکروکنترلر 8 مگاهرتز انتخاب شده است. در سطرهای بعدی دو میکروکنترلر را define کرده ایم. یک عبارت دیگر هم define کرده ایم. هرگاه قرار باشد این کد برای میکروکنترلر AVR استفاده شود جلوی عبارت McuFamily می نویسیم AVR و هرگاه این کد بخواهد برای میکروکنترلرهای LPC استفاده شود باید جلوی آن بنویسیم LPC. میکروکنترلرهای دیگر هم می توان به این نمونه کد اضافه کرد. کافی است یک define دیگر بنویسیم. به طور مثال بنویسیم: #define STM      3. در کدهای نوشته شده همان طور که خواهیم دید با توجه به این که کدام میکروکنترلر  define شده است، کد مربوط به آن میکروکنترلر کامپایل می شود. راه اندازی ADC در این نمونه کد فعلاً برای میکروکنترلر AVR انجام شده است.

در قسمت های مختلف کد نوشته شده از دستور پیش پردازندۀ #if استفاده شده است. این شرط ها بررسی می کنند که کدام میکروکنترلر define شده است. سپس در هنگام کامپایل برنامه تنها کدی که مربوط به میکروکنترلر define شده است، کامپایل می شود.

gpio.h بررسی کتابخانۀ

برای راه اندازی KS108 در AVR (یا میکروکنترلرهای دیگر) بهتر است در مقداردهی به رجیسترهای مربوط به پورت های I/O از توابع استفاده کنیم. در اینجا برای این توابع یک کتابخانه نوشتیم. توابع این کتابخانه در کتابخانۀ LCD گرافیکی استفاده شده است. ما نیز در جاهایی که نیاز داریم می توانیم از آنها استفاده کنیم. در این کتابخانه سه تابع مهم وجود دارد. این سه تابع عبارتند از: config، read و write.

void Config (unsigned char PinNum ,unsigned char PinDir , unsigned char InResOrOutTr);
void Write (unsigned char PinNum , unsigned char Status);
char Read (unsigned char PinNum);

 

تابع Config برای تنظیمات پایه هاست. این که ورودی یا خروجی باشند. وقتی خروجی هستند مقدارشان 1 باشد یا صفر و وقتی هم که ورودی هستند پول آپ باشند یا پول دون یا ... . چون کتابخانه را برای میکروکنترلرهای دیگر هم قرار است استفاده کنیم باید تمام حالاتی را که یک پایه می تواند داشته باشد، در نظر بگیریم. مثلاً برای میکروکنترلر LPC1768 هنگامی که یک پایه ورودی است چهار حالت برای آن وجود دارد: 1- پول آپ، 2- repeater، 3- float و 4- پول دون.

هر کدام از میکروکنترلرهای AVR، LPC و STM در ورودی و خروجی بودن پایه های I/O و حالات آن، شرایط متفاوتی دارند. برای اطلاع بیشتر در این باره می توانید «دورۀ آموزشی AVR»، «دورۀ آموزشی LPC1768» و یا «دورۀ آموزشی STM32F1» را از وبسایت یوبورد تهیه کرده و مشاهده بفرمایید.

در برنامه نویسی اصولی به جای مقدار دهی مستقیم به رجیسترها بهتر است از توابع استفاده کنیم. در تابع Config، ورودی اول، شمارۀ بیت رجیستر مورد نظر است. برای AVR این رجیستر DDRx نام دارد. ورودی دوم برای تعیین ورودی یا خروجی بودن پایه است. ورودی سوم نیز حالت های مختلف را در ورودی و یا خروجی بودن تعیین می کند. در AVR اگر پایه ورودی شود می تواند پول آپ و یا Tri-state باشد. اگر خروجی شود می تواند صفر یا یک شود. یعنی مقدار صفر و یا یک روی پایه قرار گیرد.

تابع write برای صفر یا یک کردن خروجی است. ورودی اول آن شماره بیت رجیستر مورد نظر است. این رجیستر در AVR، PORT نام دارد. ورودی دوم نیز برای صفر و یا یک کردن خروجی است. اگر در آن صفر بنویسیم، خروجی Low و اگر یک بنویسیم، خروجی High می شود.

تابع read نیز وضعیت صفر و یا یک بودن پایۀ مورد نظر را که در ورودی تابع نوشته شده است، برمی گرداند.

در کتابخانۀ gpio.h یک enum به صورت زیر تعریف کرده ایم:

typedef enum {

#if (McuFamily == LPC)
	P0_0, P0_1, P0_2, P0_3, P0_4, P0_5, P0_6, P0_7, P0_8, P0_9, P0_10,P0_11,P0_12,P0_13,P0_14,P0_15,
	P0_16,P0_17,P0_18,P0_19,P0_20,P0_21,P0_22,P0_23,P0_24,P0_25,P0_26,P0_27,P0_28,P0_29,P0_30,P0_31,

	P1_0, P1_1, P1_2, P1_3, P1_4, P1_5, P1_6, P1_7, P1_8, P1_9, P1_10,P1_11,P1_12,P1_13,P1_14,P1_15,
	P1_16,P1_17,P1_18,P1_19,P1_20,P1_21,P1_22,P1_23,P1_24,P1_25,P1_26,P1_27,P1_28,P1_29,P1_30,P1_31,

	P2_0, P2_1, P2_2, P2_3, P2_4, P2_5, P2_6, P2_7, P2_8, P2_9, P2_10,P2_11,P2_12,P2_13,P2_14,P2_15,
	P2_16,P2_17,P2_18,P2_19,P2_20,P2_21,P2_22,P2_23,P2_24,P2_25,P2_26,P2_27,P2_28,P2_29,P2_30,P2_31,

	P3_0, P3_1, P3_2, P3_3, P3_4, P3_5, P3_6, P3_7, P3_8, P3_9, P3_10,P3_11,P3_12,P3_13,P3_14,P3_15,
	P3_16,P3_17,P3_18,P3_19,P3_20,P3_21,P3_22,P3_23,P3_24,P3_25,P3_26,P3_27,P3_28,P3_29,P3_30,P3_31,

	P4_0, P4_1, P4_2, P4_3, P4_4, P4_5, P4_6, P4_7, P4_8, P4_9, P4_10,P4_11,P4_12,P4_13,P4_14,P4_15,
	P4_16,P4_17,P4_18,P4_19,P4_20,P4_21,P4_22,P4_23,P4_24,P4_25,P4_26,P4_27,P4_28,P4_29,P4_30,P4_31,
#elif (McuFamily == AVR)
	PA_0,PA_1,PA_2,PA_3,PA_4,PA_5,PA_6,PA_7,
	PB_0,PB_1,PB_2,PB_3,PB_4,PB_5,PB_6,PB_7,
	PC_0,PC_1,PC_2,PC_3,PC_4,PC_5,PC_6,PC_7,
	PD_0,PD_1,PD_2,PD_3,PD_4,PD_5,PD_6,PD_7,
	PE_0,PE_1,PE_2,PE_3,PE_4,PE_5,PE_6,PE_7,
	PF_0,PF_1,PF_2,PF_3,PF_4,PF_5,PF_6,PF_7,
	PG_0,PG_1,PG_2,PG_3,PG_4
	#endif
}GpioPins;

 

در این تعریف دو شرط وجود دارد. بسته به این که میکروکنترلر AVR یا LPC (LPC1768) باشد، نوع تعریف enum متفاوت است. در این تعریف برای میکروکنترلر AVR عدد صفر به PA_0 و عدد 52 به PG_4 اختصاص یافته است. در LPC پایه ها به صورت P0.0، P0.1 و ... نام گذاری می شوند که همان طور که ملاحظه می کنید در enum تعریف شده این مورد رعایت شده است. در توابع مورد استفاده شرط هایی وجود دارد که پورت و پایۀ مورد نظر از این تعریف استخراج می شود. به طور مثال اگر  ورودی PinNum درتابع read به صورت زیر باشد:

Input  = Read(PC_5);

مقدار PC_5 که در enum تعریف شده بود چک می شود. مقدار PC_5 برابر است با 21. در تابع read این دستورها اجرا می شوند:

else if( PinNum>=16  && PinNum<=23  )
	{	// PORTC
		PinNum -= 16;
		return (PINC &= (1<<PinNum));
	}

 

ابتدا 16 واحد از مقدار PinNum کم می شود تا عددی بین صفر تا 7 شود. در اینجا با این که می دانیم PC_5 برابر 21 است مقدار PinNum برابر با 5 می شود. که بیت پنجم PINC می باشد. در سطر بعد مقدار رجیستر PINC با PinNum (که در اینجا 5 است) AND بیتی (&) می شود. و جواب آن در خروجی قرار می گیرد. این یک روش برای خواندن پایه های ورودی است. در این تابع فعلاً کدهای مربوط به AVR نوشته شده است:

char Read (unsigned char PinNum)
{
	
#if (McuFamily == AVR)
	if( PinNum<=7 )
	{ // PORTA
		PinNum -= 0;
		return (PINA &= (1<<PinNum)); 
	}
	else if( PinNum>=8  && PinNum<=15  )
	{	// PORTB
		PinNum -= 8;
		return (PINB &= (1<<PinNum)); 
	}
	else if( PinNum>=16  && PinNum<=23  )
	{	// PORTC
		PinNum -= 16;
		return (PINC &= (1<<PinNum));
	}
	else if( PinNum>=24  && PinNum<=31 )
	{	// PORTD
		PinNum -= 24;
		return (PIND &= (1<<PinNum)); 
	}
	else if( PinNum>=32 && PinNum<=39 )
	{	// PORTE
		PinNum -= 32;
		return (PINE &= (1<<PinNum)); 
	}
	else if( PinNum>=40 && PinNum<=47 )
	{	// PORTF
		PinNum -= 40;
		return (PINF &= (1<<PinNum));
	}
	else if( PinNum>=48 && PinNum<=55 )
	{	// PORTG
		PinNum -= 48;
		return (PING &= (1<<PinNum)); 
	}
#endif
}

 

در تابع write نیز به همین صورت پایۀ مورد نظر از میان enum تعریف شده تشخیص داده می شود. در این تابع پس از تشخیص پایۀ مورد نظر، مقدار صفر یا یک به آن پایه اختصاص می یابد. در این تابع هم کدهای AVR و هم کدهای LPC نوشته شده است:

void Write (unsigned char PinNum , unsigned char Status)
{
#if (McuFamily == LPC)
	if( PinNum<=31 )
	{ // PORT0
		PinNum -= 0;
		if(Status)	LPC_GPIO0->FIOSET = (1<<PinNum);
		else 				LPC_GPIO0->FIOCLR = (1<<PinNum);
	}
	else if( PinNum>=32  && PinNum<=63  )
	{	// PORT1
		PinNum -= 32;
		if(Status)	LPC_GPIO1->FIOSET = (1<<PinNum);
		else 				LPC_GPIO1->FIOCLR = (1<<PinNum);
	}
	else if( PinNum>=64  && PinNum<=95  )
	{	// PORT2
		PinNum -= 64;
		if(Status)	LPC_GPIO2->FIOSET = (1<<PinNum);
		else 				LPC_GPIO2->FIOCLR = (1<<PinNum);
	}
	else if( PinNum>=96  && PinNum<=127 )
	{	// PORT3
		PinNum -= 96;
		if(Status)	LPC_GPIO3->FIOSET = (1<<PinNum);
		else 				LPC_GPIO3->FIOCLR = (1<<PinNum);
	}
	else if( PinNum>=128 && PinNum<=159 )
	{	// PORT4
		PinNum -= 128;
		if(Status)	LPC_GPIO4->FIOSET = (1<<PinNum);
		else 				LPC_GPIO4->FIOCLR = (1<<PinNum);
	}
	
	
#elif (McuFamily == AVR)
	if( PinNum<=7 )
	{ // PORTA
		PinNum -= 0;
		if(Status)	PORTA |= (1<<PinNum);
		else 		PORTA &= (~(1<<PinNum));
	}
	else if( PinNum>=8  && PinNum<=15  )
	{	// PORTB
		PinNum -= 8;
		if(Status)	PORTB |= (1<<PinNum);
		else 		PORTB &= (~(1<<PinNum));
	}
	else if( PinNum>=16  && PinNum<=23  )
	{	// PORTC
		PinNum -= 16;
		if(Status)	PORTC |= (1<<PinNum);
		else 		PORTC &= (~(1<<PinNum));
	}
	else if( PinNum>=24  && PinNum<=31 )
	{	// PORTD
		PinNum -= 24;
		if(Status)	PORTD |= (1<<PinNum);
		else 		PORTD &= (~(1<<PinNum));
	}
	else if( PinNum>=32 && PinNum<=39 )
	{	// PORTE
		PinNum -= 32;
		if(Status)	PORTE |= (1<<PinNum);
		else 		PORTE &= (~(1<<PinNum));
	}
	else if( PinNum>=40 && PinNum<=47 )
	{	// PORTF
		PinNum -= 40;
		if(Status)	PORTF |= (1<<PinNum);
		else 		PORTF &= (~(1<<PinNum));
	}
	else if( PinNum>=48 && PinNum<=55 )
	{	// PORTG
		PinNum -= 48;
		if(Status)	PORTG |= (1<<PinNum);
		else 		PORTG &= (~(1<<PinNum));
	}
#endif
}

 

به طور مثال اگر ورودی اول این تابع برابر PD_3 باشد، این مقدار در enum برابر است با 27. پس برنامه وارد شرط زیر می شود:

else if( PinNum>=24  && PinNum<=31 )
	{	// PORTD
		PinNum -= 24;
		if(Status)	PORTD |= (1<<PinNum);
		else 		PORTD &= (~(1<<PinNum));
	}

 

ابتدا از این ورودی 24 تا کم می شود تا مقداری بین 0 تا 7 بگیرد. در اینجا PinNum برابر 3 می شود. در ادامه بیت سوم رجیستر PORTD بسته به این که متغیر Status صفر یا یک باشد، صفر یا یک می شود.

تابع config نیز به صورت زیر است. فعلاً فقط کدهای AVR برای آن نوشته شده است.

void Config (unsigned char PinNum ,unsigned char PinDir , unsigned char InResOrOutTr)
{
	
#if (McuFamily == AVR)
	if( PinNum<=7 )
	{ // PORTA
		PinNum -= 0;
		if(PinDir)			DDRA |= (1<<PinNum);
		else 				DDRA &= (~(1<<PinNum));
		if(InResOrOutTr)	PORTA |= (1<<PinNum);
		else 				PORTA &= (~(1<<PinNum));
	}
	else if( PinNum>=8  && PinNum<=15  )
	{	// PORTB
		PinNum -= 8;
		if(PinDir)			DDRB |= (1<<PinNum);
		else 				DDRB &= (~(1<<PinNum));
		if(InResOrOutTr)	PORTB |= (1<<PinNum);
		else 				PORTB &= (~(1<<PinNum));
	}
	else if( PinNum>=16  && PinNum<=23  )
	{	// PORTC
		PinNum -= 16;
		if(PinDir)			DDRC |= (1<<PinNum);
		else 				DDRC &= (~(1<<PinNum));
		if(InResOrOutTr)	PORTC |= (1<<PinNum);
		else 				PORTC &= (~(1<<PinNum));
	}
	else if( PinNum>=24  && PinNum<=31 )
	{	// PORTD
		PinNum -= 24;
		if(PinDir)			DDRD |= (1<<PinNum);
		else 				DDRD &= (~(1<<PinNum));
		if(InResOrOutTr)	PORTD |= (1<<PinNum);
		else 				PORTD &= (~(1<<PinNum));
	}
	else if( PinNum>=32 && PinNum<=39 )
	{	// PORTE
		PinNum -= 32;
		if(PinDir)			DDRE |= (1<<PinNum);
		else 				DDRE &= (~(1<<PinNum));
		if(InResOrOutTr)	PORTE |= (1<<PinNum);
		else 				PORTE &= (~(1<<PinNum));
	}
	else if( PinNum>=40 && PinNum<=47 )
	{	// PORTF
		PinNum -= 40;
		if(PinDir)			DDRF |= (1<<PinNum);
		else 				DDRF &= (~(1<<PinNum));
		if(InResOrOutTr)	PORTF |= (1<<PinNum);
		else 				PORTF &= (~(1<<PinNum));
	}
	else if( PinNum>=48 && PinNum<=55 )
	{	// PORTG
		PinNum -= 48;
		if(PinDir)			DDRG |= (1<<PinNum);
		else 				DDRG &= (~(1<<PinNum));
		if(InResOrOutTr)	PORTG |= (1<<PinNum);
		else 				PORTG &= (~(1<<PinNum));
	}
#endif
}

 

در این تابع نیز ابتدا پایۀ مورد نظر تشخیص داده می شود. سپس جهت آن که ورودی یا خروجی باشد با توجه به مقدار ورودی PinDir تعیین می شود. سپس با توجه به مقدار InResOrOutTr، یکی از حالت هایی که پایۀ مورد نظر می تواند داشته باشد در بیت مربوط به آن در رجیستر مورد نظر (در AVR رجیستر PORT) قرار داده می شود. در AVR اگر پایه ورودی باشد، دو حالت برای پایه وجود دارد. اگر بیت مربوط به آن پایه در رجیستر PORT برابر صفر باشد، آن پایه Tri-state است. اگر آن بیت 1 باشد، پایۀ مورد نظر به صورت داخلی پول آپ می شود.

حالت های مختلف را برای پایه ها پیش از تابع Config به صورت کامنت آورده ایم. این حالات برای میکروکنترلرهای مختلف ممکن است متفاوت باشد. هر میکروکنترلر که به برنامه اضافه می شود بهتر است کامنت در ابتدای توابع برای راهنمایی برنامه نویس قرار گیرد.

/*
PinNum example: PA_0 For AVR and STM32, PinNum example: P0_0 For LPC
PinDir example: 0 For Input and 1 For OutPut
InResOrOutTr example:
00	Input Float      or OutPut PushPull    0
01	Input PullUp	 or OutPut PushPull    1
10	Input PullDown	 or OutPut OpenDreain  0
11	Input Repeater	 or OutPut OpenDreain  1
*/

 

GlcdKs108.h کتابخانۀ

در جایی از این کتابخانه تعیین می کنیم که پایه های LCD به کدام پایۀ میکروکنترلر متصل شوند. با توجه به ترتیب پایه های بورد توسعۀ ATmega 64pin، تعریف پایه ها را به صورت زیر انجام می دهیم:

char _DB0=PA_7, _DB1=PA_6, _DB2=PG_0, _DB3=PC_1, _DB4=PC_3, _DB5=PA_5, _DB6=PA_4, _DB7=PA_3, _DI_RS=PC_6,
_RW=PC_7, _EN=PG_2, _CS1=PB_5, _CS2=PB_4;

 

با توجه به enum تعریف شده در کتابخانۀ gpio.h، این پایه ها را به صورت بالا تعریف می کنیم.

این کتابخانه دارای توابع زیر است:

uchar strlen( uchar *Value );
uchar max( uchar Val1, uchar Val2 );
uchar min( uchar Val1, uchar Val2 );
double powerOfTen(int num);

void setDataPinValue( uchar Value );
void setDataConfig( uchar Value );
uchar getDataPinValue( void );

void glcd_on();
void glcd_off(void);
void trigger(void);
void set_start_line( uchar line );
void goto_xy( uint x, uint y );
void glcd_write( uchar b );
void glcd_clrln( uchar ln );
void glcd_clear( void );
uchar is_busy(void);
uchar glcd_read( const unsigned char column );
uchar get_point(uint x,uint y);
void point_at( uint x, uint y, uchar color );
void h_line( uint x, uint y, uchar l, uchar s, uchar c );
void v_line( uint x, uint y, uint l, uchar s, uchar c );
void line ( uint x1, uint y1, uint x2, uint y2, uchar s, uchar c );
void rectangle( uint x1, uint y1, uint x2, uint y2, uchar s, uchar c, uchar EorF/*empty(0) or full(1) circle*/ );
void cuboid( uint x11, uint y11, uint x12, uint y12, uint x21, uint y21, uint x22, uint y22, uchar s, uchar c );
void h_parallelogram( uint x1, uint y1, uint x2, uint y2, uchar l, uchar s, uchar c );
void v_parallelogram( uint x1, uint y1, uint x2, uint y2, uchar l, uchar s, uchar c );
void h_parallelepiped( uint x11, uint y11, uint x12, uint y12, uchar l1, uint x21, uint y21, uint x22, uint y22, uchar l2, uchar s, uchar c );
void v_parallelepiped( uint x11, uint y11, uint x12, uint y12, uchar l1, uint x21, uint y21, uint x22, uint y22, uchar l2, uchar s, uchar c );
void circle( uint x0, uint y0, uint r, uchar s, uchar c );
void putIt( int c, int x, int y );
void enlarge( char *large, uchar c, uchar size );
void putItSz( int c, int x, int y, uchar sz );
void glcd_putchar( uchar c, int x, int y, uchar l, uchar sz );
void glcd_puts( uchar *c, int x, int y, uchar l, uchar sz, signed char space );
void glcd_printNumber( uint data, uint base, uint x, uint y, uchar l, uchar sz, char space );
void bmp_disp(const uchar *bmp, uint x1, uint y1, uint x2, uint y2 );

 

main.cpp کدهای نوشته شده در فایل

در این برنامه از وقفۀ ADC استفاده شده است.  در ادامه می بینیم که منبع تحریک ADC را free running قرار داده ایم. هر بار که تبدیل ADC انجام می شود، این وقفه اتفاق می افتد. در این وقفه مقدار تبدیل شده را که در رجیستر دیتای ADC (ADCW) قرار می گیرد، در متغیر adc_data ریخته می شود:

ISR(ADC_vect)
{
	adc_data=ADCW;
}

 

در تابع main نیز بعد از تعریف یک رشته برای نمایش در LCD، رجیسترهای ADC مقدار دهی شده اند:

int main (void)
{
	 char LcdStr[20];
	
	// ADC initialization
	// ADC Clock frequency: 1000.000 kHz
	// ADC Voltage Reference: AREF pin
	ADMUX=0x00;
	ADCSRA=(1<<ADEN) | (0<<ADSC) | (1<<ADATE) | (0<<ADIF) | (1<<ADIE) | (0<<ADPS2) | 
        (1<<ADPS1) | (1<<ADPS0);
	SFIOR=(0<<ACME);

 

در رجیستر ADCSRA با یک کردن بیت ADATE (در میکروکنترلر ATmega128A نام این بیت ADFR است)، منبع تحریک free running فعال شده است. وقتی این منبع انتخاب می شود، تبدیل ADC به صورت دائم مقدار آنالوگ ورودی را به دیجیتال تبدیل می کند. بیت ADIE وقفۀ ADC را فعال می کند. بیت های ADPS[2:0] نیز برای انتخاب فرکانس کلاک واحد ADC مقداردهی می شوند. در ادامه LCD گرافیکی پیکربندی شده است. عبارت UBOARD.IR در آن به مدت 1 ثانیه نمایش داده می شود و سپس با دستور sei() پرچم کلی وقفه ها فعال شده است. در پایان نیز با 1 کردن بیت ADSC از رجیستر ADCSRA تبدیل ADC را شروع می کنیم. برای تبدیل ADC وقتی که منبع را free running انتخاب کرده ایم، لازم است یک بار دستور شروع تبدیل را بدهیم.

    glcd_on();
	glcd_clear();
	glcd_puts((uchar*)"UBOARD.IR",0,0,0,1,0);
	_delay_ms(1000);
	glcd_clear();
	sei();
	ADCSRA|= (1<<ADSC);

 

در حلقۀ while مقدار متغیر adc_data نمایش داده می شود:

while(1)
	{
		sprintf (LcdStr , "ADC Data = %d", adc_data );
		glcd_puts((uchar*)LcdStr,0,0,0,1,0);
		_delay_ms(1000);	
		glcd_clear();
	}

 

در کدهای نوشته دیدیم که defineهایی داشتیم:

#define AVR	   1
#define LPC	   2
#define McuFamily  AVR

 

در این نمونه کد این defineها در فایل main.cpp آورده شده اند. این کار در اینجا که نمونه کد کوتاهی است مشکلی به وجود نمی آورد. اما در برنامه هایی که نسبتاً حجم زیادی دارند از خوانایی برنامه می کاهد. در دورۀ آموزشی «زبان C و C++ ویژۀ میکروکنترلرها» به بررسی یک قالب جامع می پردازیم. در این قالب defineها و includeها و توابع و کتابخانه ها و ... را به گونه ای در کنار هم قرار  داده ایم که خواناترین حالت ممکن را داشته باشند.

 

نتیجۀ برنامه را در عمل ببینیم

برای مشاهدۀ نتیجۀ نهایی راه اندازی ADC در AVR ابتدا در این قسمت برنامه را در نرم افزار Proteus اجرا می کنیم. بعد از آن نتیجه را به صورت عملی روی بورد توسعۀ ATmega 64pin می بینیم.

تصویر 1 - شبیه سازی در نرم افزار Proteus

 

بورد توسعۀ ATmega 64pin را پروگرم می کنیم. با توجه به توضیحات user manual این بورد، باید جامپر پین هدرهای 16 به 17، 18 به 19 و 20 به 21 متصل شود. نتیجۀ راه اندازی ADC را در تصاویر زیر مشاهده می  کنید. با تغییر پتانسیومتر روی بورد مقدار رجیستر دادۀ ADC (ADCW) بین صفر تا 1023 تغییر می کند.

تصویر 2 – مقدار رجیستر ADCW

 

تصویر 3 – مقدار رجیستر ADCW

 

به عنوان تمرین برای نمایش LCD می توانید توابعی بنویسید که ولتاژ و یا دمای خوانده شده توسط ADC به صورت یک نمودار یا یک عقربه و یا هر شکل دیگری روی LCD نمایش داده شود. برای خواندن ولتاژ با ADC نیز می توانید از فرمول زیر استفاده کنید. اگر رزولوشن 10 بیتی باشد داریم:

V=(ADCW * 1023) / Vref

 

:نتیجه می گیریم AVR در ADC از راه اندازی

1 - راه اندازی ADC در AVR شامل دو مرحلۀ مقداردهی به رجیسترهای ADC (پیکربندی) و دریافت داده های خروجی ADC می شود.

2 - ADC می تواند روشی برای تحلیل سیگنال های آنالوگ توسط سیستم های دیجیتال باشد.

3 - نوشتن برنامۀ اصولی و بهینه برای راه اندازی واحدهای مختلف میکروکنترلرها اهمیت ویژه ای دارد.

4 - می توان با استفاده از دستورهای پیش پردازنده نظیر #if و #elif و ... در یک کامپایلر برای چند میکروکنترلر کد نوشت و این کد را در کامپایلرهای دیگر برای میکروکنترلرهای دیگر استفاده کرد.

5 - در کدهای نوشته شده برای میکروکنترلرهای مختلف باید همۀ ویژگی ها و تفاوت های آن میکروکنترلرها در نظر گرفته شود.

6 - در تنظیمات رجیسترها به ویژه رجیسترهای مربوط به پایه ها (در AVR رجیسترهای DDRx، PINx و PORTx) بهتر است از توابع استفاده شود.

 

برای دریافت نسخۀ PDF این نوشته، فایل پیوست را دانلود نمایید.

 


رضا اسدی
مدیر یوبرد، طراح و تولید کنندۀ محصولات مبتنی بر میکروکنترلرها، مجری پروژه و مدرس آموزش برنامه نویسی میکروکنترلرها و برد مدار چاپی PCB، فعال در طراحی خودروهای الکتریکی
اگر دیدگاهی دارید میتوانید از طریق فرم زیر دیدگاه خود را در سایت درج کنید.
نویسنده: وحید سلیمانی تاریخ: ۵ تیر ۱۳۹۹ ساعت: ۰۵:۰۰:۱۱ ب.ظ
سلام. استاد کتابخونه gpioی که توضیح دادید خیلی عالیه. دستتون درد نکنه. فقط این بررسی شرط ها توی توابع باعث کند شدن برنامه نمیشه؟ ممنون
نویسنده: رضا اسدی تاریخ: ۱۲ تیر ۱۳۹۹ ساعت: ۱۰:۱۰:۵۴ ب.ظ

سلام. خواهش میکنم. بله میتونید کتابخونتون رو محدود کنید برای یک میکروکنترلر خاص، و همچنین مشخص باشه که قراره کدام پایه را مقدار دهی کنید. در این صورت نیاز به شرط نیست.

فرم پاسخ به دیدگاه