Графический ЖКИ WG12864A и микроконтроллер ATmega8

Первый раз графические ЖКИ я увидел, когда мне было лет 12, в каталоге фирмы Bolumin. Мне тогда сильно захотелось научится управлять ими. Спустя лет 7 я написал программу на ассемблере, которая выводила точки и буквы на графическом ЖКИ. Позже, когда начал писать на С, потребовалось заново написать функции работы с графическим ЖКИ, такие как рисование точки (X,Y), линии, прямоугольника, вывод символов. Этому и посвящаю данную заметку.

Рассмотрим пациента повнимательней. Графический ЖКИ WG12864A имеет разрешение 128 на 64 точки. Управляет дисплеем 2 контроллера KS0108 (один контроллер KS0108 может управлять только областью 64х64). То есть если мы хотим вывести точку (20,10), то необходимо включить первый контроллер и вывести точку (20,10), а если мы хотим вывести точку (120,10), то необходимо включить второй контроллер, и вывести точку (120-64=56,10). Т.е. адресация точек у контроллеров независимая, от 0 до 63 в каждом. А для точек с Х-вой координатой больше 63 требуется отнимать 64, и передавать второму контроллеру. Теперь рассмотрим каждый контроллер по отдельности. KS0108 выводит на дисплей все, что записано в области отображения. Договоримся производить нумерацию пикселов с левого верхнего угла дисплея, начиная с 0. Каждый пиксел соответствует некоторому биту в памяти дисплея:
адресация памяти дисплея
Вся память разбита на байты, которые в свою очередь сгруппированы в страницы (по координате У). Например, чтобы засветить пиксел (6,14) нам потребуется выставить адрес Х=6, выставить адрес У=1= [14/8]) и выставить 14%8=6-й бит в данной ячейке памяти ([] – операция взятия целой части числа, % - остача от деления). Для этого необходимо сначала считать эту ячейку памяти (если этого не делать, мы теряем все данные, которые были там до этого выведены), выполнить выставить нужный бит в считанном байте и записать этот байт обратно. Рисование линий же будем производить с помощью алгоритма Брезенхэма адаптированного под WinAVR. Рисование же прямоугольника будем производить с помощью 4-х функций рисования линий. Т.е. для рисования графики краеугольным камнем является рисование точки. Рисование же текста будем производить побайтно, чтобы сэкономить время. Для этого был вручную «нарисован» шрифт 5х8 пикселов на букву в файле symbol.h. Если будет не хватать символов – их просто самому дорисовать.
Теперь посмотрим на интерфейс WG12864A:
распиновка дисплея WG12864A
Выводы 1,2 – питание, а выводы 19,20 – питание светодиодной подсветки, вывод 3 – настройка контраста. Чтобы передать данные или команду их следует вывести на шину данных DB0-DB7. И сделать короткий импульс продолжительностью не мене 450нс на выводе Е, по спаду сигнала данные будут записаны. Вывод D/I определяет, передается ли команда или данные, CS1, CS2 определяют с какому контроллеру предназначены команда/данные (можно передавать сразу обоим). R/W – определяет производится чтение или запись даны. Считывать данные можно после 320нс после фронта импульса на вывод Е. Вывод Reset инвертирован (активное состояние 0) и предназначен для перезагрузки дисплея. Схема включения WG12864A очень похожа на схему включения символьных ЖКИ:

