PIC18F4520 ile LCD Uygulama Sorunu

Başlatan PICaso, 25 Mart 2019, 11:59:47

PICaso

Arkadaşlar merhabalar,

PIC18f4520 ile basit bir lcd uygulaması yapıyorum. Mplab XC8 derleyicisi kullanıyorum. Lcd ye yazı gelecek gibi kareler oluşuyor fakat yazılar gelmiyor. Yardımcı olrsanız sevinirim.
İyi çalışmalar dilerim.

#define _XTAL_FREQ 20000000

#pragma config OSC = HS        // Oscillator Selection bits HS
#pragma config FCMEN = OFF       // Fail-Safe Clock Monitor Enable bit (Fail-Safe Clock Monitor enabled)
#pragma config IESO = OFF        // Internal/External Oscillator Switchover bit (Oscillator Switchover mode enabled)

// CONFIG2L
#pragma config PWRT = OFF       // Power-up Timer Enable bit (PWRT disabled)
#pragma config BOREN = ON       // Brown-out Reset Enable bits (Brown-out Reset enabled and controlled by software (SBOREN is enabled))
#pragma config BORV = 3         // Brown Out Reset Voltage bits (Minimum setting)

// CONFIG2H
#pragma config WDT = OFF        // Watchdog Timer Enable bit (WDT disabled (control is placed on the SWDTEN bit))
#pragma config WDTPS = 32768    // Watchdog Timer Postscale Select bits (1:32768)

// CONFIG3H
#pragma config CCP2MX = PORTC   // CCP2 MUX bit (CCP2 input/output is multiplexed with RC1)
#pragma config PBADEN = OFF     // PORTB A/D Enable bit (PORTB<4:0> pins are configured as digital I/O on Reset)
#pragma config LPT1OSC = OFF    // Low-Power Timer1 Oscillator Enable bit (Timer1 configured for higher power operation)
#pragma config MCLRE = ON       // MCLR Pin Enable bit (MCLR pin enabled; RE3 input pin disabled)

// CONFIG4L
#pragma config STVREN = ON      // Stack Full/Underflow Reset Enable bit (Stack full/underflow will cause Reset)
#pragma config LVP = OFF        // Single-Supply ICSP Enable bit (Single-Supply ICSP disabled)
#pragma config XINST = OFF      // Extended Instruction Set Enable bit (Instruction set extension and Indexed Addressing mode disabled (Legacy mode))

// CONFIG5L
#pragma config CP0 = OFF        // Code Protection bit (Block 0 (000800-001FFFh) not code-protected)
#pragma config CP1 = OFF        // Code Protection bit (Block 1 (002000-003FFFh) not code-protected)
#pragma config CP2 = OFF        // Code Protection bit (Block 2 (004000-005FFFh) not code-protected)
#pragma config CP3 = OFF        // Code Protection bit (Block 3 (006000-007FFFh) not code-protected)

// CONFIG5H
#pragma config CPB = OFF        // Boot Block Code Protection bit (Boot block (000000-0007FFh) not code-protected)
#pragma config CPD = OFF        // Data EEPROM Code Protection bit (Data EEPROM not code-protected)

// CONFIG6L
#pragma config WRT0 = OFF       // Write Protection bit (Block 0 (000800-001FFFh) not write-protected)
#pragma config WRT1 = OFF       // Write Protection bit (Block 1 (002000-003FFFh) not write-protected)
#pragma config WRT2 = OFF       // Write Protection bit (Block 2 (004000-005FFFh) not write-protected)
#pragma config WRT3 = OFF       // Write Protection bit (Block 3 (006000-007FFFh) not write-protected)

// CONFIG6H
#pragma config WRTC = OFF       // Configuration Register Write Protection bit (Configuration registers (300000-3000FFh) not write-protected)
#pragma config WRTB = OFF       // Boot Block Write Protection bit (Boot block (000000-0007FFh) not write-protected)
#pragma config WRTD = OFF       // Data EEPROM Write Protection bit (Data EEPROM not write-protected)

