|
- /*
- * lcd.c:
- * Text-based LCD driver.
- * This is designed to drive the parallel interface LCD drivers
- * based in the Hitachi HD44780U controller and compatables.
- *
- * Copyright (c) 2012 Gordon Henderson.
- ***********************************************************************
- * This file is part of wiringPi:
- * https://projects.drogon.net/raspberry-pi/wiringpi/
- *
- * wiringPi is free software: you can redistribute it and/or modify
- * it under the terms of the GNU Lesser General Public License as published by
- * the Free Software Foundation, either version 3 of the License, or
- * (at your option) any later version.
- *
- * wiringPi is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU Lesser General Public License for more details.
- *
- * You should have received a copy of the GNU Lesser General Public License
- * along with wiringPi. If not, see <http://www.gnu.org/licenses/>.
- ***********************************************************************
- */
-
- #include <stdio.h>
- #include <stdlib.h>
- #include <stdarg.h>
-
- #include <wiringPi.h>
-
- #include "lcd.h"
-
- #ifndef TRUE
- # define TRUE (1==1)
- # define FALSE (1==2)
- #endif
-
- // HD44780U Commands
-
- #define LCD_CLEAR 0x01
- #define LCD_HOME 0x02
- #define LCD_ENTRY 0x04
- #define LCD_CTRL 0x08
- #define LCD_CDSHIFT 0x10
- #define LCD_FUNC 0x20
- #define LCD_CGRAM 0x40
- #define LCD_DGRAM 0x80
-
- // Bits in the entry register
-
- #define LCD_ENTRY_SH 0x01
- #define LCD_ENTRY_ID 0x02
-
- // Bits in the control register
-
- #define LCD_BLINK_CTRL 0x01
- #define LCD_CURSOR_CTRL 0x02
- #define LCD_DISPLAY_CTRL 0x04
-
- // Bits in the function register
-
- #define LCD_FUNC_F 0x04
- #define LCD_FUNC_N 0x08
- #define LCD_FUNC_DL 0x10
-
- #define LCD_CDSHIFT_RL 0x04
-
- struct lcdDataStruct
- {
- int bits, rows, cols ;
- int rsPin, strbPin ;
- int dataPins [8] ;
- int cx, cy ;
- } ;
-
- struct lcdDataStruct *lcds [MAX_LCDS] ;
-
- static int lcdControl ;
-
- // Row offsets
-
- static const int rowOff [4] = { 0x00, 0x40, 0x14, 0x54 } ;
-
-
- /*
- * strobe:
- * Toggle the strobe (Really the "E") pin to the device.
- * According to the docs, data is latched on the falling edge.
- *********************************************************************************
- */
-
- static void strobe (const struct lcdDataStruct *lcd)
- {
-
- // Note timing changes for new version of delayMicroseconds ()
-
- digitalWrite (lcd->strbPin, 1) ; delayMicroseconds (50) ;
- digitalWrite (lcd->strbPin, 0) ; delayMicroseconds (50) ;
- }
-
-
- /*
- * sentDataCmd:
- * Send an data or command byte to the display.
- *********************************************************************************
- */
-
- static void sendDataCmd (const struct lcdDataStruct *lcd, unsigned char data)
- {
- register unsigned char myData = data ;
- unsigned char i, d4 ;
-
- if (lcd->bits == 4)
- {
- d4 = (myData >> 4) & 0x0F;
- for (i = 0 ; i < 4 ; ++i)
- {
- digitalWrite (lcd->dataPins [i], (d4 & 1)) ;
- d4 >>= 1 ;
- }
- strobe (lcd) ;
-
- d4 = myData & 0x0F ;
- for (i = 0 ; i < 4 ; ++i)
- {
- digitalWrite (lcd->dataPins [i], (d4 & 1)) ;
- d4 >>= 1 ;
- }
- }
- else
- {
- for (i = 0 ; i < 8 ; ++i)
- {
- digitalWrite (lcd->dataPins [i], (myData & 1)) ;
- myData >>= 1 ;
- }
- }
- strobe (lcd) ;
- }
-
-
- /*
- * putCommand:
- * Send a command byte to the display
- *********************************************************************************
- */
-
- static void putCommand (const struct lcdDataStruct *lcd, unsigned char command)
- {
- digitalWrite (lcd->rsPin, 0) ;
- sendDataCmd (lcd, command) ;
- delay (2) ;
- }
-
- static void put4Command (const struct lcdDataStruct *lcd, unsigned char command)
- {
- register unsigned char myCommand = command ;
- register unsigned char i ;
-
- digitalWrite (lcd->rsPin, 0) ;
-
- for (i = 0 ; i < 4 ; ++i)
- {
- digitalWrite (lcd->dataPins [i], (myCommand & 1)) ;
- myCommand >>= 1 ;
- }
- strobe (lcd) ;
- }
-
-
- /*
- *********************************************************************************
- * User Callable code below here
- *********************************************************************************
- */
-
- /*
- * lcdHome: lcdClear:
- * Home the cursor or clear the screen.
- *********************************************************************************
- */
-
- void lcdHome (const int fd)
- {
- struct lcdDataStruct *lcd = lcds [fd] ;
-
- putCommand (lcd, LCD_HOME) ;
- lcd->cx = lcd->cy = 0 ;
- delay (5) ;
- }
-
- void lcdClear (const int fd)
- {
- struct lcdDataStruct *lcd = lcds [fd] ;
-
- putCommand (lcd, LCD_CLEAR) ;
- putCommand (lcd, LCD_HOME) ;
- lcd->cx = lcd->cy = 0 ;
- delay (5) ;
- }
-
-
- /*
- * lcdDisplay: lcdCursor: lcdCursorBlink:
- * Turn the display, cursor, cursor blinking on/off
- *********************************************************************************
- */
-
- void lcdDisplay (const int fd, int state)
- {
- struct lcdDataStruct *lcd = lcds [fd] ;
-
- if (state)
- lcdControl |= LCD_DISPLAY_CTRL ;
- else
- lcdControl &= ~LCD_DISPLAY_CTRL ;
-
- putCommand (lcd, LCD_CTRL | lcdControl) ;
- }
-
- void lcdCursor (const int fd, int state)
- {
- struct lcdDataStruct *lcd = lcds [fd] ;
-
- if (state)
- lcdControl |= LCD_CURSOR_CTRL ;
- else
- lcdControl &= ~LCD_CURSOR_CTRL ;
-
- putCommand (lcd, LCD_CTRL | lcdControl) ;
- }
-
- void lcdCursorBlink (const int fd, int state)
- {
- struct lcdDataStruct *lcd = lcds [fd] ;
-
- if (state)
- lcdControl |= LCD_BLINK_CTRL ;
- else
- lcdControl &= ~LCD_BLINK_CTRL ;
-
- putCommand (lcd, LCD_CTRL | lcdControl) ;
- }
-
-
- /*
- * lcdSendCommand:
- * Send any arbitary command to the display
- *********************************************************************************
- */
-
- void lcdSendCommand (const int fd, unsigned char command)
- {
- struct lcdDataStruct *lcd = lcds [fd] ;
- putCommand (lcd, command) ;
- }
-
-
- /*
- * lcdPosition:
- * Update the position of the cursor on the display.
- * Ignore invalid locations.
- *********************************************************************************
- */
-
- void lcdPosition (const int fd, int x, int y)
- {
- struct lcdDataStruct *lcd = lcds [fd] ;
-
- if ((x > lcd->cols) || (x < 0))
- return ;
- if ((y > lcd->rows) || (y < 0))
- return ;
-
- putCommand (lcd, x + (LCD_DGRAM | rowOff [y])) ;
-
- lcd->cx = x ;
- lcd->cy = y ;
- }
-
-
- /*
- * lcdCharDef:
- * Defines a new character in the CGRAM
- *********************************************************************************
- */
-
- void lcdCharDef (const int fd, int index, unsigned char data [8])
- {
- struct lcdDataStruct *lcd = lcds [fd] ;
- int i ;
-
- putCommand (lcd, LCD_CGRAM | ((index & 7) << 3)) ;
-
- digitalWrite (lcd->rsPin, 1) ;
- for (i = 0 ; i < 8 ; ++i)
- sendDataCmd (lcd, data [i]) ;
- }
-
-
- /*
- * lcdPutchar:
- * Send a data byte to be displayed on the display. We implement a very
- * simple terminal here - with line wrapping, but no scrolling. Yet.
- *********************************************************************************
- */
-
- void lcdPutchar (const int fd, unsigned char data)
- {
- struct lcdDataStruct *lcd = lcds [fd] ;
-
- digitalWrite (lcd->rsPin, 1) ;
- sendDataCmd (lcd, data) ;
-
- if (++lcd->cx == lcd->cols)
- {
- lcd->cx = 0 ;
- if (++lcd->cy == lcd->rows)
- lcd->cy = 0 ;
-
- putCommand (lcd, lcd->cx + (LCD_DGRAM | rowOff [lcd->cy])) ;
- }
- }
-
-
- /*
- * lcdPuts:
- * Send a string to be displayed on the display
- *********************************************************************************
- */
-
- void lcdPuts (const int fd, const char *string)
- {
- while (*string)
- lcdPutchar (fd, *string++) ;
- }
-
-
- /*
- * lcdPrintf:
- * Printf to an LCD display
- *********************************************************************************
- */
-
- void lcdPrintf (const int fd, const char *message, ...)
- {
- va_list argp ;
- char buffer [1024] ;
-
- va_start (argp, message) ;
- vsnprintf (buffer, 1023, message, argp) ;
- va_end (argp) ;
-
- lcdPuts (fd, buffer) ;
- }
-
-
- /*
- * lcdInit:
- * Take a lot of parameters and initialise the LCD, and return a handle to
- * that LCD, or -1 if any error.
- *********************************************************************************
- */
-
- int lcdInit (const int rows, const int cols, const int bits,
- const int rs, const int strb,
- const int d0, const int d1, const int d2, const int d3, const int d4,
- const int d5, const int d6, const int d7)
- {
- static int initialised = 0 ;
-
- unsigned char func ;
- int i ;
- int lcdFd = -1 ;
- struct lcdDataStruct *lcd ;
-
- if (initialised == 0)
- {
- initialised = 1 ;
- for (i = 0 ; i < MAX_LCDS ; ++i)
- lcds [i] = NULL ;
- }
-
- // Simple sanity checks
-
- if (! ((bits == 4) || (bits == 8)))
- return -1 ;
-
- if ((rows < 0) || (rows > 20))
- return -1 ;
-
- if ((cols < 0) || (cols > 20))
- return -1 ;
-
- // Create a new LCD:
-
- for (i = 0 ; i < MAX_LCDS ; ++i)
- {
- if (lcds [i] == NULL)
- {
- lcdFd = i ;
- break ;
- }
- }
-
- if (lcdFd == -1)
- return -1 ;
-
- lcd = (struct lcdDataStruct *)malloc (sizeof (struct lcdDataStruct)) ;
- if (lcd == NULL)
- return -1 ;
-
- lcd->rsPin = rs ;
- lcd->strbPin = strb ;
- lcd->bits = 8 ; // For now - we'll set it properly later.
- lcd->rows = rows ;
- lcd->cols = cols ;
- lcd->cx = 0 ;
- lcd->cy = 0 ;
-
- lcd->dataPins [0] = d0 ;
- lcd->dataPins [1] = d1 ;
- lcd->dataPins [2] = d2 ;
- lcd->dataPins [3] = d3 ;
- lcd->dataPins [4] = d4 ;
- lcd->dataPins [5] = d5 ;
- lcd->dataPins [6] = d6 ;
- lcd->dataPins [7] = d7 ;
-
- lcds [lcdFd] = lcd ;
-
- digitalWrite (lcd->rsPin, 0) ; pinMode (lcd->rsPin, OUTPUT) ;
- digitalWrite (lcd->strbPin, 0) ; pinMode (lcd->strbPin, OUTPUT) ;
-
- for (i = 0 ; i < bits ; ++i)
- {
- digitalWrite (lcd->dataPins [i], 0) ;
- pinMode (lcd->dataPins [i], OUTPUT) ;
- }
- delay (35) ; // mS
-
-
- // 4-bit mode?
- // OK. This is a PIG and it's not at all obvious from the documentation I had,
- // so I guess some others have worked through either with better documentation
- // or more trial and error... Anyway here goes:
- //
- // It seems that the controller needs to see the FUNC command at least 3 times
- // consecutively - in 8-bit mode. If you're only using 8-bit mode, then it appears
- // that you can get away with one func-set, however I'd not rely on it...
- //
- // So to set 4-bit mode, you need to send the commands one nibble at a time,
- // the same three times, but send the command to set it into 8-bit mode those
- // three times, then send a final 4th command to set it into 4-bit mode, and only
- // then can you flip the switch for the rest of the library to work in 4-bit
- // mode which sends the commands as 2 x 4-bit values.
-
- if (bits == 4)
- {
- func = LCD_FUNC | LCD_FUNC_DL ; // Set 8-bit mode 3 times
- put4Command (lcd, func >> 4) ; delay (35) ;
- put4Command (lcd, func >> 4) ; delay (35) ;
- put4Command (lcd, func >> 4) ; delay (35) ;
- func = LCD_FUNC ; // 4th set: 4-bit mode
- put4Command (lcd, func >> 4) ; delay (35) ;
- lcd->bits = 4 ;
- }
- else
- {
- func = LCD_FUNC | LCD_FUNC_DL ;
- putCommand (lcd, func ) ; delay (35) ;
- putCommand (lcd, func ) ; delay (35) ;
- putCommand (lcd, func ) ; delay (35) ;
- }
-
- if (lcd->rows > 1)
- {
- func |= LCD_FUNC_N ;
- putCommand (lcd, func) ; delay (35) ;
- }
-
- // Rest of the initialisation sequence
-
- lcdDisplay (lcdFd, TRUE) ;
- lcdCursor (lcdFd, FALSE) ;
- lcdCursorBlink (lcdFd, FALSE) ;
- lcdClear (lcdFd) ;
-
- putCommand (lcd, LCD_ENTRY | LCD_ENTRY_ID) ;
- putCommand (lcd, LCD_CDSHIFT | LCD_CDSHIFT_RL) ;
-
- return lcdFd ;
- }
|