Refactored MDx backlight driver

This commit is contained in:
Silvano Seva 2023-07-13 17:25:47 +02:00
parent 313e9615a6
commit 163a3df607
2 changed files with 53 additions and 87 deletions

View File

@ -23,10 +23,37 @@
#include <hwconfig.h> #include <hwconfig.h>
#include "backlight.h" #include "backlight.h"
#ifndef PLATFORM_MDUV3x0 /* MD-3x0 and MD-9600 */ #if defined(PLATFORM_MDUV3x0) && defined(ENABLE_BKLIGHT_DIMMING)
/*
* Interrupt-based software PWM for backlight dimming on MD-UV3x0.
* On this family of devices the GPIO for backlight control is not connected to
* any of the available timer compare output channels, thus making impossible to
* implement an hardware-based PWM.
* However, we provide a software-base backlight dimming for experimental purposes.
*/
/* Name of interrupt handler is mangled for C++ compatibility */
void _Z29TIM1_TRG_COM_TIM11_IRQHandlerv()
{
if(TIM11->SR & TIM_SR_CC1IF)
{
gpio_clearPin(LCD_BKLIGHT); /* Clear pin on compare match */
}
if(TIM11->SR & TIM_SR_UIF)
{
gpio_setPin(LCD_BKLIGHT); /* Set pin on counter reload */
}
TIM11->SR = 0;
}
#endif
void backlight_init() void backlight_init()
{ {
#ifndef PLATFORM_MDUV3x0 /* MD-3x0 and MD-9600 */
gpio_setMode(LCD_BKLIGHT, ALTERNATE); gpio_setMode(LCD_BKLIGHT, ALTERNATE);
gpio_setAlternateFunction(LCD_BKLIGHT, 3); gpio_setAlternateFunction(LCD_BKLIGHT, 3);
@ -52,61 +79,11 @@ void backlight_init()
TIM8->CCR1 = 0; TIM8->CCR1 = 0;
TIM8->EGR = TIM_EGR_UG; /* Update registers */ TIM8->EGR = TIM_EGR_UG; /* Update registers */
TIM8->CR1 |= TIM_CR1_CEN; /* Start timer */ TIM8->CR1 |= TIM_CR1_CEN; /* Start timer */
} #else
void backlight_terminate()
{
/* Shut down backlight */
gpio_setMode(LCD_BKLIGHT, OUTPUT);
gpio_clearPin(LCD_BKLIGHT);
/* Shut down timer */
RCC->APB2ENR &= ~RCC_APB2ENR_TIM8EN;
__DSB();
}
/*
* This function is defined in platform.h
*/
void platform_setBacklightLevel(uint8_t level)
{
if(level > 100) level = 100;
// Convert value to 0 - 255
TIM8->CCR1 = (2 * level) + (level * 55)/100;
}
#elif defined(ENABLE_BKLIGHT_DIMMING) /* MD-UV3x0 AND dimming enabled */
/*
* Interrupt-based software PWM for backlight dimming on MD-UV3x0.
* On this family of devices the GPIO for backlight control is not connected to
* any of the available timer compare output channels, thus making impossible to
* implement an hardware-based PWM.
* However, we provide a software-base backlight dimming for experimental purposes.
*/
/* Name of interrupt handler is mangled for C++ compatibility */
void _Z29TIM1_TRG_COM_TIM11_IRQHandlerv()
{
if(TIM11->SR & TIM_SR_CC1IF)
{
gpio_clearPin(LCD_BKLIGHT); /* Clear pin on compare match */
}
if(TIM11->SR & TIM_SR_UIF)
{
gpio_setPin(LCD_BKLIGHT); /* Set pin on counter reload */
}
TIM11->SR = 0;
}
void backlight_init()
{
gpio_setMode(LCD_BKLIGHT, OUTPUT); gpio_setMode(LCD_BKLIGHT, OUTPUT);
gpio_clearPin(LCD_BKLIGHT); gpio_clearPin(LCD_BKLIGHT);
#ifdef ENABLE_BKLIGHT_DIMMING
/* /*
* Configure TIM11 for backlight PWM: Fpwm = 256Hz, 8 bit of resolution. * Configure TIM11 for backlight PWM: Fpwm = 256Hz, 8 bit of resolution.
* APB2 freq. is 84MHz but timer runs at twice this frequency, then: * APB2 freq. is 84MHz but timer runs at twice this frequency, then:
@ -134,14 +111,18 @@ void backlight_init()
NVIC_ClearPendingIRQ(TIM1_TRG_COM_TIM11_IRQn); NVIC_ClearPendingIRQ(TIM1_TRG_COM_TIM11_IRQn);
NVIC_SetPriority(TIM1_TRG_COM_TIM11_IRQn,15); NVIC_SetPriority(TIM1_TRG_COM_TIM11_IRQn,15);
NVIC_EnableIRQ(TIM1_TRG_COM_TIM11_IRQn); NVIC_EnableIRQ(TIM1_TRG_COM_TIM11_IRQn);
#endif
#endif
} }
void backlight_terminate() void backlight_terminate()
{ {
/* Shut down backlight */ /* Shut down backlight */
gpio_setMode(LCD_BKLIGHT, OUTPUT);
gpio_clearPin(LCD_BKLIGHT); gpio_clearPin(LCD_BKLIGHT);
/* Shut down timer */ /* Shut down timer */
RCC->APB2ENR &= ~RCC_APB2ENR_TIM8EN;
RCC->APB2ENR &= ~RCC_APB2ENR_TIM11EN; RCC->APB2ENR &= ~RCC_APB2ENR_TIM11EN;
__DSB(); __DSB();
} }
@ -151,23 +132,31 @@ void backlight_terminate()
*/ */
void platform_setBacklightLevel(uint8_t level) void platform_setBacklightLevel(uint8_t level)
{ {
if(level > 100)
level = 100;
uint8_t pwmLevel = (2 * level) + (level * 55)/100; // Convert value to 0 - 255
#ifndef PLATFORM_MDUV3x0
TIM8->CCR1 = pwmLevel;
#else
/* /*
* Little workaround for the following nasty behaviour: if CCR1 value is * If CCR1 value is zero, a waveform with 99% duty cycle is generated: to
* zero, a waveform with 99% duty cycle is generated. This is because we are * avoid this the PWM is cut off when backlight level is 1.
* emulating pwm with interrupts.
*/ */
if(level > 1) if(level > 1)
{ {
if(level > 100) level = 100; #ifdef ENABLE_BKLIGHT_DIMMING
TIM11->CCR1 = pwmLevel;
TIM11->CCR1 = (2 * level) + (level * 55)/100;
TIM11->CR1 |= TIM_CR1_CEN; TIM11->CR1 |= TIM_CR1_CEN;
#else
gpio_setPin(LCD_BKLIGHT);
#endif
} }
else else
{ {
TIM11->CR1 &= ~TIM_CR1_CEN; TIM11->CR1 &= ~TIM_CR1_CEN;
gpio_clearPin(LCD_BKLIGHT); gpio_clearPin(LCD_BKLIGHT);
} }
#endif
} }
#endif

