act_dynamic_led_display.c 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  1. /*
  2. * Copyright (c) 2021 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /*
  7. * References:
  8. *
  9. * https://www.microbit.co.uk/device/screen
  10. * https://lancaster-university.github.io/microbit-docs/ubit/display/
  11. */
  12. #include <zephyr.h>
  13. #include <init.h>
  14. #include <board.h>
  15. #include <drivers/gpio.h>
  16. #include <device.h>
  17. #include <string.h>
  18. #include <soc.h>
  19. #include <errno.h>
  20. #define LOG_LEVEL 2
  21. #include <logging/log.h>
  22. LOG_MODULE_REGISTER(led);
  23. #include <display/led_display.h>
  24. #ifdef CONFIG_CFG_DRV
  25. #include <config.h>
  26. #include <drivers/cfg_drv/driver_config.h>
  27. #endif
  28. struct pwm_chans{
  29. u8_t chan;
  30. u8_t gpio_num;
  31. };
  32. struct pwm_chans chan_gpio_num[] = {
  33. {0, 3}, {0, 4}, {0, 14}, {0, 36}, {0, 49},\
  34. {1, 5}, {1, 15}, {1, 37}, {1, 50},\
  35. {2, 6}, {2, 21}, {2, 38}, {2, 51},\
  36. {3, 7}, {3, 17}, {3, 39}, {3, 52},\
  37. {4, 8}, {4, 18}, {4, 40}, {4, 53},\
  38. {5, 9}, {5, 19}, {5, 41}, {5, 54},\
  39. {6, 10}, {6, 20}, {6, 42}, {6, 55},\
  40. {7, 11}, {7, 21}, {7, 43}, {7, 45}, {7, 56},\
  41. {8, 12}, {8, 22}, {8, 44}, {8, 46}, {8, 57},\
  42. };
  43. static u16_t get_chan(int led_index)
  44. {
  45. for(int i = 0; i < sizeof(chan_gpio_num)/sizeof(struct pwm_chans); i++)
  46. if(led_index == chan_gpio_num[i].gpio_num)
  47. return chan_gpio_num[i].chan;
  48. return -EINVAL;
  49. }
  50. int led_draw_pixel(int led_id, u32_t color, pwm_breath_ctrl_t *ctrl)
  51. {
  52. int i = 0;
  53. led_pixel_value pixel;
  54. int chan;
  55. int ret;
  56. #ifdef CONFIG_CFG_DRV
  57. CFG_Type_LED_Drive led_maps[CFG_MAX_LEDS];
  58. CFG_Type_LED_Drive *dot = NULL;
  59. struct device *op_dev;
  60. memcpy(&pixel, &color, 4);
  61. ret = cfg_get_by_key(ITEM_LED_LED,&led_maps, sizeof(led_maps));
  62. if(ret == 0) {
  63. LOG_ERR("cannot found led map\n");
  64. return -ENODEV;
  65. }
  66. for (i = 0; i < CFG_MAX_LEDS; i ++) {
  67. dot = &led_maps[i];
  68. if (dot->LED_No == (1 << led_id)) {
  69. break;
  70. }
  71. }
  72. if (!dot) {
  73. return 0;
  74. }
  75. if (dot->GPIO_Pin == 0) { //spi0_ss pin, can not used in led.
  76. return 0;
  77. }
  78. //printk("dot->LED_No:%d ,led_id:%d, gpio_pin:%d\n",dot->LED_No, led_id, dot->GPIO_Pin);
  79. switch (pixel.op_code) {
  80. case LED_OP_OFF:
  81. if(dot->GPIO_Pin/32 == 0)
  82. op_dev = device_get_binding("GPIOA");
  83. else if(dot->GPIO_Pin/32 == 1)
  84. op_dev = device_get_binding("GPIOB");
  85. else if(dot->GPIO_Pin/32 == 2)
  86. op_dev = device_get_binding("GPIOC");
  87. gpio_pin_configure(op_dev, dot->GPIO_Pin%32, GPIO_OUTPUT_ACTIVE);
  88. gpio_pin_set(op_dev, dot->GPIO_Pin, !(dot->Active_Level));
  89. break;
  90. case LED_OP_ON:
  91. if(dot->GPIO_Pin/32 == 0)
  92. op_dev = device_get_binding("GPIOA");
  93. else if(dot->GPIO_Pin/32 == 1)
  94. op_dev = device_get_binding("GPIOB");
  95. else if(dot->GPIO_Pin/32 == 2)
  96. op_dev = device_get_binding("GPIOC");
  97. gpio_pin_configure(op_dev, dot->GPIO_Pin%32, GPIO_OUTPUT_ACTIVE);
  98. gpio_pin_set(op_dev, dot->GPIO_Pin, dot->Active_Level);
  99. break;
  100. case LED_OP_BREATH:
  101. #ifdef CONFIG_PWM
  102. op_dev = device_get_binding("PWM");
  103. chan = get_chan(dot->GPIO_Pin);
  104. pwm_pin_mfp_set(op_dev, dot->GPIO_Pin);
  105. pwm_set_breath_mode(op_dev, chan, ctrl);
  106. #endif
  107. break;
  108. case LED_OP_BLINK:
  109. #ifdef CONFIG_PWM
  110. op_dev = device_get_binding("PWM");
  111. chan = get_chan(dot->GPIO_Pin);
  112. pwm_pin_mfp_set(op_dev, dot->GPIO_Pin);
  113. pwm_pin_set_cycles(op_dev, chan,8*pixel.period, 8*pixel.pulse, pixel.start_state);
  114. //pwm_pin_set_usec(dot->op_dev, dot->led_pwm, dot->pixel_value.period * 1000, dot->pixel_value.pulse * 1000, pixel.start_state);
  115. #endif
  116. }
  117. #endif
  118. return 0;
  119. }