Subversion Repositories svnkaklik

Rev

Rev 410 | Go to most recent revision | Details | Last modification | View Log

Rev Author Line No. Line
6 kaklik 1
// LCD modul pro ovladani dvouradkoveho LCD modulu se standardnim Hitachi radicem
2
// (c)miho 2002,2005
3
//
4
// Historie:
5
//
6
// 0.00  Uvodni verze se snadnou definici portu LCD displeje
7
// 0.01  Oprava portu (zapomenute stare identifikatory)
8
// 0.02  Doplnena moznost pripojeni datoveho portu LCD na libovolne porty
9
// 0.03  Doplnena procedura lcd_clr pro smazani displeje
10
//
11
//
12
// Funkce:
13
//
14
//   lcd_init()            inicializuje LCD displej a porty, nutno volat jako prvni
15
//
16
//   lcd_putc(c)           zapis snaku do lcd displeje, zpracovava nasledujici ridici znaky
17
//                         \f = \x0C   - nova stranka - smazani displeje
18
//                         \n = \x0A   - odradkovani (prechod na druhou radku)
19
//                         \b = \x08   - backspace - posunuti kurzoru o 1 pozici zpet
20
//                         \r = \x0D   - goto home to position 1,1
21
//                         \0  .. \7   - definovatelne znaky v pozicich 0 az 7 v CGRAM
22
//                         \20 .. \27  - alternativne zapsane znaky (oktalove) v pozicich 0 az 7 CGRAM
23
//                         Pozor na to, ze funkce printf konci tisk pokud narazi na \0 (konec retezce)
24
//
25
//   lcd_gotoxy(x,y)       presune kurzor na uvedenou adresu
26
//                         nekontroluje parametry
27
//
28
//   lcd_cursor_on         zapne kurzor
29
//   lcd_cursor_off        vypne kurzor
30
//
31
//   lcd_clr               smaze displej
32
//
33
//   lcd_define_char(Index, Def)       Makro, ktere definuje znaky od pozice Index obsahem definicniho
34
//                                     retezce Def. Kazdych 8 znaku retezce Def definuje dalsi znak v CGRAM.
35
//                                     Kapacita CGRAM je celkem 8 znaku s indexem 0 az 7.
36
//                                     Na konci se provede lcd_gotoxy(1,1).
37
//                                     Na konci teto knihovny je priklad pouziti definovanych znaku
38
//
39
//
40
// Definice portu:                     // Datovy port displeje pripojeny na 4 bity za sebou na jeden port
41
//
42
// #define LCD_RS          PIN_B2      // rizeni registru LCD displeje
43
// #define LCD_E           PIN_B1      // enable LCD displeje
44
// #define LCD_DATA_LSB    PIN_C2      // pripojeni LSB bitu datoveho portu LCD displeje (celkem 4 bity vzestupne za sebou)
45
//
46
//
47
// Alternativni definice:              // Datovy port displeje pripojeny na libovolne 4 bitove porty (vede na kod delsi asi o 25 slov)
48
//
49
// #define LCD_RS          PIN_B2      // rizeni registru LCD displeje
50
// #define LCD_E           PIN_B1      // enable LCD displeje
51
// #define LCD_D0          PIN_C2      // D0 - datove bity pripojene na libovolne porty
52
// #define LCD_D1          PIN_C3      // D1
53
// #define LCD_D2          PIN_C4      // D2
54
// #define LCD_D3          PIN_C5      // D3
55
 
56
 
57
 
58
 
59
// Privatni sekce, cist jen v pripade, ze neco nefunguje
60
 
61
 
62
 
63
 
64
#ifdef LCD_DATA_LSB
65
// Generovane defince portu pro ucely teto knihovny aby kod generoval spravne IO operace a soucasne
66
// bylo mozne jednoduse deklarovat pripojene piny LCD displeje pri pouziti teto knihovny. Problem spociva
67
// v tom, ze se musi spravne ridit smery portu a soucasne datovy port zabira jen 4 bity ze zadaneho portu
68
//
69
#define LCD_SHIFT (LCD_DATA_LSB&7)                 // pocet bitu posuvu datoveho kanalu v datovem portu
70
#define LCD_PORT  (LCD_DATA_LSB>>3)                // adresa LCD datoveho portu
71
#define LCD_TRIS  (LCD_PORT+0x80)                  // adresa prislusneho TRIS registru
72
#define LCD_MASK  (0xF<<LCD_SHIFT)                 // maska platnych bitu
73
//
74
#if LCD_SHIFT>4                                    // kontrola mezi
75
#error LCD data port LSB bit not in range 0..4
76
#endif
77
#endif
78
 