Источник тактирования - внутренний RC генератор на 1МГц. Резистор R1 – 10 кОм, С7 – 0,1мкФ. Светодиод D1 сигнализирует, что на схему подается питание, R2 – 1кОм. Резистор R3 ограничительный, для подсветки ЖКИ, 17 Ом, VR1 – переменный резистор на 10кОм, для регулировки контраста.
Теперь посмотрим на доступные нам команды:
команды дисплея WG12864A
Здесь все просто. Включаем дисплей командой 0x3f, устанавливаем начальный адрес отображения с помощью команды 0хс0, устанавливаем курсор в нулевой положения по Х и У координате командами 0х40, 0xb8 соответственно. После чего дисплей готов к работе.
На основе вышесказанного напишем демонстрационную программу, в которой будут рисоваться линии, точки, прямоугольники и выводиться текст:

  1. #include <avr/io.h>
  2. #include "symbol.h"
  3.  
  4. #define DI 0
  5. #define RW 1
  6. #define E 2
  7. #define CS1 3
  8. #define CS2 4
  9. #define RST 5
  10.  
  11. void pause (unsigned int a)
  12. { unsigned int i;
  13.  
  14. for (i=a;i>0;i--)
  15. ;
  16. }
  17.  
  18. void lcd_dat (unsigned char data)
  19. {
  20. PORTD=data; //Выводим данные
  21. PORTB|=_BV(DI); //Говорим дисплею что передаются данные
  22. asm("nop");
  23. PORTB|= _BV(E); //Импульс записи
  24. asm("nop");
  25. PORTB&=~_BV(E);
  26. PORTB&=~_BV(DI);
  27. }
  28.  
  29. void lcd_com (unsigned char comm)
  30. {
  31. PORTD=comm; //Выводим команду
  32. asm("nop");
  33. PORTB|= _BV(E); // Импульс записи
  34. asm("nop");
  35. PORTB&=~ _BV(E);
  36. }
  37.  
  38. void gotoxy (unsigned char x, unsigned char y)
  39. {
  40. if (x<64) PORTB|=_BV(CS1); //Если x<64 включаем CS1
  41. else {
  42. PORTB|=_BV(CS2); //else включаем CS2
  43. x=x-64;
  44. }
  45. lcd_com(0x40+x); //Выставляем X координату
  46. lcd_com(0xb8+y/8); //Выставляем Y координату
  47. }
  48.  
  49. void clear_screen (void)
  50. { unsigned char i,j;
  51.  
  52. PORTB|=_BV(CS1)|_BV(CS2); //Включаем оба чипа
  53. for (j=0;j<8;j++)
  54. {
  55. lcd_com(0xb8+j);
  56. for (i=0;i<64;i++)
  57. lcd_dat(0x00); //Очищаем
  58. }
  59. }
  60.  
  61. void wg12864_init(void)
  62. {
  63. DDRD=0xff; //PD0-7 как выхода
  64. PORTD=0x00;
  65. PORTB=_BV(RST)|_BV(CS1)|_BV(CS2); //Включаем оба чипа
  66. DDRB=_BV(DI)|_BV(RW)|_BV(E)|_BV(CS1)|_BV(CS2)|_BV(RST); //PB0-5 как выхода
  67. asm("nop");
  68. lcd_com(0x3f); //Разрешаем отображение
  69. lcd_com(0xc0); //Устанавливаем начало отображения
  70. lcd_com(0x40); //Ставим X=0
  71. lcd_com(0xb8); //Ставим Y=0
  72. clear_screen();
  73.  
  74. PORTB&=~(_BV(CS1)|_BV(CS2));
  75. }
  76.  
  77. void put_pixel (const unsigned char x, const unsigned char y, const unsigned char color)
  78. { unsigned char temp=0;
  79.  
  80. PORTB=_BV(RST);
  81. if ( (x>128)||(y>64) ) return;
  82.  
  83. gotoxy (x,y);
  84.  
  85. PORTD=0xff;
  86. DDRD=0x00; //PD0-7 как входа
  87. PORTB|=(_BV(RW)|_BV(DI)); //Считываем данные
  88. pause(1);
  89.  
  90. PORTB|= _BV(E);
  91. asm("nop");
  92. PORTB&=~_BV(E);
  93. asm("nop");
  94. PORTB|= _BV(E);
  95. asm("nop");
  96. temp=PIND;
  97. PORTB&=~_BV(E);
  98.  
  99. if (color==1) temp|= _BV(y%8); //Выставляем/снимаем нужный нам бит
  100. else temp&=~_BV(y%8);
  101.  
  102. PORTB&=~(_BV(RW)|_BV(DI));
  103. DDRD=0xff; //PD0-7 как выхода
  104. asm("nop");
  105. gotoxy(x,y); //Переходим к X,Y
  106. lcd_dat(temp); //Записываем данные
  107. PORTB=_BV(RST);
  108. }
  109.  
  110. int sign (int x)
  111. {
  112. if (x<0) return -1;
  113. if (x>0) return 1;
  114. return 0;
  115. }
  116.  
  117. int abs (int x)
  118. {
  119. if (x<0) return -x;
  120. else return x;
  121. }
  122.  
  123. void line (unsigned char x1,unsigned char y1,unsigned char x2,
  124. unsigned char y2,unsigned char color)
  125. { int dx,dy,i,sx,sy,check,e,x,y;
  126.  
  127. dx=abs(x1-x2);
  128. dy=abs(y1-y2);
  129. sx=sign(x2-x1);
  130. sy=sign(y2-y1);
  131. x=x1;
  132. y=y1;
  133. check=0;
  134. if (dy>dx) {
  135. dx=dx+dy;
  136. dy=dx-dy;
  137. dx=dx-dy;
  138. check=1;
  139. }
  140. e=2*dy - dx;
  141. for (i=0;i<=dx;i++)
  142. {
  143. put_pixel(x,y,color);
  144. if (e>=0) {
  145. if (check==1) x=x+sx;
  146. else y=y+sy;
  147. e=e-2*dx;
  148. }
  149. if (check==1) y=y+sy;
  150. else x=x+sx;
  151. e=e+2*dy;
  152. }
  153. }
  154.  
  155. void rectangle (unsigned char x1,unsigned char y1,unsigned char x2,
  156. unsigned char y2,unsigned char color)
  157. {
  158. line (x1,y1,x2,y1,color);
  159. line (x2,y1,x2,y2,color);
  160. line (x2,y2,x1,y2,color);
  161. line (x1,y2,x1,y1,color);
  162. }
  163.  
  164. void put_char (unsigned char x0, unsigned char y0, unsigned char code, unsigned char mode)
  165. { unsigned char i,x,y;
  166. x=x0;
  167. y=y0;
  168. PORTB=_BV(RST)|_BV(CS1);
  169. if ( (x>128)||(y>64) ) return;
  170. for (i=0;i<6;i++)
  171. {
  172. if (x>=64) { //Включить CS2
  173. x=x-64;
  174. PORTB|=_BV(CS2);
  175. PORTB&=~_BV(CS1);
  176. }
  177. lcd_com(0x40+x); //Перейти X,Y
  178. lcd_com(0xb8+y);
  179.  
  180. if (i<=4) { if (mode==1) lcd_dat(~sym[code][i]); //Пишем колонку
  181. else lcd_dat (sym[code][i]); //Пишем инвертированную колонку
  182. }
  183. if (i==5) { if (mode==1) lcd_dat(0xff); //Рисуем разделитель
  184. else lcd_dat(0x00); //Рисуем инвертированный разделитель
  185. }
  186. x=x+1;
  187. }
  188.  
  189. }
  190.  
  191. int main(void)
  192. { unsigned char data[10]={0x24,0x39,0x35,0x2f,0x24,0x25,0x86,0x26,0x32,0x30}; //"avrlab.com"
  193. unsigned char data2[8]={0x20,0x10,0x01,0x02,0x08,0x06,0x04,0x0a}; //"WG12864A"
  194. unsigned char i;
  195.  
  196. pause(5000); //Задержка для включения ЖКИ
  197. wg12864_init(); //Инициализация ЖКИ
  198.  
  199. rectangle(0,0,127,63,1); //Рисуем контур
  200. line (127,0,0,63,1); //Рисуем линии
  201. line (127,5,5,63,1);
  202. line (127,10,10,63,1);
  203. line (127,15,15,63,1);
  204. line (127,20,20,63,1);
  205. line (127,25,25,63,1);
  206. line (127,30,30,63,1);
  207. line (127,35,35,63,1);
  208. line (127,40,40,63,1);
  209. line (127,45,45,63,1);
  210. line (127,50,50,63,1);
  211. line (127,55,55,63,1);
  212. line (127,60,60,63,1);
  213.  
  214. for (i=0;i<10;i++) //Пишем "avrlab.com"
  215. put_char (20+6*i,1,data[i],1);
  216.  
  217. for (i=0;i<8;i++)
  218. put_char (10+6*i,3,data2[i],0); //Пишем "WG12864A"
  219.  
  220. while (1)
  221. ;
  222.  
  223. return 1;
  224. }

