|
2 | 2 | #include <hal/nrf_gpio.h>
|
3 | 3 | #include "displayapp/screens/Symbols.h"
|
4 | 4 | #include "drivers/PinMap.h"
|
| 5 | +#include <libraries/delay/nrf_delay.h> |
5 | 6 | using namespace Pinetime::Controllers;
|
6 | 7 |
|
| 8 | +namespace { |
| 9 | + // reinterpret_cast is not constexpr so this is the best we can do |
| 10 | + static NRF_RTC_Type* const RTC = reinterpret_cast<NRF_RTC_Type*>(NRF_RTC2_BASE); |
| 11 | +} |
| 12 | + |
7 | 13 | void BrightnessController::Init() {
|
8 | 14 | nrf_gpio_cfg_output(PinMap::LcdBacklightLow);
|
9 | 15 | nrf_gpio_cfg_output(PinMap::LcdBacklightMedium);
|
10 | 16 | nrf_gpio_cfg_output(PinMap::LcdBacklightHigh);
|
| 17 | + |
| 18 | + nrf_gpio_pin_clear(PinMap::LcdBacklightLow); |
| 19 | + nrf_gpio_pin_clear(PinMap::LcdBacklightMedium); |
| 20 | + nrf_gpio_pin_clear(PinMap::LcdBacklightHigh); |
| 21 | + |
| 22 | + static_assert(timerFrequency == 32768, "Change the prescaler below"); |
| 23 | + RTC->PRESCALER = 0; |
| 24 | + // CC1 switches the backlight on (pin transitions from high to low) and resets the counter to 0 |
| 25 | + RTC->CC[1] = timerPeriod; |
| 26 | + // Enable compare events for CC0,CC1 |
| 27 | + RTC->EVTEN = 0b0000'0000'0000'0011'0000'0000'0000'0000; |
| 28 | + // Disable all interrupts |
| 29 | + RTC->INTENCLR = 0b0000'0000'0000'1111'0000'0000'0000'0011; |
11 | 30 | Set(level);
|
12 | 31 | }
|
13 | 32 |
|
| 33 | +void BrightnessController::ApplyBrightness(uint16_t rawBrightness) { |
| 34 | + // The classic off, low, medium, high brightnesses are at {0, timerPeriod, timerPeriod*2, timerPeriod*3} |
| 35 | + // These brightness levels do not use PWM: they only set/clear the corresponding pins |
| 36 | + // Any brightness level between the above levels is achieved with efficient RTC based PWM on the next pin up |
| 37 | + // E.g 2.5*timerPeriod corresponds to medium brightness with 50% PWM on the high pin |
| 38 | + // Note: Raw brightness does not necessarily correspond to a linear perceived brightness |
| 39 | + |
| 40 | + uint8_t pin; |
| 41 | + if (rawBrightness > 2 * timerPeriod) { |
| 42 | + rawBrightness -= 2 * timerPeriod; |
| 43 | + pin = PinMap::LcdBacklightHigh; |
| 44 | + } else if (rawBrightness > timerPeriod) { |
| 45 | + rawBrightness -= timerPeriod; |
| 46 | + pin = PinMap::LcdBacklightMedium; |
| 47 | + } else { |
| 48 | + pin = PinMap::LcdBacklightLow; |
| 49 | + } |
| 50 | + if (rawBrightness == timerPeriod || rawBrightness == 0) { |
| 51 | + if (lastPin != UNSET) { |
| 52 | + RTC->TASKS_STOP = 1; |
| 53 | + nrf_delay_us(rtcStopTime); |
| 54 | + nrf_ppi_channel_disable(ppiBacklightOff); |
| 55 | + nrf_ppi_channel_disable(ppiBacklightOn); |
| 56 | + nrfx_gpiote_out_uninit(lastPin); |
| 57 | + nrf_gpio_cfg_output(lastPin); |
| 58 | + } |
| 59 | + lastPin = UNSET; |
| 60 | + if (rawBrightness == 0) { |
| 61 | + nrf_gpio_pin_set(pin); |
| 62 | + } else { |
| 63 | + nrf_gpio_pin_clear(pin); |
| 64 | + } |
| 65 | + } else { |
| 66 | + // If the pin on which we are doing PWM is changing |
| 67 | + // Disable old PWM channel (if exists) and set up new one |
| 68 | + if (lastPin != pin) { |
| 69 | + if (lastPin != UNSET) { |
| 70 | + RTC->TASKS_STOP = 1; |
| 71 | + nrf_delay_us(rtcStopTime); |
| 72 | + nrf_ppi_channel_disable(ppiBacklightOff); |
| 73 | + nrf_ppi_channel_disable(ppiBacklightOn); |
| 74 | + nrfx_gpiote_out_uninit(lastPin); |
| 75 | + nrf_gpio_cfg_output(lastPin); |
| 76 | + } |
| 77 | + nrfx_gpiote_out_config_t gpioteCfg = {.action = NRF_GPIOTE_POLARITY_TOGGLE, |
| 78 | + .init_state = NRF_GPIOTE_INITIAL_VALUE_LOW, |
| 79 | + .task_pin = true}; |
| 80 | + APP_ERROR_CHECK(nrfx_gpiote_out_init(pin, &gpioteCfg)); |
| 81 | + nrfx_gpiote_out_task_enable(pin); |
| 82 | + nrf_ppi_channel_endpoint_setup(ppiBacklightOff, |
| 83 | + reinterpret_cast<uint32_t>(&RTC->EVENTS_COMPARE[0]), |
| 84 | + nrfx_gpiote_out_task_addr_get(pin)); |
| 85 | + nrf_ppi_channel_endpoint_setup(ppiBacklightOn, |
| 86 | + reinterpret_cast<uint32_t>(&RTC->EVENTS_COMPARE[1]), |
| 87 | + nrfx_gpiote_out_task_addr_get(pin)); |
| 88 | + nrf_ppi_fork_endpoint_setup(ppiBacklightOn, reinterpret_cast<uint32_t>(&RTC->TASKS_CLEAR)); |
| 89 | + nrf_ppi_channel_enable(ppiBacklightOff); |
| 90 | + nrf_ppi_channel_enable(ppiBacklightOn); |
| 91 | + } else { |
| 92 | + // If the pin used for PWM isn't changing, we only need to set the pin state to the initial value (low) |
| 93 | + RTC->TASKS_STOP = 1; |
| 94 | + nrf_delay_us(rtcStopTime); |
| 95 | + // Due to errata 20,179 and the intricacies of RTC timing, keep it simple: override the pin state |
| 96 | + nrfx_gpiote_out_task_force(pin, false); |
| 97 | + } |
| 98 | + // CC0 switches the backlight off (pin transitions from low to high) |
| 99 | + RTC->CC[0] = rawBrightness; |
| 100 | + RTC->TASKS_CLEAR = 1; |
| 101 | + RTC->TASKS_START = 1; |
| 102 | + lastPin = pin; |
| 103 | + } |
| 104 | + switch (pin) { |
| 105 | + case PinMap::LcdBacklightHigh: |
| 106 | + nrf_gpio_pin_clear(PinMap::LcdBacklightLow); |
| 107 | + nrf_gpio_pin_clear(PinMap::LcdBacklightMedium); |
| 108 | + break; |
| 109 | + case PinMap::LcdBacklightMedium: |
| 110 | + nrf_gpio_pin_clear(PinMap::LcdBacklightLow); |
| 111 | + nrf_gpio_pin_set(PinMap::LcdBacklightHigh); |
| 112 | + break; |
| 113 | + case PinMap::LcdBacklightLow: |
| 114 | + nrf_gpio_pin_set(PinMap::LcdBacklightMedium); |
| 115 | + nrf_gpio_pin_set(PinMap::LcdBacklightHigh); |
| 116 | + } |
| 117 | +} |
| 118 | + |
14 | 119 | void BrightnessController::Set(BrightnessController::Levels level) {
|
15 | 120 | this->level = level;
|
16 | 121 | switch (level) {
|
17 | 122 | default:
|
18 | 123 | case Levels::High:
|
19 |
| - nrf_gpio_pin_clear(PinMap::LcdBacklightLow); |
20 |
| - nrf_gpio_pin_clear(PinMap::LcdBacklightMedium); |
21 |
| - nrf_gpio_pin_clear(PinMap::LcdBacklightHigh); |
| 124 | + ApplyBrightness(3 * timerPeriod); |
22 | 125 | break;
|
23 | 126 | case Levels::Medium:
|
24 |
| - nrf_gpio_pin_clear(PinMap::LcdBacklightLow); |
25 |
| - nrf_gpio_pin_clear(PinMap::LcdBacklightMedium); |
26 |
| - nrf_gpio_pin_set(PinMap::LcdBacklightHigh); |
| 127 | + ApplyBrightness(2 * timerPeriod); |
27 | 128 | break;
|
28 | 129 | case Levels::Low:
|
29 |
| - nrf_gpio_pin_clear(PinMap::LcdBacklightLow); |
30 |
| - nrf_gpio_pin_set(PinMap::LcdBacklightMedium); |
31 |
| - nrf_gpio_pin_set(PinMap::LcdBacklightHigh); |
| 130 | + ApplyBrightness(timerPeriod); |
| 131 | + break; |
| 132 | + case Levels::AlwaysOn: |
| 133 | + ApplyBrightness(timerPeriod / 10); |
32 | 134 | break;
|
33 | 135 | case Levels::Off:
|
34 |
| - nrf_gpio_pin_set(PinMap::LcdBacklightLow); |
35 |
| - nrf_gpio_pin_set(PinMap::LcdBacklightMedium); |
36 |
| - nrf_gpio_pin_set(PinMap::LcdBacklightHigh); |
| 136 | + ApplyBrightness(0); |
37 | 137 | break;
|
38 | 138 | }
|
39 | 139 | }
|
|
0 commit comments