79
 
80
// Definice konstant pro LCD display
81
//
82
#define LCD_CURSOR_ON_  0x0E     // kurzor jako blikajici radka pod znakem
83
#define LCD_CURSOR_OFF_ 0x0C     // zadny kurzor
84
#define LCD_LINE_2      0x40     // adresa 1. znaku 2. radky
85
 
86
 
87
// Definice rezimu LCD displeje
88
//
89
BYTE const LCD_INIT_STRING[4] =
90
{
91
   0x28,                         // intrfejs 4 bity, 2 radky, font 5x7
92
   LCD_CURSOR_OFF_,              // display on, kurzor off,
93
   0x01,                         // clear displeje
94
   0x06                          // inkrement pozice kurzoru (posun kurzoru doprava)
95
};
96
 
97
 
98
// Odesle nibble do displeje (posle data a klikne signalem e)
99
//
100
void lcd_send_nibble( BYTE n )
101
{
102
   #ifdef LCD_DATA_LSB
103
      // data jsou za sebou na 4 bitech jednoho portu
104
      *LCD_PORT = (*LCD_PORT & ~LCD_MASK) | ((n << LCD_SHIFT) & LCD_MASK);      // nastav datove bity portu a ostatni zachovej
105
   #else
106
      // data jsou na libovolnych 4 bitech libovolnych portu
107
      output_bit(LCD_D0,bit_test(n,0));
108
      output_bit(LCD_D1,bit_test(n,1));
109
      output_bit(LCD_D2,bit_test(n,2));
110
      output_bit(LCD_D3,bit_test(n,3));
111
   #endif
112
   output_bit(LCD_E,1);       // vzestupna hrana
113
   delay_us(1);               // pockej alespon 450ns od e nebo alespon 195ns od dat
114
   output_bit(LCD_E,0);       // sestupna hrana (minimalni perioda e je 1us)
115
}
116
 
117
 
118
// Odesle bajt do registru LCD
119
//
120
// Pokud je Adr=0 .. instrukcni registr
121
// Pokud je Adr=1 .. datovy registr
122
//
123
void lcd_send_byte( BOOLEAN Adr, BYTE n )
124
{
125
   output_bit(LCD_RS,Adr);    // vyber registr
126
   swap(n);
127
   lcd_send_nibble(n);        // posli horni pulku bajtu
128
   swap(n);
129
   lcd_send_nibble(n);        // posli spodni pulku bajtu
130
   delay_us(40);              // minimalni doba na provedeni prikazu
131
}
132
 
133
 
134
// Provede inicializaci LCD displeje, smaze obsah a nastavi mod displeje
135
//
136
// Tato procedura se musi volat pred pouzitim ostatnich lcd_ procedur
137
//
138
void lcd_init()
139
{
140
 
141
   int i;                              // pocitadlo cyklu
142
 
143
   delay_ms(20);                       // spozdeni pro provedeni startu displeje po zapnuti napajeni
144
 
145
#ifdef LCD_DATA_LSB
146
   // data jsou na 4 bitech za sebou, nastav smer pro vsechny dalsi prenosy
147
   *LCD_TRIS = *LCD_TRIS & ~LCD_MASK;  // nuluj odpovidajici bity tris registru datoveho portu LCD
148
#endif
149
 
150
   output_bit(LCD_RS,0);               // nastav jako vystup a nastav klidovy stav
151
   output_bit(LCD_E, 0);               // nastav jako vystup a nastav klidovy stav
152
 
153
   for (i=0; i<3; i++)                 // nastav lcd do rezimu 8 bitu sbernice
154
   {
155
      delay_ms(2);                     // muze byt rozdelany prenos dat (2x 4 bity) nebo pomaly povel
156
      lcd_send_nibble(3);              // rezim 8 bitu
157
   }
158
 
159
   delay_us(40);                       // cas na zpracovani
160
   lcd_send_nibble(2);                 // nastav rezim 4 bitu (plati od nasledujiciho prenosu)
161
   delay_us(40);                       // cas na zpracovani
162
 
163
   for (i=0;i<3;i++)                   // proved inicializaci (nastaveni modu, smazani apod)
164
   {
165
      lcd_send_byte(0,LCD_INIT_STRING[i]);
166
      delay_ms(2);
167
   }
168
}
169
 
170
 