// CONFIG7L
#pragma config EBTR0 = OFF      // Table Read Protection bit (Block 0 (000800-001FFFh) not protected from table reads executed in other blocks)
#pragma config EBTR1 = OFF      // Table Read Protection bit (Block 1 (002000-003FFFh) not protected from table reads executed in other blocks)
#pragma config EBTR2 = OFF      // Table Read Protection bit (Block 2 (004000-005FFFh) not protected from table reads executed in other blocks)
#pragma config EBTR3 = OFF      // Table Read Protection bit (Block 3 (006000-007FFFh) not protected from table reads executed in other blocks)

// CONFIG7H
#pragma config EBTRB = OFF      // Boot Block Table Read Protection bit (Boot block (000000-0007FFh) not protected from table reads executed in other blocks)

 
//LCD module connections
#define LCD_RS       RD0
#define LCD_EN       RD1
#define LCD_D4       RD2
#define LCD_D5       RD3
#define LCD_D6       RD4
#define LCD_D7       RD5
#define LCD_RS_DIR   TRISD0
#define LCD_EN_DIR   TRISD1
#define LCD_D4_DIR   TRISD2
#define LCD_D5_DIR   TRISD3
#define LCD_D6_DIR   TRISD4
#define LCD_D7_DIR   TRISD5
//End LCD module connections
 
 
#include <xc.h>

#include <stdio.h>            // for sprintf
#include "LCD_Lib.c"          // include LCD driver source file
 
char i = 0, text[4];
 
void main(void)
{
  
 
    LCD_Begin();       // initialize LCD module
 
    LCD_Goto(4, 1);           // go to column 4, row 1
    LCD_Print("MPLAB XC8");
    LCD_Goto(3, 2);           // go to column 3, row 2
    LCD_Print("LCD Example");
 
    __delay_ms(5000);         // wait 5 seconds
 
    LCD_Cmd(LCD_CLEAR);       // clear the whole screen
    LCD_Goto(3, 1);           // go to column 3, row 1
    LCD_Print("Hello world!");
 
    while (1)
    {
      sprintf(text, "%03u", i);
 
      LCD_Goto(7, 2);      // go to column 7, row 2
      LCD_Print(text);
 
      i++;                 // increment i
      __delay_ms(500);     // wait 1/2 second
    }
}



LCD_Lib.c dosyası:
*******************************************************************************
 * LCD_Lib.c                                                                   *
 * MPLAB XC8 compiler LCD driver for LCDs with HD44780 compliant controllers.  *
 * https://simple-circuit.com/                                                 *
 *                                                                             *
 ******************************************************************************/





#pragma warning disable 520

#include <stdint.h>

#define LCD_FIRST_ROW          0x80
#define LCD_SECOND_ROW         0xC0
#define LCD_THIRD_ROW          0x94
#define LCD_FOURTH_ROW         0xD4
#define LCD_CLEAR              0x01
#define LCD_RETURN_HOME        0x02
#define LCD_ENTRY_MODE_SET     0x04
#define LCD_CURSOR_OFF         0x0C
#define LCD_UNDERLINE_ON       0x0E
#define LCD_BLINK_CURSOR_ON    0x0F
#define LCD_MOVE_CURSOR_LEFT   0x10
#define LCD_MOVE_CURSOR_RIGHT  0x14
#define LCD_TURN_ON            0x0C
#define LCD_TURN_OFF           0x08
#define LCD_SHIFT_LEFT         0x18
#define LCD_SHIFT_RIGHT        0x1E

#ifndef LCD_TYPE
   #define LCD_TYPE 2           // 0=5x7, 1=5x10, 2=2 lines
#endif

__bit RS;

void LCD_Write_Nibble(uint8_t n);
void LCD_Cmd(uint8_t Command);
void LCD_Goto(uint8_t col, uint8_t row);
void LCD_PutC(char LCD_Char);
void LCD_Print(char* LCD_Str);
void LCD_Begin();

void LCD_Write_Nibble(uint8_t n)
{
  LCD_RS = RS;
  LCD_D4 = n & 0x01;
  LCD_D5 = (n >> 1) & 0x01;
  LCD_D6 = (n >> 2) & 0x01;
  LCD_D7 = (n >> 3) & 0x01;

  // send enable pulse
  LCD_EN = 0;
  __delay_us(1);
  LCD_EN = 1;
  __delay_us(1);
  LCD_EN = 0;
  __delay_us(100);
}

