Pohyblivý text
Zo stránky SensorWiki
Záverečný projekt predmetu MIPS / LS2026 - Dariia Dordiai
Zadanie
Cieľom projektu je zobraziť text na LCD displeji EA-DOGM163, ktorý je pripojený k vývojovej doske ACROB s mikrokontrolérom ATmega328P. Text je možné posúvať vľavo a vpravo po riadku displeja pomocou potenciometra. Poloha potenciometra je snímaná analógovo-digitálnym prevodníkom (ADC) a prepočítaná na pozíciu kurzora na displeji. Program umožňuje zobraziť ľubovoľný text.
Literatúra:
Analýza a opis riešenia
Systém sa skladá z dvoch hlavných častí: vývojovej dosky ACROB s mikrokontrolérom ATmega328P a rozširujúceho modulu s LCD displejom EA-DOGM163. Externe je pripojená ovládacia krabička s potenciometrom, ktorá je k doske prepojená páskowym káblom. Otočením potenciometra sa mení napätie na analógovom vstupe, ktoré mikrokontrolér prevedie na pozíciu textu a zobrazí ho na displeji.

Použité komponenty
Vývojová doska ACROB — doska postavená okolo mikrokontroléra ATmega328P.

LCD modul EA-DOGM163 — rozširujúci modul pre dosku ACROB s displejom 3×16 znakov, oranžovým podsvietením, troma tlačidlami a jednou LED diódou.

