2017-03-25 11:00:58 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2017 HAW Hamburg
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup drivers_hd44780
|
|
|
|
*
|
|
|
|
* @{
|
|
|
|
* @file
|
|
|
|
* @brief Driver for the HD44780 LCD
|
|
|
|
*
|
|
|
|
* @note The display is also known as LCM1602C from Arduino kits
|
|
|
|
*
|
|
|
|
* @author Sebastian Meiling <s@mlng.net>
|
2017-04-21 11:30:15 +02:00
|
|
|
*
|
|
|
|
* @}
|
2017-03-25 11:00:58 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "log.h"
|
2019-11-07 17:30:46 +01:00
|
|
|
#ifdef MODULE_PCF857X
|
|
|
|
#include "pcf857x.h"
|
|
|
|
#include "pcf857x_params.h"
|
|
|
|
#else
|
2017-03-25 11:00:58 +01:00
|
|
|
#include "periph/gpio.h"
|
2019-11-07 17:30:46 +01:00
|
|
|
#endif
|
2017-03-25 11:00:58 +01:00
|
|
|
#include "xtimer.h"
|
|
|
|
|
|
|
|
#include "hd44780.h"
|
|
|
|
#include "hd44780_internal.h"
|
|
|
|
|
2019-11-07 17:30:46 +01:00
|
|
|
#ifdef MODULE_PCF857X
|
|
|
|
static pcf857x_t _pcf857x_dev;
|
|
|
|
#endif
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
static inline void _command(const hd44780_t *dev, uint8_t value);
|
|
|
|
static void _pulse(const hd44780_t *dev);
|
2018-10-11 08:21:28 +02:00
|
|
|
static void _send(const hd44780_t *dev, uint8_t value, hd44780_state_t state);
|
2017-06-20 17:32:45 +02:00
|
|
|
static void _write_bits(const hd44780_t *dev, uint8_t bits, uint8_t value);
|
2017-03-25 11:00:58 +01:00
|
|
|
|
2019-11-07 17:30:46 +01:00
|
|
|
static inline void _gpio_set(gpio_t pin);
|
|
|
|
static inline void _gpio_clear(gpio_t pin);
|
|
|
|
static inline int _gpio_init(gpio_t pin, gpio_mode_t mode);
|
|
|
|
|
2017-03-25 11:00:58 +01:00
|
|
|
/**
|
|
|
|
* @brief Send a command to the display
|
|
|
|
*
|
|
|
|
* @param[in] dev device descriptor of display to initialize
|
|
|
|
* @param[in] value the command
|
|
|
|
*/
|
2017-06-20 17:32:45 +02:00
|
|
|
static inline void _command(const hd44780_t *dev, uint8_t value)
|
2017-03-25 11:00:58 +01:00
|
|
|
{
|
|
|
|
_send(dev, value, HD44780_OFF);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Send a pulse to the display to enable new state
|
|
|
|
*
|
|
|
|
* @param[in] dev device descriptor of display to initialize
|
|
|
|
*/
|
2017-06-20 17:32:45 +02:00
|
|
|
static void _pulse(const hd44780_t *dev)
|
2017-03-25 11:00:58 +01:00
|
|
|
{
|
2019-11-07 17:30:46 +01:00
|
|
|
_gpio_clear(dev->p.enable);
|
2017-03-25 11:00:58 +01:00
|
|
|
xtimer_usleep(HD44780_PULSE_WAIT_SHORT);
|
2019-11-07 17:30:46 +01:00
|
|
|
_gpio_set(dev->p.enable);
|
2017-03-25 11:00:58 +01:00
|
|
|
xtimer_usleep(HD44780_PULSE_WAIT_SHORT);
|
2019-11-07 17:30:46 +01:00
|
|
|
_gpio_clear(dev->p.enable);
|
2017-03-25 11:00:58 +01:00
|
|
|
xtimer_usleep(HD44780_PULSE_WAIT_LONG);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Send a data value to the display
|
|
|
|
*
|
|
|
|
* @param[in] dev device descriptor of display to initialize
|
|
|
|
* @param[in] value the data value, either char or command
|
|
|
|
* @param[in] state send state, to distinguish chars and commands
|
|
|
|
*/
|
2017-06-20 17:32:45 +02:00
|
|
|
static void _send(const hd44780_t *dev, uint8_t value, hd44780_state_t state)
|
2017-03-25 11:00:58 +01:00
|
|
|
{
|
2019-11-07 17:30:46 +01:00
|
|
|
(state == HD44780_ON) ? _gpio_set(dev->p.rs) : _gpio_clear(dev->p.rs);
|
2017-03-25 11:00:58 +01:00
|
|
|
/* if RW pin is available, set it to LOW */
|
2020-01-17 12:45:13 +01:00
|
|
|
if (gpio_is_valid(dev->p.rw)) {
|
2019-11-07 17:30:46 +01:00
|
|
|
_gpio_clear(dev->p.rw);
|
2017-03-25 11:00:58 +01:00
|
|
|
}
|
|
|
|
/* write data in 8Bit or 4Bit mode */
|
|
|
|
if (dev->flag & HD44780_8BITMODE) {
|
|
|
|
_write_bits(dev, 8, value);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
_write_bits(dev, 4, value>>4);
|
|
|
|
_write_bits(dev, 4, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
static void _write_bits(const hd44780_t *dev, uint8_t bits, uint8_t value)
|
2017-03-25 11:00:58 +01:00
|
|
|
{
|
|
|
|
for (unsigned i = 0; i < bits; ++i) {
|
|
|
|
if ((value >> i) & 0x01) {
|
2019-11-07 17:30:46 +01:00
|
|
|
_gpio_set(dev->p.data[i]);
|
2017-03-25 11:00:58 +01:00
|
|
|
}
|
|
|
|
else {
|
2019-11-07 17:30:46 +01:00
|
|
|
_gpio_clear(dev->p.data[i]);
|
2017-03-25 11:00:58 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
_pulse(dev);
|
|
|
|
}
|
|
|
|
|
2019-11-07 17:30:46 +01:00
|
|
|
static inline int _gpio_init(gpio_t pin, gpio_mode_t mode)
|
|
|
|
{
|
|
|
|
#ifdef MODULE_PCF857X
|
|
|
|
return pcf857x_gpio_init(&_pcf857x_dev, pin, mode);
|
|
|
|
#else
|
|
|
|
return gpio_init(pin, mode);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void _gpio_set(gpio_t pin)
|
|
|
|
{
|
|
|
|
#ifdef MODULE_PCF857X
|
|
|
|
pcf857x_gpio_set(&_pcf857x_dev, pin);
|
|
|
|
#else
|
|
|
|
gpio_set(pin);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void _gpio_clear(gpio_t pin)
|
|
|
|
{
|
|
|
|
#ifdef MODULE_PCF857X
|
|
|
|
pcf857x_gpio_clear(&_pcf857x_dev, pin);
|
|
|
|
#else
|
|
|
|
gpio_clear(pin);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-03-25 11:00:58 +01:00
|
|
|
int hd44780_init(hd44780_t *dev, const hd44780_params_t *params)
|
|
|
|
{
|
|
|
|
/* write config params to device descriptor */
|
2019-01-06 22:54:44 +01:00
|
|
|
dev->p = *params;
|
2017-03-25 11:00:58 +01:00
|
|
|
/* verify cols and rows */
|
|
|
|
if ((dev->p.cols > HD44780_MAX_COLS) || (dev->p.rows > HD44780_MAX_ROWS)
|
|
|
|
|| (dev->p.rows * dev->p.cols > 80)) {
|
|
|
|
LOG_ERROR("hd44780_init: invalid LCD size!\n");
|
|
|
|
return -1;
|
|
|
|
}
|
2019-11-06 16:14:39 +01:00
|
|
|
dev->flag = 0;
|
2019-11-06 16:27:39 +01:00
|
|
|
/* set mode depending on configured pins */
|
2020-01-17 12:45:13 +01:00
|
|
|
if (gpio_is_valid(dev->p.data[4])) {
|
2019-11-06 16:27:39 +01:00
|
|
|
dev->flag |= HD44780_8BITMODE;
|
2017-03-25 11:00:58 +01:00
|
|
|
}
|
|
|
|
else {
|
2019-11-06 16:27:39 +01:00
|
|
|
dev->flag |= HD44780_4BITMODE;
|
2017-03-25 11:00:58 +01:00
|
|
|
}
|
|
|
|
/* set flag for 1 or 2 row mode, 4 rows are 2 rows split half */
|
|
|
|
if (dev->p.rows > 1) {
|
|
|
|
dev->flag |= HD44780_2LINE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dev->flag |= HD44780_1LINE;
|
|
|
|
}
|
|
|
|
/* set char size to 5x8 as default, 5x10 is hardly used by LCDs anyway */
|
|
|
|
dev->flag |= HD44780_5x8DOTS;
|
|
|
|
/* calc and set row offsets, depending on number of columns */
|
|
|
|
dev->roff[0] = 0x00;
|
|
|
|
dev->roff[1] = 0x40;
|
|
|
|
dev->roff[2] = 0x00 + dev->p.cols;
|
|
|
|
dev->roff[3] = 0x40 + dev->p.cols;
|
|
|
|
|
2019-11-07 17:30:46 +01:00
|
|
|
#ifdef MODULE_PCF857X
|
|
|
|
/*
|
|
|
|
* TODO: We need an approach for defining and initializing the PCF8574.
|
|
|
|
* With this approach, only one PCF8574 could exist in the system
|
|
|
|
*/
|
|
|
|
pcf857x_init(&_pcf857x_dev, &pcf857x_params[0]);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
_gpio_init(dev->p.rs, GPIO_OUT);
|
2019-10-22 16:33:34 +02:00
|
|
|
/* RW (read/write) of LCD not required, set it to GPIO_UNDEF */
|
2020-01-17 12:45:13 +01:00
|
|
|
if (gpio_is_valid(dev->p.rw)) {
|
2019-11-07 17:30:46 +01:00
|
|
|
_gpio_init(dev->p.rw, GPIO_OUT);
|
2017-03-25 11:00:58 +01:00
|
|
|
}
|
2019-11-07 17:30:46 +01:00
|
|
|
_gpio_init(dev->p.enable, GPIO_OUT);
|
2017-03-25 11:00:58 +01:00
|
|
|
/* configure all data pins as output */
|
|
|
|
for (int i = 0; i < ((dev->flag & HD44780_8BITMODE) ? 8 : 4); ++i) {
|
2019-11-07 17:30:46 +01:00
|
|
|
_gpio_init(dev->p.data[i], GPIO_OUT);
|
2017-03-25 11:00:58 +01:00
|
|
|
}
|
|
|
|
/* see hitachi HD44780 datasheet pages 45/46 for init specs */
|
|
|
|
xtimer_usleep(HD44780_INIT_WAIT_XXL);
|
2019-11-07 17:30:46 +01:00
|
|
|
_gpio_clear(dev->p.rs);
|
|
|
|
_gpio_clear(dev->p.enable);
|
2020-01-17 12:45:13 +01:00
|
|
|
if (gpio_is_valid(dev->p.rw)) {
|
2019-11-07 17:30:46 +01:00
|
|
|
_gpio_clear(dev->p.rw);
|
2017-03-25 11:00:58 +01:00
|
|
|
}
|
|
|
|
/* put the LCD into 4 bit or 8 bit mode */
|
|
|
|
if (!(dev->flag & HD44780_8BITMODE)) {
|
|
|
|
/* see hitachi HD44780 datasheet figure 24, pg 46 */
|
|
|
|
_write_bits(dev, 4, 0x03);
|
|
|
|
xtimer_usleep(HD44780_INIT_WAIT_LONG);
|
|
|
|
|
|
|
|
_write_bits(dev, 4, 0x03);
|
|
|
|
xtimer_usleep(HD44780_INIT_WAIT_LONG);
|
|
|
|
|
|
|
|
_write_bits(dev, 4, 0x03);
|
|
|
|
xtimer_usleep(HD44780_INIT_WAIT_SHORT);
|
|
|
|
|
|
|
|
_write_bits(dev, 4, 0x02);
|
|
|
|
} else {
|
|
|
|
/* see hitachi HD44780 datasheet page 45 figure 23 */
|
|
|
|
_command(dev, HD44780_FUNCTIONSET | dev->flag);
|
2018-02-05 11:40:02 +01:00
|
|
|
xtimer_usleep(HD44780_INIT_WAIT_LONG); /* wait more than 4.1ms */
|
2017-03-25 11:00:58 +01:00
|
|
|
|
|
|
|
_command(dev, HD44780_FUNCTIONSET | dev->flag);
|
|
|
|
xtimer_usleep(HD44780_INIT_WAIT_SHORT);
|
|
|
|
|
|
|
|
_command(dev, HD44780_FUNCTIONSET | dev->flag);
|
|
|
|
}
|
|
|
|
_command(dev, HD44780_FUNCTIONSET | dev->flag);
|
|
|
|
|
|
|
|
/* turn the display on with no cursor or blinking default, and clear */
|
|
|
|
dev->ctrl = HD44780_DISPLAYON | HD44780_CURSOROFF | HD44780_BLINKOFF;
|
|
|
|
hd44780_display(dev, HD44780_ON);
|
|
|
|
hd44780_clear(dev);
|
|
|
|
/* Initialize to default text direction for western languages */
|
|
|
|
dev->mode = HD44780_ENTRYLEFT | HD44780_ENTRYSHIFTDECREMENT;
|
|
|
|
_command(dev, HD44780_ENTRYMODESET | dev->mode);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
void hd44780_clear(const hd44780_t *dev)
|
2017-03-25 11:00:58 +01:00
|
|
|
{
|
|
|
|
_command(dev, HD44780_CLEARDISPLAY);
|
|
|
|
xtimer_usleep(HD44780_CMD_WAIT);
|
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
void hd44780_home(const hd44780_t *dev)
|
2017-03-25 11:00:58 +01:00
|
|
|
{
|
|
|
|
_command(dev, HD44780_RETURNHOME);
|
|
|
|
xtimer_usleep(HD44780_CMD_WAIT);
|
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
void hd44780_set_cursor(const hd44780_t *dev, uint8_t col, uint8_t row)
|
2017-03-25 11:00:58 +01:00
|
|
|
{
|
|
|
|
if (row >= dev->p.rows) {
|
|
|
|
row = dev->p.rows - 1;
|
|
|
|
}
|
|
|
|
_command(dev, HD44780_SETDDRAMADDR | (col + dev->roff[row]));
|
|
|
|
}
|
|
|
|
|
|
|
|
void hd44780_display(hd44780_t *dev, hd44780_state_t state)
|
|
|
|
{
|
|
|
|
if (state == HD44780_ON) {
|
|
|
|
dev->ctrl |= HD44780_DISPLAYON;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dev->ctrl &= ~HD44780_DISPLAYON;
|
|
|
|
}
|
|
|
|
_command(dev, HD44780_DISPLAYCONTROL | dev->ctrl);
|
|
|
|
}
|
|
|
|
|
|
|
|
void hd44780_cursor(hd44780_t *dev, hd44780_state_t state)
|
|
|
|
{
|
|
|
|
if (state == HD44780_ON) {
|
|
|
|
dev->ctrl |= HD44780_CURSORON;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dev->ctrl &= ~HD44780_CURSORON;
|
|
|
|
}
|
|
|
|
_command(dev, HD44780_DISPLAYCONTROL | dev->ctrl);
|
|
|
|
}
|
|
|
|
|
|
|
|
void hd44780_blink(hd44780_t *dev, hd44780_state_t state)
|
|
|
|
{
|
|
|
|
if (state == HD44780_ON) {
|
|
|
|
dev->ctrl |= HD44780_BLINKON;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dev->ctrl &= ~HD44780_BLINKON;
|
|
|
|
}
|
|
|
|
_command(dev, HD44780_DISPLAYCONTROL | dev->ctrl);
|
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
void hd44780_scroll_left(const hd44780_t *dev)
|
2017-03-25 11:00:58 +01:00
|
|
|
{
|
|
|
|
_command(dev, HD44780_CURSORSHIFT | HD44780_DISPLAYMOVE | HD44780_MOVELEFT);
|
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
void hd44780_scroll_right(const hd44780_t *dev) {
|
2017-03-25 11:00:58 +01:00
|
|
|
_command(dev, HD44780_CURSORSHIFT | HD44780_DISPLAYMOVE | HD44780_MOVERIGHT);
|
|
|
|
}
|
|
|
|
|
|
|
|
void hd44780_left2right(hd44780_t *dev)
|
|
|
|
{
|
|
|
|
dev->mode |= HD44780_ENTRYLEFT;
|
|
|
|
_command(dev, HD44780_ENTRYMODESET | dev->mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
void hd44780_right2left(hd44780_t *dev)
|
|
|
|
{
|
|
|
|
dev->mode &= ~HD44780_ENTRYLEFT;
|
|
|
|
_command(dev, HD44780_ENTRYMODESET | dev->mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
void hd44780_autoscroll(hd44780_t *dev, hd44780_state_t state)
|
|
|
|
{
|
|
|
|
if (state == HD44780_ON) {
|
|
|
|
dev->mode |= HD44780_ENTRYSHIFTINCREMENT;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dev->mode &= ~HD44780_ENTRYSHIFTINCREMENT;
|
|
|
|
}
|
|
|
|
_command(dev, HD44780_ENTRYMODESET | dev->mode);
|
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
void hd44780_create_char(const hd44780_t *dev, uint8_t location, uint8_t charmap[])
|
2017-03-25 11:00:58 +01:00
|
|
|
{
|
|
|
|
location &= 0x7; /* 8 locations (0-7) possible */
|
|
|
|
_command(dev, HD44780_SETCGRAMADDR | (location << 3));
|
|
|
|
for (unsigned i = 0; i < HD44780_CGRAM_SIZE; ++i) {
|
|
|
|
hd44780_write(dev, charmap[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
void hd44780_write(const hd44780_t *dev, uint8_t value)
|
2017-03-25 11:00:58 +01:00
|
|
|
{
|
|
|
|
_send(dev, value, HD44780_ON);
|
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
void hd44780_print(const hd44780_t *dev, const char *data )
|
2017-03-25 11:00:58 +01:00
|
|
|
{
|
|
|
|
while (*data != '\0') {
|
|
|
|
hd44780_write(dev, *data++);
|
|
|
|
}
|
|
|
|
}
|