void LCD_Cmd(uint8_t Command)
{
  RS = 0;
  LCD_Write_Nibble(Command >> 4);
  LCD_Write_Nibble(Command);
  if((Command == LCD_CLEAR) || (Command == LCD_RETURN_HOME))
    __delay_ms(2);
}

void LCD_Goto(uint8_t col, uint8_t row)
{
  switch(row)
  {
    case 2:
      LCD_Cmd(LCD_SECOND_ROW + col - 1);
      break;
    case 3:
      LCD_Cmd(LCD_THIRD_ROW  + col - 1);
      break;
    case 4:
      LCD_Cmd(LCD_FOURTH_ROW + col - 1);
    break;
    default:      // case 1:
      LCD_Cmd(LCD_FIRST_ROW  + col - 1);
  }

}

void LCD_PutC(char LCD_Char)
{
  RS = 1;
  LCD_Write_Nibble(LCD_Char >> 4);
  LCD_Write_Nibble(LCD_Char );
}

void LCD_Print(char* LCD_Str)
{
  uint8_t i = 0;
  RS = 1;
  while(LCD_Str[i] != '\0')
  {
    LCD_Write_Nibble(LCD_Str[i] >> 4);
    LCD_Write_Nibble(LCD_Str[i++] );
  }
}

void LCD_Begin()
{
  RS = 0;

  LCD_RS     = 0;
  LCD_EN     = 0;
  LCD_D4     = 0;
  LCD_D5     = 0;
  LCD_D6     = 0;
  LCD_D7     = 0;
  LCD_RS_DIR = 0;
  LCD_EN_DIR = 0;
  LCD_D4_DIR = 0;
  LCD_D5_DIR = 0;
  LCD_D6_DIR = 0;
  LCD_D7_DIR = 0;

  __delay_ms(40);
  LCD_Cmd(3);
  __delay_ms(5);
  LCD_Cmd(3);
  __delay_ms(5);
  LCD_Cmd(3);
  __delay_ms(5);
  LCD_Cmd(LCD_RETURN_HOME);
  __delay_ms(5);
  LCD_Cmd(0x20 | (LCD_TYPE << 2));
  __delay_ms(50);
  LCD_Cmd(LCD_TURN_ON);
  __delay_ms(50);
  LCD_Cmd(LCD_CLEAR);
  __delay_ms(50);
  LCD_Cmd(LCD_ENTRY_MODE_SET | LCD_RETURN_HOME);
  __delay_ms(50);
}

Tagli

Eğer LCD'nin ilk satırı tamamen karelerle doluyorsa bu durum LCD'nin init yapılamadığını gösterir. LCD'yi işlemciye bağlamadan sadece besleme ve kontrast bağlantılarını yaparsan da bu durumla karşılaşırsın. Bu sebeple, eğer durum buysa sorunu kestirmek zor, yani daha LCD_Begin() fonksiyonunda sorun çıkıyor demektir. Kütüphanenin doğru çalıştığından eminsen (yani bir yerden hazır aldıysan) fiziksel bağlantıları ve işlemcinin clock ayarlarını kontrol etmeni öneririm. LCD_Begin()'i çağırmadan önce bir LED'i yarım saniye yak söndür. Bakalım gerçekten de zamanlama doğru mu?
Gökçe Tağlıoğlu

gokhangokcen

