связь SPI LCD st9720 с ATmega328P

#c #spi #atmega #atmel #lcd

#c #spi #атмега #atmel #ЖК-дисплей #atmega

Вопрос:

В настоящее время я пытаюсь записать простую строку на свой ЖК-дисплей, который использует ST9720, когда я запускаю код, который я получил для печати простой строки, я получаю довольно много шума на ЖК-дисплее. Я не уверен, почему, я думал, может быть, значение тактовой частоты, но тогда я очень не уверен, на что оно должно быть установлено, чтобы избавиться от шума на ЖК-дисплее. ЖК-дисплей подключен к микросхеме ATmega328P, которая подключена к программатору avr, который обменивается данными через spi. Это также использует atmel, я получил код из https://github.com/olikraus/u8g2/wiki/u8g2as7 Он использует библиотеку u8g2. Любая помощь приветствуется.

 #include <avr/io.h>
#include <u8g2.h>
#include <util/delay.h>

#define DISPLAY_CLK_DIR DDRB
#define DISPLAY_CLK_PORT PORTB
#define DISPLAY_CLK_PIN 5

#define DISPLAY_DATA_DIR DDRB
#define DISPLAY_DATA_PORT PORTB
#define DISPLAY_DATA_PIN 3

#define DISPLAY_CS_DIR DDRB
#define DISPLAY_CS_PORT PORTB
#define DISPLAY_CS_PIN 2

//#define DISPLAY_DC_DIR DDRB
//#define DISPLAY_DC_PORT PORTB
//#define DISPLAY_DC_PIN 1

#define DISPLAY_RESET_DIR DDRB
#define DISPLAY_RESET_PORT PORTB
#define DISPLAY_RESET_PIN 0


#define P_CPU_NS (1000000000UL / F_CPU)

u8g2_t u8g2;

