IO Lcd HD44780.C
/ * Driver module for Hitachi HD44780 based Optrex DMC-20481 LCD display
* The module is operated in it's 4 bit-mode to be connected to a single
* 8 bit-port
*
* Modified 22-24 Dec 1999 Andrew McmMeikan
* Added support for '8-Bit WinAmp' Style of wiring
*
* Modified 16-19 Nov 1999 Andrew McMeikan
* fixed hard coded geometry issues, finished support for split screens
*
* Modified 10 Nov 1999 Andrew McMeikan
* put back in Benjamin Tse's LCD_Pause to avoid timing errors
* 13 Nov, fixed delay times in old 4 bit code
* Modified 19-26 Oct 1999 Andrew McMeikan <andrewm@engineer.com>
* to support connection via a shift register with keypad
* 4094 conection details at http://members.xoom.com/andrewmuck/LCD.htm
* TODO: document connections
* also started mods for two lcd enable lines (40x4) or 2x(2x20)
*
* Copyright (c) 1998 Richard Rognlie GNU Public License
* <rrognlie@gamerz.net>
*
* Large quantities of this code lifted (nearly verbatim) from
* the lcd4.c module of lcdtext. Copyright (C) 1997 Matthias Prinke
* <m.prinke@trashcan.mcnet.de> and covered by GNU's GPL.
* In particular, this program is free software and comes WITHOUT
* ANY WARRANTY.
*
* Matthias stole (er, adapted) the code from the package lcdtime by
* Benjamin Tse (blt@mundil.cs.mu.oz.au), August/October 1995
* which uses the LCD-controller's 8 bit-mode.
* References: port.h by <damianf@wpi.edu>
* Data Sheet LTN211, Philips
* Various FAQs and TXTs about Hitachi's LCD Controller HD44780 -
* http://www.paranoia.com/~filipg is a good starting point ???
*/
#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
#include <termios.h>
#include <fcntl.h>
#include <string.h>
#include <errno.h>
#include <sys/perm.h>
#include "../../shared/str.h"
#include "lcd.h"
#include "hd44780.h"
#include "drv_base.h"
lcd_logical_driver *HD44780;
/* shift register code by andrewm@engineer.com *
* uncomment the next line if you do want to use *
* a shift register to connect to the LCD */
//#define USE_SHIFT_REG
//uncomment the next line if you use the winamp style 8 bit interface
//#define WINAMP_STYLE
//#define SPLIT_SCREENS
/* If you are using two LCD Displays or an 8 line or a *
* 40x4 you want to define SPLIT_SCREENS */
#define FONTSIZE 0
// 4= 7line font 0=10 line font (default was 0=10 line andrewm)
#ifndef USE_SHIFT_REG
#ifdef WINAMP_STYLE
#define INV_MASK 0xb
#define RS 4
#define LCDEN 1
#else
#define LCDEN 64
#define RS 16
#endif
#endif
#ifndef USE_SHIFT_REG
#ifdef SPLIT_SCREENS
#ifdef WINAMP_STYLE
#define SECONDLCDEN 8
#else
#define SECONDLCDEN 128
#endif
#endif
#endif
#ifdef USE_SHIFT_REG
#define RS 16
#define LCDDATA 8
#define LCDCLOCK 16
#define LCDEN 4
#ifdef SPLIT_SCREENS
#define SECONDLCDEN 32
#endif
#endif
#ifndef SECONDLCDEN
#define SECONDLCDEN 0
#endif
#ifndef LPTPORT
unsigned int port = 0x378;
#else
unsigned int port = LPTPORT;
#endif
static int lcd_x=0;
static int lcd_y=0;
static void HD44780_linewrap(int on);
static void HD44780_autoscroll(int on);
static int lp;
// IO delay to avoid a task switch
void LCD_Pause(int delayCalls)
{
int i;
for (i = 0; i < delayCalls; ++i)
port_in(port);
//TODO: put in option for nanosleep rather than dummy I/O call
}
#ifdef USE_SHIFT_REG
static char lastkey=0;
static char HD44780_getkey();
void rawshift (unsigned char r) /* this function sends r out *
* onto the shift register */
{
int i;
for (i=7;i>=0;i--) /* MSB first */
{
port_out(lp,((r>>i)&1)*LCDDATA); /*set up data */
port_out(lp,(((r>>i)&1)*LCDDATA)|LCDCLOCK);/*rising edge of clock */
}
}
void shiftreg (unsigned char displayID, unsigned char r)
{
rawshift(r);
port_out(lp,displayID); //latch it, to correct display
port_out(lp,0);
}
#endif
static void HD44780_senddata(unsigned char displayID, unsigned char flags, unsigned char ch)
{
#ifndef WINAMP_STYLE
unsigned char h = ch >> 4;
unsigned char l = ch & 15;
#endif
#ifndef USE_SHIFT_REG
#ifdef WINAMP_STYLE
port_out(lp,ch);
port_out(lp+2,(displayID|flags)^INV_MASK);
port_out(lp+2,flags^INV_MASK);
#else
port_out(lp, displayID | flags | h);LCD_Pause(1);
port_out(lp, flags | h);
port_out(lp, displayID | flags | l);LCD_Pause(1);
port_out(lp, flags | l);
#endif
#else
shiftreg(displayID,flags|h);
shiftreg(displayID,flags|l);
/*all actions take at least 40us to execute TODO:instead of delay read keys here */
#endif
LCD_Pause(50);
}
/////////////////////////////////////////////////////////////////
// Opens com port and sets baud correctly...
//
int HD44780_init(lcd_logical_driver *driver, char *args)
{
char *argv[64];
int argc;
int i;
char displayID = LCDEN|SECONDLCDEN;
HD44780 = driver;
argc = get_args(argv, args, 64);
for(i=0; i<argc; i++)
{
//printf("Arg(%i): %s\n", i, argv[i]);
if(0 == strcmp(argv[i], "-p") ||
0 == strcmp(argv[i], "--port"))
{
if(i + 1 > argc) {
fprintf(stderr, "HD44780_init: %s requires an argument\n",
argv[i]);
return -1;
}
printf("Sorry, runtime lpt-port switching not yet implemented...\n");
//strcpy(device, argv[++i]);
}
else if(0 == strcmp(argv[i], "-c") ||
0 == strcmp(argv[i], "--contrast"))
{
if(i + 1 > argc) {
fprintf(stderr, "HD44780_init: %s requires an argument\n",
argv[i]);
return -1;
}
printf("Sorry, contrast changing not yet implemented...\n");
//contrast = atoi(argv[++i]);
}
else if(0 == strcmp(argv[i], "-h") ||
0 == strcmp(argv[i], "--help"))
{
printf("LCDproc HD44780 driver\n"
"\t-p\t--port\tSelect the output device to use [0x%x]\n"
"\t-c\t--contrast\tSet the initial contrast [default]\n"
"\t-h\t--help\t\tShow this help information\n",
port);
return -1;
}
else
{
printf("Invalid parameter: %s\n", argv[i]);
}
}
// Set up io port correctly, and open it...
if ((ioperm(port,3,255)) == -1) {
fprintf(stderr, "HD44780_init: failed (%s)\n", strerror(errno));
return -1;
}
lp = port;
// init HD44780
#ifndef USE_SHIFT_REG
#ifdef WINAMP_STYLE
port_out(lp,3<<4);
port_out(lp+2,displayID^INV_MASK);port_out(lp+2,INV_MASK);
LCD_Pause(4100);
port_out(lp+2,displayID^INV_MASK);port_out(lp+2,INV_MASK);
LCD_Pause(100);
port_out(lp+2,displayID^INV_MASK);port_out(lp+2,INV_MASK);
LCD_Pause(40);
// now in 8-bit mode...
#else
port_out(lp,displayID | 3);port_out(lp,3);LCD_Pause(4100);
port_out(lp,displayID | 3);port_out(lp,3);LCD_Pause(100);
port_out(lp,displayID | 3);port_out(lp,3);LCD_Pause(40);
// now in 8-bit mode... set 4-bit mode
port_out(lp,displayID | 2);port_out(lp,2);LCD_Pause(40);
#endif
#else
shiftreg(displayID,3);
LCD_Pause(4100);
shiftreg(displayID,3);
LCD_Pause(100);
shiftreg(displayID,3);
LCD_Pause(40);
shiftreg(displayID,2);
LCD_Pause(40);
#endif
#ifndef WINAMP_STYLE
// now in 4-bit mode...
// set display type function set(32) data8/4 (16) set 2 line(8), small char mode(4)
HD44780_senddata(displayID,0, 32 | 8 | FONTSIZE);
#else
// in 8-bit mode...
// set display type function set(32) data8/4 (16) set 2 line(8), small char mode(4)
HD44780_senddata(displayID,0, 32 | 16 | 8 | FONTSIZE);
#endif
//clear
HD44780_senddata(displayID,0, 1);LCD_Pause(1600);
// set lcd on (4), cursor_on (2), and cursor_blink(1)
HD44780_senddata(displayID,0, 8 | 4 | 0 | 0);LCD_Pause(1600);
// Set display-specific stuff..
//HD44780_linewrap(1);
//HD44780_autoscroll(1);
// Make sure the frame buffer is there...
if (!HD44780->framebuf)
HD44780->framebuf = (unsigned char *) malloc(lcd.wid * lcd.hgt);
if (!HD44780->framebuf) {
//HD44780_close();
return -1;
}
// Set the functions the driver supports...
driver->clear = (void *)-1;
driver->string = (void *)-1;
driver->chr = (void *)-1;
driver->vbar = HD44780_vbar;
driver->init_vbar = HD44780_init_vbar;
driver->hbar = HD44780_hbar;
driver->init_hbar = HD44780_init_hbar;
driver->num = HD44780_num;
driver->init_num = HD44780_init_num;
driver->init = HD44780_init;
driver->close = (void *)-1;
driver->flush = HD44780_flush;
driver->flush_box = HD44780_flush_box;
driver->contrast = HD44780_contrast;
driver->backlight = HD44780_backlight;
driver->set_char = HD44780_set_char;
driver->icon = HD44780_icon;
driver->draw_frame = HD44780_draw_frame;
#ifndef USE_SHIFT_REG
driver->getkey = NULL;
#else
driver->getkey = HD44780_getkey;
#endif
return lp;
}
/////////////////////////////////////////////////////////////////
// Clean-up
//
/*
void HD44780_close()
{
drv_base_close();
}
*/
static void HD44780_position(int x, int y)
{
int val = x + (y%2) * 0x40;
if ((y%4)>=2) val += lcd.wid;
if (lcd.hgt==1) val += (x%8) * (0x40-8); //cope with 16x1 display
#ifndef SPLIT_SCREENS
HD44780_senddata(LCDEN,0,128|val);
#else
if (y<(lcd.hgt/2))
{
if (lcd.hgt==2) val += (x%8) * (0x40-8);// two 16x1 displays
HD44780_senddata(LCDEN,0,128|val);
}
else
{
val = x + ((y-(lcd.hgt/2))%2) * 0x40;
if (((y-(lcd.hgt/2))%4)>=2) val += lcd.wid;
if (lcd.hgt==2) val += (x%8) * (0x40-8);// two 16x1 displays
HD44780_senddata(SECONDLCDEN,0,128|val);
}
#endif
lcd_x = x;
lcd_y = y;
}
void HD44780_flush()
{
HD44780_draw_frame(lcd.framebuf);
}
void HD44780_flush_box(int lft, int top, int rgt, int bot)
{
int x,y;
// printf("Flush (%i,%i)-(%i,%i)\n", lft, top, rgt, bot);
for (y=top; y<=bot; y++) {
HD44780_position(lft,y);
//printf("\n%d,%d :",lft,y);
for (x=lft ; x<=rgt ; x++) {
if ((lcd.hgt==1)&&(x==8)) HD44780_position(x,y);
//cope with 16x1 display
#ifdef SPLIT_SCREENS
if ((lcd.hgt==2)&&(x==8)) HD44780_position(x,y); //two 16x1's
if (y<(lcd.hgt/2))
HD44780_senddata(LCDEN,RS,lcd.framebuf[(y*lcd.wid)+x]);
else
HD44780_senddata(SECONDLCDEN,RS,lcd.framebuf[(y*lcd.wid)+x]);
//printf("%c",lcd.framebuf[(y*lcd.wid)+x]);
#else
HD44780_senddata(LCDEN,RS,lcd.framebuf[(y*lcd.wid)+x]);
#endif
}
//write(fd, lcd.framebuf[(y*lcd.wid)+lft, rgt-lft+1]);
}
}
/////////////////////////////////////////////////////////////////
// Changes screen contrast (0-255; 140 seems good)
//
int HD44780_contrast(int contrast)
{
return 0;
}
/////////////////////////////////////////////////////////////////
// Sets the backlight on or off -- can be done quickly for
// an intermediate brightness...
//
void HD44780_backlight(int on)
{
}
/////////////////////////////////////////////////////////////////
// Toggle the built-in linewrapping feature
//
static void HD44780_linewrap(int on)
{
}
/////////////////////////////////////////////////////////////////
// Toggle the built-in automatic scrolling feature
//
static void HD44780_autoscroll(int on)
{
HD44780_senddata(LCDEN|SECONDLCDEN,0,4 | on * 2);
LCD_Pause(1600); //this can take time to do on some displays
}
/////////////////////////////////////////////////////////////////
// Sets up for vertical bars. Call before lcd.vbar()
//
void HD44780_init_vbar()
{
char a[] = {
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
1,1,1,1,1,
};
char b[] = {
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
1,1,1,1,1,
1,1,1,1,1,
};
char c[] = {
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
1,1,1,1,1,
1,1,1,1,1,
1,1,1,1,1,
};
char d[] = {
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
1,1,1,1,1,
1,1,1,1,1,
1,1,1,1,1,
1,1,1,1,1,
};
char e[] = {
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
1,1,1,1,1,
1,1,1,1,1,
1,1,1,1,1,
1,1,1,1,1,
1,1,1,1,1,
};
char f[] = {
0,0,0,0,0,
0,0,0,0,0,
1,1,1,1,1,
1,1,1,1,1,
1,1,1,1,1,
1,1,1,1,1,
1,1,1,1,1,
1,1,1,1,1,
};
char g[] = {
0,0,0,0,0,
1,1,1,1,1,
1,1,1,1,1,
1,1,1,1,1,
1,1,1,1,1,
1,1,1,1,1,
1,1,1,1,1,
1,1,1,1,1,
};
HD44780_set_char(1,a);
HD44780_set_char(2,b);
HD44780_set_char(3,c);
HD44780_set_char(4,d);
HD44780_set_char(5,e);
HD44780_set_char(6,f);
HD44780_set_char(7,g);
}
/////////////////////////////////////////////////////////////////
// Inits horizontal bars...
//
void HD44780_init_hbar()
{
char a[] = {
1,0,0,0,0,
1,0,0,0,0,
1,0,0,0,0,
1,0,0,0,0,
1,0,0,0,0,
1,0,0,0,0,
1,0,0,0,0,
1,0,0,0,0,
};
char b[] = {
1,1,0,0,0,
1,1,0,0,0,
1,1,0,0,0,
1,1,0,0,0,
1,1,0,0,0,
1,1,0,0,0,
1,1,0,0,0,
1,1,0,0,0,
};
char c[] = {
1,1,1,0,0,
1,1,1,0,0,
1,1,1,0,0,
1,1,1,0,0,
1,1,1,0,0,
1,1,1,0,0,
1,1,1,0,0,
1,1,1,0,0,
};
char d[] = {
1,1,1,1,0,
1,1,1,1,0,
1,1,1,1,0,
1,1,1,1,0,
1,1,1,1,0,
1,1,1,1,0,
1,1,1,1,0,
1,1,1,1,0,
};
HD44780_set_char(1,a);
HD44780_set_char(2,b);
HD44780_set_char(3,c);
HD44780_set_char(4,d);
}
/////////////////////////////////////////////////////////////////
// Draws a vertical bar...
//
void HD44780_vbar(int x, int len)
{
char map[9] = {32, 1, 2, 3, 4, 5, 6, 7, 255 };
int y;
for(y=lcd.hgt; y > 0 && len>0; y--) {
if (len >= lcd.cellhgt)
drv_base_chr(x, y, 255);
else
drv_base_chr(x, y, map[len]);
len -= lcd.cellhgt;
}
}
/////////////////////////////////////////////////////////////////
// Draws a horizontal bar to the right.
//
void HD44780_hbar(int x, int y, int len)
{
char map[6] = { 32, 1, 2, 3, 4, 255 };
for (; x<=lcd.wid && len>0; x++) {
if (len >= lcd.cellwid)
drv_base_chr(x,y,255);
else
drv_base_chr(x, y, map[len]);
len -= lcd.cellwid;
}
}
/////////////////////////////////////////////////////////////////
// Sets up for big numbers.
//
void HD44780_init_num()
{
char out[3];
sprintf(out, "%cn", 254);
//write(fd, out, 2);
}
/////////////////////////////////////////////////////////////////
// Writes a big number.
//
void HD44780_num(int x, int num)
{
char out[5];
sprintf(out, "%c#%c%c", 254, x, num);
//write(fd, out, 4);
}
/////////////////////////////////////////////////////////////////
// Sets a custom character from 0-7...
//
// For input, values > 0 mean "on" and values <= 0 are "off".
//
// The input is just an array of characters...
//
void HD44780_set_char(int n, char *dat)
{
int row, col;
int letter;
if(n < 0 || n > 7) return;
if(!dat) return;
HD44780_senddata(LCDEN|SECONDLCDEN,0, 64 | n*8);
for(row=0; row<lcd.cellhgt; row++) {
letter = 0;
for(col=0; col<lcd.cellwid; col++) {
letter <<= 1;
letter |= (dat[(row*lcd.cellwid) + col] > 0);
}
HD44780_senddata(LCDEN|SECONDLCDEN,RS, letter);
}
}
void HD44780_icon(int which, char dest)
{
char icons[3][5*8] = {
{
1,1,1,1,1, // Empty Heart
1,0,1,0,1,
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
1,0,0,0,1,
1,1,0,1,1,
1,1,1,1,1,
},
{
1,1,1,1,1, // Filled Heart
1,0,1,0,1,
0,1,0,1,0,
0,1,1,1,0,
0,1,1,1,0,
1,0,1,0,1,
1,1,0,1,1,
1,1,1,1,1,
},
{
0,0,0,0,0, // Ellipsis
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
0,0,0,0,0,
1,0,1,0,1,
},
};
HD44780_set_char(dest, &icons[which][0]);
}
/////////////////////////////////////////////////////////////
// Blasts a single frame onscreen, to the lcd...
//
// Input is a character array, sized lcd.wid*lcd.hgt
//
void HD44780_draw_frame(char *dat)
{
int x,y;
if (!dat) return;
for (y=0; y<lcd.hgt; y++) {
HD44780_position(0,y);
//printf("\n%d :",y);
for (x=0 ; x<lcd.wid ; x++) {
if ((lcd.hgt==1)&&(x==8)) HD44780_position(x,y);//for 16x1
#ifdef SPLIT_SCREENS
if ((lcd.hgt==2)&&(x==8)) HD44780_position(x,y);// two 16x1's
if (y<(lcd.hgt/2))
HD44780_senddata(LCDEN,RS,dat[(y*lcd.wid)+x]);
else
HD44780_senddata(SECONDLCDEN,RS,dat[(y*lcd.wid)+x]);
//printf("%c",dat[(y*lcd.wid)+x]);
#else
HD44780_senddata(LCDEN,RS,dat[(y*lcd.wid)+x]);
#endif
}
}
}
#ifdef USE_SHIFT_REG
char HD44780_getkey()
{
// TODO: a keypad scan that does not use shift register
// define a key table
char keytr[]="ABCDEFGH";
int n;
rawshift(0);LCD_Pause(1); //send all line on shift register low
if ((port_in(lp+1)&32)==0) //test if line back is low - if not return(0)
{ //else
//start walking a single zero across the eight lines
for(n=7;n>=0;n--)
{
rawshift(255 - (1<<n));LCD_Pause(1);
if ((port_in(lp+1)&32)==0) // check if line back is low if yes debounce
{
if (lastkey==keytr[n])
return(0);
// printf("key is %c %d\n",keytr[n],n);
lastkey=keytr[n];
return(keytr[n]); //return correct key code.
}
}
} //else fall out and return(0) [too transient a keypress]
lastkey=0;
return(0);
}
#endif
file: /Techref/io/lcd/hd44780.c, 17KB, , updated: 2000/1/28 10:52, local time: 2025/1/25 22:37,
|
| ©2025 These pages are served without commercial sponsorship. (No popup ads, etc...).Bandwidth abuse increases hosting cost forcing sponsorship or shutdown. This server aggressively defends against automated copying for any reason including offline viewing, duplication, etc... Please respect this requirement and DO NOT RIP THIS SITE. Questions? <A HREF="http://linistepper.com/techref/io/lcd/hd44780.c"> io lcd hd44780</A> |
Did you find what you needed?
|