Так случилось, что прикупил я тут себе поприколу LCD дисплейчик две строки по восемь символов. Валялся он в ящике валялся, да чегото поперло меня и решил я его заюзать, попутно вкурив в его работу. О том как подключить к AVR LCD дисплей я вам сейчас и поведаю.
Для начала оговорюсь сразу, что речь тут пойдет о LCD индикаторах на контроллере HD44780, который стал промышленным стандартом де-факто на рынке цифро-буквенных дисплеев. Продается везде где только можно, стоит недорого (8х2 мне обошелся порядка 150 рублей), а также под него написана куча кода. Я же, как обычно, решил изобрести велосипед и сварганить свою собственную тру-библиотеку для работы с этим типом индикаторов. Разумеется на ассемблере, а на чем же еще? ;)
Подключение.
LCD
на базе HD44780
подключается к AVR
микроконтроллеру напрямую к портам. Есть два способа подключения — на 8 бит и на 4 бита. В восьмибитном режиме немножко проще закидывать байты — не нужно сдвигать байт, зато в четырех битном резко нужно тратить на целых четыре ножки контроллера меньше. Есть еще одна особенность работы в 8-битном режиме — к некоторым контроллерам можно подрубить этот дисплей как внешнее ОЗУ
и засылать данные простыми командами пересылки. Лично я подключил его в режиме полного порта у меня один фиг выводы уже девать некуда было, так что не жалко.
- Выводы DB7…DB0 это шина данных/адреса.
- E — стробирующий вход. Дрыгом напряжения на этой линии мы даем понять дисплею что нужно забирать/отдавать данные с/на шину данных.
- RW — определяет в каком направлении у нас движутся данные. Если 1 — то на чтение из дисплея, если 0 то на запись в дисплей.
- RS — определяет что у нас передается, команда (RS=0) или данные (RS=1). Данные будут записаны в память по текущему адресу, а команда исполнена контроллером.
Со стороны питания все еще проще:
- GND — минус, он же общий.
- Vcc — плюс питания, обычно 5V
- V0 — вход контрастности. Сюда нужно подавать напряжение от нуля до напряжения питания, тем самым задается контрастность изображения. Можно поставить переменный резистор, включенный потенциометром и крутить в свое удовольствие. Главное поймать значение максимального контраста, но чтобы не было видно знакомест (серый ореол из квадратов вокруг символа). Если же выставить слишком малый контраст, то символы будут переключаться лениво и задумчиво. Примерно как в калькуляторе у которого сели батарейки.
- А — это вход Анода светодиодной подсветки. Короче плюс.
- К — соответственно Катод, он же минус. Подсветка хавает примерно 100мА и поэтому нужно выставить туда токоограничительный резистор на 100 Ом. Кстати, многие ЖК дисплеи имеют на плате пятачки для припайки резисторов. Если прозвонить, то можно убедиться в том, что эти линии ведут на входы питания LCD, поэтому, впаяв резисторы, можно не заморачиваться на запитку подстветки, она будет подключена к питанию контроллера.
Контроллер имеет свой блок управления, который обрабатывает команды и память. Она делится на три вида:
DDRAM — память дисплея. Все что запишется в DDRAM будет выведено на экран. То есть, например, записали мы туда код 0x31 — на экране выскочит символ «1» т.к. 0х31 это ASCII код цифры 1. Но есть тут одна особенность — DDRAM память гораздо больше чем видимая область экрана. Как правило, DDRAM содержит 80 ячеек — 40 в первой строке и 40 во второй, а на дисплей может двигаться по этой линейке как окошко на логарифмической линейке, высвечивая видимую область. То есть, например, можно засунуть в DDRAM сразу пять пунктов меню, а потом просто гонять дисплей туда сюда, показывая по одному пункту. Для перемещения дисплея есть спец команда. Также есть понятие курсора — это место в которое будет записан следующий символ, т.е. текущее значение счетчика адреса. Курсор не обязательно может быть на экране, он может располагаться и за экраном или быть отключен вовсе.
CGROM — таблица символов. Когда мы записываем в ячейку DDRAM байт, то из таблицы берется символ и рисуется на экране. CGROM нельзя изменить, поэтому важно, чтобы она имела на борту русские буквы. Если, конечно, планируется русскоязычный интерфейс.
CGRAM — тоже таблица символов, но ее мы можем менять, создавая свои символы. Адресуется она линейно, то есть вначале идет 8 байт одного символа, построчно, снизу вверх — один бит равен одной точке на экране. Потом второй символ тем же макаром. Поскольку знакоместо у нас 5 на 8 точек, то старшие три бита роли не играют . Всего в CGRAM может быть 8 символов, соответственно CGRAM имеет 64 байта памяти. Эти программируемые символы имеют коды от 0х00 до 0х07. Так что, закинув, например, в первые 8 байт CGRAM (первый символ с кодом 00) какую нибудь фигню, и записав в DDRAM нуль (код первого символа в CGRAM ) мы увидим на экране нашу хрень.
Доступ к памяти.
Тут все просто. Мы командой выбираем в какую именно память и начиная с какого адреса будем писать. А потом просто шлем байты. Если указано, что записываем в DDRAM то на экран (или в скрытую область) полезут символы, если в CGRAM то байты полезут уже в память знакогенератора. Главное потом не забыть переключится обратно на область DDRAM
Система команд.
Система команд проста как мычание. О том, что передается команда контроллеру дисплея сообщит нога RS
=0. Сама команда состоит из старшего бита, определяющего за что отвечает данная команда и битов параметров, указывающих контроллеру HD44780 как дальше жить.
Таблица команд:
DB7 | DB6 | DB5 | DB4 | DB3 | DB2 | DB1 | DB0 | Значение |
0 | 0 | 0 | 0 | 0 | 0 | 0 | 1 | Очистка экрана. Счетчик адреса на 0 позицию DDRAM |
0 | 0 | 0 | 0 | 0 | 0 | 1 | — | Адресация на DDRAM сброс сдвигов, Счетчик адреса на 0 |
0 | 0 | 0 | 0 | 0 | 1 | I/D | S | Настройка сдвига экрана и курсора |
0 | 0 | 0 | 0 | 1 | D | C | B | Настройка режима отображения |
0 | 0 | 0 | 1 | S/C | R/L | — | — | Сдвиг курсора или экрана, в зависимости от битов |
0 | 0 | 1 | DL | N | F | — | — | Выбор числа линий, ширины шины и размера символа |
0 | 1 | AG | AG | AG | AG | AG | AG | Переключить адресацию на SGRAM и задать адрес в SGRAM |
1 | AD | AD | AD | AD | AD | AD | AD | Переключить адресацию на DDRAM и задать адрес в DDRAM |
Теперь поясню что значат отдельные биты:
- I/D — инкремент или декремент счетчика адреса. По дефолту стоит 0 — Декремент. Т.е. каждый следующий байт будет записан в n-1 ячейку. Если поставить 1 — будет Инкремент.
- S — сдвиг экрана, если поставить 1 то с каждым новым символом будет сдвигаться окно экрана, пока не достигнет конца DDRAM, наверное удобно будет когда выводишь на экран здоровенную строку, на все 40 символов, чтобы не убегала за экран.
- D — включить дисплей. Если поставить туда 0 то изображение исчезнет, а мы в это время можем в видеопамяти творить всякие непотребства и они не будут мозолить глаза. А чтобы картинка появилась в эту позицию надо записать 1.
- С — включить курсор в виде прочерка. Все просто, записали сюда 1 — включился курсор.
- B — сделать курсор в виде мигающего черного квадрата.
- S/C сдвиг курсора или экрана. Если стоит 0, то сдвигается курсор. Если 1, то экран. По одному разу за команду
- R/L — определяет направление сдвига курсора и экрана. 0 — влево, 1 — вправо.
- D/L — бит определяющий ширину шины данных. 1-8 бит, 0-4 бита
- N — число строк. 0 — одна строка, 1 — две строки.
- F — размер символа 0 — 5х8 точек. 1 — 5х10 точек (встречается крайне редко)
- AG — адрес в памяти CGRAM
- АD — адрес в памяти DDRAM
Я сам долго тупил в эту табличку, пытаясь понять, что же от меня хотят. Видимо был невыспавшийся, но и вправду, она на первый взгляд не очевидна, поэтому подкреплю все примером.
Задача:
- Включить дисплей.
- Очистить содержимое.
- Сдвинуть курсор на одну позицию.
- И записать туда «1».
Первым делом Инициализация
дисплея без которой большая часть дисплеев на HD44780 просто откажется работать. Некоторые виды имеют дефолтные состояние (шина 8 бит, курсор в 0) и им только дисплей включить. Но все же ее лучше сделать, мало ли что там намудрил разработчик. Лишней не будет.
- 001 11000 Шина 8 бит, 2 строки
- 00000001 Очистка экрана
- 000001 10 Инкремент адреса. Экран не движется
- 00001 100 Включили дисплей (D=1)
- 00000001 Очистили дисплей. Указатель встал на DDRAM
- 0001 0100 Сдвинули курсор (S/C=0) вправо (R/L=1)
- 00110001 — это мы уже записали данные (ножка RS=1) код «1» 0х31
Задача: создать свой символ. С кодом 01 и вывести его на экран.
Считаем, что дисплей у нас уже инициализирован и готов к приему данных.
Решение:
- 01 001000 Выбираем в CGRAM адрес 0х08 — как раз начало второго символа (напомню, что на один символ уходит 8 байт)
- 00000001 Это пошли 8 байт данных. (RS=1 )
- 0000001 0 Рисуем значок молнии, ну или
- 000001 00 ССовскую Зиг руну, кому как
- 00001 000 больше нравится.
- 00011111 Старшие три бита не действуют
- 0000001 0 Туда можно писать что угодно, на
- 000001 00 результат влиять не будет.
- 00001 000 Последний байт данных
- 1 0000000 А это уже команда — переключение адреса на DDRAM и указатель на адрес 0000000 — первый символ в первой строке.
- 00000001 И снова данные (RS=1 ), код 01 — именно в него мы засунули нашу молнию.
Так, с логикой разобрались, пора вкуривать в физику протокола общения. Код я приведу несколько позже, когда вылижу свою библиотеку и заоптимизирую до состояния идеала. Пока же дам алгоритм, а его уж на любом языке программирования реализовать можно. Хоть на ассемблере, хоть на Сях, да хоть на Васике:)
Алгоритм чтения/записи в LCD контроллер HD44780
Направление, а также команда/данные определяются ножками, а чтение и запись осуществляется по переходу строба (вывод Е) из 1 в 0
Инициализация портов
- RS, RW, E — в режим выхода.
- DB7..DB0 в режим входа. Впрочем, можно их не трогать, дальше переопределим.
- RS=0 (команда)
- RW=1 (чтение)
- E=1 (Готовьсь!!!)
- Пауза (14 тактов процессора на 8МГЦ хватало)
- Е=0 (Пли!)
- Читаем из порта. Если бит 7 (Busy flag) установлен, то повторяем все заново, пока не сбросится.
- Ожидание готовности
- RS=0 (команда)
- RW=0 (запись)
- Е=1 (Готовьсь!!!)
- Порт на выход
- Вывести в порт код команды
- Пауза
- Е=0 (Пли!)
- Орудие на плечо Порт на вход, на всякий случай.
- Ожидание готовности
- RS=1 (Данные)
- RW=0 (запись)
- Е=1 (Готовьсь!!!)
- Порт на выход
- Вывести в порт код команды
- Пауза
- Е=0 (Пли!)
- Порт на вход, на всякий случай.
- Ожидание готовности
- Порт данных на вход с подтяжкой (DDR=0, PORT=1)
- RS=0 (команда)
- RW=1 (чтение)
- Пауза
- Считываем данные с порта
- E=0 (Ать!)
- Ожидание готовности
- Порт данных на вход с подтяжкой (DDR=0, PORT=1)
- RS=1 (Данные)
- RW=1 (чтение)
- Е = 1 (Готовьсь! В этот момент данные из LCD вылазят на шину)
- Пауза
- Считываем данные с порта
- E=0 (Ать!)
С четырех разрядной шиной все точно также, только там каждая операция чтения/записи делается за два дрыга строба.
Запись:
- Пауза
- Выставили в порт старшую тетраду
- Пауза
- Пауза
- Выставили в порт младшую тетраду
- Пауза
- Читаем из порта старшую тетраду
- Пауза
- Пауза
- Читаем из порта младшую тетраду
Ждите код:) Скоро будет:)
UPD:
Для работы с символьными графическими дисплеями предлагаем воспользоваться библиотекой LiquidCrystal которая входит в стандартный набор Arduino IDE и предназначена для работы по 8-битному (4-битному) параллельному интерфейсу. Если Ваш дисплей подключается к Arduino по шине I2, то Вам нужно установить библиотеку LiquidCrystal_I2C (большинство функций которой повторяют функции первой библиотеки).
Поддерживаемые дисплеи:
Дисплей | Подключение и инициализация |
---|---|
LCD1602 - символьный дисплей (16x02 символов), |
#include [ , 8 , 9 , 10 , 11 ]); void setup(){ lcd.begin(16 , 2); } // Пояснение:
|
с интерфейсом I2C (синий) |
#include #include LiquidCrystal_I2C lcd(0x27 или 0x3F , 16 , 2); void setup(){ lcd.init(); } // Пояснение: |
LCD1602 I2C - символьный дисплей (16x02 символов), с интерфейсом I2C (зелёный) |
#include #include LiquidCrystal_I2C lcd(0x27 или 0x3F , 16 , 2); void setup(){ lcd.init(); } // Пояснение: |
LCD2004 - символьный дисплей (20x04 символов), с параллельным интерфейсом (синий) |
#include LiquidCrystal lcd(2 , 3 , 4 , 5 , 6 , 7 [ , 8 , 9 , 10 , 11 ]); void setup(){ lcd.begin(20 , 4); } // Пояснение: // Если используется 8 проводов шины данных, то указываем их все |
LCD2004 I2C - символьный дисплей (20x04 символов), с интерфейсом I2C (синий) |
#include #include LiquidCrystal_I2C lcd(0x27 или 0x3F , 20 , 4); void setup(){ lcd.init(); } // Пояснение: |
#1 Пример
Выводим надпись на дисплей LCD1602 подключённый по шине I2C. Для работы с дисплеем LCD2004 нужно изменить 3 строку на LiquidCrystal_I2C lcd(0x27,20,4);
#include
#2 Пример
Выводим надпись на дисплей LCD1602 подключённый по 4-битной параллельной шине. Для работы с дисплеем LCD2004 нужно изменить 5 строку на lcd.begin(20, 4);
#include
#3 Пример
Выводим надпись «Русский язык» на дисплей LCD1602 подключённый по шине I2C:
#include
Функции, общие для библиотек LiquidCrystal и LiquidCrystal_I2C:
- begin(cols,rows,); – Инициализация дисплея с указанием количества столбцов, строк и размера символа.
- clear(); – Очистка дисплея с установкой курсора в положение 0,0 (Занимает много времени!).
- home(); – Установка курсора в положение 0,0 (Занимает много времени!).
- display(); – Быстрое включение дисплея (без изменения данных в ОЗУ).
- noDisplay(); – Быстрое выключение дисплея (без изменения данных в ОЗУ).
- blink(); – Включение мигающего курсора (с частотой около 1 Гц).
- noBlink(); – Выключение мигающего курсора.
- cursor(); – Включение подчеркивания курсора.
- noCursor(); – Выключение подчеркивания курсора.
- scrollDisplayLeft(); – Прокрутка дисплея влево. Сдвиг координат дисплея на один столбец влево (без изменения ОЗУ).
- scrollDisplayRight(); – Прокрутка дисплея вправо. Сдвиг координат дисплея на один столбец вправо (без изменения ОЗУ).
- leftToRight(); – Указывает в дальнейшем сдвигать положение курсора, после вывода очередного символа, на один столбец вправо.
- rightToLeft(); – Указывает в дальнейшем сдвигать положение курсора, после вывода очередного символа, на один столбец влево.
- noAutoscroll(); – Указывает в дальнейшем выравнивать текст по левому краю от позиции курсора (как обычно).
- autoscroll(); – Указывает в дальнейшем выравнивать текст по правому краю от позиции курсора.
- createChar(num,array); – Запись пользовательского символа в CGRAM дисплея под указанным номером.
- setCursor(col,row); – Установка курсора в позицию указанную номером колонки и строки.
- print(text); – Вывод текста, символов или цифр на экран дисплея. Синтаксис схож с одноимённой функцией класса Serial.
Функции, реализованные только в библиотеке LiquidCrystal_I2C:
- init(); – Инициализация дисплея. Должна быть первой командой библиотеки LiquidCrystal_I2C после создания объекта. На самом деле данная функция есть и в библиотеке LiquidCrystal, но в той библиотеке она вызывается автоматически (по умолчанию) при создании объекта.
- backlight(); – Включение подсветки дисплея.
- noBacklight(); – Выключение подсветки дисплея.
- setBacklight(flag); – Управление подсветкой (true - включить / false - выключить), используется вместо функций noBacklight и backlight.
Подключение:
// Для шины I2C:
|
Параметр:
|
// Для параллельной шины из 4 проводов:
#include LiquidCrystal lcd( RS , E , D4 , D5 , D6 , D7 ); void setup(){ lcd.begin( col , row ); } |
Параметр:
|
// Для параллельной шины из 8 проводов:
#include LiquidCrystal lcd( RS , E , D0 , D1 , D2 , D3 , D4 , D5 , D6 , D7 ); void setup(){ lcd.begin( col , row ); } |
|
begin(col ,
row ,
);
Инициализация дисплея с указанием размеров экрана и символов. |
Параметр:
|
Функции управления дисплеем:
display();
Включает дисплей после того как он был выключен функцией noDisplay. |
Примечание: Функция выполняется быстро и без изменений в ОЗУ дисплея. |
noDisplay();
Выключает дисплей. Данные на дисплее не будут отображаться до вызова функции display, но и не сотрутся из памяти ОЗУ, а после вызова функции display, опять будут отображаться. |
Примечание: Функция выполняется быстро и без изменений в ОЗУ дисплея. |
scrollDisplayLeft();
Сдвигает координаты дисплея на один столбец влево. |
|
scrollDisplayRight();
Сдвигает координаты дисплея на один столбец вправо. Постоянный вызов данной функции создаст эффект бегущей строки. Координаты сдвигаются как для имеющейся на дисплее информации, так и для той, которая будет выведена после. |
Примечание: Функция выполняется без изменений ОЗУ дисплея. Если вызвать функцию 40 раз подряд, то координата вернётся в изначальную точку |
clear();
Очистка дисплея с установкой курсора в положение 0,0. Информация имеющаяся на дисплее безвозвратно сотрётся. |
Примечание: Занимает много времени. |
backlight();
Включение подсветки дисплея. |
|
noBacklight();
Выключение подсветки дисплея. |
Примечание: Функция реализована только в библиотеке LiquidCrystal_I2C. |
setBacklight(flag );
Управление подсветкой (вместо функций noBacklight и backlight). |
Параметр:
|
Функции управления курсором:
setCursor(col ,
row );
Установка курсора в указанную позицию. |
Параметр:
|
home();
Установка курсора в позицию 0,0. Работает как функция setCursor(0,0); |
Примечание: Занимает много времени. |
blink();
Включение мигающего курсора. |
Примечание: Курсор занимает всё поле символа и мигает с частотой около 1 Гц, в той позиции где он был установлен ранее. |
noBlink();
Выключение мигающего курсора. |
Примечание: Курсор становится невидим, но его позиция сохраняется. |
cursor();
Включение подчеркивания курсора. |
Примечание: Курсор принимает вид символа подчёркивания и находится в той позиции, где он был установлен ранее. |
noCursor();
Выключение подчеркивания курсора. |
Примечание: Курсор становится невидим, но его позиция сохраняется. |
Функции указывающие направление и выравнивание:
leftToRight();
Указывает, что после каждого нового символа, положение курсора должно сдвигаться на один столбец вправо. |
Примечание: Если вывести текст "abc" на дисплее отобразится "abc" и текст будет находиться правее от изначального положения курсора. (Как обычно) |
rightToLeft();
Указывает, что после каждого нового символа, положение курсора должно сдвигаться на один столбец влево. |
Примечание: Если вывести текст "abc" на дисплее отобразится "cba" и текст будет находиться левее от изначального положения курсора. (Письменность справа налево) |
noAutoscroll();
Указывает, что в дальнейшем, текст нужно выравнивать по левому краю от изначальной позиции курсора. |
Примечание: если установить курсор в позицию 10,0 и вывести текст, то в данной позиции будет находиться первый символ выведенного текста. (Как обычно) |
autoscroll();
Указывает, что в дальнейшем, текст нужно выравнивать по правому краю от изначальной позиции курсора. |
Примечание: если установить курсор в позицию 10,0 и вывести текст, то в данной позиции будет находиться курсор. (Координаты дисплея будут сдвинуты влево, как будто Вы вызвали функцию scrollDisplayLeft столько раз, сколько букв в выведенном тексте) |
Функции ввода текста и символов:
createChar(num,array);
Запись пользовательского символа в CGRAM дисплея под указанным номером. Если Вы хотите вывести текст (функцией print) в котором должен находиться установленный Вами символ, укажите слэш и номер под которым был записан этот символ: print("C\1MBO\2"). |
Параметр:
|
print(text);
Вывод текста, символов или цифр на экран дисплея. |
Параметр:
|
Сегодня все чаще для отображения простой символьной информации используют символьные жидкокристаллические индикаторы ЖКИ. О том, как с ними работать мы и поговорим. В этой части статьи, мы детально рассмотрим символьные ЖКИ на базе контроллера HITACHI HD44780 (или совместимым с ним SAMSUNG KS0066). Статья - попытка систематизировать найденную мной информацию при работе с этими ЖКИ.
Символьный ЖКИ с контроллером HD44780 (KS0066). Интерфейс
Символьный ЖКИ есть ничем другим, как матрицей из точек, разделенной на строки и поля символов:
Для управления этой матрицей и вывода собственно символов используется специальный контроллер.
HD44780 (и совместимый с ним KS0066) – есть де-факто стандартом на контроллеры монохромных жидкокристаллических знакосинтезирующих дисплеев с параллельным 4-х или 8-битным интерфейсом. На базе этого контроллера выпускается огромное количество моделей с различным конструктивном и разрешением, начиная с 8x1 (восемь символов в одной строке), и заканчивая 40x4 (содержащих два независимых управляющих чипа). Типичная частота работы контроллера – 270 кГц.
Контролер ЖКИ оперирует 3-мя блоками памяти:
1. Для вывода символа контроллер использует память DDRAM (Display Data RAM), где хранятся ASCII-коды символов, которые мы хотим видеть на ЖКИ. Под нее отведено 80 ячеек памяти. Понятно, что на ЖКИ мы увидим лишь часть символов, которые находятся в DDRAM - если наш ЖКИ 1 или 2-строчный и отображает 8 символов в строке, то так:
Рабочую область дисплея, как видно, можно смещать по ячейкам DDRAM (получается эффект бегущей строки).
2. Шаблоны самих символов контроллер берет из CGROM (Character Generator ROM) – памяти знакогенератора. Таблицу символов можно посмотреть в спецификации на HD44780.
3. Для хранения пользовательских символов (их шаблонов) предусмотрена память CGRAM (Character Generator RAM).
Также, контроллер в зависимости от некоторых условий распределяет пришедшие в него данные в регистр инструкций или регистр данных.
Типичный 14-пиновый интерфейс контроллеров HD44780 :
Земля, общий провод, GND |
|
Напряжение питания, Vcc (+5V) |
|
Настройка контрастности (Vo) |
|
Выбор регистра (R/S для HD44780, A0 для KS0066) |
|
Чтение/запись (R/W) |
|
Строб по спаду Е (Enable) |
|
Bit 0 (младший для 8мибитного интерфейса) |
|
Линия данных |
|
DB 4 (младший для 4х битного интерфейса) |
|
DB 7 (старший для 8-х (4х) битного интерфейса) |
|
Для дисплея с подсветкой |
|
Питание подсветки для дисплеев с подсветкой (анод) |
|
Питание подсветки для дисплеев с подсветкой (катод) |
Нумерацию пинов на конкретном ЖКИ смотрим в datasheet.
Контрастность изображения на ЖКИ можно изменять, подключив дополнительно построечный резистор на 10 кОм по схеме:
Но, следует смотреть в спецификацию на свой контроллер (например у ЖКИ Klsn10294v-0 на чипе KS0066 1-Vcc, а 2-GND). Подача питания подсветки может различаться от модели к модели в зависимости от её типа. Обычно подсветка питается от 5 вольт, токоограничительный резистор (50-100 Ом) обычно не обязателен.
Назначение выводов R/S, R/W, E:
При переходе Е с высокого лог. уровня на низкий данные, которые уже «висят» на выводах DB0..DB7, записываются в память контроллера ЖКИ для последующей обработки.
При высоком лог. уровне на R/S(Register Select) контроллер ЖКИ воспринимает этот набор битов как данные(код символа), а при низком – как инструкцию и направляет их в соответствующий регистр.
R/W определяет направление работы выводов DB0..DB7 – если на R/W «0», то мы можем только писать в порт DB, а если R/W = «1», то можем прочитать с него (например узнать занят контроллер или свободен для приема новых данных). Если мы не будем читать данные из ЖКИ, то можно «посадить» R/W на землю.
Набор команд HD44780
Для того чтобы начать выводить информацию на ЖКИ, его контроллер надо проинициализировать (сообщить ему об интерфейсе, шрифте, смещениях и т.д.). Контроллер может воспринимать всего 11 команд:
Название инструкции |
Состояние выводов |
Время исполнения f раб. =270 кГц |
||||||||||
Очистка Всего ЖКИ установка адреса DDRAM в 0 |
||||||||||||
Установка текущего адреса DDRAM в 0 (курсор – домой) Данные DDRAM не меняются |
||||||||||||
Установка направления движения курсора (I/D) и смещения дисплея (S) при выводе данных |
||||||||||||
Display on/off control |
Вкл/выкл. дисплей(D), курсор(C) и его мерцание(B) |
|||||||||||
Cursor or display shift |
Двигает курсор и смещает дисплей по DDRAM |
|||||||||||
Установка интерфейса(DL), число строк(N) и шрифт символов(F) |
||||||||||||
Set CGRAM address |
Установка счетчика адреса CGRAM. После этого можно записывать данные в CGRAM |
|||||||||||
Set DDRAM address |
Установка счетчика адреса DDRAM |
|||||||||||
Read busy flag & address |
Если BF = 1 то контроллер ЖКИ выполняет внутреннюю операцию (занят). АC6-АC0 – текущее значение адреса DDRAM |
|||||||||||
Write data to RAM |
Запись данных в RAM |
|||||||||||
Read data from RAM |
Чтение данных из RAM |
I/D = 1: адрес DDRAM увеличивается I/D = 0: уменьшается
S = 1: сдвиг рабочей области дисплея по DDRAM разрешен
D = 1: дисплей (изображение) включен
C = 1: курсор включен
В = 1: мерцание курсора включено
S/C = 1: сдвинуть дисплей S/C = 0: переместить курсор
R/L = 1: вправо R/L = 0: влево
DL = 1: 8 bit DL = 0: 4 bits
N = 1: 2 lines N = 0: 1 line
F = 1: 5х10 F = 0: 5х8
ACG: CGRAM address
ADD: DDRAM address (адрес курсора)
AC: Address counter DD и CGRAM адресов
Инициализация ЖКИ
Есть 2 способа инициализации контроллера ЖКИ:
1. Через внутреннюю схему сброса.
2. В ручном режиме (через посылку в него ряда команд, которыми мы задаем режим работы ЖКИ)
Внутренняя схема сброса контроллера начинает работать сразу после включения питания. В этом есть один минус – если питание у нас «ползет» до рабочего уровня медленно (медленнее, чем за 10 мс), то самоинициализация контроллера может пройти некорректно. При этом способе инициализации контроллер сам исполняет следующие команды:
1. Display clear
2. Function set:
DL = 1; 8-bit interface data
N = 0; 1-line display
F = 0; 5х8 dot character font
3. Display on/off control:
D = 0; Display off
C = 0; Cursor off
B = 0; Blinking off
4. Entry mode set:
I/D = 1; Increment by 1
S = 0; No shift
Второй способ исключает зависимость схемы от источника питания. Для инициализации контроллера ЖКИ в ручном режиме необходимо исполнить следующий алгоритм:
Как видно, здесь нет ничего сложного: посылаем в ЖКИ команду за командой, учитывая время их исполнения (около 40 мкс) или проверяя флаг занятости контроллера ЖКИ (тогда надо посадить пин RW на лапку микроконтроллера и выставлять его в «1», когда хотим узнать, занят ЖКИ или нет).
Вот, собственно, все, что касается теории работы с символьными ЖКИ. Если что-то пропустил или ошибся, читаем спецификацию на контроллер или .
Во второй части рассмотрим аппаратную и программную реализацию связи PIC-микроконтроллера с ЖКИ.
Для активизации четырехбитового режима надо программно сформировать сигналы управления согласно временным диаграммам на рис.1. По структуре они совпадают с диаграммой 8-ми разрядной шины за исключением удвоенного числа импульсов "Е". Линии связи проходят через старшие разряды шины данных DB4-DB7, младшие DB0-DB3 остаются не задействованными.
Рис.1
Достоинство режима - малое число проводников, упрощение топологии печатной платы, экономия линий портов МК. Недостаток - пониженная скорость передачи данных в ЖКИ, так как приходится информацию передавать двумя порциями (нибблами или тетрадами) по 4 бита в каждой. Однако, учитывая обязательные задержки времени в программе и физическую инерционность "жидких кристаллов", снижение скорости почти не чувствуется.
Принцип работы 4-х разрядной шины рассмотрим на примере тестовой программы для нашего LCD. На дисплей будут с секундными паузами выводиться цифры десятичного адреса знакоместа 0-255 и графические образы содержащихся в них символов.
Рис. 2
Как известно, каждый LCD имеет встроенный знакогенератор, представляющий собой область ПЗУ объемом более 8 Кб, которая прошивается на заводе-изготовителе. Традиционно первая половина ПЗУ с адресами 00-7Fh содержит начертания цифр, знаков препинания, а также заглавных и строчных букв латинского алфавита. Все как в IBM PC. Вторая половина "отдана на откуп" национальным алфавитам. В связи с этим HD44780 имеет модификации исполнения с тремя основными вариантами зашивки знакогенератора:
латиница и европейские языки (European standard font или Euro)
латиница и японские иероглифы (Japanese standard font или Japan)
латиница и кириллица (Custom font или Russian, рис.2)
Не все из ячеек знакогенератора заполнены. При обращении к "пустым" ячейкам на экране будет выведена произвольная информация, чаще всего состоящая из засвеченных точек. Первые 8 символов с адресами 0х00-0х07 отмечены "звездочкой". При желании они могут быть самостоятельно запрограммированы пользователем.
Какой знакогенератор имеется в конкретном LCD, должно быть указано в его условном обозначении или в технических параметрах, хотя на практике приходится верить честному слову продавца. Другой подход воочию увидеть на экране LCD все возможные начертания символов. Напишем, откомпилируем программу и запрограммируем контроллер, после чего в верхней строке экрана LCD будут с секундными паузами будут выводиться цифры десятичного адреса знакоместа 0-255 и графические образы содержащихся в них символов. Если графика и очередность появления символов соответствует рис.2, значит, LCD в порядке.
class="eliadunit"> |
Далее собираем схему согласно рис.3, ее отличие только в том что шина данных подключена по 4-х проводной линии, т.е. DB4-DB7 подключены, а DB0-DB3 остаются не задействованными. Вывод R/W дисплея подключен на минус, т.к. дисплей у нас является приемником данных.
Код программы проверки знакогенератора LCD приведен ниже.
// Программа проверки знакогенератора LCD
#include