1 /*
2 * Copyright (c) 2022 Andreas Sandberg
3 * Copyright (c) 2020 PHYTEC Messtechnik GmbH
4 * Copyright 2024 NXP
5 *
6 * SPDX-License-Identifier: Apache-2.0
7 */
8
9 #include <string.h>
10 #include <zephyr/device.h>
11 #include <zephyr/init.h>
12 #include <zephyr/drivers/display.h>
13 #include <zephyr/drivers/gpio.h>
14 #include <zephyr/drivers/mipi_dbi.h>
15 #include <zephyr/sys/byteorder.h>
16
17 #include "uc81xx_regs.h"
18
19 #include <zephyr/logging/log.h>
20 LOG_MODULE_REGISTER(uc81xx, CONFIG_DISPLAY_LOG_LEVEL);
21
22 /**
23 * UC81XX compatible EPD controller driver.
24 *
25 * Currently only the black/white panels are supported (KW mode),
26 * also first gate/source should be 0.
27 */
28
29 #define UC81XX_PIXELS_PER_BYTE 8U
30
31 struct uc81xx_dt_array {
32 uint8_t *data;
33 uint8_t len;
34 };
35
36 enum uc81xx_profile_type {
37 UC81XX_PROFILE_FULL = 0,
38 UC81XX_PROFILE_PARTIAL,
39 UC81XX_NUM_PROFILES,
40 UC81XX_PROFILE_INVALID = UC81XX_NUM_PROFILES,
41 };
42
43 struct uc81xx_profile {
44 struct uc81xx_dt_array pwr;
45
46 uint8_t cdi;
47 bool override_cdi;
48 uint8_t tcon;
49 bool override_tcon;
50 uint8_t pll;
51 bool override_pll;
52 uint8_t vdcs;
53 bool override_vdcs;
54
55 const struct uc81xx_dt_array lutc;
56 const struct uc81xx_dt_array lutww;
57 const struct uc81xx_dt_array lutkw;
58 const struct uc81xx_dt_array lutwk;
59 const struct uc81xx_dt_array lutkk;
60 const struct uc81xx_dt_array lutbd;
61 };
62
63 struct uc81xx_quirks {
64 uint16_t max_width;
65 uint16_t max_height;
66
67 bool auto_copy;
68
69 int (*set_cdi)(const struct device *dev, bool border);
70 int (*set_tres)(const struct device *dev);
71 int (*set_ptl)(const struct device *dev, uint16_t x, uint16_t y,
72 uint16_t x_end_idx, uint16_t y_end_idx,
73 const struct display_buffer_descriptor *desc);
74 };
75
76 struct uc81xx_config {
77 const struct uc81xx_quirks *quirks;
78
79 const struct device *mipi_dev;
80 const struct mipi_dbi_config dbi_config;
81 struct gpio_dt_spec busy_gpio;
82
83 uint16_t height;
84 uint16_t width;
85
86 struct uc81xx_dt_array softstart;
87
88 const struct uc81xx_profile *profiles[UC81XX_NUM_PROFILES];
89 };
90
91 struct uc81xx_data {
92 bool blanking_on;
93 enum uc81xx_profile_type profile;
94 };
95
96
uc81xx_busy_wait(const struct device * dev)97 static inline void uc81xx_busy_wait(const struct device *dev)
98 {
99 const struct uc81xx_config *config = dev->config;
100 int pin = gpio_pin_get_dt(&config->busy_gpio);
101
102 while (pin > 0) {
103 __ASSERT(pin >= 0, "Failed to get pin level");
104 k_sleep(K_MSEC(UC81XX_BUSY_DELAY));
105 pin = gpio_pin_get_dt(&config->busy_gpio);
106 }
107 }
108
uc81xx_write_cmd(const struct device * dev,uint8_t cmd,const uint8_t * data,size_t len)109 static inline int uc81xx_write_cmd(const struct device *dev, uint8_t cmd,
110 const uint8_t *data, size_t len)
111 {
112 const struct uc81xx_config *config = dev->config;
113 int err;
114
115 uc81xx_busy_wait(dev);
116
117 err = mipi_dbi_command_write(config->mipi_dev, &config->dbi_config,
118 cmd, data, len);
119 mipi_dbi_release(config->mipi_dev, &config->dbi_config);
120 return err;
121 }
122
uc81xx_write_cmd_pattern(const struct device * dev,uint8_t cmd,uint8_t pattern,size_t len)123 static inline int uc81xx_write_cmd_pattern(const struct device *dev,
124 uint8_t cmd,
125 uint8_t pattern, size_t len)
126 {
127 const struct uc81xx_config *config = dev->config;
128 struct display_buffer_descriptor mipi_desc;
129 int err;
130 uint8_t data[64];
131
132 uc81xx_busy_wait(dev);
133
134 err = mipi_dbi_command_write(config->mipi_dev, &config->dbi_config,
135 cmd, NULL, 0);
136 if (err < 0) {
137 return err;
138 }
139
140 /*
141 * MIPI display write API requires a display buffer descriptor.
142 * Create one that describes the buffer we are writing
143 */
144 mipi_desc.height = 1;
145
146 memset(data, pattern, sizeof(data));
147 while (len) {
148 mipi_desc.buf_size = mipi_desc.width = mipi_desc.pitch =
149 MIN(len, sizeof(data));
150
151 err = mipi_dbi_write_display(config->mipi_dev,
152 &config->dbi_config,
153 data, &mipi_desc,
154 PIXEL_FORMAT_MONO10);
155 if (err < 0) {
156 goto out;
157 }
158
159 len -= mipi_desc.buf_size;
160 }
161
162 out:
163 mipi_dbi_release(config->mipi_dev, &config->dbi_config);
164 return err;
165 }
166
uc81xx_write_cmd_uint8(const struct device * dev,uint8_t cmd,uint8_t data)167 static inline int uc81xx_write_cmd_uint8(const struct device *dev, uint8_t cmd,
168 uint8_t data)
169 {
170 return uc81xx_write_cmd(dev, cmd, &data, 1);
171 }
172
uc81xx_write_array_opt(const struct device * dev,uint8_t cmd,const struct uc81xx_dt_array * array)173 static inline int uc81xx_write_array_opt(const struct device *dev, uint8_t cmd,
174 const struct uc81xx_dt_array *array)
175 {
176 if (array->len && array->data) {
177 return uc81xx_write_cmd(dev, cmd, array->data, array->len);
178 } else {
179 return 0;
180 }
181 }
182
uc81xx_have_profile(const struct device * dev,enum uc81xx_profile_type type)183 static int uc81xx_have_profile(const struct device *dev,
184 enum uc81xx_profile_type type)
185 {
186 const struct uc81xx_config *config = dev->config;
187
188 return type < UC81XX_NUM_PROFILES &&
189 config->profiles[type];
190 }
191
uc81xx_set_profile(const struct device * dev,enum uc81xx_profile_type type)192 static int uc81xx_set_profile(const struct device *dev,
193 enum uc81xx_profile_type type)
194 {
195 const struct uc81xx_config *config = dev->config;
196 const struct uc81xx_profile *p;
197 struct uc81xx_data *data = dev->data;
198 uint8_t psr =
199 UC81XX_PSR_KW_R |
200 UC81XX_PSR_UD |
201 UC81XX_PSR_SHL |
202 UC81XX_PSR_SHD |
203 UC81XX_PSR_RST;
204
205 if (type >= UC81XX_NUM_PROFILES) {
206 return -EINVAL;
207 }
208
209 /* No need to update the current profile, so do nothing */
210 if (data->profile == type) {
211 return 0;
212 }
213
214 p = config->profiles[type];
215 data->profile = type;
216
217 LOG_DBG("Initialize UC81XX controller with profile %d", type);
218
219 if (p) {
220 LOG_HEXDUMP_DBG(p->pwr.data, p->pwr.len, "PWR");
221 if (uc81xx_write_array_opt(dev, UC81XX_CMD_PWR, &p->pwr)) {
222 return -EIO;
223 }
224
225 if (uc81xx_write_array_opt(dev, UC81XX_CMD_BTST,
226 &config->softstart)) {
227 return -EIO;
228 }
229
230 /*
231 * Enable LUT overrides if a LUT has been provided by
232 * the user.
233 */
234 if (p->lutc.len || p->lutww.len || p->lutkw.len ||
235 p->lutwk.len || p->lutbd.len) {
236 LOG_DBG("Using LUT from registers");
237 psr |= UC81XX_PSR_REG;
238 }
239 }
240
241 /* Panel settings, KW mode and soft reset */
242 LOG_DBG("PSR: %#hhx", psr);
243 if (uc81xx_write_cmd_uint8(dev, UC81XX_CMD_PSR, psr)) {
244 return -EIO;
245 }
246
247 /* Set panel resolution */
248 if (config->quirks->set_tres(dev)) {
249 return -EIO;
250 }
251
252 /* Set CDI and enable border output */
253 if (config->quirks->set_cdi(dev, true)) {
254 return -EIO;
255 }
256
257 /*
258 * The rest of the configuration is optional and depends on
259 * having profile overrides specified in the device tree.
260 */
261 if (!p) {
262 return 0;
263 }
264
265 if (uc81xx_write_array_opt(dev, UC81XX_CMD_LUTC, &p->lutc)) {
266 return -EIO;
267 }
268
269 if (uc81xx_write_array_opt(dev, UC81XX_CMD_LUTWW, &p->lutww)) {
270 return -EIO;
271 }
272
273 if (uc81xx_write_array_opt(dev, UC81XX_CMD_LUTKW, &p->lutkw)) {
274 return -EIO;
275 }
276
277 if (uc81xx_write_array_opt(dev, UC81XX_CMD_LUTWK, &p->lutwk)) {
278 return -EIO;
279 }
280
281 if (uc81xx_write_array_opt(dev, UC81XX_CMD_LUTKK, &p->lutkk)) {
282 return -EIO;
283 }
284
285 if (uc81xx_write_array_opt(dev, UC81XX_CMD_LUTBD, &p->lutbd)) {
286 return -EIO;
287 }
288
289 if (p->override_pll) {
290 LOG_DBG("PLL: %#hhx", p->pll);
291 if (uc81xx_write_cmd_uint8(dev, UC81XX_CMD_PLL, p->pll)) {
292 return -EIO;
293 }
294 }
295
296 if (p->override_vdcs) {
297 LOG_DBG("VDCS: %#hhx", p->vdcs);
298 if (uc81xx_write_cmd_uint8(dev, UC81XX_CMD_VDCS, p->vdcs)) {
299 return -EIO;
300 }
301 }
302
303 if (p->override_tcon) {
304 if (uc81xx_write_cmd_uint8(dev, UC81XX_CMD_TCON, p->tcon)) {
305 return -EIO;
306 }
307 }
308
309 return 0;
310 }
311
uc81xx_update_display(const struct device * dev)312 static int uc81xx_update_display(const struct device *dev)
313 {
314 LOG_DBG("Trigger update sequence");
315
316 /* Turn on: booster, controller, regulators, and sensor. */
317 if (uc81xx_write_cmd(dev, UC81XX_CMD_PON, NULL, 0)) {
318 return -EIO;
319 }
320
321 k_sleep(K_MSEC(UC81XX_PON_DELAY));
322
323 if (uc81xx_write_cmd(dev, UC81XX_CMD_DRF, NULL, 0)) {
324 return -EIO;
325 }
326
327 k_sleep(K_MSEC(UC81XX_BUSY_DELAY));
328
329 /* Turn on: booster, controller, regulators, and sensor. */
330 if (uc81xx_write_cmd(dev, UC81XX_CMD_POF, NULL, 0)) {
331 return -EIO;
332 }
333
334 return 0;
335 }
336
uc81xx_blanking_off(const struct device * dev)337 static int uc81xx_blanking_off(const struct device *dev)
338 {
339 struct uc81xx_data *data = dev->data;
340
341 if (data->blanking_on) {
342 /* Update EPD panel in normal mode */
343 if (uc81xx_update_display(dev)) {
344 return -EIO;
345 }
346 }
347
348 data->blanking_on = false;
349
350 return 0;
351 }
352
uc81xx_blanking_on(const struct device * dev)353 static int uc81xx_blanking_on(const struct device *dev)
354 {
355 struct uc81xx_data *data = dev->data;
356
357 if (!data->blanking_on) {
358 if (uc81xx_set_profile(dev, UC81XX_PROFILE_FULL)) {
359 return -EIO;
360 }
361 }
362
363 data->blanking_on = true;
364
365 return 0;
366 }
367
uc81xx_write(const struct device * dev,const uint16_t x,const uint16_t y,const struct display_buffer_descriptor * desc,const void * buf)368 static int uc81xx_write(const struct device *dev, const uint16_t x, const uint16_t y,
369 const struct display_buffer_descriptor *desc,
370 const void *buf)
371 {
372 const struct uc81xx_config *config = dev->config;
373 struct uc81xx_data *data = dev->data;
374
375 uint16_t x_end_idx = x + desc->width - 1;
376 uint16_t y_end_idx = y + desc->height - 1;
377 size_t buf_len;
378 const uint8_t back_buffer = data->blanking_on ?
379 UC81XX_CMD_DTM1 : UC81XX_CMD_DTM2;
380
381 LOG_DBG("x %u, y %u, height %u, width %u, pitch %u",
382 x, y, desc->height, desc->width, desc->pitch);
383
384 buf_len = MIN(desc->buf_size,
385 desc->height * desc->width / UC81XX_PIXELS_PER_BYTE);
386 __ASSERT(desc->width <= desc->pitch, "Pitch is smaller than width");
387 __ASSERT(buf != NULL, "Buffer is not available");
388 __ASSERT(buf_len != 0U, "Buffer of length zero");
389 __ASSERT(!(desc->width % UC81XX_PIXELS_PER_BYTE),
390 "Buffer width not multiple of %d", UC81XX_PIXELS_PER_BYTE);
391
392 if ((y_end_idx > (config->height - 1)) ||
393 (x_end_idx > (config->width - 1))) {
394 LOG_ERR("Position out of bounds");
395 return -EINVAL;
396 }
397
398 if (!data->blanking_on) {
399 /* Blanking isn't on, so this is a partial
400 * refresh. Request the partial profile if it
401 * exists. If a partial profile hasn't been provided,
402 * we continue to use the full refresh profile. Note
403 * that the controller still only scans a partial
404 * window.
405 *
406 * This operation becomes a no-op if the profile is
407 * already active
408 */
409 if (uc81xx_have_profile(dev, UC81XX_PROFILE_PARTIAL) &&
410 uc81xx_set_profile(dev, UC81XX_PROFILE_PARTIAL)) {
411 return -EIO;
412 }
413 }
414
415 if (uc81xx_write_cmd(dev, UC81XX_CMD_PTIN, NULL, 0)) {
416 return -EIO;
417 }
418
419 if (config->quirks->set_ptl(dev, x, y, x_end_idx, y_end_idx, desc)) {
420 return -EIO;
421 }
422
423 if (uc81xx_write_cmd(dev, UC81XX_CMD_DTM2, (uint8_t *)buf, buf_len)) {
424 return -EIO;
425 }
426
427 /* Update the display */
428 if (data->blanking_on == false) {
429 /* Disable border output */
430 if (config->quirks->set_cdi(dev, false)) {
431 return -EIO;
432 }
433
434 if (uc81xx_update_display(dev)) {
435 return -EIO;
436 }
437
438 /* Enable border output */
439 if (config->quirks->set_cdi(dev, true)) {
440 return -EIO;
441 }
442 }
443
444 if (!config->quirks->auto_copy) {
445 /* Some controllers don't copy the new data to the old
446 * data buffer on refresh. Do that manually here if
447 * needed.
448 */
449
450 if (config->quirks->set_ptl(dev, x, y, x_end_idx, y_end_idx, desc)) {
451 return -EIO;
452 }
453
454 if (uc81xx_write_cmd(dev, back_buffer,
455 (uint8_t *)buf, buf_len)) {
456 return -EIO;
457 }
458 }
459
460 if (uc81xx_write_cmd(dev, UC81XX_CMD_PTOUT, NULL, 0)) {
461 return -EIO;
462 }
463
464 return 0;
465 }
466
uc81xx_get_capabilities(const struct device * dev,struct display_capabilities * caps)467 static void uc81xx_get_capabilities(const struct device *dev,
468 struct display_capabilities *caps)
469 {
470 const struct uc81xx_config *config = dev->config;
471
472 memset(caps, 0, sizeof(struct display_capabilities));
473 caps->x_resolution = config->width;
474 caps->y_resolution = config->height;
475 caps->supported_pixel_formats = PIXEL_FORMAT_MONO10;
476 caps->current_pixel_format = PIXEL_FORMAT_MONO10;
477 caps->screen_info = SCREEN_INFO_MONO_MSB_FIRST | SCREEN_INFO_EPD;
478 }
479
uc81xx_set_pixel_format(const struct device * dev,const enum display_pixel_format pf)480 static int uc81xx_set_pixel_format(const struct device *dev,
481 const enum display_pixel_format pf)
482 {
483 if (pf == PIXEL_FORMAT_MONO10) {
484 return 0;
485 }
486
487 LOG_ERR("not supported");
488 return -ENOTSUP;
489 }
490
uc81xx_clear_and_write_buffer(const struct device * dev,uint8_t pattern,bool update)491 static int uc81xx_clear_and_write_buffer(const struct device *dev,
492 uint8_t pattern, bool update)
493 {
494 const struct uc81xx_config *config = dev->config;
495 const int size = config->width * config->height
496 / UC81XX_PIXELS_PER_BYTE;
497
498 if (uc81xx_write_cmd_pattern(dev, UC81XX_CMD_DTM1, pattern, size)) {
499 return -EIO;
500 }
501
502 if (uc81xx_write_cmd_pattern(dev, UC81XX_CMD_DTM2, pattern, size)) {
503 return -EIO;
504 }
505
506 if (update == true) {
507 if (uc81xx_update_display(dev)) {
508 return -EIO;
509 }
510 }
511
512 return 0;
513 }
514
uc81xx_controller_init(const struct device * dev)515 static int uc81xx_controller_init(const struct device *dev)
516 {
517 const struct uc81xx_config *config = dev->config;
518 struct uc81xx_data *data = dev->data;
519
520 mipi_dbi_reset(config->mipi_dev, UC81XX_RESET_DELAY);
521 k_sleep(K_MSEC(UC81XX_RESET_DELAY));
522 uc81xx_busy_wait(dev);
523
524 data->blanking_on = true;
525 data->profile = UC81XX_PROFILE_INVALID;
526
527 if (uc81xx_set_profile(dev, UC81XX_PROFILE_FULL)) {
528 return -EIO;
529 }
530
531 if (uc81xx_clear_and_write_buffer(dev, 0xff, false)) {
532 return -EIO;
533 }
534
535 return 0;
536 }
537
uc81xx_init(const struct device * dev)538 static int uc81xx_init(const struct device *dev)
539 {
540 const struct uc81xx_config *config = dev->config;
541
542 LOG_DBG("");
543
544 if (!device_is_ready(config->mipi_dev)) {
545 LOG_ERR("MIPI device not ready");
546 return -ENODEV;
547 }
548
549 if (!gpio_is_ready_dt(&config->busy_gpio)) {
550 LOG_ERR("Busy GPIO device not ready");
551 return -ENODEV;
552 }
553
554 gpio_pin_configure_dt(&config->busy_gpio, GPIO_INPUT);
555
556 if (config->width > config->quirks->max_width ||
557 config->height > config->quirks->max_height) {
558 LOG_ERR("Display size out of range.");
559 return -EINVAL;
560 }
561
562 return uc81xx_controller_init(dev);
563 }
564
565 #if DT_HAS_COMPAT_STATUS_OKAY(ultrachip_uc8175)
uc81xx_set_tres_8(const struct device * dev)566 static int uc81xx_set_tres_8(const struct device *dev)
567 {
568 const struct uc81xx_config *config = dev->config;
569 const struct uc81xx_tres8 tres = {
570 .hres = config->width,
571 .vres = config->height,
572 };
573
574 LOG_HEXDUMP_DBG(&tres, sizeof(tres), "TRES");
575
576 return uc81xx_write_cmd(dev, UC81XX_CMD_TRES, (const void *)&tres, sizeof(tres));
577 }
578
uc81xx_set_ptl_8(const struct device * dev,uint16_t x,uint16_t y,uint16_t x_end_idx,uint16_t y_end_idx,const struct display_buffer_descriptor * desc)579 static inline int uc81xx_set_ptl_8(const struct device *dev, uint16_t x, uint16_t y,
580 uint16_t x_end_idx, uint16_t y_end_idx,
581 const struct display_buffer_descriptor *desc)
582 {
583 const struct uc81xx_ptl8 ptl = {
584 .hrst = x,
585 .hred = x_end_idx,
586 .vrst = y,
587 .vred = y_end_idx,
588 .flags = UC81XX_PTL_FLAG_PT_SCAN,
589 };
590
591 /* Setup Partial Window and enable Partial Mode */
592 LOG_HEXDUMP_DBG(&ptl, sizeof(ptl), "ptl");
593
594 return uc81xx_write_cmd(dev, UC81XX_CMD_PTL, (const void *)&ptl, sizeof(ptl));
595 }
596 #endif
597
598 #if DT_HAS_COMPAT_STATUS_OKAY(ultrachip_uc8176) || DT_HAS_COMPAT_STATUS_OKAY(ultrachip_uc8179)
uc81xx_set_tres_16(const struct device * dev)599 static int uc81xx_set_tres_16(const struct device *dev)
600 {
601 const struct uc81xx_config *config = dev->config;
602 const struct uc81xx_tres16 tres = {
603 .hres = sys_cpu_to_be16(config->width),
604 .vres = sys_cpu_to_be16(config->height),
605 };
606
607 LOG_HEXDUMP_DBG(&tres, sizeof(tres), "TRES");
608
609 return uc81xx_write_cmd(dev, UC81XX_CMD_TRES, (const void *)&tres, sizeof(tres));
610 }
611
uc81xx_set_ptl_16(const struct device * dev,uint16_t x,uint16_t y,uint16_t x_end_idx,uint16_t y_end_idx,const struct display_buffer_descriptor * desc)612 static inline int uc81xx_set_ptl_16(const struct device *dev, uint16_t x, uint16_t y,
613 uint16_t x_end_idx, uint16_t y_end_idx,
614 const struct display_buffer_descriptor *desc)
615 {
616 const struct uc81xx_ptl16 ptl = {
617 .hrst = sys_cpu_to_be16(x),
618 .hred = sys_cpu_to_be16(x_end_idx),
619 .vrst = sys_cpu_to_be16(y),
620 .vred = sys_cpu_to_be16(y_end_idx),
621 .flags = UC81XX_PTL_FLAG_PT_SCAN,
622 };
623
624 /* Setup Partial Window and enable Partial Mode */
625 LOG_HEXDUMP_DBG(&ptl, sizeof(ptl), "ptl");
626
627 return uc81xx_write_cmd(dev, UC81XX_CMD_PTL, (const void *)&ptl, sizeof(ptl));
628 }
629 #endif
630
631 #if DT_HAS_COMPAT_STATUS_OKAY(ultrachip_uc8175) || DT_HAS_COMPAT_STATUS_OKAY(ultrachip_uc8176)
uc8176_set_cdi(const struct device * dev,bool border)632 static int uc8176_set_cdi(const struct device *dev, bool border)
633 {
634 const struct uc81xx_config *config = dev->config;
635 const struct uc81xx_data *data = dev->data;
636 const struct uc81xx_profile *p = config->profiles[data->profile];
637 uint8_t cdi = UC8176_CDI_VBD1 | UC8176_CDI_DDX0 |
638 (p ? (p->cdi & UC8176_CDI_CDI_MASK) : 0);
639
640 if (!p || !p->override_cdi) {
641 return 0;
642 }
643
644 if (!border) {
645 /* Floating border */
646 cdi |= UC8176_CDI_VBD1 | UC8176_CDI_VBD0;
647 }
648
649 LOG_DBG("CDI: %#hhx", cdi);
650 return uc81xx_write_cmd_uint8(dev, UC81XX_CMD_CDI, cdi);
651 }
652 #endif
653
654 #if DT_HAS_COMPAT_STATUS_OKAY(ultrachip_uc8175)
655 static const struct uc81xx_quirks uc8175_quirks = {
656 .max_width = 80,
657 .max_height = 160,
658
659 .auto_copy = false,
660
661 .set_cdi = uc8176_set_cdi,
662 .set_tres = uc81xx_set_tres_8,
663 .set_ptl = uc81xx_set_ptl_8,
664 };
665 #endif
666
667 #if DT_HAS_COMPAT_STATUS_OKAY(ultrachip_uc8176)
668 static const struct uc81xx_quirks uc8176_quirks = {
669 .max_width = 400,
670 .max_height = 300,
671
672 .auto_copy = false,
673
674 .set_cdi = uc8176_set_cdi,
675 .set_tres = uc81xx_set_tres_16,
676 .set_ptl = uc81xx_set_ptl_16,
677 };
678 #endif
679
680 #if DT_HAS_COMPAT_STATUS_OKAY(ultrachip_uc8179)
uc8179_set_cdi(const struct device * dev,bool border)681 static int uc8179_set_cdi(const struct device *dev, bool border)
682 {
683 const struct uc81xx_config *config = dev->config;
684 const struct uc81xx_data *data = dev->data;
685 const struct uc81xx_profile *p = config->profiles[data->profile];
686 uint8_t cdi[UC8179_CDI_REG_LENGTH] = {
687 UC8179_CDI_BDV1 | UC8179_CDI_N2OCP | UC8179_CDI_DDX0,
688 p ? p->cdi : 0,
689 };
690
691 if (!p || !p->override_cdi) {
692 return 0;
693 }
694
695 cdi[UC8179_CDI_BDZ_DDX_IDX] |= border ? 0 : UC8179_CDI_BDZ;
696
697 LOG_HEXDUMP_DBG(cdi, sizeof(cdi), "CDI");
698 return uc81xx_write_cmd(dev, UC81XX_CMD_CDI, cdi, sizeof(cdi));
699 }
700
701 static const struct uc81xx_quirks uc8179_quirks = {
702 .max_width = 800,
703 .max_height = 600,
704
705 .auto_copy = true,
706
707 .set_cdi = uc8179_set_cdi,
708 .set_tres = uc81xx_set_tres_16,
709 .set_ptl = uc81xx_set_ptl_16,
710 };
711 #endif
712
713 static DEVICE_API(display, uc81xx_driver_api) = {
714 .blanking_on = uc81xx_blanking_on,
715 .blanking_off = uc81xx_blanking_off,
716 .write = uc81xx_write,
717 .get_capabilities = uc81xx_get_capabilities,
718 .set_pixel_format = uc81xx_set_pixel_format,
719 };
720
721 #define UC81XX_MAKE_ARRAY_OPT(n, p) \
722 static uint8_t data_ ## n ## _ ## p[] = DT_PROP_OR(n, p, {})
723
724 #define UC81XX_MAKE_ARRAY(n, p) \
725 static uint8_t data_ ## n ## _ ## p[] = DT_PROP(n, p)
726
727 #define UC81XX_ASSIGN_ARRAY(n, p) \
728 { \
729 .data = data_ ## n ## _ ## p, \
730 .len = sizeof(data_ ## n ## _ ## p), \
731 }
732
733 #define UC81XX_PROFILE(n) \
734 UC81XX_MAKE_ARRAY_OPT(n, pwr); \
735 UC81XX_MAKE_ARRAY_OPT(n, lutc); \
736 UC81XX_MAKE_ARRAY_OPT(n, lutww); \
737 UC81XX_MAKE_ARRAY_OPT(n, lutkw); \
738 UC81XX_MAKE_ARRAY_OPT(n, lutwk); \
739 UC81XX_MAKE_ARRAY_OPT(n, lutkk); \
740 UC81XX_MAKE_ARRAY_OPT(n, lutbd); \
741 \
742 static const struct uc81xx_profile uc81xx_profile_ ## n = { \
743 .pwr = UC81XX_ASSIGN_ARRAY(n, pwr), \
744 .cdi = DT_PROP_OR(n, cdi, 0), \
745 .override_cdi = DT_NODE_HAS_PROP(n, cdi), \
746 .tcon = DT_PROP_OR(n, tcon, 0), \
747 .override_tcon = DT_NODE_HAS_PROP(n, tcon), \
748 .pll = DT_PROP_OR(n, pll, 0), \
749 .override_pll = DT_NODE_HAS_PROP(n, pll), \
750 .vdcs = DT_PROP_OR(n, vdcs, 0), \
751 .override_vdcs = DT_NODE_HAS_PROP(n, vdcs), \
752 \
753 .lutc = UC81XX_ASSIGN_ARRAY(n, lutc), \
754 .lutww = UC81XX_ASSIGN_ARRAY(n, lutww), \
755 .lutkw = UC81XX_ASSIGN_ARRAY(n, lutkw), \
756 .lutwk = UC81XX_ASSIGN_ARRAY(n, lutwk), \
757 .lutkk = UC81XX_ASSIGN_ARRAY(n, lutkk), \
758 .lutbd = UC81XX_ASSIGN_ARRAY(n, lutbd), \
759 };
760
761 #define _UC81XX_PROFILE_PTR(n) &uc81xx_profile_ ## n
762
763 #define UC81XX_PROFILE_PTR(n) \
764 COND_CODE_1(DT_NODE_EXISTS(n), \
765 (_UC81XX_PROFILE_PTR(n)), \
766 NULL)
767
768 #define UC81XX_DEFINE(n, quirks_ptr) \
769 UC81XX_MAKE_ARRAY_OPT(n, softstart); \
770 \
771 DT_FOREACH_CHILD(n, UC81XX_PROFILE); \
772 \
773 static const struct uc81xx_config uc81xx_cfg_ ## n = { \
774 .quirks = quirks_ptr, \
775 .mipi_dev = DEVICE_DT_GET(DT_PARENT(n)), \
776 .dbi_config = { \
777 .mode = MIPI_DBI_MODE_SPI_4WIRE, \
778 .config = MIPI_DBI_SPI_CONFIG_DT(n, \
779 SPI_OP_MODE_MASTER | \
780 SPI_LOCK_ON | SPI_WORD_SET(8), \
781 0), \
782 }, \
783 .busy_gpio = GPIO_DT_SPEC_GET(n, busy_gpios), \
784 \
785 .height = DT_PROP(n, height), \
786 .width = DT_PROP(n, width), \
787 \
788 .softstart = UC81XX_ASSIGN_ARRAY(n, softstart), \
789 \
790 .profiles = { \
791 [UC81XX_PROFILE_FULL] = \
792 UC81XX_PROFILE_PTR(DT_CHILD(n, full)), \
793 [UC81XX_PROFILE_PARTIAL] = \
794 UC81XX_PROFILE_PTR(DT_CHILD(n, partial)), \
795 }, \
796 }; \
797 \
798 static struct uc81xx_data uc81xx_data_##n = {}; \
799 \
800 DEVICE_DT_DEFINE(n, uc81xx_init, NULL, \
801 &uc81xx_data_ ## n, \
802 &uc81xx_cfg_ ## n, \
803 POST_KERNEL, \
804 CONFIG_DISPLAY_INIT_PRIORITY, \
805 &uc81xx_driver_api);
806
807 DT_FOREACH_STATUS_OKAY_VARGS(ultrachip_uc8175, UC81XX_DEFINE,
808 &uc8175_quirks);
809
810 DT_FOREACH_STATUS_OKAY_VARGS(ultrachip_uc8176, UC81XX_DEFINE,
811 &uc8176_quirks);
812
813 DT_FOREACH_STATUS_OKAY_VARGS(ultrachip_uc8179, UC81XX_DEFINE,
814 &uc8179_quirks);
815