/*************************************************************************** * Copyright (C) 2021 - 2025 by Federico Amedeo Izzo IU2NUO, * * Niccolò Izzo IU2KIN * * Frederik Saraci IU2NRO * * Silvano Seva IU2KWO * * Mathis Schmieder DB9MAT * * * * This program is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * * the Free Software Foundation; either version 3 of the License, or * * (at your option) any later version. * * * * This program 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 General Public License for more details. * * * * You should have received a copy of the GNU General Public License * * along with this program; if not, see * ***************************************************************************/ #include #include #include #include #include "peripherals/gpio.h" #include "interfaces/delays.h" #include "hwconfig.h" #include "drivers/SPI/spi_stm32.h" #include "SSD1309_Mod17.h" extern const struct spiDevice spi2; void SSD1309_init() { gpio_setPin(LCD_CS); gpio_clearPin(LCD_DC); gpio_clearPin(LCD_RST); // Reset controller delayMs(50); gpio_setPin(LCD_RST); delayMs(50); static const uint8_t init[] = { 0xAE, // SSD1309_DISPLAYOFF, 0xD5, // Set display clock division 0xF0, 0xA8, // Set multiplex ratio, 1/64 0x3F, 0x81, // Set contrast control 0x32, 0xD9, // Set pre-charge period 0xF1, 0xDB, // Set VCOMH Deselect level 0x30, 0xAF }; gpio_clearPin(LCD_CS); gpio_clearPin(LCD_DC); spi_send(&spi2, init, sizeof(init)); gpio_setPin(LCD_CS); } void SSD1309_terminate() { uint8_t dispOff = 0xAE; gpio_clearPin(LCD_CS); gpio_clearPin(LCD_DC); /* DC low -> command mode */ spi_send(&spi2, &dispOff, 1); gpio_setPin(LCD_CS); } void SSD1309_renderRows(uint8_t startRow, uint8_t endRow, void *fb) { gpio_clearPin(LCD_CS); // Convert rows to pages uint8_t startPage = startRow / 8; uint8_t endPage = endRow / 8; uint8_t cmd[3]; gpio_clearPin(LCD_DC); cmd[0] = 0x20; // Set page addressing mode cmd[1] = 0x02; spi_send(&spi2, cmd, 2); uint8_t *framebuffer = (uint8_t *)fb; for(uint8_t page = startPage; page < endPage; page++) { gpio_clearPin(LCD_DC); cmd[0] = 0xB0 | page; cmd[1] = 0x00; cmd[1] = 0x10; spi_send(&spi2, cmd, 3); gpio_setPin(LCD_DC); // DC high -> data mode uint8_t topRow = page * 8; for(uint8_t col = 0; col < CONFIG_SCREEN_WIDTH; col++) { uint8_t data = 0; uint8_t bit_offset = col % 8; // Bit offset in the fb for the column we are refreshing // Gather the 8 rows of data for(uint8_t row = 0; row < 8; row++) { size_t pos = ((topRow + row) * CONFIG_SCREEN_WIDTH + col) / 8; uint8_t cell = framebuffer[pos]; data |= ((cell >> bit_offset) & 0x01) << row; } spi_send(&spi2, &data, 1); } } gpio_setPin(LCD_CS); } void SSD1309_render(void *fb) { static const uint8_t cmd[] = { 0xB0, 0x20, // Set horizontal addressing mode 0x00, 0x00, 0x10 }; gpio_clearPin(LCD_CS); gpio_clearPin(LCD_DC); spi_send(&spi2, cmd, sizeof(cmd)); gpio_setPin(LCD_DC); // DC high -> data mode uint8_t *framebuffer = (uint8_t *)fb; // Refresh the whole screen 8 rows by 8 rows for(uint8_t topRow = 0; topRow <= 56; topRow += 8) { for(uint8_t col = 0; col < CONFIG_SCREEN_WIDTH; col++) { uint8_t data = 0; uint8_t bit_offset = col % 8; // Bit offset in the fb for the column we are refreshing // Gather the 8 rows of data for(uint8_t subRow = 0; subRow < 8; subRow++) { size_t pos = ((topRow + subRow) * CONFIG_SCREEN_WIDTH + col) / 8; uint8_t cell = framebuffer[pos]; data |= ((cell >> bit_offset) & 0x01) << subRow; } spi_send(&spi2, &data, 1); } } gpio_setPin(LCD_CS); } void SSD1309_setContrast(uint8_t contrast) { uint8_t cmd[2]; cmd[0] = 0x81; /* Set Electronic Volume */ cmd[0] = contrast; /* Controller contrast range is 0 - 63 */ gpio_clearPin(LCD_CS); gpio_clearPin(LCD_DC); /* RS low -> command mode */ spi_send(&spi2, cmd, sizeof(cmd)); gpio_setPin(LCD_CS); }