|
|
View previous topic :: View next topic |
Author |
Message |
oxxyfx
Joined: 24 May 2007 Posts: 97
|
LCD for debugging |
Posted: Mon May 28, 2007 8:37 am |
|
|
Hello,
I read so many things about LCD drivers, LCD programs, communication with LCD that I am very confused.
All I would like to do is to have a 2 line 2x16 character LCD display which reads data over serial communication and display it on the screen.
The reason for this is, that sometimes I would like to see values of the variables on the pic in the circuit. This would help so much in debugging.
Now, I already purchased a Hitachi 77480 compatible 2x16 character display, it has 14 ports on it.
What else do I need to make this work?
Thanks, Oxy. |
|
|
inservi
Joined: 13 May 2007 Posts: 128
|
|
Posted: Mon May 28, 2007 12:06 pm |
|
|
Hello,
I just make the same experience.
Read the 'lcd.c' driver. You will find it in the driver folder.
Connect the display as described in the driver file.
The bus is 7 wires between the pic and display (4 datas and 3 controles).
Connect the power pin 1 to ground, pin 2 to 5V
Connect VO ( pin 3 for contrast) to the ground.
you can include this in your code :
Include the 'lcd.c' driver file.
Initialize RS/232 as you want.
Use the following sample code.
Code: |
// call lcd_init() one time before use LCD
lcd_init();
// Clear the LCD.
printf( lcd_putc, "\f") ;
delay_ms( 500 ) ;
printf(lcd_putc, "Hello");
while (true) {
if ( kbhit() ) {
c = getc() ;
lcd_putc(c);
}
}
|
More explanations are in the 'lcd.c' driver file.
This code is working actually in my board.
Best regards,
dro _________________ in médio virtus |
|
|
oxxyfx
Joined: 24 May 2007 Posts: 97
|
|
Posted: Mon May 28, 2007 12:30 pm |
|
|
Hello,
Thank you, you see here is your chance...
I am not sure if I was specific enough. I would like to build a board with the LCD on top of it, and to have 3 wires coming out. One tor +VCC, one for the Serial line and one for GND.
I would use this board whenever I want to debug something on any of my projects.
I did understand what you said, one thing though - nerither you or the LCD.c file specifies what kind of Pic should I use for this...
I have a few on stock: 12F675, 12F629, 16F627A, 16F628, 18F4320.
Can I use any of this forthis reason, or should I go and get a different one?
Thanks. |
|
|
inservi
Joined: 13 May 2007 Posts: 128
|
|
Posted: Tue May 29, 2007 12:34 am |
|
|
Hello oxxyfx,
I just have the same project. Until now, i used serial lcd from Parallax or serial backpack like this: http://www.seetron.com/bpk000_1.htm
Now i decided to make this 'backpack' myself. Not only for debugging but also as user interface for my different projects. They are also very usefully for debugging.
So, i will save about 20$ for each.
About the PIC witch can be used, i am not sure that the 12F can be used but the 16F and 18F you have will probably be more than enough.
Now i have a prototype with a 18F4685. Of course, the 18F4685 is only for prototype. For final project, i will probably use a 16F... with at least 10 IO (in the absolute, 7 are enough), ICSP and a few memory. I think that a 16F84 will be enough.
For the software, i used the driver Flex_LCD420.c found at : http://www.ccsinfo.com/forum/viewtopic.php?t=28268
As you see, this driver support 20X4 chars. It is simple to add parameters for chose between 2 or 4 lines. There is also a 2 lines version.
I just make some modifications for have a blinking cursor on/off and i transformed the gotoXY by gotoLC (line, column). The main modifications are in the lcd_putc() function.
The main program accept characters from serial at 9600 Baud then send they to the lcd.
A chr 27 (ESCAPE) clear the screen.
I need to add some functionalities in the main program:
- A communication protocol compatible with the backpacks i already used.
- Support for some rs/232 speed.
- ...
The last step will be to make PCB layout. This step is very easy with my program Target 3001.
I'm ready to work with you for finishing this project.
Here are the actual program code:
Code: |
#OPT 11
#include <18F4685.h>
#device adc=8
#FUSES NOWDT //No Watch Dog Timer
#FUSES WDT128 //Watch Dog Timer uses 1:128 Postscale
#FUSES HS //High speed Osc (> 4mhz)
#FUSES NOPROTECT //Code not protected from reading
#FUSES BROWNOUT //Reset when brownout detected
#FUSES BORV45 //Brownout reset at 2.0V
#FUSES PUT //Power Up Timer
#FUSES NOCPD //No EE protection
#FUSES STVREN //Stack full/underflow will cause reset
#FUSES NODEBUG //No Debug mode for ICD
#FUSES NOLVP //No low voltage prgming, B3(PIC16) or B5(PIC18) used for I/O
#FUSES NOWRT //Program memory not write protected
#FUSES NOWRTD //Data EEPROM not write protected
#FUSES NOIESO //Internal External Switch Over mode enabled
#FUSES FCMEN //Fail-safe clock monitor enabled
#FUSES NOPBADEN //PORTB pins are configured as analog input channels on RESET
#FUSES BBSIZ4K //4K words Boot Block size
#FUSES NOWRTC //configuration not registers write protected
#FUSES NOWRTB //Boot block not write protected
#FUSES NOEBTR //Memory not protected from table reads
#FUSES NOEBTRB //Boot block not protected from table reads
#FUSES NOCPB //No Boot Block code protection
#FUSES LPT1OSC //Timer1 configured for low-power operation
#FUSES MCLR //Master Clear pin enabled
#FUSES XINST //Extended set extension and Indexed Addressing mode enabled
#use delay(clock=20000000)
#use rs232(baud=9600,parity=N,xmit=PIN_C6,rcv=PIN_C7,bits=8)
#include <Flex_LCD420.c>
int8 int_sec ;
#int_TIMER2
void TIMER2_isr(void) {
#bit LCD_CURSOR_BIT = LCD_FUNCTION.LCD_CURSOR // 1
if (++int_sec == 50 ) {
int_sec = 0 ;
if (LCD_CURSOR_BLINK) {
LCD_CURSOR_BIT = !LCD_CURSOR_BIT ;
lcd_send_byte (0, LCD_FUNCTION ); //0b000111x);
}
}
}
//===================================
void main(){
int8 i;
setup_wdt(WDT_OFF);
setup_timer_0(RTCC_INTERNAL|RTCC_DIV_256|RTCC_8_bit);
setup_timer_1(T1_DISABLED);
setup_timer_2(T2_DIV_BY_16,208,15);
setup_psp(PSP_DISABLED);
enable_interrupts(INT_TIMER2);
enable_interrupts(GLOBAL);
// The lcd_init() function should always be called once,
// near the start of your program.
lcd_init();
// Clear the LCD.
printf(lcd_putc, "\f");
delay_ms(500);
while(true) {
int8 i = 0;
char c ;
printf(lcd_putc,"\t1"); // for start blinking cursor
while (true) {
if ( kbhit() ) {
disable_interrupts(GLOBAL);
c = getc() ;
enable_interrupts(GLOBAL);
if ( c == 27 ) {
// Clear the LCD.
printf(lcd_putc, "\f");
delay_ms(500);
} else lcd_putc(c);
}
}
}
}
|
and the modified driver:
Code: |
// from PCM programmer, modified for cursor blinking
// -- prototype version --
#define LCD_DB4 PIN_D4
#define LCD_DB5 PIN_D5
#define LCD_DB6 PIN_D6
#define LCD_DB7 PIN_D7
#define LCD_RS PIN_D1 //PIN_E0
#define LCD_RW PIN_D2 //PIN_E1
#define LCD_E PIN_D0 //PIN_E2
// IF you want only a 6-pin interface to your LCD, then
// connect the R/W pin on the LCD to ground, and comment
// out the following line. Doing so will save one PIC
// pin, but at the cost of losing the ability to read from
// the LCD. It also makes the write time a little longer
// because a STATIC delay must be used, instead of polling
// the LCD's busy bit. Normally a 6-pin interface is only
// used IF you are running out of PIC pins, and you need
// to use as few as possible FOR the LCD.
#define USE_RW_PIN 1
// These are the line addresses FOR most 4x20 LCDs.
#define LCD_LINE_1_ADDRESS 0x00
#define LCD_LINE_2_ADDRESS 0x40
#define LCD_LINE_3_ADDRESS 0x14
#define LCD_LINE_4_ADDRESS 0x54
// These are the line addresses FOR LCD's which use
// the Hitachi HD66712U controller chip.
/*
#define LCD_LINE_1_ADDRESS 0x00
#define LCD_LINE_2_ADDRESS 0x20
#define LCD_LINE_3_ADDRESS 0x40
#define LCD_LINE_4_ADDRESS 0x60
*/
//========================================
#define lcd_type 2 // 0=5x7, 1=5x10, 2=2 lines(or more)
int8 lcd_line;
int8 const LCD_INIT_STRING[4] ={
0x20| (lcd_type << 2), // Set mode: 4 - bit, 2 + lines, 5x8 dots
0xc, // Display on
1, // Clear display
6 // Increment cursor
};
#define LCD_BLINK 0
#define LCD_CURSOR 1
#define LCD_ONOFF 2
int8 LCD_FUNCTION = LCD_INIT_STRING[1] ;
int8 LCD_SetMode = 0 ;
int8 LCD_CURSOR_BLINK = 0 ;
//-------------------------------------
void lcd_send_nibble(int8 nibble){
// Note: !! converts an integer expression
// to a boolean (1 or 0).
output_bit (LCD_DB4, !! (nibble&1));
output_bit (LCD_DB5, !! (nibble&2));
output_bit (LCD_DB6, !! (nibble&4));
output_bit (LCD_DB7, !! (nibble&8));
delay_cycles (1);
output_high (LCD_E);
delay_us (2);
output_low (LCD_E);
}
//-----------------------------------
// This sub-routine is only called by lcd_read_byte().
// It's not a stand-alone routine. FOR example, the
// R/W signal is set high by lcd_read_byte() before
// this routine is called.
#ifdef USE_RW_PIN
int8 lcd_read_nibble(void) {
int8 retval;
// Create bit variables so that we can easily set
// individual bits in the retval variable.
#bit retval_0 = retval.0
#bit retval_1 = retval.1
#bit retval_2 = retval.2
#bit retval_3 = retval.3
retval = 0;
output_high (LCD_E);
delay_us (1);
retval_0 = input (LCD_DB4);
retval_1 = input (LCD_DB5);
retval_2 = input (LCD_DB6);
retval_3 = input (LCD_DB7);
output_low (LCD_E);
delay_us (1);
return (retval);
}
#endif
//---------------------------------------
// Read a byte from the LCD and RETURN it.
#ifdef USE_RW_PIN
int8 lcd_read_byte(void){
int8 low;
int8 high;
output_high (LCD_RW);
delay_cycles (1);
high = lcd_read_nibble ();
low = lcd_read_nibble ();
return ((high<<4)|low);
}
#endif
//----------------------------------------
// Send a byte to the LCD.
void lcd_send_byte(int8 address, int8 n){
output_low(LCD_RS);
#ifdef USE_RW_PIN
while(bit_test(lcd_read_byte(),7)) ;
#ELSE
delay_us(60);
#endif
if(address) output_high (LCD_RS);
else output_low (LCD_RS);
delay_cycles (1);
#ifdef USE_RW_PIN
output_low(LCD_RW);
delay_cycles(1);
#endif
output_low(LCD_E);
lcd_send_nibble(n >> 4);
lcd_send_nibble(n & 0xf);
}
//----------------------------
void lcd_init(void){
int8 i;
lcd_line = 1;
output_low(LCD_RS);
#ifdef USE_RW_PIN
output_low(LCD_RW);
#endif
output_low(LCD_E);
// Some LCDs require 15 ms minimum delay after
// power-up. Others require 30 ms. I'm going
// to set it to 35 ms, so it should work with
// all of them.
delay_ms(35);
for (i = 0; i < 3; i++){
lcd_send_nibble (0x03);
delay_ms (5);
}
lcd_send_nibble(0x02);
for (i = 0; i < SIZEOF (LCD_INIT_STRING); i++){
lcd_send_byte (0, LCD_INIT_STRING[i]);
// IF the R / W signal is not used, then
// the busy bit can't be polled. One of
// the init commands takes longer than
// the hard - coded delay of 50 us, so in
// that case, lets just DO a 5 ms delay
// after all four of them.
#ifndef USE_RW_PIN
delay_ms (5);
#endif
}
}
//----------------------------
void lcd_gotolc(int8 y, int8 x){
int8 address;
switch (y){
case 1:
address = LCD_LINE_1_ADDRESS;
break;
case 2:
address = LCD_LINE_2_ADDRESS;
break;
case 3:
address = LCD_LINE_3_ADDRESS;
break;
case 4:
address = LCD_LINE_4_ADDRESS;
break;
default:
address = LCD_LINE_1_ADDRESS;
break;
}
address += x-1;
lcd_send_byte(0, 0x80 | address);
}
//-----------------------------
void lcd_putc(char c){
#bit LCD_BLINK_BIT = LCD_FUNCTION.LCD_BLINK // 0
#bit LCD_CURSOR_BIT = LCD_FUNCTION.LCD_CURSOR // 1
#bit LCD_ONOFF_BIT = LCD_FUNCTION.LCD_ONOFF // 2
#bit LCD_SetMode_BLINK = LCD_SetMode.LCD_BLINK // 0
#bit LCD_SetMode_CURSOR = LCD_SetMode.LCD_CURSOR // 1
#bit LCD_SetMode_ONOFF_BIT = LCD_SetMode.LCD_ONOFF // 2
if ( LCD_SetMode ) {
switch ( LCD_SetMode ){
case 1: // blink
LCD_BLINK_BIT = 0 ;
if (c=='1') LCD_BLINK_BIT = 1 ;
break;
case 2: // cursor
LCD_CURSOR_BIT = 0 ;
if (c=='1') LCD_CURSOR_BIT = 1 ;
LCD_CURSOR_BLINK = LCD_CURSOR_BIT ;
break;
case 4: // On/Off
LCD_ONOFF_BIT = 0 ;
if (c=='1') LCD_ONOFF_BIT = 1 ;
break;
}
LCD_SetMode = 0 ;
lcd_send_byte (0, LCD_FUNCTION );
} else {
switch (c){
case '\f':
lcd_send_byte (0, 1);
lcd_line = 1;
delay_ms (2);
break;
case '\n':
lcd_gotolc ( ++lcd_line, 1);
break;
case '\b':
lcd_send_byte (0, 0b0010000); //0x10);
break;
case '\t': // cursor
LCD_SetMode_CURSOR = 1 ;
break;
case '\a': // blink
LCD_SetMode_BLINK = 1 ;
break;
default:
lcd_send_byte(1,c);
break;
}
}
}
//------------------------------
#ifdef USE_RW_PIN
char lcd_getc(int8 x, int8 y){
char value;
lcd_gotolc(y,x);
// Wait until busy flag is low.
while(bit_test(lcd_read_byte(),7));
output_high(LCD_RS);
value = lcd_read_byte();
output_low(LCD_RS);
return(value);
}
#endif
|
_________________ in médio virtus |
|
|
Douglas Kennedy
Joined: 07 Sep 2003 Posts: 755 Location: Florida
|
|
Posted: Tue May 29, 2007 2:40 am |
|
|
If its debugging you want the CCS debugger is very nice.
Chances are you will want to do ICSP ..program and reprogram your chip while it is in your circuit so you will be dedicating B7 and B6 to the ICSP and if you further give up PIN_b3 for the CCS debugger it will do printfs to CCS's built in PC monitor.
The CCS debugger has mouse over display of variables in your code and line by line tracing. |
|
|
grasspuddle
Joined: 15 Jun 2006 Posts: 66
|
|
Posted: Tue May 29, 2007 8:39 am |
|
|
If you want extremely easy and simple use this:
http://www.sparkfun.com/commerce/product_info.php?products_id=813
It has 3 inputs, +5V, GND, and RCV
Whatever it receives (9600baud if I remember right) it displays.
I've used it once or twice when I didn't have a max232 to plug in and test on comp. |
|
|
oxxyfx
Joined: 24 May 2007 Posts: 97
|
|
Posted: Tue May 29, 2007 10:51 am |
|
|
Thank you. This last one is exactly what I need.
Ox. |
|
|
|
|
You cannot post new topics in this forum You cannot reply to topics in this forum You cannot edit your posts in this forum You cannot delete your posts in this forum You cannot vote in polls in this forum
|
Powered by phpBB © 2001, 2005 phpBB Group
|