LCD ekranın kontrast ayarında sorun olabilir direnç değerini değiştir bakalım. (LCD'nin 3 nolu pinine giden bağlantı)
Bildiğini paylaşmak, Allah'ın verdiği öğrenme yeteneğinin zekatıdır.

PICaso

Alıntı yapılan: gokhangokcen - 25 Mart 2019, 16:48:03LCD ekranın kontrast ayarında sorun olabilir direnç değerini değiştir bakalım. (LCD'nin 3 nolu pinine giden bağlantı)
10K pot kullandım, herhangi bir değişiklik gözlemleyemedim.

PICaso

Alıntı yapılan: Tagli - 25 Mart 2019, 16:41:59Eğer LCD'nin ilk satırı tamamen karelerle doluyorsa bu durum LCD'nin init yapılamadığını gösterir. LCD'yi işlemciye bağlamadan sadece besleme ve kontrast bağlantılarını yaparsan da bu durumla karşılaşırsın. Bu sebeple, eğer durum buysa sorunu kestirmek zor, yani daha LCD_Begin() fonksiyonunda sorun çıkıyor demektir. Kütüphanenin doğru çalıştığından eminsen (yani bir yerden hazır aldıysan) fiziksel bağlantıları ve işlemcinin clock ayarlarını kontrol etmeni öneririm. LCD_Begin()'i çağırmadan önce bir LED'i yarım saniye yak söndür. Bakalım gerçekten de zamanlama doğru mu?
evet dediğiniz gibi ilk satırda bu durum oluyor. Dediğiniz şekilde değişiklikleri yapıp sonucu bildireceğim. Teşekkürler.

PICaso

Alıntı yapılan: Tagli - 25 Mart 2019, 16:41:59Eğer LCD'nin ilk satırı tamamen karelerle doluyorsa bu durum LCD'nin init yapılamadığını gösterir. LCD'yi işlemciye bağlamadan sadece besleme ve kontrast bağlantılarını yaparsan da bu durumla karşılaşırsın. Bu sebeple, eğer durum buysa sorunu kestirmek zor, yani daha LCD_Begin() fonksiyonunda sorun çıkıyor demektir. Kütüphanenin doğru çalıştığından eminsen (yani bir yerden hazır aldıysan) fiziksel bağlantıları ve işlemcinin clock ayarlarını kontrol etmeni öneririm. LCD_Begin()'i çağırmadan önce bir LED'i yarım saniye yak söndür. Bakalım gerçekten de zamanlama doğru mu?
Hocam dediğiniz şekilde ledi yan sön yaptırabiliyorum. Clock ayaralarında sorun görünmüyor. Bağlantıları da kontrol ettim. Şöyle bir durum var. Kütüphaneyi aldığım sitede ki arkadaş 16f887 kullanmıştı benim işlemcim 18f4520. Bu sorun oluşturabilir mi? Kütüphaneyi aldığım site: https://simple-circuit.com/pic-microcontroller-mplab-xc8-lcd/

ahuramazda

18 serisi piclerle uğraşırken Lcd bağlantısı tanımlamasında(çıkış olan portlar için) LAT register kullanmalısınız.

PICaso

#define LCD_RS_DIR   TRISD0
#define LCD_EN_DIR   TRISD1
#define LCD_D4_DIR   TRISD2
#define LCD_D5_DIR   TRISD3
#define LCD_D6_DIR   TRISD4
#define LCD_D7_DIR   TRISD5 
Kodu
#define LCD_RS_DIR   LATD0
#define LCD_EN_DIR   LATD1
#define LCD_D4_DIR   LATD2
#define LCD_D5_DIR   LATD3
#define LCD_D6_DIR   LATD4
#define LCD_D7_DIR   LATD5
Bu şekilde değiştirdim. Lcd de kareler dahi oluşmadı bu değişimden sonra.

Tagli

TRIS'ler değil, PORT'lar yerine LAT kullanılır.
Gökçe Tağlıoğlu

PICaso

Kodun içerisinde port komutu kullanılmıyor. Sizin önerebileceğiniz kütüphane dosyası var mıdır?

ahuramazda

#define LCD_RS       RD0
#define LCD_EN       RD1
#define LCD_D4       RD2
#define LCD_D5       RD3
#define LCD_D6       RD4
#define LCD_D7       RD5

Değişim yapacağınız yer yukarıdaki kısım. Tris registerına dokunmayın

Tagli

Yıllar önce yazdığım bir kodu forumda paylaşmıştım. Buradan ulaşabilirsin. Ancak çalışması için biraz kurcalamak gerekecek sanırım. Ayrıca kod "busy flag" (BF) okuması yaparak çalışıyor. Yani RW bacağı da kullanılıyor. Bunun iyi bir yaklaşım olmadığını yıllar sonra tatsız bir deneyimle öğrendim. BF okumayı iptal edip oraya bir delay koymakta fayda var. Ayrıca o zamanlar XC8'in delay kodu biraz sorunlu olduğu için kendim bir şeyler yazmak zorunda kalmıştım. Buna artık gerek kalmadı sanırım. Programdaki delay'ler XC8'in kendi delay fonksiyonları ile değiştirilebilir.
Gökçe Tağlıoğlu