View File

@ -66,23 +66,13 @@ void platform_init()
rtc_init(); /* Initialise RTC */ rtc_init(); /* Initialise RTC */
chSelector_init(); /* Initialise channel selector handler */ chSelector_init(); /* Initialise channel selector handler */
audio_init(); /* Initialise audio management module */ audio_init(); /* Initialise audio management module */
#ifdef ENABLE_BKLIGHT_DIMMING
backlight_init(); /* Initialise backlight driver */ backlight_init(); /* Initialise backlight driver */
#else
gpio_setMode(LCD_BKLIGHT, OUTPUT);
gpio_clearPin(LCD_BKLIGHT);
#endif
} }
void platform_terminate() void platform_terminate()
{ {
/* Shut down backlight */ /* Shut down backlight */
#ifdef ENABLE_BKLIGHT_DIMMING
backlight_terminate(); backlight_terminate();
#else
gpio_clearPin(LCD_BKLIGHT);
#endif
/* Shut down LEDs */ /* Shut down LEDs */
gpio_clearPin(GREEN_LED); gpio_clearPin(GREEN_LED);
@ -212,20 +202,7 @@ const hwInfo_t *platform_getHwInfo()
// int8_t platform_getChSelector() // int8_t platform_getChSelector()
/* /*
* NOTE: when backligth dimming is enabled, the implementation of this API * NOTE: implementation of this API function is provided in
* function is provided in platform/drivers/backlight/backlight_MDx.c to avoid * platform/drivers/backlight/backlight_MDx.c
* an useless function call.
*/ */
#ifndef ENABLE_BKLIGHT_DIMMING // void platform_setBacklightLevel(uint8_t level)
void platform_setBacklightLevel(uint8_t level)
{
if(level > 1)
{
gpio_setPin(LCD_BKLIGHT);
}
else
{
gpio_clearPin(LCD_BKLIGHT);
}
}
#endif