Программа в принципе проста. Сервисные функции lcd_dat(unsigned char data), lcd_com(unsigned char data) служат для выведения на ЖКИ данных и команды соответственно. Функция gotoxy(unsigned char x, unsigned char y) устанавливает курсор в положение Х,У (У – страница памяти, где находиться наш пиксел). сlear_scrren(void) очищает экран, а wg12864_init(void) – инициализирует дисплей. Точка выводиться как мы и писали выше, сперва считываются байт в который будут вноситься изменения, производиться изменения нужного бита и записывается обратно. В процессе написания этой функции я столкнулся с проблемой считать данные с ЖКИ, проблема решился тем, что для считывания 2 раза вывожу импульс на Е, а во время второго импульса считываю данные (делал разные задержки и разную длительность импульса на Е, но считать данные так и не получилось, так что оставил рабочую версию). Функции sign(x), abs(x) – написаны для реализации алгоритма рисования линии Брезенхэма. Рисование же текста функцией put_char (unsigned char x, unsigned char y, unsigned char code, unsigned char mode) проивзодиться побайтным вбитием в RAM дисплея буквы, которая соответствует параметру code в файле шрифта symbol.h Также в функциях рисования точки, линии, прямоугольника и буквы я сделал параметр color (mode), который отвечает, будет ли рисоваться объект черным на белом фоне или наоборот, белым на черном фоне. На этом думаю можно завершить описание программы.
А вот как выглядит работа нашего макета:
фото работающего дисплея WG12864A

Скачать прошивку в виде проекта под AVR studio 4.