From 8b14855007b08b5b97798304471d5ade8de2f103 Mon Sep 17 00:00:00 2001 From: Kevin <68612569+diyelectromusic@users.noreply.github.com> Date: Fri, 3 May 2024 14:06:22 -0700 Subject: [PATCH] Initial build with basic structure to support st7789device once added to circle. --- src/config.cpp | 49 ++++++++++++++++++++++++++++++++++ src/config.h | 27 ++++++++++++++++++- src/kernel.cpp | 25 +++++++++++++++++- src/kernel.h | 2 ++ src/minidexed.cpp | 4 +-- src/minidexed.h | 3 ++- src/minidexed.ini | 14 ++++++++++ src/userinterface.cpp | 61 +++++++++++++++++++++++++++++++++++++++---- src/userinterface.h | 7 ++++- 9 files changed, 181 insertions(+), 11 deletions(-) diff --git a/src/config.cpp b/src/config.cpp index bb65840..93ecc87 100644 --- a/src/config.cpp +++ b/src/config.cpp @@ -102,6 +102,15 @@ void CConfig::Load (void) m_bSSD1306LCDRotate = m_Properties.GetNumber ("SSD1306LCDRotate", 0) != 0; m_bSSD1306LCDMirror = m_Properties.GetNumber ("SSD1306LCDMirror", 0) != 0; + m_nSPIBus = m_Properties.GetNumber ("SPIBus", SPI_INACTIVE); // Disabled by default + m_bST7789Enabled = m_Properties.GetNumber ("ST7789Enabled", 0) != 0; + m_nST7789Data = m_Properties.GetNumber ("ST7789Data", 0); + m_nST7789Select = m_Properties.GetNumber ("ST7789Select", 0); + m_nST7789Reset = m_Properties.GetNumber ("ST7789Reset", 0); // optional + m_nST7789Backlight = m_Properties.GetNumber ("ST7789Backlight", 0); // optional + m_nST7789Width = m_Properties.GetNumber ("ST7789Width", 240); + m_nST7789Height = m_Properties.GetNumber ("ST7789Height", 240); + m_nLCDColumns = m_Properties.GetNumber ("LCDColumns", 16); m_nLCDRows = m_Properties.GetNumber ("LCDRows", 2); @@ -299,6 +308,46 @@ bool CConfig::GetSSD1306LCDMirror (void) const return m_bSSD1306LCDMirror; } +unsigned CConfig::GetSPIBus (void) const +{ + return m_nSPIBus; +} + +bool CConfig::GetST7789Enabled (void) const +{ + return m_bST7789Enabled; +} + +unsigned CConfig::GetST7789Data (void) const +{ + return m_nST7789Data; +} + +unsigned CConfig::GetST7789Select (void) const +{ + return m_nST7789Select; +} + +unsigned CConfig::GetST7789Reset (void) const +{ + return m_nST7789Reset; +} + +unsigned CConfig::GetST7789Backlight (void) const +{ + return m_nST7789Backlight; +} + +unsigned CConfig::GetST7789Width (void) const +{ + return m_nST7789Width; +} + +unsigned CConfig::GetST7789Height (void) const +{ + return m_nST7789Height; +} + unsigned CConfig::GetLCDColumns (void) const { return m_nLCDColumns; diff --git a/src/config.h b/src/config.h index 55c038e..bb2c167 100644 --- a/src/config.h +++ b/src/config.h @@ -28,6 +28,12 @@ #include #include +#define SPI_INACTIVE 255 +#define SPI_CLOCK_SPEED 15000000 // Hz +#define SPI_CPOL 1 // Taken from circle sample application +#define SPI_CPHA 0 + + class CConfig // Configuration for MiniDexed { public: @@ -103,6 +109,16 @@ public: bool GetSSD1306LCDRotate (void) const; bool GetSSD1306LCDMirror (void) const; + // ST7789 LCD + unsigned GetSPIBus (void) const; + bool GetST7789Enabled (void) const; + unsigned GetST7789Data (void) const; + unsigned GetST7789Select (void) const; + unsigned GetST7789Reset (void) const; + unsigned GetST7789Backlight (void) const; + unsigned GetST7789Width (void) const; + unsigned GetST7789Height (void) const; + unsigned GetLCDColumns (void) const; unsigned GetLCDRows (void) const; @@ -204,7 +220,16 @@ private: unsigned m_nSSD1306LCDHeight; bool m_bSSD1306LCDRotate; bool m_bSSD1306LCDMirror; - + + unsigned m_nSPIBus; + bool m_bST7789Enabled; + unsigned m_nST7789Data; + unsigned m_nST7789Select; + unsigned m_nST7789Reset; + unsigned m_nST7789Backlight; + unsigned m_nST7789Width; + unsigned m_nST7789Height; + unsigned m_nLCDColumns; unsigned m_nLCDRows; diff --git a/src/kernel.cpp b/src/kernel.cpp index 3ff835c..65c72c8 100644 --- a/src/kernel.cpp +++ b/src/kernel.cpp @@ -33,6 +33,7 @@ CKernel::CKernel (void) m_Config (&mFileSystem), m_GPIOManager (&mInterrupt), m_I2CMaster (CMachineInfo::Get ()->GetDevice (DeviceI2CMaster), TRUE), + m_pSPIMaster (nullptr), m_pDexed (0) { s_pThis = this; @@ -66,6 +67,28 @@ bool CKernel::Initialize (void) m_Config.Load (); + unsigned nSPIMaster = m_Config.GetSPIBus(); +#if RASPPI<4 + // By default older RPI versions use SPI 0. + // It is possible to build circle to support SPI 1 for + // devices that use the 40-pin header, but that isn't + // enabled at present... + if (nSPIMaster == 0) +#else + // RPI 4+ has several possible SPI Bus Configurations. + // As mentioned above, SPI 1 is not built by default. + // See circle/include/circle/spimaster.h + if (nSPIMaster == 0 || nSPIMaster == 3 || nSPIMaster == 4 || nSPIMaster == 5 || nSPIMaster == 6) +#endif + { + m_pSPIMaster = new CSPIMaster (SPI_CLOCK_SPEED, SPI_CPOL, SPI_CPHA, nSPIMaster); + if (!m_pSPIMaster->Initialize()) + { + delete (m_pSPIMaster); + m_pSPIMaster = nullptr; + } + } + if (m_Config.GetUSBGadgetMode()) { #if RASPPI==5 @@ -86,7 +109,7 @@ bool CKernel::Initialize (void) return FALSE; } - m_pDexed = new CMiniDexed (&m_Config, &mInterrupt, &m_GPIOManager, &m_I2CMaster, + m_pDexed = new CMiniDexed (&m_Config, &mInterrupt, &m_GPIOManager, &m_I2CMaster, m_pSPIMaster, &mFileSystem); assert (m_pDexed); diff --git a/src/kernel.h b/src/kernel.h index 7d2f346..efe5f4f 100644 --- a/src/kernel.h +++ b/src/kernel.h @@ -24,6 +24,7 @@ #include #include #include +#include #include #include "config.h" #include "minidexed.h" @@ -54,6 +55,7 @@ private: CCPUThrottle m_CPUThrottle; CGPIOManager m_GPIOManager; CI2CMaster m_I2CMaster; + CSPIMaster *m_pSPIMaster; CMiniDexed *m_pDexed; CUSBController *m_pUSB; diff --git a/src/minidexed.cpp b/src/minidexed.cpp index e2576b5..9547baf 100644 --- a/src/minidexed.cpp +++ b/src/minidexed.cpp @@ -31,13 +31,13 @@ LOGMODULE ("minidexed"); CMiniDexed::CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, - CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, FATFS *pFileSystem) + CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, CSPIMaster *pSPIMaster, FATFS *pFileSystem) : #ifdef ARM_ALLOW_MULTI_CORE CMultiCoreSupport (CMemorySystem::Get ()), #endif m_pConfig (pConfig), - m_UI (this, pGPIOManager, pI2CMaster, pConfig), + m_UI (this, pGPIOManager, pI2CMaster, pSPIMaster, pConfig), m_PerformanceConfig (pFileSystem), m_PCKeyboard (this, pConfig, &m_UI), m_SerialMIDI (this, pInterrupt, pConfig, &m_UI), diff --git a/src/minidexed.h b/src/minidexed.h index 407d5db..8ca74c8 100644 --- a/src/minidexed.h +++ b/src/minidexed.h @@ -36,6 +36,7 @@ #include #include #include +#include #include #include #include @@ -51,7 +52,7 @@ class CMiniDexed { public: CMiniDexed (CConfig *pConfig, CInterruptSystem *pInterrupt, - CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, FATFS *pFileSystem); + CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, CSPIMaster *pSPIMaster, FATFS *pFileSystem); bool Initialize (void); diff --git a/src/minidexed.ini b/src/minidexed.ini index 84d76ac..792ef58 100644 --- a/src/minidexed.ini +++ b/src/minidexed.ini @@ -55,6 +55,20 @@ SSD1306LCDHeight=32 SSD1306LCDRotate=0 SSD1306LCDMirror=0 +# ST7789 LCD +# SPIBus=0 for any RPi (GPIO 10,11,8,7) +# Select=0|1 for CE0 or CE1 +# Data = GPIO pin number +# Optional: Reset, Backlight = GPIO pin numbers +SPIBus= +ST7789Enabled=0 +ST7789Data= +ST7789Select= +ST7789Reset= +ST7789Backlight= +ST7789Width=240 +ST7789Height=240 + # Default is 16x2 display (e.g. HD44780) LCDColumns=16 LCDRows=2 diff --git a/src/userinterface.cpp b/src/userinterface.cpp index 241605e..c26b223 100644 --- a/src/userinterface.cpp +++ b/src/userinterface.cpp @@ -27,10 +27,11 @@ LOGMODULE ("ui"); -CUserInterface::CUserInterface (CMiniDexed *pMiniDexed, CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, CConfig *pConfig) +CUserInterface::CUserInterface (CMiniDexed *pMiniDexed, CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, CSPIMaster *pSPIMaster, CConfig *pConfig) : m_pMiniDexed (pMiniDexed), m_pGPIOManager (pGPIOManager), m_pI2CMaster (pI2CMaster), + m_pSPIMaster (pSPIMaster), m_pConfig (pConfig), m_pLCD (0), m_pLCDBuffered (0), @@ -57,17 +58,65 @@ bool CUserInterface::Initialize (void) { unsigned i2caddr = m_pConfig->GetLCDI2CAddress (); unsigned ssd1306addr = m_pConfig->GetSSD1306LCDI2CAddress (); + bool st7789 = m_pConfig->GetST7789Enabled (); if (ssd1306addr != 0) { m_pSSD1306 = new CSSD1306Device (m_pConfig->GetSSD1306LCDWidth (), m_pConfig->GetSSD1306LCDHeight (), m_pI2CMaster, ssd1306addr, m_pConfig->GetSSD1306LCDRotate (), m_pConfig->GetSSD1306LCDMirror ()); - LOGDBG ("LCD: SSD1306"); if (!m_pSSD1306->Initialize ()) { + LOGDBG("LCD: SSD1306 initialization failed"); return false; } + LOGDBG ("LCD: SSD1306"); m_pLCD = m_pSSD1306; - } else if (i2caddr == 0) + } + else if (st7789) + { + if (m_pSPIMaster == nullptr) + { + LOGDBG("LCD: ST7789 Enabled but SPI Initialisation Failed"); + return false; + } + + LOGDBG ("ST7789 Params: dc=%d, cs=%d", m_pConfig->GetST7789Data(), m_pConfig->GetST7789Select()); + + m_pST7789Display = new CST7789Display (m_pSPIMaster, + m_pConfig->GetST7789Data(), + m_pConfig->GetST7789Reset(), + m_pConfig->GetST7789Backlight(), + m_pConfig->GetST7789Width(), + m_pConfig->GetST7789Height(), + SPI_CPOL, SPI_CPHA, SPI_CLOCK_SPEED, + m_pConfig->GetST7789Select()); + LOGDBG ("ST7789 Display OK"); + if (m_pST7789Display->Initialize()) + { + m_pST7789 = new CST7789Device (m_pSPIMaster, m_pST7789Display, m_pConfig->GetLCDColumns (), m_pConfig->GetLCDRows ()); + if (m_pST7789->Initialize()) + { + LOGDBG ("LCD: ST7789"); + m_pLCD = m_pST7789; + } + else + { + LOGDBG ("LCD: Failed to initalize ST7789 character device"); + delete (m_pST7789); + delete (m_pST7789Display); + m_pST7789 = nullptr; + m_pST7789Display = nullptr; + return false; + } + } + else + { + LOGDBG ("LCD: Failed to initialize ST7789 display"); + delete (m_pST7789Display); + m_pST7789Display = nullptr; + return false; + } + } + else if (i2caddr == 0) { m_pHD44780 = new CHD44780Device (m_pConfig->GetLCDColumns (), m_pConfig->GetLCDRows (), m_pConfig->GetLCDPinData4 (), @@ -77,22 +126,24 @@ bool CUserInterface::Initialize (void) m_pConfig->GetLCDPinEnable (), m_pConfig->GetLCDPinRegisterSelect (), m_pConfig->GetLCDPinReadWrite ()); - LOGDBG ("LCD: HD44780"); if (!m_pHD44780->Initialize ()) { + LOGDBG("LCD: HD44780 initialization failed"); return false; } + LOGDBG ("LCD: HD44780"); m_pLCD = m_pHD44780; } else { m_pHD44780 = new CHD44780Device (m_pI2CMaster, i2caddr, m_pConfig->GetLCDColumns (), m_pConfig->GetLCDRows ()); - LOGDBG ("LCD: HD44780 I2C"); if (!m_pHD44780->Initialize ()) { + LOGDBG("LCD: HD44780 (I2C) initialization failed"); return false; } + LOGDBG ("LCD: HD44780 I2C"); m_pLCD = m_pHD44780; } assert (m_pLCD); diff --git a/src/userinterface.h b/src/userinterface.h index 5de2846..a8026db 100644 --- a/src/userinterface.h +++ b/src/userinterface.h @@ -26,16 +26,18 @@ #include #include #include +#include #include #include #include +#include class CMiniDexed; class CUserInterface { public: - CUserInterface (CMiniDexed *pMiniDexed, CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, CConfig *pConfig); + CUserInterface (CMiniDexed *pMiniDexed, CGPIOManager *pGPIOManager, CI2CMaster *pI2CMaster, CSPIMaster *pSPIMaster, CConfig *pConfig); ~CUserInterface (void); bool Initialize (void); @@ -68,11 +70,14 @@ private: CMiniDexed *m_pMiniDexed; CGPIOManager *m_pGPIOManager; CI2CMaster *m_pI2CMaster; + CSPIMaster *m_pSPIMaster; CConfig *m_pConfig; CCharDevice *m_pLCD; CHD44780Device *m_pHD44780; CSSD1306Device *m_pSSD1306; + CST7789Display *m_pST7789Display; + CST7789Device *m_pST7789; CWriteBufferDevice *m_pLCDBuffered; CUIButtons *m_pUIButtons;