u8x8_d_custom.c 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248
  1. /*
  2. u8x8_d_custom.c
  3. Universal 8bit Graphics Library (https://github.com/olikraus/u8g2/)
  4. */
  5. #include "luat_u8g2.h"
  6. #include "luat_timer.h"
  7. static luat_u8g2_conf_t* conf = NULL;
  8. /*
  9. (24)
  10. (21), (0xae)
  11. (21), (0xd5), (22), (0x80)
  12. (21), (0xa8), (22), (0x3f)
  13. (21), (0xd3), (22), (0x00)
  14. (21), (0x40)
  15. (21), (0x8d), (22), (0x14)
  16. (21), (0x20), (22), (0x00)
  17. (21), (0xa1)
  18. (21), (0xc8)
  19. (21), (0xda), (22), (0x12)
  20. (21), (0x81), (22), (0xcf)
  21. (21), (0xd9), (22), (0xf1)
  22. (21), (0xdb), (22), (0x40)
  23. (21), (0x2e)
  24. (21), (0xa4)
  25. (21), (0xa6)
  26. (25)
  27. (0xff)
  28. */
  29. /* more or less generic setup of all these small OLEDs */
  30. // static const uint8_t u8x8_d_custom_noname_init_seq[] = {
  31. // U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  32. // U8X8_C(0x0ae), /* display off */
  33. // U8X8_CA(0x0d5, 0x080), /* clock divide ratio (0x00=1) and oscillator frequency (0x8) */
  34. // U8X8_CA(0x0a8, 0x03f), /* multiplex ratio */
  35. // U8X8_CA(0x0d3, 0x000), /* display offset */
  36. // U8X8_C(0x040), /* set display start line to 0 */
  37. // U8X8_CA(0x08d, 0x014), /* [2] charge pump setting (p62): 0x014 enable, 0x010 disable, SSD1306 only, should be removed for SH1106 */
  38. // U8X8_CA(0x020, 0x000), /* horizontal addressing mode */
  39. // U8X8_C(0x0a1), /* segment remap a0/a1*/
  40. // U8X8_C(0x0c8), /* c0: scan dir normal, c8: reverse */
  41. // // Flipmode
  42. // // U8X8_C(0x0a0), /* segment remap a0/a1*/
  43. // // U8X8_C(0x0c0), /* c0: scan dir normal, c8: reverse */
  44. // U8X8_CA(0x0da, 0x012), /* com pin HW config, sequential com pin config (bit 4), disable left/right remap (bit 5) */
  45. // U8X8_CA(0x081, 0x0cf), /* [2] set contrast control */
  46. // U8X8_CA(0x0d9, 0x0f1), /* [2] pre-charge period 0x022/f1*/
  47. // U8X8_CA(0x0db, 0x040), /* vcomh deselect level */
  48. // // if vcomh is 0, then this will give the biggest range for contrast control issue #98
  49. // // restored the old values for the noname constructor, because vcomh=0 will not work for all OLEDs, #116
  50. // U8X8_C(0x02e), /* Deactivate scroll */
  51. // U8X8_C(0x0a4), /* output ram to display */
  52. // U8X8_C(0x0a6), /* none inverted normal display mode */
  53. // U8X8_END_TRANSFER(), /* disable chip */
  54. // U8X8_END() /* end of sequence */
  55. // };
  56. static const uint8_t u8x8_d_custom_noname_flip0_seq[] = {
  57. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  58. U8X8_C(0x0a1), /* segment remap a0/a1*/
  59. U8X8_C(0x0c8), /* c0: scan dir normal, c8: reverse */
  60. U8X8_END_TRANSFER(), /* disable chip */
  61. U8X8_END() /* end of sequence */
  62. };
  63. static const uint8_t u8x8_d_custom_noname_flip1_seq[] = {
  64. U8X8_START_TRANSFER(), /* enable chip, delay is part of the transfer start */
  65. U8X8_C(0x0a0), /* segment remap a0/a1*/
  66. U8X8_C(0x0c0), /* c0: scan dir normal, c8: reverse */
  67. U8X8_END_TRANSFER(), /* disable chip */
  68. U8X8_END() /* end of sequence */
  69. };
  70. static uint8_t u8x8_d_custom_generic(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
  71. {
  72. uint8_t x, c;
  73. uint8_t *ptr;
  74. switch(msg)
  75. {
  76. /* handled by the calling function
  77. case U8X8_MSG_DISPLAY_SETUP_MEMORY:
  78. u8x8_d_helper_display_setup_memory(u8x8, &u8x8_custom_noname_display_info);
  79. break;
  80. */
  81. /* handled by the calling function
  82. case U8X8_MSG_DISPLAY_INIT:
  83. u8x8_d_helper_display_init(u8x8);
  84. u8x8_cad_SendSequence(u8x8, u8x8_d_custom_noname_init_seq);
  85. break;
  86. */
  87. case U8X8_MSG_DISPLAY_SET_POWER_SAVE:
  88. if ( arg_int == 0 ){
  89. u8x8_cad_StartTransfer(u8x8);
  90. u8x8_cad_SendCmd(u8x8, conf->wakecmd);
  91. u8x8_cad_EndTransfer(u8x8);
  92. }
  93. else{
  94. u8x8_cad_StartTransfer(u8x8);
  95. u8x8_cad_SendCmd(u8x8, conf->sleepcmd);
  96. u8x8_cad_EndTransfer(u8x8);
  97. }
  98. break;
  99. case U8X8_MSG_DISPLAY_SET_FLIP_MODE:
  100. if ( arg_int == 0 )
  101. {
  102. u8x8_cad_SendSequence(u8x8, u8x8_d_custom_noname_flip0_seq);
  103. u8x8->x_offset = u8x8->display_info->default_x_offset;
  104. }
  105. else
  106. {
  107. u8x8_cad_SendSequence(u8x8, u8x8_d_custom_noname_flip1_seq);
  108. u8x8->x_offset = u8x8->display_info->flipmode_x_offset;
  109. }
  110. break;
  111. #ifdef U8X8_WITH_SET_CONTRAST
  112. case U8X8_MSG_DISPLAY_SET_CONTRAST:
  113. u8x8_cad_StartTransfer(u8x8);
  114. u8x8_cad_SendCmd(u8x8, 0x081 );
  115. u8x8_cad_SendArg(u8x8, arg_int ); /* ssd1306 has range from 0 to 255 */
  116. u8x8_cad_EndTransfer(u8x8);
  117. break;
  118. #endif
  119. case U8X8_MSG_DISPLAY_DRAW_TILE:
  120. u8x8_cad_StartTransfer(u8x8);
  121. x = ((u8x8_tile_t *)arg_ptr)->x_pos;
  122. x *= 8;
  123. x += u8x8->x_offset;
  124. u8x8_cad_SendCmd(u8x8, 0x040 ); /* set line offset to 0 */
  125. u8x8_cad_SendCmd(u8x8, 0x010 | (x>>4) );
  126. u8x8_cad_SendArg(u8x8, 0x000 | ((x&15))); /* probably wrong, should be SendCmd */
  127. u8x8_cad_SendArg(u8x8, 0x0b0 | (((u8x8_tile_t *)arg_ptr)->y_pos)); /* probably wrong, should be SendCmd */
  128. do
  129. {
  130. c = ((u8x8_tile_t *)arg_ptr)->cnt;
  131. ptr = ((u8x8_tile_t *)arg_ptr)->tile_ptr;
  132. u8x8_cad_SendData(u8x8, c*8, ptr); /* note: SendData can not handle more than 255 bytes */
  133. /*
  134. do
  135. {
  136. u8x8_cad_SendData(u8x8, 8, ptr);
  137. ptr += 8;
  138. c--;
  139. } while( c > 0 );
  140. */
  141. arg_int--;
  142. } while( arg_int > 0 );
  143. u8x8_cad_EndTransfer(u8x8);
  144. break;
  145. default:
  146. return 0;
  147. }
  148. return 1;
  149. }
  150. static u8x8_display_info_t u8x8_custom_noname_display_info =
  151. {
  152. /* chip_enable_level = */ 0,
  153. /* chip_disable_level = */ 1,
  154. /* post_chip_enable_wait_ns = */ 20,
  155. /* pre_chip_disable_wait_ns = */ 10,
  156. /* reset_pulse_width_ms = */ 100, /* SSD1306: 3 us */
  157. /* post_reset_wait_ms = */ 100, /* far east OLEDs need much longer setup time */
  158. /* sda_setup_time_ns = */ 50, /* SSD1306: 15ns, but cycle time is 100ns, so use 100/2 */
  159. /* sck_pulse_width_ns = */ 50, /* SSD1306: 20ns, but cycle time is 100ns, so use 100/2, AVR: below 70: 8 MHz, >= 70 --> 4MHz clock */
  160. /* sck_clock_hz = */ 8000000UL, /* since Arduino 1.6.0, the SPI bus speed in Hz. Should be 1000000000/sck_pulse_width_ns */
  161. /* spi_mode = */ 0, /* active high, rising edge */
  162. /* i2c_bus_clock_100kHz = */ 4,
  163. /* data_setup_time_ns = */ 40,
  164. /* write_pulse_width_ns = */ 150, /* SSD1306: cycle time is 300ns, so use 300/2 = 150 */
  165. /* tile_width = */ 16,
  166. /* tile_height = */ 8,
  167. /* default_x_offset = */ 0,
  168. /* flipmode_x_offset = */ 0,
  169. /* pixel_width = */ 128,
  170. /* pixel_height = */ 64
  171. };
  172. uint8_t u8x8_d_custom_noname(u8x8_t *u8x8, uint8_t msg, uint8_t arg_int, void *arg_ptr)
  173. {
  174. uint32_t cmd = 0;
  175. if ( u8x8_d_custom_generic(u8x8, msg, arg_int, arg_ptr) != 0 )
  176. return 1;
  177. luat_u8g2_custom_t* u8g2_custom = NULL;
  178. // printf("u8x8_d_custom_noname msg:%d\n",msg);
  179. switch(msg)
  180. {
  181. case U8X8_MSG_DISPLAY_INIT:
  182. u8x8_d_helper_display_init(u8x8);
  183. // u8x8_cad_SendSequence(u8x8, u8x8_d_custom_noname_init_seq);
  184. u8g2_custom = (luat_u8g2_custom_t*)conf->userdata;
  185. u8x8_cad_StartTransfer(u8x8);
  186. for (size_t i = 0; i < u8g2_custom->init_cmd_count; i++){
  187. cmd = u8g2_custom->initcmd[i];
  188. switch(((cmd >> 16) & 0xFF)) {
  189. case 0x0000 :
  190. u8x8->cad_cb(u8x8, U8X8_MSG_CAD_SEND_CMD, (uint8_t)(cmd & 0xFF), NULL);
  191. break;
  192. case 0x0001 :
  193. luat_timer_mdelay(cmd & 0xFF);
  194. break;
  195. case 0x0002 :
  196. u8x8->cad_cb(u8x8, U8X8_MSG_CAD_SEND_CMD, (uint8_t)(cmd & 0xFF), NULL);
  197. break;
  198. case 0x0003 :
  199. u8x8->cad_cb(u8x8, U8X8_MSG_CAD_SEND_CMD, (uint8_t)(cmd & 0xFF), NULL);
  200. break;
  201. default:
  202. break;
  203. }
  204. }
  205. u8x8_cad_EndTransfer(u8x8);
  206. break;
  207. case U8X8_MSG_DISPLAY_SETUP_MEMORY:
  208. conf = (luat_u8g2_conf_t*)u8x8->user_ptr;
  209. u8x8_custom_noname_display_info.pixel_width = conf->w;
  210. u8x8_custom_noname_display_info.pixel_height = conf->h;
  211. u8x8_custom_noname_display_info.tile_width = conf->w/8;
  212. u8x8_custom_noname_display_info.tile_height = conf->h/8;
  213. u8x8_d_helper_display_setup_memory(u8x8, &u8x8_custom_noname_display_info);
  214. break;
  215. default:
  216. return 0;
  217. }
  218. return 1;
  219. }