u8x8_d_st7565.c 46 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243
  1. /*
  2. u8x8_d_st7565.c
  3. also includes support for nt7534
  4. Universal 8bit Graphics Library (https://github.com/olikraus/u8g2/)
  5. Copyright (c) 2016, olikraus@gmail.com
  6. All rights reserved.
  7. Redistribution and use in source and binary forms, with or without modification,
  8. are permitted provided that the following conditions are met:
  9. * Redistributions of source code must retain the above copyright notice, this list
  10. of conditions and the following disclaimer.
  11. * Redistributions in binary form must reproduce the above copyright notice, this
  12. list of conditions and the following disclaimer in the documentation and/or other
  13. materials provided with the distribution.
  14. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
  15. CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
  16. INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
  17. MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  18. DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
  19. CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  20. SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
  21. NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  22. LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
  23. CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
  24. STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
  25. ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
  26. ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  27. */
  28. #include "u8x8.h"
  29. static const uint8_t u8x8_d_st7565_powersave0_seq[] = {
  30. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  31. U8X8_C(0x0a4), /* all pixel off, issue 142 */
  32. U8X8_C(0x0af), /* display on */
  33. U8X8_END_TRANSFER(), /* disable chip */
  34. U8X8_END() /* end of sequence */
  35. };
  36. static const uint8_t u8x8_d_st7565_powersave1_seq[] = {
  37. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  38. U8X8_C(0x0ae), /* display off */
  39. U8X8_C(0x0a5), /* enter powersafe: all pixel on, issue 142 */
  40. U8X8_END_TRANSFER(), /* disable chip */
  41. U8X8_END() /* end of sequence */
  42. };
  43. static const uint8_t u8x8_d_st7565_flip0_seq[] = {
  44. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  45. U8X8_C(0x0a1), /* segment remap a0/a1*/
  46. U8X8_C(0x0c0), /* c0: scan dir normal, c8: reverse */
  47. U8X8_END_TRANSFER(), /* disable chip */
  48. U8X8_END() /* end of sequence */
  49. };
  50. static const uint8_t u8x8_d_st7565_flip1_seq[] = {
  51. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  52. U8X8_C(0x0a0), /* segment remap a0/a1*/
  53. U8X8_C(0x0c8), /* c0: scan dir normal, c8: reverse */
  54. U8X8_END_TRANSFER(), /* disable chip */
  55. U8X8_END() /* end of sequence */
  56. };
  57. static const uint8_t u8x8_d_st7565_zflip0_seq[] = {
  58. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  59. U8X8_C(0x0a1), /* segment remap a0/a1*/
  60. U8X8_C(0x0c8), /* c0: scan dir normal, c8: reverse */
  61. U8X8_END_TRANSFER(), /* disable chip */
  62. U8X8_END() /* end of sequence */
  63. };
  64. static const uint8_t u8x8_d_st7565_zflip1_seq[] = {
  65. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  66. U8X8_C(0x0a0), /* segment remap a0/a1*/
  67. U8X8_C(0x0c0), /* c0: scan dir normal, c8: reverse */
  68. U8X8_END_TRANSFER(), /* disable chip */
  69. U8X8_END() /* end of sequence */
  70. };
  71. static const u8x8_display_info_t u8x8_st7565_128x64_display_info =
  72. {
  73. /* chip_enable_level = */ 0,
  74. /* chip_disable_level = */ 1,
  75. /* post_chip_enable_wait_ns = */ 150, /* st7565 datasheet, table 26, tcsh */
  76. /* pre_chip_disable_wait_ns = */ 50, /* st7565 datasheet, table 26, tcss */
  77. /* reset_pulse_width_ms = */ 1,
  78. /* post_reset_wait_ms = */ 1,
  79. /* sda_setup_time_ns = */ 50, /* st7565 datasheet, table 26, tsds */
  80. /* sck_pulse_width_ns = */ 120, /* half of cycle time (100ns according to datasheet), AVR: below 70: 8 MHz, >= 70 --> 4MHz clock */
  81. /* sck_clock_hz = */ 4000000UL, /* since Arduino 1.6.0, the SPI bus speed in Hz. Should be 1000000000/sck_pulse_width_ns */
  82. /* spi_mode = */ 0, /* active high, rising edge */
  83. /* i2c_bus_clock_100kHz = */ 4,
  84. /* data_setup_time_ns = */ 40, /* st7565 datasheet, table 24, tds8 */
  85. /* write_pulse_width_ns = */ 80, /* st7565 datasheet, table 24, tcclw */
  86. /* tile_width = */ 16, /* width of 16*8=128 pixel */
  87. /* tile_hight = */ 8,
  88. /* default_x_offset = */ 0,
  89. /* flipmode_x_offset = */ 4,
  90. /* pixel_width = */ 128,
  91. /* pixel_height = */ 64
  92. };
  93. uint8_t u8x8_d_st7565_common(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
  94. {
  95. uint8_t x, c;
  96. uint8_t *ptr;
  97. switch(msg)
  98. {
  99. case U8X8_MSG_DISPLAY_DRAW_TILE:
  100. u8x8_cad_StartTransfer(u8x8);
  101. x = ((u8x8_tile_t *)arg_ptr)->x_pos;
  102. x *= 8;
  103. x += u8x8->x_offset;
  104. u8x8_cad_SendCmd(u8x8, 0x010 | (x>>4) );
  105. u8x8_cad_SendCmd(u8x8, 0x000 | ((x&15)));
  106. u8x8_cad_SendCmd(u8x8, 0x0b0 | (((u8x8_tile_t *)arg_ptr)->y_pos));
  107. c = ((u8x8_tile_t *)arg_ptr)->cnt;
  108. c *= 8;
  109. ptr = ((u8x8_tile_t *)arg_ptr)->tile_ptr;
  110. /*
  111. The following if condition checks the hardware limits of the st7565
  112. controller: It is not allowed to write beyond the display limits.
  113. This is in fact an issue within flip mode.
  114. */
  115. if ( c + x > 132u )
  116. {
  117. c = 132u;
  118. c -= x;
  119. }
  120. do
  121. {
  122. u8x8_cad_SendData(u8x8, c, ptr); /* note: SendData can not handle more than 255 bytes */
  123. arg_int--;
  124. } while( arg_int > 0 );
  125. u8x8_cad_EndTransfer(u8x8);
  126. break;
  127. /* handled in the calling procedure
  128. case U8X8_MSG_DISPLAY_SETUP_MEMORY:
  129. u8x8_d_helper_display_setup_memory(u8x8, &u8x8_st7565_128x64_display_info);
  130. break;
  131. case U8X8_MSG_DISPLAY_INIT:
  132. u8x8_d_helper_display_init(u8x8);
  133. u8x8_cad_SendSequence(u8x8, u8x8_d_uc1701_dogs102_init_seq);
  134. break;
  135. */
  136. case U8X8_MSG_DISPLAY_SET_POWER_SAVE:
  137. if ( arg_int == 0 )
  138. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_powersave0_seq);
  139. else
  140. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_powersave1_seq);
  141. break;
  142. #ifdef U8X8_WITH_SET_CONTRAST
  143. case U8X8_MSG_DISPLAY_SET_CONTRAST:
  144. u8x8_cad_StartTransfer(u8x8);
  145. u8x8_cad_SendCmd(u8x8, 0x081 );
  146. u8x8_cad_SendArg(u8x8, arg_int >> 2 ); /* st7565 has range from 0 to 63 */
  147. u8x8_cad_EndTransfer(u8x8);
  148. break;
  149. #endif
  150. default:
  151. return 0;
  152. }
  153. return 1;
  154. }
  155. /*================================================*/
  156. /* DOGM128 */
  157. static const uint8_t u8x8_d_st7565_dogm128_init_seq[] = {
  158. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  159. U8X8_C(0x0e2), /* soft reset */
  160. U8X8_C(0x0ae), /* display off */
  161. U8X8_C(0x040), /* set display start line to 0 */
  162. U8X8_C(0x0a1), /* ADC set to reverse */
  163. U8X8_C(0x0c0), /* common output mode */
  164. // Flipmode
  165. // U8X8_C(0x0a0), /* ADC set to reverse */
  166. // U8X8_C(0x0c8), /* common output mode */
  167. U8X8_C(0x0a6), /* display normal, bit val 0: LCD pixel off. */
  168. U8X8_C(0x0a2), /* LCD bias 1/9 */
  169. U8X8_C(0x02f), /* all power control circuits on (regulator, booster and follower) */
  170. U8X8_CA(0x0f8, 0x000), /* set booster ratio to 4x */
  171. U8X8_C(0x027), /* set V0 voltage resistor ratio to max */
  172. U8X8_CA(0x081, 0x018), /* set contrast, contrast value, EA default: 0x016 */
  173. U8X8_C(0x0ae), /* display off */
  174. U8X8_C(0x0a5), /* enter powersafe: all pixel on, issue 142 */
  175. U8X8_END_TRANSFER(), /* disable chip */
  176. U8X8_END() /* end of sequence */
  177. };
  178. uint8_t u8x8_d_st7565_ea_dogm128(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
  179. {
  180. /* call common procedure first and handle messages there */
  181. if ( u8x8_d_st7565_common(u8x8, msg, arg_int, arg_ptr) == 0 )
  182. {
  183. /* msg not handled, then try here */
  184. switch(msg)
  185. {
  186. case U8X8_MSG_DISPLAY_SETUP_MEMORY:
  187. u8x8_d_helper_display_setup_memory(u8x8, &u8x8_st7565_128x64_display_info);
  188. break;
  189. case U8X8_MSG_DISPLAY_INIT:
  190. u8x8_d_helper_display_init(u8x8);
  191. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_dogm128_init_seq);
  192. break;
  193. case U8X8_MSG_DISPLAY_SET_FLIP_MODE:
  194. if ( arg_int == 0 )
  195. {
  196. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip0_seq);
  197. u8x8->x_offset = u8x8->display_info->default_x_offset;
  198. }
  199. else
  200. {
  201. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip1_seq);
  202. u8x8->x_offset = u8x8->display_info->flipmode_x_offset;
  203. }
  204. break;
  205. default:
  206. return 0; /* msg unknown */
  207. }
  208. }
  209. return 1;
  210. }
  211. /*================================================*/
  212. /* LM6063 https://github.com/olikraus/u8g2/issues/893 */
  213. static const uint8_t u8x8_d_st7565_lm6063_init_seq[] = {
  214. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  215. U8X8_C(0x0e2), /* soft reset */
  216. U8X8_C(0x0ae), /* display off */
  217. U8X8_C(0x040), /* set display start line to 0 */
  218. U8X8_C(0x0a1), /* ADC set to reverse */
  219. U8X8_C(0x0c0), /* common output mode */
  220. // Flipmode
  221. // U8X8_C(0x0a0), /* ADC set to reverse */
  222. // U8X8_C(0x0c8), /* common output mode */
  223. U8X8_C(0x0a6), /* display normal, bit val 0: LCD pixel off. */
  224. U8X8_C(0x0a2), /* LCD bias 1/9 */
  225. U8X8_C(0x02f), /* all power control circuits on (regulator, booster and follower) */
  226. U8X8_CA(0x0f8, 0x000), /* set booster ratio to 4x */
  227. U8X8_C(0x027), /* set V0 voltage resistor ratio to max */
  228. U8X8_CA(0x081, 50/4), /* set contrast, contrast value, 40..60 seems to be good */
  229. U8X8_C(0x0ae), /* display off */
  230. U8X8_C(0x0a5), /* enter powersafe: all pixel on, issue 142 */
  231. U8X8_END_TRANSFER(), /* disable chip */
  232. U8X8_END() /* end of sequence */
  233. };
  234. uint8_t u8x8_d_st7565_lm6063(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
  235. {
  236. /* call common procedure first and handle messages there */
  237. if ( u8x8_d_st7565_common(u8x8, msg, arg_int, arg_ptr) == 0 )
  238. {
  239. /* msg not handled, then try here */
  240. switch(msg)
  241. {
  242. case U8X8_MSG_DISPLAY_SETUP_MEMORY:
  243. u8x8_d_helper_display_setup_memory(u8x8, &u8x8_st7565_128x64_display_info);
  244. break;
  245. case U8X8_MSG_DISPLAY_INIT:
  246. u8x8_d_helper_display_init(u8x8);
  247. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_lm6063_init_seq);
  248. break;
  249. case U8X8_MSG_DISPLAY_SET_FLIP_MODE:
  250. if ( arg_int == 0 )
  251. {
  252. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip0_seq);
  253. u8x8->x_offset = u8x8->display_info->default_x_offset;
  254. }
  255. else
  256. {
  257. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip1_seq);
  258. u8x8->x_offset = u8x8->display_info->flipmode_x_offset;
  259. }
  260. break;
  261. default:
  262. return 0; /* msg unknown */
  263. }
  264. }
  265. return 1;
  266. }
  267. /*================================================*/
  268. /* Displaytech 64128n */
  269. static const uint8_t u8x8_d_st7565_64128n_init_seq[] = {
  270. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  271. #ifdef NOT_WORKING
  272. U8X8_C(0x0e2), /* soft reset */
  273. U8X8_C(0x0ae), /* display off */
  274. U8X8_C(0x040), /* set display start line to 0 */
  275. U8X8_C(0x0a1), /* ADC set to reverse */
  276. U8X8_C(0x0c0), /* common output mode */
  277. // Flipmode
  278. // U8X8_C(0x0a0), /* ADC set to reverse */
  279. // U8X8_C(0x0c8), /* common output mode */
  280. U8X8_C(0x0a6), /* display normal, bit val 0: LCD pixel off. */
  281. U8X8_C(0x0a2), /* LCD bias 1/9 */
  282. U8X8_C(0x02f), /* all power control circuits on */
  283. //U8X8_CA(0x0f8, 0x000), /* set booster ratio to 4x */
  284. //U8X8_C(0x027), /* set V0 voltage resistor ratio to max */
  285. U8X8_C(0x010), /* Set V0 voltage resistor ratio. Setting for controlling brightness of Displaytech 64128N */
  286. U8X8_CA(0x081, 0x01e), /* set contrast, contrast value */
  287. U8X8_C(0x0ae), /* display off */
  288. U8X8_C(0x0a5), /* enter powersafe: all pixel on, issue 142 */
  289. #else
  290. U8X8_C(0x0e2), /* soft reset */
  291. U8X8_C(0x0A2), /* 0x0a2: LCD bias 1/9 (according to Displaytech 64128N datasheet) */
  292. U8X8_C(0x0a1), /* ADC set to reverse */
  293. U8X8_C(0x0c0), /* common output mode */
  294. //U8X8_C(0x0A0), /* Normal ADC Select (according to Displaytech 64128N datasheet) */
  295. //U8X8_C(0x0c8), /* common output mode: set scan direction normal operation/SHL Select, 0x0c0 --> SHL = 0, normal, 0x0c8 --> SHL = 1 */
  296. U8X8_C(0x040), /* Display start line for Displaytech 64128N */
  297. U8X8_C(0x028 | 0x04), /* power control: turn on voltage converter */
  298. U8X8_C(0x028 | 0x06), /* power control: turn on voltage regulator */
  299. U8X8_C(0x028 | 0x07), /* power control: turn on voltage follower */
  300. U8X8_C(0x010), /* Set V0 voltage resistor ratio. Setting for controlling brightness of Displaytech 64128N */
  301. /* 19 Jul 17: Not sure if this is true, cmd 0x1? is used to set the column */
  302. U8X8_C(0x0a6), /* display normal, bit val 0: LCD pixel off. */
  303. U8X8_C(0x081), /* set contrast */
  304. U8X8_C(0x01e), /* Contrast value. Setting for controlling brightness of Displaytech 64128N */
  305. //U8X8_C(0x0af), /* display on */
  306. //U8X8_C(0x0a5), /* display all points, ST7565 */
  307. //U8X8_C(0x0a4), /* normal display */
  308. U8X8_C(0x0ae), /* display off */
  309. #endif
  310. U8X8_END_TRANSFER(), /* disable chip */
  311. U8X8_END() /* end of sequence */
  312. };
  313. static const u8x8_display_info_t u8x8_st7565_64128n_display_info =
  314. {
  315. /* chip_enable_level = */ 0,
  316. /* chip_disable_level = */ 1,
  317. /* post_chip_enable_wait_ns = */ 150, /* st7565 datasheet, table 26, tcsh */
  318. /* pre_chip_disable_wait_ns = */ 50, /* st7565 datasheet, table 26, tcss */
  319. /* reset_pulse_width_ms = */ 1,
  320. /* post_reset_wait_ms = */ 1,
  321. /* sda_setup_time_ns = */ 50, /* st7565 datasheet, table 26, tsds */
  322. /* sck_pulse_width_ns = */ 120, /* half of cycle time (100ns according to datasheet), AVR: below 70: 8 MHz, >= 70 --> 4MHz clock */
  323. /* sck_clock_hz = */ 4000000UL, /* since Arduino 1.6.0, the SPI bus speed in Hz. Should be 1000000000/sck_pulse_width_ns */
  324. /* spi_mode = */ 0, /* active high, rising edge */
  325. /* i2c_bus_clock_100kHz = */ 4,
  326. /* data_setup_time_ns = */ 40, /* st7565 datasheet, table 24, tds8 */
  327. /* write_pulse_width_ns = */ 80, /* st7565 datasheet, table 24, tcclw */
  328. /* tile_width = */ 16, /* width of 16*8=128 pixel */
  329. /* tile_hight = */ 8,
  330. /* default_x_offset = */ 4,
  331. /* flipmode_x_offset = */ 0,
  332. /* pixel_width = */ 128,
  333. /* pixel_height = */ 64
  334. };
  335. uint8_t u8x8_d_st7565_64128n(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
  336. {
  337. /* call common procedure first and handle messages there */
  338. if ( u8x8_d_st7565_common(u8x8, msg, arg_int, arg_ptr) == 0 )
  339. {
  340. /* msg not handled, then try here */
  341. switch(msg)
  342. {
  343. case U8X8_MSG_DISPLAY_SETUP_MEMORY:
  344. u8x8_d_helper_display_setup_memory(u8x8, &u8x8_st7565_64128n_display_info);
  345. break;
  346. case U8X8_MSG_DISPLAY_INIT:
  347. u8x8_d_helper_display_init(u8x8);
  348. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_64128n_init_seq);
  349. break;
  350. case U8X8_MSG_DISPLAY_SET_FLIP_MODE:
  351. if ( arg_int == 0 )
  352. {
  353. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip0_seq);
  354. u8x8->x_offset = u8x8->display_info->default_x_offset;
  355. }
  356. else
  357. {
  358. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip1_seq);
  359. u8x8->x_offset = u8x8->display_info->flipmode_x_offset;
  360. }
  361. break;
  362. default:
  363. return 0; /* msg unknown */
  364. }
  365. }
  366. return 1;
  367. }
  368. /*================================================*/
  369. /* ZOLEN 128x64 */
  370. static const uint8_t u8x8_d_st7565_zolen_128x64_init_seq[] = {
  371. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  372. U8X8_C(0x0e2), /* soft reset */
  373. U8X8_C(0x0ae), /* display off */
  374. U8X8_C(0x040), /* set display start line to 0 */
  375. U8X8_C(0x0a1), /* ADC set to reverse */
  376. U8X8_C(0x0c8), /* common output mode */
  377. // Flipmode
  378. // U8X8_C(0x0a0), /* ADC set to reverse */
  379. // U8X8_C(0x0c0), /* common output mode */
  380. U8X8_C(0x0a6), /* display normal, bit val 0: LCD pixel off. */
  381. U8X8_C(0x0a2), /* LCD bias 1/9 */
  382. U8X8_C(0x02f), /* all power control circuits on (regulator, booster and follower) */
  383. U8X8_CA(0x0f8, 0x000), /* set booster ratio to 4x */
  384. U8X8_C(0x027), /* set V0 voltage resistor ratio to max */
  385. U8X8_CA(0x081, 0x007), /* set contrast, contrast value, EA default: 0x016 */
  386. U8X8_C(0x0ae), /* display off */
  387. U8X8_C(0x0a5), /* enter powersafe: all pixel on, issue 142 */
  388. U8X8_END_TRANSFER(), /* disable chip */
  389. U8X8_END() /* end of sequence */
  390. };
  391. uint8_t u8x8_d_st7565_zolen_128x64(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
  392. {
  393. /* call common procedure first and handle messages there */
  394. if ( u8x8_d_st7565_common(u8x8, msg, arg_int, arg_ptr) == 0 )
  395. {
  396. /* msg not handled, then try here */
  397. switch(msg)
  398. {
  399. case U8X8_MSG_DISPLAY_SETUP_MEMORY:
  400. u8x8_d_helper_display_setup_memory(u8x8, &u8x8_st7565_128x64_display_info);
  401. break;
  402. case U8X8_MSG_DISPLAY_INIT:
  403. u8x8_d_helper_display_init(u8x8);
  404. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_zolen_128x64_init_seq);
  405. break;
  406. case U8X8_MSG_DISPLAY_SET_FLIP_MODE:
  407. if ( arg_int == 0 )
  408. {
  409. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_zflip0_seq);
  410. u8x8->x_offset = u8x8->display_info->default_x_offset;
  411. }
  412. else
  413. {
  414. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_zflip1_seq);
  415. u8x8->x_offset = u8x8->display_info->flipmode_x_offset;
  416. }
  417. break;
  418. default:
  419. return 0; /* msg unknown */
  420. }
  421. }
  422. return 1;
  423. }
  424. /*================================================*/
  425. /* NHD-C12832 */
  426. static const u8x8_display_info_t u8x8_st7565_128x32_display_info =
  427. {
  428. /* chip_enable_level = */ 0,
  429. /* chip_disable_level = */ 1,
  430. /* post_chip_enable_wait_ns = */ 150, /* st7565 datasheet, table 26, tcsh */
  431. /* pre_chip_disable_wait_ns = */ 50, /* st7565 datasheet, table 26, tcss */
  432. /* reset_pulse_width_ms = */ 1,
  433. /* post_reset_wait_ms = */ 1,
  434. /* sda_setup_time_ns = */ 50, /* st7565 datasheet, table 26, tsds */
  435. /* sck_pulse_width_ns = */ 120, /* half of cycle time (100ns according to datasheet), AVR: below 70: 8 MHz, >= 70 --> 4MHz clock */
  436. /* sck_clock_hz = */ 4000000UL, /* since Arduino 1.6.0, the SPI bus speed in Hz. Should be 1000000000/sck_pulse_width_ns */
  437. /* spi_mode = */ 0, /* active high, rising edge */
  438. /* i2c_bus_clock_100kHz = */ 4,
  439. /* data_setup_time_ns = */ 40, /* st7565 datasheet, table 24, tds8 */
  440. /* write_pulse_width_ns = */ 80, /* st7565 datasheet, table 24, tcclw */
  441. /* tile_width = */ 16, /* width of 16*8=128 pixel */
  442. /* tile_hight = */ 4,
  443. /* default_x_offset = */ 4,
  444. /* flipmode_x_offset = */ 0,
  445. /* pixel_width = */ 128,
  446. /* pixel_height = */ 32
  447. };
  448. static const uint8_t u8x8_d_st7565_nhd_c12832_init_seq[] = {
  449. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  450. U8X8_C(0x0e2), /* soft reset */
  451. U8X8_C(0x0ae), /* display off */
  452. U8X8_C(0x040), /* set display start line to 0 */
  453. U8X8_C(0x0a1), /* ADC set to reverse */
  454. U8X8_C(0x0c0), /* common output mode */
  455. // Flipmode
  456. //U8X8_C(0x0a0), /* ADC set to reverse */
  457. //U8X8_C(0x0c8), /* common output mode */
  458. U8X8_C(0x0a6), /* display normal, bit val 0: LCD pixel off. */
  459. U8X8_C(0x0a2), /* LCD bias 1/9 */
  460. U8X8_C(0x02f), /* all power control circuits on */
  461. U8X8_CA(0x0f8, 0x000), /* set booster ratio to 4x */
  462. U8X8_C(0x023), /* set V0 voltage resistor ratio to large*/
  463. U8X8_CA(0x081, 0x00a), /* set contrast, contrast value NHD C12832 */
  464. U8X8_C(0x0ae), /* display off */
  465. U8X8_C(0x0a5), /* enter powersafe: all pixel on, issue 142 */
  466. U8X8_END_TRANSFER(), /* disable chip */
  467. U8X8_END() /* end of sequence */
  468. };
  469. uint8_t u8x8_d_st7565_nhd_c12832(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
  470. {
  471. /* call common procedure first and handle messages there */
  472. if ( u8x8_d_st7565_common(u8x8, msg, arg_int, arg_ptr) == 0 )
  473. {
  474. /* msg not handled, then try here */
  475. switch(msg)
  476. {
  477. case U8X8_MSG_DISPLAY_SETUP_MEMORY:
  478. u8x8_d_helper_display_setup_memory(u8x8, &u8x8_st7565_128x32_display_info);
  479. break;
  480. case U8X8_MSG_DISPLAY_INIT:
  481. u8x8_d_helper_display_init(u8x8);
  482. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_nhd_c12832_init_seq);
  483. break;
  484. case U8X8_MSG_DISPLAY_SET_FLIP_MODE:
  485. if ( arg_int == 0 )
  486. {
  487. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip0_seq);
  488. u8x8->x_offset = u8x8->display_info->default_x_offset;
  489. }
  490. else
  491. {
  492. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip1_seq);
  493. u8x8->x_offset = u8x8->display_info->flipmode_x_offset;
  494. }
  495. break;
  496. default:
  497. return 0; /* msg unknown */
  498. }
  499. }
  500. return 1;
  501. }
  502. /*================================================*/
  503. /* NHD-C12864 */
  504. static const u8x8_display_info_t u8x8_st7565_nhd_c12864_display_info =
  505. {
  506. /* chip_enable_level = */ 0,
  507. /* chip_disable_level = */ 1,
  508. /* post_chip_enable_wait_ns = */ 150, /* st7565 datasheet, table 26, tcsh */
  509. /* pre_chip_disable_wait_ns = */ 50, /* st7565 datasheet, table 26, tcss */
  510. /* reset_pulse_width_ms = */ 1,
  511. /* post_reset_wait_ms = */ 1,
  512. /* sda_setup_time_ns = */ 50, /* st7565 datasheet, table 26, tsds */
  513. /* sck_pulse_width_ns = */ 120, /* half of cycle time (100ns according to datasheet), AVR: below 70: 8 MHz, >= 70 --> 4MHz clock */
  514. /* sck_clock_hz = */ 4000000UL, /* since Arduino 1.6.0, the SPI bus speed in Hz. Should be 1000000000/sck_pulse_width_ns */
  515. /* spi_mode = */ 0, /* active high, rising edge */
  516. /* i2c_bus_clock_100kHz = */ 4,
  517. /* data_setup_time_ns = */ 40, /* st7565 datasheet, table 24, tds8 */
  518. /* write_pulse_width_ns = */ 80, /* st7565 datasheet, table 24, tcclw */
  519. /* tile_width = */ 16, /* width of 16*8=128 pixel */
  520. /* tile_hight = */ 8,
  521. /* default_x_offset = */ 4,
  522. /* flipmode_x_offset = */ 0,
  523. /* pixel_width = */ 128,
  524. /* pixel_height = */ 64
  525. };
  526. static const uint8_t u8x8_d_st7565_nhd_c12864_init_seq[] = {
  527. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  528. U8X8_C(0x0e2), /* soft reset */
  529. U8X8_C(0x0ae), /* display off */
  530. U8X8_C(0x040), /* set display start line to 0 */
  531. U8X8_C(0x0a1), /* ADC set to reverse */
  532. U8X8_C(0x0c0), /* common output mode */
  533. // Flipmode
  534. //U8X8_C(0x0a0), /* ADC set to reverse */
  535. //U8X8_C(0x0c8), /* common output mode */
  536. U8X8_C(0x0a6), /* display normal, bit val 0: LCD pixel off. */
  537. U8X8_C(0x0a2), /* LCD bias 1/9 */
  538. U8X8_C(0x02f), /* all power control circuits on */
  539. U8X8_CA(0x0f8, 0x000), /* set booster ratio to 4x */
  540. U8X8_C(0x023), /* set V0 voltage resistor ratio to large*/
  541. U8X8_CA(0x081, 180), /* set contrast, contrast value NHD C12864, see issue 186, increased contrast to 180 (issue 219) */
  542. U8X8_C(0x0ae), /* display off */
  543. U8X8_C(0x0a5), /* enter powersafe: all pixel on, issue 142 */
  544. U8X8_END_TRANSFER(), /* disable chip */
  545. U8X8_END() /* end of sequence */
  546. };
  547. uint8_t u8x8_d_st7565_nhd_c12864(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
  548. {
  549. /* call common procedure first and handle messages there */
  550. if ( u8x8_d_st7565_common(u8x8, msg, arg_int, arg_ptr) == 0 )
  551. {
  552. /* msg not handled, then try here */
  553. switch(msg)
  554. {
  555. case U8X8_MSG_DISPLAY_SETUP_MEMORY:
  556. u8x8_d_helper_display_setup_memory(u8x8, &u8x8_st7565_nhd_c12864_display_info);
  557. break;
  558. case U8X8_MSG_DISPLAY_INIT:
  559. u8x8_d_helper_display_init(u8x8);
  560. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_nhd_c12864_init_seq);
  561. break;
  562. case U8X8_MSG_DISPLAY_SET_FLIP_MODE:
  563. if ( arg_int == 0 )
  564. {
  565. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip0_seq);
  566. u8x8->x_offset = u8x8->display_info->default_x_offset;
  567. }
  568. else
  569. {
  570. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip1_seq);
  571. u8x8->x_offset = u8x8->display_info->flipmode_x_offset;
  572. }
  573. break;
  574. default:
  575. return 0; /* msg unknown */
  576. }
  577. }
  578. return 1;
  579. }
  580. /*================================================*/
  581. /* JLX12864 */
  582. uint8_t u8x8_d_st7565_jlx12864(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
  583. {
  584. return u8x8_d_st7565_nhd_c12864(u8x8, msg, arg_int, arg_ptr);
  585. }
  586. /*================================================*/
  587. /* LM6059 (Adafruit)... probably this is a ST7567 display */
  588. static const uint8_t u8x8_d_st7565_lm6059_init_seq[] = {
  589. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  590. U8X8_C(0x0e2), /* soft reset */
  591. U8X8_C(0x0ae), /* display off */
  592. U8X8_C(0x060), /* set display start line to ... */
  593. U8X8_C(0x0a0), /* ADC set to reverse */
  594. U8X8_C(0x0c8), /* common output mode */
  595. //U8X8_C(0x0a1), /* ADC set to reverse */
  596. //U8X8_C(0x0c0), /* common output mode */
  597. // Flipmode
  598. // U8X8_C(0x0a0), /* ADC set to reverse */
  599. // U8X8_C(0x0c8), /* common output mode */
  600. U8X8_C(0x0a6), /* display normal, bit val 0: LCD pixel off. */
  601. U8X8_C(0x0a3), /* LCD bias 1/9 */
  602. U8X8_C(0x02f), /* all power control circuits on (regulator, booster and follower) */
  603. U8X8_CA(0x0f8, 0x000), /* set booster ratio to 4x (ST7567 feature) */
  604. U8X8_C(0x027), /* set V0 voltage resistor ratio to max */
  605. U8X8_CA(0x081, 0x018), /* set contrast, contrast value, EA default: 0x016 */
  606. U8X8_C(0x0ae), /* display off */
  607. U8X8_C(0x0a5), /* enter powersafe: all pixel on, issue 142 */
  608. U8X8_END_TRANSFER(), /* disable chip */
  609. U8X8_END() /* end of sequence */
  610. };
  611. static const u8x8_display_info_t u8x8_st7565_lm6059_display_info =
  612. {
  613. /* chip_enable_level = */ 0,
  614. /* chip_disable_level = */ 1,
  615. /* post_chip_enable_wait_ns = */ 150, /* st7565 datasheet, table 26, tcsh */
  616. /* pre_chip_disable_wait_ns = */ 50, /* st7565 datasheet, table 26, tcss */
  617. /* reset_pulse_width_ms = */ 1,
  618. /* post_reset_wait_ms = */ 1,
  619. /* sda_setup_time_ns = */ 50, /* st7565 datasheet, table 26, tsds */
  620. /* sck_pulse_width_ns = */ 120, /* half of cycle time (100ns according to datasheet), AVR: below 70: 8 MHz, >= 70 --> 4MHz clock */
  621. /* sck_clock_hz = */ 4000000UL, /* since Arduino 1.6.0, the SPI bus speed in Hz. Should be 1000000000/sck_pulse_width_ns */
  622. /* spi_mode = */ 0, /* active high, rising edge */
  623. /* i2c_bus_clock_100kHz = */ 4,
  624. /* data_setup_time_ns = */ 40, /* st7565 datasheet, table 24, tds8 */
  625. /* write_pulse_width_ns = */ 80, /* st7565 datasheet, table 24, tcclw */
  626. /* tile_width = */ 16, /* width of 16*8=128 pixel */
  627. /* tile_hight = */ 8,
  628. /* default_x_offset = */ 1, /* not sure... */
  629. /* flipmode_x_offset = */ 3,
  630. /* pixel_width = */ 128,
  631. /* pixel_height = */ 64
  632. };
  633. uint8_t u8x8_d_st7565_lm6059(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
  634. {
  635. /* call common procedure first and handle messages there */
  636. if ( u8x8_d_st7565_common(u8x8, msg, arg_int, arg_ptr) == 0 )
  637. {
  638. /* msg not handled, then try here */
  639. switch(msg)
  640. {
  641. case U8X8_MSG_DISPLAY_SETUP_MEMORY:
  642. u8x8_d_helper_display_setup_memory(u8x8, &u8x8_st7565_lm6059_display_info);
  643. break;
  644. case U8X8_MSG_DISPLAY_INIT:
  645. u8x8_d_helper_display_init(u8x8);
  646. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_lm6059_init_seq);
  647. break;
  648. case U8X8_MSG_DISPLAY_SET_FLIP_MODE:
  649. if ( arg_int == 0 )
  650. {
  651. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip1_seq);
  652. u8x8->x_offset = u8x8->display_info->default_x_offset;
  653. }
  654. else
  655. {
  656. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip0_seq);
  657. u8x8->x_offset = u8x8->display_info->flipmode_x_offset;
  658. }
  659. break;
  660. default:
  661. return 0; /* msg unknown */
  662. }
  663. }
  664. return 1;
  665. }
  666. /*================================================*/
  667. /* LX12864 issue 576 */
  668. static const uint8_t u8x8_d_st7565_lx12864_init_seq[] = {
  669. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  670. U8X8_C(0x0e2), /* soft reset */
  671. U8X8_C(0x0ae), /* display off */
  672. U8X8_C(0x060), /* set display start line to ... */
  673. U8X8_C(0x0a0), /* ADC set to reverse */
  674. U8X8_C(0x0c8), /* common output mode */
  675. //U8X8_C(0x0a1), /* ADC set to reverse */
  676. //U8X8_C(0x0c0), /* common output mode */
  677. // Flipmode
  678. // U8X8_C(0x0a0), /* ADC set to reverse */
  679. // U8X8_C(0x0c8), /* common output mode */
  680. U8X8_C(0x0a6), /* display normal, bit val 0: LCD pixel off. */
  681. U8X8_C(0x0a2), /* LCD bias 1/9 */
  682. U8X8_C(0x02f), /* all power control circuits on (regulator, booster and follower) */
  683. U8X8_CA(0x0f8, 0x000), /* set booster ratio to 4x (ST7567 feature) */
  684. U8X8_C(0x027), /* set V0 voltage resistor ratio to max */
  685. U8X8_CA(0x081, 0x008), /* set contrast, contrast value */
  686. U8X8_C(0x0ae), /* display off */
  687. U8X8_C(0x0a5), /* enter powersafe: all pixel on, issue 142 */
  688. U8X8_END_TRANSFER(), /* disable chip */
  689. U8X8_END() /* end of sequence */
  690. };
  691. static const u8x8_display_info_t u8x8_st7565_lx12864_display_info =
  692. {
  693. /* chip_enable_level = */ 0,
  694. /* chip_disable_level = */ 1,
  695. /* post_chip_enable_wait_ns = */ 150, /* st7565 datasheet, table 26, tcsh */
  696. /* pre_chip_disable_wait_ns = */ 50, /* st7565 datasheet, table 26, tcss */
  697. /* reset_pulse_width_ms = */ 1,
  698. /* post_reset_wait_ms = */ 1,
  699. /* sda_setup_time_ns = */ 50, /* st7565 datasheet, table 26, tsds */
  700. /* sck_pulse_width_ns = */ 120, /* half of cycle time (100ns according to datasheet), AVR: below 70: 8 MHz, >= 70 --> 4MHz clock */
  701. /* sck_clock_hz = */ 4000000UL, /* since Arduino 1.6.0, the SPI bus speed in Hz. Should be 1000000000/sck_pulse_width_ns */
  702. /* spi_mode = */ 0, /* active high, rising edge */
  703. /* i2c_bus_clock_100kHz = */ 4,
  704. /* data_setup_time_ns = */ 40, /* st7565 datasheet, table 24, tds8 */
  705. /* write_pulse_width_ns = */ 80, /* st7565 datasheet, table 24, tcclw */
  706. /* tile_width = */ 16, /* width of 16*8=128 pixel */
  707. /* tile_hight = */ 8,
  708. /* default_x_offset = */ 1, /* not sure... */
  709. /* flipmode_x_offset = */ 3,
  710. /* pixel_width = */ 128,
  711. /* pixel_height = */ 64
  712. };
  713. uint8_t u8x8_d_st7565_lx12864(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
  714. {
  715. /* call common procedure first and handle messages there */
  716. if ( u8x8_d_st7565_common(u8x8, msg, arg_int, arg_ptr) == 0 )
  717. {
  718. /* msg not handled, then try here */
  719. switch(msg)
  720. {
  721. case U8X8_MSG_DISPLAY_SETUP_MEMORY:
  722. u8x8_d_helper_display_setup_memory(u8x8, &u8x8_st7565_lx12864_display_info);
  723. break;
  724. case U8X8_MSG_DISPLAY_INIT:
  725. u8x8_d_helper_display_init(u8x8);
  726. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_lx12864_init_seq);
  727. break;
  728. case U8X8_MSG_DISPLAY_SET_FLIP_MODE:
  729. if ( arg_int == 0 )
  730. {
  731. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip1_seq);
  732. u8x8->x_offset = u8x8->display_info->default_x_offset;
  733. }
  734. else
  735. {
  736. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip0_seq);
  737. u8x8->x_offset = u8x8->display_info->flipmode_x_offset;
  738. }
  739. break;
  740. default:
  741. return 0; /* msg unknown */
  742. }
  743. }
  744. return 1;
  745. }
  746. /*================================================*/
  747. /* ERC12864-1 (buydisplay.com) */
  748. static const uint8_t u8x8_d_st7565_erc12864_init_seq[] = {
  749. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  750. U8X8_C(0x0e2), /* soft reset */
  751. U8X8_C(0x0ae), /* display off */
  752. U8X8_C(0x040), /* set display start line to ... */
  753. U8X8_C(0x0a0), /* ADC set to reverse */
  754. U8X8_C(0x0c8), /* common output mode */
  755. //U8X8_C(0x0a1), /* ADC set to reverse */
  756. //U8X8_C(0x0c0), /* common output mode */
  757. // Flipmode
  758. // U8X8_C(0x0a0), /* ADC set to reverse */
  759. // U8X8_C(0x0c8), /* common output mode */
  760. U8X8_C(0x0a6), /* display normal, bit val 0: LCD pixel off. */
  761. U8X8_C(0x0a3), /* LCD bias 1/9 */
  762. U8X8_C(0x02f), /* all power control circuits on (regulator, booster and follower) */
  763. U8X8_CA(0x0f8, 0x000), /* set booster ratio to 4x (ST7567 feature)*/
  764. U8X8_C(0x027), /* set V0 voltage resistor ratio to max */
  765. U8X8_CA(0x081, 0x018), /* set contrast, contrast value, EA default: 0x016 */
  766. U8X8_C(0x0ae), /* display off */
  767. U8X8_C(0x0a5), /* enter powersafe: all pixel on, issue 142 */
  768. U8X8_END_TRANSFER(), /* disable chip */
  769. U8X8_END() /* end of sequence */
  770. };
  771. static const u8x8_display_info_t u8x8_st7565_erc12864_display_info =
  772. {
  773. /* chip_enable_level = */ 0,
  774. /* chip_disable_level = */ 1,
  775. /* post_chip_enable_wait_ns = */ 150, /* st7565 datasheet, table 26, tcsh */
  776. /* pre_chip_disable_wait_ns = */ 50, /* st7565 datasheet, table 26, tcss */
  777. /* reset_pulse_width_ms = */ 1,
  778. /* post_reset_wait_ms = */ 1,
  779. /* sda_setup_time_ns = */ 50, /* st7565 datasheet, table 26, tsds */
  780. /* sck_pulse_width_ns = */ 120, /* half of cycle time (100ns according to datasheet), AVR: below 70: 8 MHz, >= 70 --> 4MHz clock */
  781. /* sck_clock_hz = */ 4000000UL, /* since Arduino 1.6.0, the SPI bus speed in Hz. Should be 1000000000/sck_pulse_width_ns */
  782. /* spi_mode = */ 0, /* active high, rising edge */
  783. /* i2c_bus_clock_100kHz = */ 4,
  784. /* data_setup_time_ns = */ 40, /* st7565 datasheet, table 24, tds8 */
  785. /* write_pulse_width_ns = */ 80, /* st7565 datasheet, table 24, tcclw */
  786. /* tile_width = */ 16, /* width of 16*8=128 pixel */
  787. /* tile_hight = */ 8,
  788. /* default_x_offset = */ 0,
  789. /* flipmode_x_offset = */ 4,
  790. /* pixel_width = */ 128,
  791. /* pixel_height = */ 64
  792. };
  793. uint8_t u8x8_d_st7565_erc12864(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
  794. {
  795. /* call common procedure first and handle messages there */
  796. if ( u8x8_d_st7565_common(u8x8, msg, arg_int, arg_ptr) == 0 )
  797. {
  798. /* msg not handled, then try here */
  799. switch(msg)
  800. {
  801. case U8X8_MSG_DISPLAY_SETUP_MEMORY:
  802. u8x8_d_helper_display_setup_memory(u8x8, &u8x8_st7565_erc12864_display_info);
  803. break;
  804. case U8X8_MSG_DISPLAY_INIT:
  805. u8x8_d_helper_display_init(u8x8);
  806. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_erc12864_init_seq);
  807. break;
  808. case U8X8_MSG_DISPLAY_SET_FLIP_MODE:
  809. if ( arg_int == 0 )
  810. {
  811. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip1_seq);
  812. u8x8->x_offset = u8x8->display_info->default_x_offset;
  813. }
  814. else
  815. {
  816. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip0_seq);
  817. u8x8->x_offset = u8x8->display_info->flipmode_x_offset;
  818. }
  819. break;
  820. default:
  821. return 0; /* msg unknown */
  822. }
  823. }
  824. return 1;
  825. }
  826. /*================================================*/
  827. /* ERC12864-1 alternative version, suggested in issue 790 */
  828. static const uint8_t u8x8_d_st7565_erc12864_alt_init_seq[] = {
  829. // original sequence
  830. // U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  831. // U8X8_C(0x0e2), /* soft reset */
  832. // U8X8_C(0x0ae), /* display off */
  833. // U8X8_C(0x040), /* set display start line to ... */
  834. // U8X8_C(0x0a0), /* ADC set to reverse */
  835. // U8X8_C(0x0c8), /* common output mode */
  836. // U8X8_C(0x0a6), /* display normal, bit val 0: LCD pixel off. */
  837. // U8X8_C(0x0a3), /* LCD bias 1/9 */
  838. // U8X8_C(0x02f), /* all power control circuits on (regulator, booster and follower) */
  839. // U8X8_CA(0x0f8, 0x000), /* set booster ratio to 4x (ST7567 feature)*/
  840. // U8X8_C(0x027), /* set V0 voltage resistor ratio to max */
  841. // U8X8_CA(0x081, 0x018), /* set contrast, contrast value, EA default: 0x016 */
  842. // U8X8_C(0x0ae), /* display off */
  843. // U8X8_C(0x0a5), /* enter powersafe: all pixel on, issue 142 */
  844. // U8X8_END_TRANSFER(), /* disable chip */
  845. // U8X8_END() /* end of sequence */
  846. // suggested in https://github.com/olikraus/u8g2/issues/790
  847. // U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  848. // U8X8_C(0x0e2), /* soft reset */
  849. // U8X8_C(0x0ae), /* display off */
  850. // U8X8_C(0x040), /* set display start line to ... */
  851. // U8X8_C(0x0a0), /* ADC set to reverse */
  852. // U8X8_C(0x0c8), /* common output mode */
  853. // U8X8_C(0x0a6), /* display normal, bit val 0: LCD pixel off. */
  854. // U8X8_C(0x0a2), /* LCD bias 1/9 - *** Changed by Ismail - was 0xa3 - 1/7 bias we were getting dark pixel off */
  855. // U8X8_C(0x02f), /* all power control circuits on (regulator, booster and follower) */
  856. // U8X8_CA(0x0f8, 0x000), /* set booster ratio to 4x (ST7567 feature)*/
  857. // U8X8_C(0x027), /* set V0 voltage resistor ratio to max */
  858. // U8X8_CA(0x081, 0x07), /* set contrast, contrast value, EA default: 0x016 - *** Changed by Ismail to 0x05 */
  859. // U8X8_C(0x0ae), /* display off */
  860. // U8X8_C(0x0a5), /* enter powersafe: all pixel on, issue 142 */
  861. // U8X8_END_TRANSFER(), /* disable chip */
  862. // U8X8_END() /* end of sequence */
  863. // flipper zero sequence
  864. U8X8_START_TRANSFER(),
  865. U8X8_C(0x0e2), // soft reset
  866. U8X8_C(0xA3), // ST7565_st7565_command(CMD_SET_BIAS_7);
  867. U8X8_C(0xA0), // ST7565_st7565_command(CMD_SET_ADC_NORMAL);
  868. U8X8_C(0xC8), // ST7565_st7565_command(CMD_SET_COM_REVERSE);
  869. U8X8_C(0x40), // ST7565_st7565_command(CMD_SET_DISP_START_LINE);
  870. U8X8_C(0x28 | 0x4), // ST7565_st7565_command(CMD_SET_POWER_CONTROL | 0x4);
  871. U8X8_DLY(50),
  872. U8X8_C(0x28 | 0x6), // ST7565_st7565_command(CMD_SET_POWER_CONTROL | 0x6);
  873. U8X8_DLY(50),
  874. U8X8_C(0x28 | 0x7), // ST7565_st7565_command(CMD_SET_POWER_CONTROL | 0x7);
  875. U8X8_DLY(50),
  876. U8X8_C(0x20 | 0x6), // ST7565_st7565_command(CMD_SET_RESISTOR_RATIO | 0x6);
  877. U8X8_END_TRANSFER(), // disable chip
  878. U8X8_END() // end of sequence
  879. };
  880. uint8_t u8x8_d_st7565_erc12864_alt(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
  881. {
  882. /* call common procedure first and handle messages there */
  883. if ( u8x8_d_st7565_common(u8x8, msg, arg_int, arg_ptr) == 0 )
  884. {
  885. /* msg not handled, then try here */
  886. switch(msg)
  887. {
  888. case U8X8_MSG_DISPLAY_SETUP_MEMORY:
  889. u8x8_d_helper_display_setup_memory(u8x8, &u8x8_st7565_erc12864_display_info);
  890. break;
  891. case U8X8_MSG_DISPLAY_INIT:
  892. u8x8_d_helper_display_init(u8x8);
  893. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_erc12864_alt_init_seq);
  894. break;
  895. case U8X8_MSG_DISPLAY_SET_FLIP_MODE:
  896. if ( arg_int == 0 )
  897. {
  898. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip1_seq);
  899. u8x8->x_offset = u8x8->display_info->default_x_offset;
  900. }
  901. else
  902. {
  903. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip0_seq);
  904. u8x8->x_offset = u8x8->display_info->flipmode_x_offset;
  905. }
  906. break;
  907. default:
  908. return 0; /* msg unknown */
  909. }
  910. }
  911. return 1;
  912. }
  913. /*================================================*/
  914. /* NT7534, TG12864R */
  915. /* The NT7534 has an extended command set for the ST7565, however this is not used. */
  916. /* The TG12864R display is also shifted in lines, like the LM6059/Adafruit display */
  917. /* However contrast seems to be different */
  918. static const uint8_t u8x8_d_nt7534_tg12864r_init_seq[] = {
  919. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  920. U8X8_C(0x0e2), /* soft reset */
  921. U8X8_C(0x0ae), /* display off */
  922. U8X8_C(0x060), /* set display start line to ... */
  923. U8X8_C(0x0a0), /* ADC set to reverse */
  924. U8X8_C(0x0c8), /* common output mode */
  925. // Flipmode
  926. //U8X8_C(0x0a1), /* ADC set to reverse */
  927. //U8X8_C(0x0c0), /* common output mode */
  928. U8X8_C(0x0a6), /* display normal, bit val 0: LCD pixel off. */
  929. U8X8_C(0x0a3), /* LCD bias 1/9 */
  930. U8X8_C(0x02f), /* all power control circuits on (regulator, booster and follower) */
  931. //U8X8_CA(0x0f8, 0x000), /* set booster ratio to 4x (ST7567 feature)*/
  932. U8X8_C(0x027), /* set V0 voltage resistor ratio to max */
  933. U8X8_CA(0x081, 0x009), /* set contrast, contrast value, EA default: 0x016 */
  934. U8X8_C(0x0ae), /* display off */
  935. U8X8_C(0x0a5), /* enter powersafe: all pixel on, issue 142 */
  936. U8X8_END_TRANSFER(), /* disable chip */
  937. U8X8_END() /* end of sequence */
  938. };
  939. uint8_t u8x8_d_nt7534_tg12864r(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
  940. {
  941. /* call common procedure first and handle messages there */
  942. if ( u8x8_d_st7565_common(u8x8, msg, arg_int, arg_ptr) == 0 )
  943. {
  944. /* msg not handled, then try here */
  945. switch(msg)
  946. {
  947. case U8X8_MSG_DISPLAY_SETUP_MEMORY:
  948. /* reuse the LM6059 data structure... this display seems to have similar shifts and offsets */
  949. u8x8_d_helper_display_setup_memory(u8x8, &u8x8_st7565_lm6059_display_info);
  950. break;
  951. case U8X8_MSG_DISPLAY_INIT:
  952. u8x8_d_helper_display_init(u8x8);
  953. //u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_lm6059_init_seq);
  954. u8x8_cad_SendSequence(u8x8, u8x8_d_nt7534_tg12864r_init_seq);
  955. break;
  956. case U8X8_MSG_DISPLAY_SET_FLIP_MODE:
  957. if ( arg_int == 0 )
  958. {
  959. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip1_seq);
  960. u8x8->x_offset = u8x8->display_info->default_x_offset;
  961. }
  962. else
  963. {
  964. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip0_seq);
  965. u8x8->x_offset = u8x8->display_info->flipmode_x_offset;
  966. }
  967. break;
  968. default:
  969. return 0; /* msg unknown */
  970. }
  971. }
  972. return 1;
  973. }
  974. /*================================================*/
  975. /* EA DOGM132 */
  976. static const u8x8_display_info_t u8x8_st7565_dogm132_display_info =
  977. {
  978. /* chip_enable_level = */ 0,
  979. /* chip_disable_level = */ 1,
  980. /* post_chip_enable_wait_ns = */ 150, /* st7565 datasheet, table 26, tcsh */
  981. /* pre_chip_disable_wait_ns = */ 50, /* st7565 datasheet, table 26, tcss */
  982. /* reset_pulse_width_ms = */ 1,
  983. /* post_reset_wait_ms = */ 1,
  984. /* sda_setup_time_ns = */ 50, /* st7565 datasheet, table 26, tsds */
  985. /* sck_pulse_width_ns = */ 120, /* half of cycle time (100ns according to datasheet), AVR: below 70: 8 MHz, >= 70 --> 4MHz clock */
  986. /* sck_clock_hz = */ 4000000UL, /* since Arduino 1.6.0, the SPI bus speed in Hz. Should be 1000000000/sck_pulse_width_ns */
  987. /* spi_mode = */ 0, /* active high, rising edge */
  988. /* i2c_bus_clock_100kHz = */ 4,
  989. /* data_setup_time_ns = */ 40, /* st7565 datasheet, table 24, tds8 */
  990. /* write_pulse_width_ns = */ 80, /* st7565 datasheet, table 24, tcclw */
  991. /* tile_width = */ 17, /* width of 16*8=136 pixel */
  992. /* tile_hight = */ 4,
  993. /* default_x_offset = */ 0,
  994. /* flipmode_x_offset = */ 0,
  995. /* pixel_width = */ 132,
  996. /* pixel_height = */ 32
  997. };
  998. static const uint8_t u8x8_d_st7565_dogm132_init_seq[] = {
  999. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  1000. U8X8_C(0x0e2), /* soft reset */
  1001. U8X8_C(0x0ae), /* display off */
  1002. U8X8_C(0x040), /* set display start line to 0 */
  1003. U8X8_C(0x0a1), /* ADC set to reverse */
  1004. U8X8_C(0x0c0), /* common output mode */
  1005. // Flipmode
  1006. //U8X8_C(0x0a0), /* ADC set to reverse */
  1007. //U8X8_C(0x0c8), /* common output mode */
  1008. U8X8_C(0x0a6), /* display normal, bit val 0: LCD pixel off. */
  1009. U8X8_C(0x0a2), /* LCD bias 1/9 */
  1010. U8X8_C(0x02f), /* all power control circuits on */
  1011. U8X8_CA(0x0f8, 0x000), /* set booster ratio to 4x */
  1012. U8X8_C(0x023), /* set V0 voltage resistor ratio to large*/
  1013. U8X8_CA(0x081, 0x01f), /* set contrast, contrast value EA DOGM132 */
  1014. U8X8_C(0x0ae), /* display off */
  1015. U8X8_C(0x0a5), /* enter powersafe: all pixel on, issue 142 */
  1016. U8X8_END_TRANSFER(), /* disable chip */
  1017. U8X8_END() /* end of sequence */
  1018. };
  1019. uint8_t u8x8_d_st7565_ea_dogm132(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
  1020. {
  1021. /* call common procedure first and handle messages there */
  1022. if ( u8x8_d_st7565_common(u8x8, msg, arg_int, arg_ptr) == 0 )
  1023. {
  1024. /* msg not handled, then try here */
  1025. switch(msg)
  1026. {
  1027. case U8X8_MSG_DISPLAY_SETUP_MEMORY:
  1028. u8x8_d_helper_display_setup_memory(u8x8, &u8x8_st7565_dogm132_display_info);
  1029. break;
  1030. case U8X8_MSG_DISPLAY_INIT:
  1031. u8x8_d_helper_display_init(u8x8);
  1032. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_dogm132_init_seq);
  1033. break;
  1034. case U8X8_MSG_DISPLAY_SET_FLIP_MODE:
  1035. if ( arg_int == 0 )
  1036. {
  1037. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip0_seq);
  1038. u8x8->x_offset = u8x8->display_info->default_x_offset;
  1039. }
  1040. else
  1041. {
  1042. u8x8_cad_SendSequence(u8x8, u8x8_d_st7565_flip1_seq);
  1043. u8x8->x_offset = u8x8->display_info->flipmode_x_offset;
  1044. }
  1045. break;
  1046. default:
  1047. return 0; /* msg unknown */
  1048. }
  1049. }
  1050. return 1;
  1051. }