logo

qmk_firmware

custom branch of QMK firmware git clone https://anongit.hacktivis.me/git/qmk_firmware.git

qp.c (10186B)


  1. // Copyright 2021 Nick Brassel (@tzarc)
  2. // SPDX-License-Identifier: GPL-2.0-or-later
  3. #include <quantum.h>
  4. #include <utf8.h>
  5. #include "qp_internal.h"
  6. #include "qp_comms.h"
  7. #include "qp_draw.h"
  8. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  9. // Internal driver validation
  10. static bool validate_driver_vtable(painter_driver_t *driver) {
  11. return (driver && driver->driver_vtable && driver->driver_vtable->init && driver->driver_vtable->power && driver->driver_vtable->clear && driver->driver_vtable->viewport && driver->driver_vtable->pixdata && driver->driver_vtable->palette_convert && driver->driver_vtable->append_pixels && driver->driver_vtable->append_pixdata) ? true : false;
  12. }
  13. static bool validate_comms_vtable(painter_driver_t *driver) {
  14. return (driver && driver->comms_vtable && driver->comms_vtable->comms_init && driver->comms_vtable->comms_start && driver->comms_vtable->comms_stop && driver->comms_vtable->comms_send) ? true : false;
  15. }
  16. static bool validate_driver_integrity(painter_driver_t *driver) {
  17. return validate_driver_vtable(driver) && validate_comms_vtable(driver);
  18. }
  19. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  20. // Quantum Painter External API: qp_init
  21. bool qp_init(painter_device_t device, painter_rotation_t rotation) {
  22. qp_dprintf("qp_init: entry\n");
  23. painter_driver_t *driver = (painter_driver_t *)device;
  24. if (!driver) {
  25. qp_dprintf("qp_init: fail (pointer to NULL)\n");
  26. return false;
  27. }
  28. driver->validate_ok = false;
  29. if (!validate_driver_integrity(driver)) {
  30. qp_dprintf("Failed to validate driver integrity in qp_init\n");
  31. return false;
  32. }
  33. driver->validate_ok = true;
  34. if (!qp_comms_init(device)) {
  35. driver->validate_ok = false;
  36. qp_dprintf("qp_init: fail (could not init comms)\n");
  37. return false;
  38. }
  39. if (!qp_comms_start(device)) {
  40. qp_dprintf("qp_init: fail (could not start comms)\n");
  41. return false;
  42. }
  43. // Set the rotation before init
  44. driver->rotation = rotation;
  45. // Invoke init
  46. bool ret = driver->driver_vtable->init(device, rotation);
  47. qp_comms_stop(device);
  48. qp_dprintf("qp_init: %s\n", ret ? "ok" : "fail");
  49. return ret;
  50. }
  51. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  52. // Quantum Painter External API: qp_power
  53. bool qp_power(painter_device_t device, bool power_on) {
  54. qp_dprintf("qp_power: entry\n");
  55. painter_driver_t *driver = (painter_driver_t *)device;
  56. if (!driver || !driver->validate_ok) {
  57. qp_dprintf("qp_power: fail (validation_ok == false)\n");
  58. return false;
  59. }
  60. if (!qp_comms_start(device)) {
  61. qp_dprintf("qp_power: fail (could not start comms)\n");
  62. return false;
  63. }
  64. bool ret = driver->driver_vtable->power(device, power_on);
  65. qp_comms_stop(device);
  66. qp_dprintf("qp_power: %s\n", ret ? "ok" : "fail");
  67. return ret;
  68. }
  69. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  70. // Quantum Painter External API: qp_clear
  71. bool qp_clear(painter_device_t device) {
  72. qp_dprintf("qp_clear: entry\n");
  73. painter_driver_t *driver = (painter_driver_t *)device;
  74. if (!driver || !driver->validate_ok) {
  75. qp_dprintf("qp_clear: fail (validation_ok == false)\n");
  76. return false;
  77. }
  78. if (!qp_comms_start(device)) {
  79. qp_dprintf("qp_clear: fail (could not start comms)\n");
  80. return false;
  81. }
  82. bool ret = driver->driver_vtable->clear(device);
  83. qp_comms_stop(device);
  84. qp_dprintf("qp_clear: %s\n", ret ? "ok" : "fail");
  85. return ret;
  86. }
  87. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  88. // Quantum Painter External API: qp_flush
  89. bool qp_flush(painter_device_t device) {
  90. qp_dprintf("qp_flush: entry\n");
  91. painter_driver_t *driver = (painter_driver_t *)device;
  92. if (!driver || !driver->validate_ok) {
  93. qp_dprintf("qp_flush: fail (validation_ok == false)\n");
  94. return false;
  95. }
  96. if (!qp_comms_start(device)) {
  97. qp_dprintf("qp_flush: fail (could not start comms)\n");
  98. return false;
  99. }
  100. bool ret = driver->driver_vtable->flush(device);
  101. qp_comms_stop(device);
  102. qp_dprintf("qp_flush: %s\n", ret ? "ok" : "fail");
  103. return ret;
  104. }
  105. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  106. // Quantum Painter External API: qp_get_*
  107. uint16_t qp_get_width(painter_device_t device) {
  108. qp_dprintf("qp_get_width: entry\n");
  109. painter_driver_t *driver = (painter_driver_t *)device;
  110. if (!driver || !driver->validate_ok) {
  111. qp_dprintf("qp_get_width: fail (invalid driver)\n");
  112. return 0;
  113. }
  114. uint16_t width;
  115. switch (driver->rotation) {
  116. default:
  117. case QP_ROTATION_0:
  118. case QP_ROTATION_180:
  119. width = driver->panel_width;
  120. break;
  121. case QP_ROTATION_90:
  122. case QP_ROTATION_270:
  123. width = driver->panel_height;
  124. break;
  125. }
  126. qp_dprintf("qp_get_width: ok\n");
  127. return width;
  128. }
  129. uint16_t qp_get_height(painter_device_t device) {
  130. qp_dprintf("qp_get_height: entry\n");
  131. painter_driver_t *driver = (painter_driver_t *)device;
  132. if (!driver || !driver->validate_ok) {
  133. qp_dprintf("qp_get_height: fail (invalid driver)\n");
  134. return 0;
  135. }
  136. uint16_t height;
  137. switch (driver->rotation) {
  138. default:
  139. case QP_ROTATION_0:
  140. case QP_ROTATION_180:
  141. height = driver->panel_height;
  142. break;
  143. case QP_ROTATION_90:
  144. case QP_ROTATION_270:
  145. height = driver->panel_width;
  146. break;
  147. }
  148. qp_dprintf("qp_get_height: ok\n");
  149. return height;
  150. }
  151. painter_rotation_t qp_get_rotation(painter_device_t device) {
  152. qp_dprintf("qp_get_rotation: entry\n");
  153. painter_driver_t *driver = (painter_driver_t *)device;
  154. if (!driver || !driver->validate_ok) {
  155. qp_dprintf("qp_get_rotation: fail (invalid driver)\n");
  156. return QP_ROTATION_0;
  157. }
  158. qp_dprintf("qp_get_rotation: ok\n");
  159. return driver->rotation;
  160. }
  161. uint16_t qp_get_offset_x(painter_device_t device) {
  162. qp_dprintf("qp_get_offset_x: entry\n");
  163. painter_driver_t *driver = (painter_driver_t *)device;
  164. if (!driver || !driver->validate_ok) {
  165. qp_dprintf("qp_get_offset_x: fail (invalid driver)\n");
  166. return 0;
  167. }
  168. qp_dprintf("qp_get_offset_x: ok\n");
  169. return driver->offset_x;
  170. }
  171. uint16_t qp_get_offset_y(painter_device_t device) {
  172. qp_dprintf("qp_get_offset_y: entry\n");
  173. painter_driver_t *driver = (painter_driver_t *)device;
  174. if (!driver || !driver->validate_ok) {
  175. qp_dprintf("qp_get_offset_y: fail (invalid driver)\n");
  176. return 0;
  177. }
  178. qp_dprintf("qp_get_offset_y: ok\n");
  179. return driver->offset_y;
  180. }
  181. void qp_get_geometry(painter_device_t device, uint16_t *width, uint16_t *height, painter_rotation_t *rotation, uint16_t *offset_x, uint16_t *offset_y) {
  182. qp_dprintf("qp_geometry: entry\n");
  183. painter_driver_t *driver = (painter_driver_t *)device;
  184. if (!driver || !driver->validate_ok) {
  185. qp_dprintf("qp_geometry: fail (invalid driver)\n");
  186. return;
  187. }
  188. if (width) {
  189. *width = qp_get_width(device);
  190. }
  191. if (height) {
  192. *height = qp_get_height(device);
  193. }
  194. if (rotation) {
  195. *rotation = qp_get_rotation(device);
  196. }
  197. if (offset_x) {
  198. *offset_x = qp_get_offset_x(device);
  199. }
  200. if (offset_y) {
  201. *offset_y = qp_get_offset_y(device);
  202. }
  203. qp_dprintf("qp_get_geometry: ok\n");
  204. }
  205. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  206. // Quantum Painter External API: qp_set_viewport_offsets
  207. void qp_set_viewport_offsets(painter_device_t device, uint16_t offset_x, uint16_t offset_y) {
  208. qp_dprintf("qp_set_viewport_offsets: entry\n");
  209. painter_driver_t *driver = (painter_driver_t *)device;
  210. if (!driver) {
  211. qp_dprintf("qp_set_viewport_offsets: fail (pointer to NULL)\n");
  212. return;
  213. }
  214. driver->offset_x = offset_x;
  215. driver->offset_y = offset_y;
  216. qp_dprintf("qp_set_viewport_offsets: ok\n");
  217. }
  218. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  219. // Quantum Painter External API: qp_viewport
  220. bool qp_viewport(painter_device_t device, uint16_t left, uint16_t top, uint16_t right, uint16_t bottom) {
  221. qp_dprintf("qp_viewport: entry\n");
  222. painter_driver_t *driver = (painter_driver_t *)device;
  223. if (!driver || !driver->validate_ok) {
  224. qp_dprintf("qp_viewport: fail (validation_ok == false)\n");
  225. return false;
  226. }
  227. if (!qp_comms_start(device)) {
  228. qp_dprintf("qp_viewport: fail (could not start comms)\n");
  229. return false;
  230. }
  231. // Set the viewport
  232. bool ret = driver->driver_vtable->viewport(device, left, top, right, bottom);
  233. qp_dprintf("qp_viewport: %s\n", ret ? "ok" : "fail");
  234. qp_comms_stop(device);
  235. return ret;
  236. }
  237. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  238. // Quantum Painter External API: qp_pixdata
  239. bool qp_pixdata(painter_device_t device, const void *pixel_data, uint32_t native_pixel_count) {
  240. qp_dprintf("qp_pixdata: entry\n");
  241. painter_driver_t *driver = (painter_driver_t *)device;
  242. if (!driver || !driver->validate_ok) {
  243. qp_dprintf("qp_pixdata: fail (validation_ok == false)\n");
  244. return false;
  245. }
  246. if (!qp_comms_start(device)) {
  247. qp_dprintf("qp_pixdata: fail (could not start comms)\n");
  248. return false;
  249. }
  250. bool ret = driver->driver_vtable->pixdata(device, pixel_data, native_pixel_count);
  251. qp_dprintf("qp_pixdata: %s\n", ret ? "ok" : "fail");
  252. qp_comms_stop(device);
  253. return ret;
  254. }