uint8_t u8x8_avr_delay(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
{
    uint8_t cycles;

    switch(msg)
    {
        case U8X8_MSG_DELAY_NANO:     // delay arg_int * 1 nano second
            // At 20Mhz, each cycle is 50ns, the call itself is slower.
            break;
        case U8X8_MSG_DELAY_100NANO:    // delay arg_int * 100 nano seconds
            // Approximate best case values...
#define CALL_CYCLES 26UL
#define CALC_CYCLES 4UL
#define RETURN_CYCLES 4UL
#define CYCLES_PER_LOOP 4UL

            cycles = (100UL * arg_int) / (P_CPU_NS * CYCLES_PER_LOOP);

            if(cycles > CALL_CYCLES   RETURN_CYCLES   CALC_CYCLES) 
                break;

            __asm__ __volatile__ (
            "1: sbiw %0,1" "nt" // 2 cycles
            "brne 1b" : "=w" (cycles) : "0" (cycles) // 2 cycles
            );
            break;
        case U8X8_MSG_DELAY_10MICRO:    // delay arg_int * 10 micro seconds
            for(int i=0 ; i < arg_int ; i  )
                _delay_us(10);
            break;
        case U8X8_MSG_DELAY_MILLI:      // delay arg_int * 1 milli second
            for(int i=0 ; i < arg_int ; i  )
                _delay_ms(1);
            break;
        default:
            return 0;
    }
    return 1;
}


uint8_t u8x8_avr_gpio_and_delay(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
{
    // Re-use library for delays

    switch(msg)
    {
        case U8X8_MSG_GPIO_AND_DELAY_INIT:  // called once during init phase of u8g2/u8x8
            DISPLAY_CLK_DIR |= 1<<DISPLAY_CLK_PIN;
            DISPLAY_DATA_DIR |= 1<<DISPLAY_DATA_PIN;
            DISPLAY_CS_DIR |= 1<<DISPLAY_CS_PIN;
            //DISPLAY_DC_DIR |= 1<<DISPLAY_DC_PIN;
            DISPLAY_RESET_DIR |= 1<<DISPLAY_RESET_PIN;
            break;              // can be used to setup pins
        case U8X8_MSG_GPIO_SPI_CLOCK:        // Clock pin: Output level in arg_int
            if(arg_int)
                DISPLAY_CLK_PORT |= (1<<DISPLAY_CLK_PIN);
            else
                DISPLAY_CLK_PORT amp;= ~(1<<DISPLAY_CLK_PIN);
            break;
        case U8X8_MSG_GPIO_SPI_DATA:        // MOSI pin: Output level in arg_int
            if(arg_int)
                DISPLAY_DATA_PORT |= (1<<DISPLAY_DATA_PIN);
            else
                DISPLAY_DATA_PORT amp;= ~(1<<DISPLAY_DATA_PIN);
            break;
        case U8X8_MSG_GPIO_CS:        // CS (chip select) pin: Output level in arg_int
            if(arg_int)
                DISPLAY_CS_PORT |= (1<<DISPLAY_CS_PIN);
            else
                DISPLAY_CS_PORT amp;= ~(1<<DISPLAY_CS_PIN);
            break;
        //case U8X8_MSG_GPIO_DC:        // DC (data/cmd, A0, register select) pin: Output level in arg_int
        //  if(arg_int)
        //      DISPLAY_DC_PORT |= (1<<DISPLAY_DC_PIN);
        //  else
        //      DISPLAY_DC_PORT amp;= ~(1<<DISPLAY_DC_PIN);
        //  break;
        
        case U8X8_MSG_GPIO_RESET:     // Reset pin: Output level in arg_int
            if(arg_int)
                DISPLAY_RESET_PORT |= (1<<DISPLAY_RESET_PIN);
            else
                DISPLAY_RESET_PORT amp;= ~(1<<DISPLAY_RESET_PIN);
            break;
        default:
            if (u8x8_avr_delay(u8x8, msg, arg_int, arg_ptr))    // check for any delay msgs
                return 1;
            u8x8_SetGPIOResult(u8x8, 1);      // default return value
            break;
    }
    return 1;
}

int main(void)
{
    /*
        Select a setup procedure for your display from here: https://github.com/olikraus/u8g2/wiki/u8g2setupc
        1. Arg: Address of an empty u8g2 structure
        2. Arg: Usually U8G2_R0, others are listed here: https://github.com/olikraus/u8g2/wiki/u8g2reference#carduino-example
        3. Arg: Protocol procedure (u8g2-byte), list is here: https://github.com/olikraus/u8g2/wiki/Porting-to-new-MCU-platform#communication-callback-eg-u8x8_byte_hw_i2c
        4. Arg: Defined in this code itself (see above)
    */
    u8g2_Setup_st7920_s_128x64_1(amp;u8g2, U8G2_R0, u8x8_byte_4wire_sw_spi, u8x8_avr_gpio_and_delay);
    u8g2_InitDisplay(amp;u8g2);
    u8g2_SetPowerSave(amp;u8g2, 0);
    
    /* full buffer example, setup procedure ends in _f */
    u8g2_ClearBuffer(amp;u8g2);
    u8g2_SetFont(amp;u8g2, u8g2_font_ncenB14_tr);
    u8g2_DrawStr(amp;u8g2, 1, 18, "U8g2 on AVR");
    u8g2_SendBuffer(amp;u8g2);
        
    while(1){
    }
}
  

Комментарии:

1. Какую скорость передачи данных вы используете и сколько данных вам нужно отправить? Что заставляет вас думать, что это программная проблема?

2. @Lundin используя скорость передачи данных 9600, нам не нужно отправлять много данных, просто строку на этом этапе. Я не совсем уверен, что это проблема с программным обеспечением, просто одна из первых вещей, о которых я подумал, что это может быть, поскольку он печатает часть слова, а остальная часть экрана — шум

3. @Lundin в настоящее время я подключил следующее на аппаратном уровне, на ЖК-дисплее у нас есть CLK_dir, подключенный к контакту включения на ЖК-дисплее. Data_dir подключен к RW, cs_dir подключен к CS, reset_dir подключен к RST. но я прокомментировал вывод DC_dir как неуверенный, нужен ли он и к какому выводу на ЖК-дисплее он должен быть подключен

4. Это мне ничего не говорит. Стандартов для ЖК-дисплеев столько же, сколько и запчастей на рынке, в значительной степени. Если вы хотите устранить неполадки в оборудовании, вам необходимо опубликовать схему на electronics.stackexchange.com

Ответ №1:

Использовать u8g2_Setup_st7920_s_128x64_f() вместо использования u8g2_Setup_st7920_s_128x64_1() .