171
// Proved presun kurzoru
172
//
173
// Pozice 1.1 je domu
174
//
175
void lcd_gotoxy( BYTE x, BYTE y)
176
{
177
 
178
   BYTE Adr;
179
 
180
   Adr=x-1;
181
   if(y==2)
182
     Adr+=LCD_LINE_2;
183
 
184
   lcd_send_byte(0,0x80|Adr);
185
}
186
 
187
 
188
// Zapis znaku na displej, zpracovani ridicich znaku
189
//
190
void lcd_putc( char c)
191
{
192
 
193
   switch (c)
194
   {
195
      case '\f'   : lcd_send_byte(0,1);            // smaz displej
196
                    delay_ms(2);
197
                                            break;
198
      case '\n'   : lcd_gotoxy(1,2);        break; // presun se na 1. znak 2. radky
199
      case '\r'   : lcd_gotoxy(1,1);        break; // presun home
200
      case '\b'   : lcd_send_byte(0,0x10);  break; // posun kurzor o 1 zpet
201
      default     : if (c<0x20) c&=0x7;            // preklopeni definovatelnych znaku na rozsah 0 az 0x1F
202
                    lcd_send_byte(1,c);     break; // zapis znak
203
   }
204
}
205
 
206
 
207
// Zapni kurzor
208
//
209
void lcd_cursor_on()
210
{
211
   lcd_send_byte(0,LCD_CURSOR_ON_);
212
}
213
 
214
 
215
// Vypni kurzor
216
//
217
void lcd_cursor_off()
218
{
219
   lcd_send_byte(0,LCD_CURSOR_OFF_);
220
}
221
 
222
 
223
// Smaz displej
224
//
225
void lcd_clr()
226
{
227
   lcd_putc('\f');
228
}
229
 
230
 
231
// Definice vlastnich fontu
232
//
233
// Vlastnich definic muze byt jen 8 do pozic 0 az 7 pameti CGRAM radice lcd displeje
234
// Pro snadne definovani jsou pripraveny nasledujici definice a na konci souboru je uveden
235
// priklad pouziti definovanych znaku.
236
 
237
 
238
// Pomocna procedura pro posilani ridicich dat do radice displeje
239
//
240
void lcd_putc2(int Data)
241
{
242
   lcd_send_byte(1,Data);
243
}
244
 
245
 
246
// Pomocne definice pro programovani obsahu CGRAM
247
//
248
#define lcd_define_start(Code)      lcd_send_byte(0,0x40+(Code<<3)); delay_ms(2)
249
#define lcd_define_def(String)      printf(lcd_putc2,String);
250
#define lcd_define_end()            lcd_send_byte(0,3); delay_ms(2)
251
 
252
 
253
// Vlastni vykonne makro pro definovani fontu do pozice Index CGRAM s definicnim retezcem Def
254
//
255
#define lcd_define_char(Index, Def) lcd_define_start(Index); lcd_define_def(Def); lcd_define_end();
256
 
257
 