Potenciometer — externý otočný potenciometer umiestnený v krabičke so stupnicou 0–10. Slúži ako vstupné zariadenie pre nastavenie polohy textu. Prepojený s doskou páskowym káblom.
Algoritmus a program
Program je napísaný v jazyku C pre mikrokontrolér ATmega328P. Po inicializácii UART, LCD displeja a ADC vstupuje program do nekonečnej slučky. V každom cykle sa načíta hodnota z potenciometra. Aby sa eliminoval šum, meranie sa opakuje päťkrát s prestávkou 5 ms a výsledok sa spriemeruje. Ak je výsledná hodnota väčšia ako 1000, automaticky sa zaokrúhli na 1023 — táto mŕtva zóna zabraňuje chveniu textu pri krajnej polohe potenciometra. Hodnota ADC sa prepočíta na pozíciu kurzora v rozsahu 0–10 podľa vzorca:
pos = ((1023 - val) × MAX_COL) / 1023
Text sa prepíše na displej len vtedy, keď sa pozícia zmení oproti predchádzajúcemu cyklu. Tým sa predchádza zbytočnému blikaniu displeja. Riadok sa najprv vymaže funkciou lcd_clearline(), následne sa nastaví kurzor na novú pozíciu a vypíše sa text pomocou lcd_puts().
#include <avr/io.h>
#include <util/delay.h>
#include <stdio.h>
#include "uart.h"
#include "lcd.h"
#include "adc.h"
FILE mystdout = FDEV_SETUP_STREAM(uart_putc, NULL, _FDEV_SETUP_WRITE);
#define TEXT "DARIIA" // zobrazovaný text
#define TEXT_LEN 6 // dĺžka textu
#define MAX_COL 10 // maximálna pozícia kurzora
int main(void)
{
uart_init(); // inicializácia UART
stdout = &mystdout;
lcd_init(); // inicializácia LCD displeja
lcd_bklt(1); // zapnutie podsvietenia
adc_init(); // inicializácia ADC
int last_pos = -1; // posledná pozícia textu
while(1)
{
// načítanie hodnoty z potenciometra (priemer 5 meraní)
unsigned int val = 0;
for (uint8_t i = 0; i < 5; i++) {
val += adc_read(4); // čítanie ADC kanála 4
_delay_ms(5);
}
val /= 5; // výpočet priemeru
// mŕtva zóna — stabilizácia pri krajnej polohe
if (val > 1000) val = 1023;
// prepočet hodnoty ADC na pozíciu kurzora (0 - 10)
int pos = ((1023 - val) * MAX_COL) / 1023;
// aktualizácia displeja len pri zmene pozície
if (pos != last_pos)
{
lcd_clearline(1); // vymazanie riadku
_delay_ms(2); // krátka pauza po vymazaní
lcd_setCursor(1, pos); // nastavenie kurzora na novú pozíciu
lcd_puts(TEXT); // vypísanie textu
last_pos = pos; // uloženie aktuálnej pozície
}
_delay_ms(150); // pauza pred ďalším cyklom
}
return 0;
}
#define BAUD 9600
#define F_CPU 16000000UL
#include <avr/io.h>
#include <util/setbaud.h>
void uart_init( void )
{
UBRR0H = UBRRH_VALUE;
UBRR0L = UBRRL_VALUE;
#if USE_2X
UCSR0A |= _BV(U2X0);
#else
UCSR0A &= ~(_BV(U2X0));
#endif
UCSR0C = _BV(UCSZ01) | _BV(UCSZ00);
UCSR0B = _BV(RXEN0) | _BV(TXEN0);
}
void uart_putc(char c)
{
if (c == '\n')
{
uart_putc('\r');
}
loop_until_bit_is_set(UCSR0A, UDRE0);
UDR0 = c;
}
void uart_puts(const char *s)
{
int i=0;
for(i=0;s[i]!='\0';i++)
uart_putc(s[i]);
}
char uart_getc(void) {
loop_until_bit_is_set(UCSR0A, RXC0);
return UDR0;
}
#ifndef UART_H_
#define UART_H_
void uart_init( void );
void uart_putc( char c );
void uart_puts( const char *s );
char uart_getc( void );
#endif /* UART_H_ */
#include "lcd.h"
/* Primitívne funkcie, nepredpokladá sa ich využitie užívateľom */
/* Funkcia zapíše jeden bajt po SPI zbernici do zariadenia */
void lcd_write( char data )
{
signed char index = 8;
char c_data;
DISPLAY_CSB_PORT &= ~(1<<DISPLAY_CSB_PIN); // Chip-Select do log.0
c_data = data;
do
{
_delay_us(6);
if ( c_data & 0x80 ) // najvyšší bit zamaskujeme
PORTB |= (1<<3); // a pošleme ho na zbernicu
else
PORTB &= ~(1<<3); // cez vodič MOSI
_delay_us(5); // vygenerujeme jeden hodinový pulz
PORTB &= ~(1<<5); // na vývod CLK
_delay_us(6);
PORTB |= (1<<5);
c_data = c_data << 1; // na najvyššie miesto posunieme ďalší bit
index--;
} while (index > 0); // opakujeme 8-krát
_delay_ms( 2 );
DISPLAY_CSB_PORT |= (1<<DISPLAY_CSB_PIN); // zdvihneme /CS do log.1
}
/* Funkcia zapíše jeden bajt do riadiaceho (Control) registra */
void lcd_command( char instruction )
{
DISPLAY_RS_PORT &= ~(1<<DISPLAY_RS_PIN); // RS = 0 → príkaz
_delay_us( 1 );
lcd_write( instruction );
}
/* Funkcia zapíše jeden bajt do dátového (Data) registra */
void lcd_data( char data )
{
DISPLAY_RS_PORT |= 1<<DISPLAY_RS_PIN; // RS = 1 → dáta
_delay_us( 7 );
lcd_write( data );
}
/* Užívateľské funkcie */
/* Inicializácia displeja podľa datasheetu pre radič ST7036 */
void lcd_init(void)
{
DDRB |= (1<<PB3) | (1<<PB5); // MOSI + SCK ako výstupy
DISPLAY_RS_DDR |= 1<<DISPLAY_RS_PIN; // RS ako výstup
DISPLAY_CSB_DDR |= 1<<DISPLAY_CSB_PIN; // CSB ako výstup
DISPLAY_BKLT_DDR|= 1<<DISPLAY_BKLT_PIN; // podsvietenie ako výstup
PORTB |= (1<<PB5); // CLK do log.1
DISPLAY_CSB_PORT |= (1<<DISPLAY_CSB_PIN); // CSB do log.1
DISPLAY_RS_PORT &= ~(1<<DISPLAY_RS_PIN); // RS do log.0
_delay_ms(50); // čakáme viac ako 40ms na ustálenie napájacieho napätia
lcd_command( 0x39 ); // nastavenie funkcie: 8-bit, 2 riadky, tabuľka inštrukcií 1
_delay_us(50);
lcd_command( 0x1d ); // nastavenie Bias: BS 1/5, 3-riadkový displej
_delay_us(50);
lcd_command( 0x50 ); // Booster vypnutý, nastavenie kontrastu C5, C4
_delay_us(50);
lcd_command( 0x6c ); // nastavenie napäťového sledovača a zosilnenia
_delay_ms( 500 ); // čakáme viac ako 200ms
lcd_command( 0x7c ); // nastavenie kontrastu C3, C2, C1
_delay_us(50);
lcd_command( 0x38 ); // zapnutie displeja
_delay_us(50);
lcd_command( 0x0c ); // displej zapnutý, kurzor vypnutý
_delay_us(50);
lcd_command( 0x01 ); // vymazanie displeja, kurzor na začiatok
_delay_ms(400);
lcd_command( 0x06 ); // automatický posun kurzora doprava
_delay_us(50);
}
/* Funkcia zobrazí na pozícii kurzora jeden znak */
void lcd_putc( char znak )
{
lcd_data(znak);
}
/* Funkcia vypíše reťazec znakov na displej */
void lcd_puts(char *string)
{
while (*string) {
lcd_data(*string); // posiela znaky jeden po druhom
string++;
}
}
/* Funkcia nastaví kurzor na pozíciu riadok, stĺpec */
void lcd_setCursor(char row, char col)
{
// adresy riadkov pre 3-riadkový displej 16x3
const char row_offsets[] = {0x00, 0x10, 0x20};
lcd_command(0x80 | (row_offsets[row] + col));
}
/* Funkcia vymaže jeden riadok displeja */
void lcd_clearline( unsigned char riadok )
{
unsigned char index;
lcd_setCursor( riadok, 0 );
for (index=1; index<20; index++) lcd_data( ' ' ); // vypĺňame medzerami
}
/* Funkcia vymaže celý displej */
void lcd_clear( void )
{
lcd_clearline( 0 ); // vymazanie 1. riadku
lcd_clearline( 1 ); // vymazanie 2. riadku
lcd_clearline( 2 ); // vymazanie 3. riadku
}
/* Funkcia zapne alebo vypne podsvietenie displeja */
void lcd_bklt( char OnOff)
{
if (OnOff)
DISPLAY_BKLT_high; // podsvietenie zapnuté
else
DISPLAY_BKLT_low; // podsvietenie vypnuté
}
/* Funkcia uloží užívateľom definovaný znak do CG RAM na adresu */
void def_znak(unsigned char *ZnakArray, unsigned char Address)
{
lcd_command(0x40|(Address<<3)); //nastavenie adresy znaku v CGRAM
for(unsigned char i = 0;i<8; i++) lcd_data( *(ZnakArray + i));
lcd_command(0x80); // zmena na DD RAM
}
#ifndef LCD_H_
#define LCD_H_
#include <stdlib.h>
#include <util/delay.h>
#include <avr/interrupt.h>
#include <avr/pgmspace.h>
#define CHARACTER_BUFFER_BASE_ADDRESS 0x80
#define CHARACTERS_PER_ROW 16
#define DISPLAY_RS_PORT PORTD
#define DISPLAY_RS_DDR DDRD
#define DISPLAY_RS_PIN 2
#define DISPLAY_CSB_PORT PORTD
#define DISPLAY_CSB_DDR DDRD
#define DISPLAY_CSB_PIN 4
#define DISPLAY_BKLT_PORT PORTD
#define DISPLAY_BKLT_DDR DDRD
#define DISPLAY_BKLT_PIN 6
#define DISPLAY_RS_low (DISPLAY_RS_PORT &= ~(1<<DISPLAY_RS_PIN))
#define DISPLAY_RS_high (DISPLAY_RS_PORT |= (1<<DISPLAY_RS_PIN))
#define DISPLAY_CSB_low (DISPLAY_CSB_PORT &= ~(1<<DISPLAY_CSB_PIN))
#define DISPLAY_CSB_high (DISPLAY_CSB_PORT |= (1<<DISPLAY_CSB_PIN))
#define DISPLAY_BKLT_low (DISPLAY_BKLT_PORT &= ~(1<<DISPLAY_BKLT_PIN))
#define DISPLAY_BKLT_high (DISPLAY_BKLT_PORT |= (1<<DISPLAY_BKLT_PIN))
#define INSTRUCTION_CLEAR_DISPLAY 0b00000001
#define INSTRUCTION_FUNCTION_SET_INIT_0 0b00110011
#define INSTRUCTION_FUNCTION_SET_INIT_1 0b00110010
#define INSTRUCTION_FUNCTION_SET_INIT_2 0b00101001
#define INSTRUCTION_INSTRUCTION_SET_0 0b00101000
#define INSTRUCTION_INSTRUCTION_SET_1 0b00101001
#define INSTRUCTION_BIAS_SET 0b00010101
#define INSTRUCTION_POWER_CONTROL 0b01010011
#define INSTRUCTION_FOLLOWER_CONTROL 0b01101100
#define INSTRUCTION_CONTRAST_SET 0b01111111
#define INSTRUCTION_DISPLAY_ON 0b00001100
#define INSTRUCTION_ENTRY_MODE 0b00000110
void lcd_init(void);
void lcd_write( char data );
void lcd_data( char data );
void lcd_command( char instruction );
void lcd_putc( char znak );
void lcd_puts( char *string );
void lcd_clear( void );
void lcd_clearline( unsigned char zeile );
void lcd_setCursor(char row, char col);
void lcd_bklt( char OnOff);
void def_znak(unsigned char *ZnakArray, unsigned char Address);
#endif /* LCD_H_ */
#include <avr/io.h>
void adc_init(void)
{
ADMUX = (1<<REFS0);
ADCSRA = (1<<ADEN)|(1<<ADPS2)|(1<<ADPS1)|(1<<ADPS0);
}
unsigned int adc_read(char channel)
{
channel &= 0x0F;
ADMUX = (ADMUX & 0xF0)|channel;
ADCSRA |= (1<<ADSC);
while(ADCSRA & (1<<ADSC))
{ /* wait */ }
return (ADC);
}
#include <avr/io.h>
void adc_init(void);
unsigned int adc_read(char a_pin);
Zdrojový kód: Dordiai_Dariia_projekt.zip
Overenie
Funkčnosť zariadenia bola overená manuálnym testovaním. Po zapnutí napájania sa na LCD displeji zobrazí text na predvolenej pozícii. Otočením potenciometra vľavo sa text posúva smerom doľava, otočením vpravo sa posúva doprava. Text sa aktualizuje plynule a bez blikania. Overenie prebehlo úspešne — text sa zobrazuje správne na všetkých pozíciách v rozsahu 0–10 a zariadenie reaguje na zmenu polohy potenciometra bez oneskorenia.
Video:
Čo by som urobila inak
Do budúcna by som implementovala plynulejší pohyb textu pomocou prerušení namiesto oneskorení _delay_ms(). Použitie časovača s prerušením by umožnilo presnejšie časovanie a mikrokontrolér by mohol počas čakania vykonávať iné úlohy.