ssd13x6.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298
  1. /**
  2. * Copyright (c) 2017-2018 Tara Keeling
  3. *
  4. * This software is released under the MIT License.
  5. * https://opensource.org/licenses/MIT
  6. */
  7. #include <stdio.h>
  8. #include <string.h>
  9. #include <stdint.h>
  10. #include <stdbool.h>
  11. #include <stdlib.h>
  12. #include <math.h>
  13. #include <esp_heap_caps.h>
  14. #include "ssd13x6.h"
  15. #define COM_Disable_LR_Remap 0
  16. #define COM_Enable_LR_Remap BIT( 5 )
  17. #define COM_Pins_Sequential 0
  18. #define COM_Pins_Alternative BIT( 4 )
  19. #define COM_ScanDir_LR 0
  20. #define COM_ScanDir_RL 1
  21. // used by both but different
  22. static uint8_t SSDCmd_Set_Display_Start_Line;
  23. static uint8_t SSDCmd_Set_Display_Offset;
  24. static uint8_t SSDCmd_Set_Column_Address;
  25. static uint8_t SSDCmd_Set_Display_CLK;
  26. static uint8_t SSDCmd_Set_Page_Address;
  27. // misc boundaries
  28. static uint8_t SSD13x6_Max_Col;
  29. static const uint8_t SSD13x6_Max_Row = 7;
  30. static bool SSD13x6_Init( struct SSD13x6_Device* DeviceHandle, int Width, int Height );
  31. bool SSD13x6_WriteCommand( struct SSD13x6_Device* DeviceHandle, SSDCmd SSDCommand ) {
  32. NullCheck( DeviceHandle->WriteCommand, return false );
  33. return ( DeviceHandle->WriteCommand ) ( DeviceHandle, SSDCommand );
  34. }
  35. bool SSD13x6_WriteData( struct SSD13x6_Device* DeviceHandle, uint8_t* Data, size_t DataLength ) {
  36. NullCheck( DeviceHandle->WriteData, return false );
  37. return ( DeviceHandle->WriteData ) ( DeviceHandle, Data, DataLength );
  38. }
  39. void SSD13x6_SetMuxRatio( struct SSD13x6_Device* DeviceHandle, uint8_t Ratio ) {
  40. SSD13x6_WriteCommand( DeviceHandle, 0xA8 );
  41. SSD13x6_WriteCommand( DeviceHandle, Ratio );
  42. }
  43. void SSD13x6_SetDisplayOffset( struct SSD13x6_Device* DeviceHandle, uint8_t Offset ) {
  44. SSD13x6_WriteCommand( DeviceHandle, SSDCmd_Set_Display_Offset );
  45. SSD13x6_WriteCommand( DeviceHandle, Offset );
  46. }
  47. void SSD13x6_SetDisplayStartLine( struct SSD13x6_Device* DeviceHandle, int Line ) {
  48. SSD13x6_WriteCommand( DeviceHandle, SSDCmd_Set_Display_Start_Line + ( uint32_t ) ( Line & 0x1F ) );
  49. }
  50. void SSD13x6_SetContrast( struct SSD13x6_Device* DeviceHandle, uint8_t Contrast ) {
  51. SSD13x6_WriteCommand( DeviceHandle, 0x81 );
  52. SSD13x6_WriteCommand( DeviceHandle, Contrast );
  53. }
  54. void SSD13x6_EnableDisplayRAM( struct SSD13x6_Device* DeviceHandle ) {
  55. SSD13x6_WriteCommand( DeviceHandle, 0xA4 );
  56. }
  57. void SSD13x6_DisableDisplayRAM( struct SSD13x6_Device* DeviceHandle ) {
  58. SSD13x6_WriteCommand( DeviceHandle, 0xA5 );
  59. }
  60. void SSD13x6_SetInverted( struct SSD13x6_Device* DeviceHandle, bool Inverted ) {
  61. SSD13x6_WriteCommand( DeviceHandle, Inverted ? 0xA7 : 0xA6 );
  62. }
  63. void SSD13x6_SetDisplayClocks( struct SSD13x6_Device* DeviceHandle, uint32_t DisplayClockDivider, uint32_t OSCFrequency ) {
  64. DisplayClockDivider&= 0x0F;
  65. OSCFrequency&= 0x0F;
  66. SSD13x6_WriteCommand( DeviceHandle, SSDCmd_Set_Display_CLK );
  67. SSD13x6_WriteCommand( DeviceHandle, ( ( OSCFrequency << 4 ) | DisplayClockDivider ) );
  68. }
  69. void SSD13x6_DisplayOn( struct SSD13x6_Device* DeviceHandle ) {
  70. SSD13x6_WriteCommand( DeviceHandle, 0xAF );
  71. }
  72. void SSD13x6_DisplayOff( struct SSD13x6_Device* DeviceHandle ) {
  73. SSD13x6_WriteCommand( DeviceHandle, 0xAE );
  74. }
  75. void SSD132x_ReMap( struct SSD13x6_Device* DeviceHandle ) {
  76. SSD13x6_WriteCommand( DeviceHandle, 0xA0 );
  77. SSD13x6_WriteCommand( DeviceHandle, DeviceHandle->ReMap );
  78. }
  79. void SSD13x6_SetDisplayAddressMode( struct SSD13x6_Device* DeviceHandle, SSD13x6_AddressMode AddressMode ) {
  80. switch (DeviceHandle->Model) {
  81. case SSD1306:
  82. SSD13x6_WriteCommand( DeviceHandle, 0x20 );
  83. SSD13x6_WriteCommand( DeviceHandle, AddressMode );
  84. break;
  85. case SSD1326:
  86. DeviceHandle->ReMap = (AddressMode == AddressMode_Horizontal) ? (DeviceHandle->ReMap & ~0x80) : (DeviceHandle->ReMap | 0x80);
  87. SSD132x_ReMap(DeviceHandle);
  88. break;
  89. }
  90. }
  91. void SSD13x6_Update( struct SSD13x6_Device* DeviceHandle ) {
  92. SSD13x6_SetColumnAddress( DeviceHandle, 0, DeviceHandle->Width - 1);
  93. SSD13x6_SetPageAddress( DeviceHandle, 0, DeviceHandle->Height / 8 - 1);
  94. SSD13x6_WriteData( DeviceHandle, DeviceHandle->Framebuffer, DeviceHandle->FramebufferSize );
  95. }
  96. void SSD13x6_WriteRawData( struct SSD13x6_Device* DeviceHandle, uint8_t* Data, size_t DataLength ) {
  97. NullCheck( Data, return );
  98. DataLength = DataLength > DeviceHandle->FramebufferSize ? DeviceHandle->FramebufferSize : DataLength;
  99. if ( DataLength > 0 ) SSD13x6_WriteData( DeviceHandle, Data, DataLength );
  100. }
  101. void SSD13x6_SetHFlip( struct SSD13x6_Device* DeviceHandle, bool On ) {
  102. switch (DeviceHandle->Model) {
  103. case SSD1306:
  104. SSD13x6_WriteCommand( DeviceHandle, On ? 0xA1 : 0xA0 );
  105. break;
  106. case SSD1326:
  107. DeviceHandle->ReMap = On ? (DeviceHandle->ReMap | 0x01) : (DeviceHandle->ReMap & ~0x01);
  108. SSD132x_ReMap(DeviceHandle);
  109. break;
  110. }
  111. }
  112. void SSD13x6_SetVFlip( struct SSD13x6_Device* DeviceHandle, bool On ) {
  113. switch (DeviceHandle->Model) {
  114. case SSD1306:
  115. SSD13x6_WriteCommand( DeviceHandle, On ? 0xC8 : 0xC0 );
  116. break;
  117. case SSD1326:
  118. DeviceHandle->ReMap = On ? (DeviceHandle->ReMap | 0x05) : (DeviceHandle->ReMap & ~0x05);
  119. SSD132x_ReMap( DeviceHandle );
  120. break;
  121. }
  122. }
  123. void SSD13x6_SetColumnAddress( struct SSD13x6_Device* DeviceHandle, uint8_t Start, uint8_t End ) {
  124. CheckBounds( Start > SSD13x6_Max_Col, return );
  125. CheckBounds( End > SSD13x6_Max_Col, return );
  126. SSD13x6_WriteCommand( DeviceHandle, SSDCmd_Set_Column_Address );
  127. SSD13x6_WriteCommand( DeviceHandle, Start );
  128. SSD13x6_WriteCommand( DeviceHandle, End );
  129. }
  130. void SSD13x6_SetPageAddress( struct SSD13x6_Device* DeviceHandle, uint8_t Start, uint8_t End ) {
  131. NullCheck( DeviceHandle, return );
  132. CheckBounds( Start > SSD13x6_Max_Row, return );
  133. CheckBounds( End > SSD13x6_Max_Row, return );
  134. // in case of SSD1326, this is sub-optimal as it can address by line, not by page
  135. if (DeviceHandle->Model != SSD1306) {
  136. Start *= 8;
  137. End = (End + 1) * 8 - 1;
  138. }
  139. SSD13x6_WriteCommand( DeviceHandle, SSDCmd_Set_Page_Address );
  140. SSD13x6_WriteCommand( DeviceHandle, Start );
  141. SSD13x6_WriteCommand( DeviceHandle, End );
  142. }
  143. bool SSD13x6_HWReset( struct SSD13x6_Device* DeviceHandle ) {
  144. NullCheck( DeviceHandle, return 0 );
  145. if ( DeviceHandle->Reset != NULL ) {
  146. return ( DeviceHandle->Reset ) ( DeviceHandle );
  147. }
  148. /* This should always return true if there is no reset callback as
  149. * no error would have occurred during the non existant reset.
  150. */
  151. return true;
  152. }
  153. /*
  154. * This is all a big giant mystery that I have yet to figure out.
  155. * Beware all ye who enter.
  156. */
  157. static void SetCOMPinConfiguration( struct SSD13x6_Device* DeviceHandle, uint32_t RemapCFG, uint32_t PinCFG, int ScanDir ) {
  158. SSD13x6_WriteCommand( DeviceHandle, 0xDA );
  159. SSD13x6_WriteCommand( DeviceHandle, ( uint8_t ) ( RemapCFG | PinCFG | BIT( 1 ) ) );
  160. SSD13x6_WriteCommand( DeviceHandle,
  161. ( ScanDir == COM_ScanDir_LR ) ? 0xC0 : 0xC8
  162. );
  163. }
  164. static bool SSD13x6_Init( struct SSD13x6_Device* DeviceHandle, int Width, int Height ) {
  165. DeviceHandle->Width = Width;
  166. DeviceHandle->Height = Height;
  167. DeviceHandle->FramebufferSize = ( DeviceHandle->Width * Height ) / 8;
  168. // DeviceHandle->Framebuffer = heap_caps_calloc( 1, DeviceHandle->FramebufferSize, MALLOC_CAP_INTERNAL );
  169. DeviceHandle->Framebuffer = calloc( 1, DeviceHandle->FramebufferSize );
  170. NullCheck( DeviceHandle->Framebuffer, return false );
  171. SSD13x6_HWReset( DeviceHandle );
  172. if (DeviceHandle->Model == SSD1306) {
  173. SSDCmd_Set_Display_Start_Line = 0x40;
  174. SSDCmd_Set_Display_Offset = 0xD3;
  175. SSDCmd_Set_Column_Address = 0x21,
  176. SSDCmd_Set_Display_CLK = 0xD5;
  177. SSDCmd_Set_Page_Address = 0x22;
  178. SSD13x6_Max_Col = 127;
  179. // charge pump regulator, do direct init
  180. SSD13x6_WriteCommand( DeviceHandle, 0x8D );
  181. SSD13x6_WriteCommand( DeviceHandle, 0x14 ); /* MAGIC NUMBER */
  182. if ( Height == 64 ) {
  183. SetCOMPinConfiguration( DeviceHandle, COM_Disable_LR_Remap, COM_Pins_Alternative, COM_ScanDir_LR );
  184. } else {
  185. SetCOMPinConfiguration( DeviceHandle, COM_Disable_LR_Remap, COM_Pins_Sequential, COM_ScanDir_LR );
  186. }
  187. } else if (DeviceHandle->Model == SSD1326) {
  188. SSDCmd_Set_Display_Start_Line = 0xA1;
  189. SSDCmd_Set_Display_Offset = 0xA2;
  190. SSDCmd_Set_Column_Address = 0x15;
  191. SSDCmd_Set_Display_CLK = 0xB3;
  192. SSDCmd_Set_Page_Address = 0x75; // not really a page but a row
  193. SSD13x6_Max_Col = 255;
  194. // no gray scale
  195. DeviceHandle->ReMap |= 0x10;
  196. SSD132x_ReMap( DeviceHandle );
  197. SSD13x6_SetHFlip( DeviceHandle, false );
  198. SSD13x6_SetVFlip( DeviceHandle, false );
  199. }
  200. SSD13x6_SetMuxRatio( DeviceHandle, Height - 1 );
  201. SSD13x6_SetDisplayOffset( DeviceHandle, 0x00 );
  202. SSD13x6_SetDisplayStartLine( DeviceHandle, 0 );
  203. SSD13x6_SetContrast( DeviceHandle, 0x7F );
  204. SSD13x6_DisableDisplayRAM( DeviceHandle );
  205. SSD13x6_SetInverted( DeviceHandle, false );
  206. SSD13x6_SetDisplayClocks( DeviceHandle, 0, 8 );
  207. SSD13x6_SetDisplayAddressMode( DeviceHandle, AddressMode_Vertical );
  208. SSD13x6_SetColumnAddress( DeviceHandle, 0, DeviceHandle->Width - 1 );
  209. SSD13x6_SetPageAddress( DeviceHandle, 0, ( DeviceHandle->Height / 8 ) - 1 );
  210. SSD13x6_EnableDisplayRAM( DeviceHandle );
  211. SSD13x6_DisplayOn( DeviceHandle );
  212. SSD13x6_Update( DeviceHandle );
  213. return true;
  214. }
  215. bool SSD13x6_Init_I2C( struct SSD13x6_Device* DeviceHandle, int Width, int Height, int I2CAddress, int ResetPin, WriteCommandProc WriteCommand, WriteDataProc WriteData, ResetProc Reset ) {
  216. NullCheck( DeviceHandle, return false );
  217. NullCheck( WriteCommand, return false );
  218. NullCheck( WriteData, return false );
  219. memset( DeviceHandle, 0, sizeof( struct SSD13x6_Device ) );
  220. DeviceHandle->WriteCommand = WriteCommand;
  221. DeviceHandle->WriteData = WriteData;
  222. DeviceHandle->Reset = Reset;
  223. DeviceHandle->Address = I2CAddress;
  224. DeviceHandle->RSTPin = ResetPin;
  225. return SSD13x6_Init( DeviceHandle, Width, Height );
  226. }
  227. bool SSD13x6_Init_SPI( struct SSD13x6_Device* DeviceHandle, int Width, int Height, int ResetPin, int CSPin, spi_device_handle_t SPIHandle, WriteCommandProc WriteCommand, WriteDataProc WriteData, ResetProc Reset ) {
  228. NullCheck( DeviceHandle, return false );
  229. NullCheck( WriteCommand, return false );
  230. NullCheck( WriteData, return false );
  231. memset( DeviceHandle, 0, sizeof( struct SSD13x6_Device ) );
  232. DeviceHandle->WriteCommand = WriteCommand;
  233. DeviceHandle->WriteData = WriteData;
  234. DeviceHandle->Reset = Reset;
  235. DeviceHandle->SPIHandle = SPIHandle;
  236. DeviceHandle->RSTPin = ResetPin;
  237. DeviceHandle->CSPin = CSPin;
  238. return SSD13x6_Init( DeviceHandle, Width, Height );
  239. }