258
// Pripravene definice fontu vybranych znaku
259
// V tabulce nesmi byt 00 (konec retezce v printf()), misto toho davame 80
260
//
261
#define LCD_CHAR_BAT100 "\x0E\x1F\x1F\x1F\x1F\x1F\x1F\x1F"      /* symbol plne baterie       */
262
#define LCD_CHAR_BAT50  "\x0E\x1F\x11\x11\x13\x17\x1F\x1F"      /* symbol polovicni baterie  */
263
#define LCD_CHAR_BAT0   "\x0E\x1F\x11\x11\x11\x11\x11\x1F"      /* symbol vybite baterie     */
264
#define LCD_CHAR_UP     "\x80\x04\x0E\x15\x04\x04\x04\x80"      /* symbol sipka nahoru       */
265
#define LCD_CHAR_DOWN   "\x80\x04\x04\x04\x15\x0E\x04\x80"      /* symbol Sipka dolu         */
266
#define LCD_CHAR_LUA    "\x04\x0E\x11\x11\x1F\x11\x11\x80"      /* A s carkou                */
267
#define LCD_CHAR_LLA    "\x01\x02\x0E\x01\x1F\x11\x0F\x80"      /* a s carkou                */
268
#define LCD_CHAR_HUC    "\x0A\x0E\x11\x10\x10\x11\x0E\x80"      /* C s hackem                */
269
#define LCD_CHAR_HLC    "\x0A\x04\x0E\x10\x10\x11\x0E\x80"      /* c s hackem                */
270
#define LCD_CHAR_HUD    "\x0A\x1C\x12\x11\x11\x12\x1C\x80"      /* D s hackem                */
271
#define LCD_CHAR_HLD    "\x05\x03\x0D\x13\x11\x11\x0F\x80"      /* d s hackem                */
272
#define LCD_CHAR_LUE    "\x04\x1F\x10\x10\x1E\x10\x1F\x80"      /* E s carkou                */
273
#define LCD_CHAR_LLE    "\x01\x02\x0E\x11\x1F\x10\x0E\x80"      /* e s carkou                */
274
#define LCD_CHAR_HUE    "\x0A\x1F\x10\x1E\x10\x10\x1F\x80"      /* E s hackem                */
275
#define LCD_CHAR_HLE    "\x0A\x04\x0E\x11\x1F\x10\x0E\x80"      /* e s hackem                */
276
#define LCD_CHAR_LUI    "\x04\x0E\x04\x04\x04\x04\x0E\x80"      /* I s carkou                */
277
#define LCD_CHAR_LLI    "\x02\x04\x80\x0C\x04\x04\x0E\x80"      /* i s carkou                */
278
#define LCD_CHAR_HUN    "\x0A\x15\x11\x19\x15\x13\x11\x80"      /* N s hackem                */
279
#define LCD_CHAR_HLN    "\x0A\x04\x16\x19\x11\x11\x11\x80"      /* n s hackem                */
280
#define LCD_CHAR_LUO    "\x04\x0E\x11\x11\x11\x11\x0E\x80"      /* O s carkou                */
281
#define LCD_CHAR_LLO    "\x02\x04\x0E\x11\x11\x11\x0E\x80"      /* o s carkou                */
282
#define LCD_CHAR_HUR    "\x0A\x1E\x11\x1E\x14\x12\x11\x80"      /* R s hackem                */
283
#define LCD_CHAR_HLR    "\x0A\x04\x16\x19\x10\x10\x10\x80"      /* r s hackem                */
284
#define LCD_CHAR_HUS    "\x0A\x0F\x10\x0E\x01\x01\x1E\x80"      /* S s hackem                */
285
#define LCD_CHAR_HLS    "\x0A\x04\x0E\x10\x0E\x01\x1E\x80"      /* s s hackem                */
286
#define LCD_CHAR_HUT    "\x0A\x1F\x04\x04\x04\x04\x04\x80"      /* T s hackem                */
287
#define LCD_CHAR_HLT    "\x0A\x0C\x1C\x08\x08\x09\x06\x80"      /* t s hackem                */
288
#define LCD_CHAR_LUU    "\x02\x15\x11\x11\x11\x11\x0E\x80"      /* U s carkou                */
289
#define LCD_CHAR_LLU    "\x02\x04\x11\x11\x11\x13\x0D\x80"      /* u s carkou                */
290
#define LCD_CHAR_CUU    "\x06\x17\x11\x11\x11\x11\x0E\x80"      /* U s krouzkem              */
291
#define LCD_CHAR_CLU    "\x06\x06\x11\x11\x11\x11\x0E\x80"      /* u s krouzkem              */
292
#define LCD_CHAR_LUY    "\x02\x15\x11\x0A\x04\x04\x04\x80"      /* Y s carkou                */
293
#define LCD_CHAR_LLY    "\x02\x04\x11\x11\x0F\x01\x0E\x80"      /* y s carkou                */
294
#define LCD_CHAR_HUZ    "\x0A\x1F\x01\x02\x04\x08\x1F\x80"      /* Z s hackem                */
295
#define LCD_CHAR_HLZ    "\x0A\x04\x1F\x02\x04\x08\x1F\x80"      /* z s hackem                */
296
 
297
 
298
// Priklad pouziti definovanych znaku
299
//
300
//
301
//void lcd_sample()
302
//{
303
//   lcd_define_char(0,LCD_CHAR_BAT50);                 // Priklad definice znaku baterie do pozice 0
304
//   lcd_define_char(2,LCD_CHAR_HLE LCD_CHAR_LUI);      // Priklad definice znaku e s hackem a I s carkou od pozice 2
305
//                                                      // vsimnete si, ze neni carka mezi retezci s definici (oba retezce definuji
306
//                                                      // jediny definicni retezec)
307
//   printf(lcd_putc,"\fZnaky:\20\22\23");              // priklad vypisu znaku z pozice 0, 2 a 3
308
//   delay_ms(1000);
309
//   lcd_define_char(0,LCD_CHAR_BAT0);                  // Predefinovani tvaru znaku v pozici 0
310
//   delay_ms(1000);
311
//}