Функция обратного вызова таймера Micrium OS -III не вызывается
Я использую компилятор Keil для руки (STM32F4). В своем коде я создал таймер для вызова функции, которая вызывает функцию из файла *.lib.
Функция вызывается, если я использую оптимизацию компоновщика уровня 0 (O-0) и приложение работает нормально. Если я изменяю уровень оптимизации на (O-1), функция не включается в исполняемый файл.
Я проверил структуру таймера в окне просмотра, обновляются значения счетчика и присутствует адрес функции обратного вызова, который совпадает с адресом в файле карты.
Из помощи Кейла я узнал, что при оптимизации уровня 1-o1 выполняются следующие оптимизации.
- Точки останова не могут быть установлены на мертвом коде.
- Значения переменных могут быть недоступны в их области действия после
они были инициализированы. Например, если их назначенное местоположение
был повторно использован. - Функции без побочных эффектов могут быть вызваны не по порядку или могут быть опущены, если результат не требуется.
Я пытался вызвать функцию lib из задачи. Но все же происходит то же самое. Поскольку на функцию ссылались напрямую, я подозреваю, что компилятор считает, что эта функция не имеет побочных эффектов.
Как я могу решить это?
void LCDVsync_TMRHandler(void *ptmr, void *parg)
{
OS_ERR err;
OSWrappers::signalVSync();
}
void LCDFrontPorch_TMRHandler(void *ptmr,void *arg)
{
OS_ERR err;
HAL::getInstance()->frontPorchEntered();
OSTmrStart(&ostmr_LCDVsync,&err);
}
Код, который создает таймер.
OSTmrCreate(&ostmr_LCDVsync,"LCD vsync signalling",10,0,OS_OPT_TMR_ONE_SHOT,LCDVsync_TMRHandler,(void *)0,&err);
if(OS_ERR_NONE != err)
{
DEBUG_ERROR_APP("ERROR: APP.C : AppTmrCreate : LCD vsync signalling tmr create failure");
}
OSTmrCreate(&ostmr_LCDFrontPorch,"LCD Front porch",0,20,OS_OPT_TMR_PERIODIC,LCDFrontPorch_TMRHandler,(void *)0,&err);
if(OS_ERR_NONE != err)
{
DEBUG_ERROR_APP("ERROR: APP.C : AppTmrCreate : LCD vsync signalling tmr create failure");
}
Код, запускающий периодический таймер.
OSTmrStart(&ostmr_LCDFrontPorch,&err);
Строка управления компилятором.
c --cpu Cortex-M4.fp -g -O0 --apcs=interwork -I..\platform\3rd_Party_ST\Drivers\CMSIS\Include
-I C:\Users\bro\Desktop\Project\Charger\Workspace\some\yeah\RTE
-I C:\Keil_v5\ARM\PACK\Keil\STM32F4xx_DFP\2.5.0
-I C:\Keil_v5\ARM\CMSIS\Include
-I C:\Keil_v5\ARM\PACK\Keil\STM32F4xx_DFP\2.5.0\Drivers\CMSIS\Device\ST\STM32F4xx\Include
-D__UVISION_VERSION="513" -DSTM32F429xx -DUSE_HAL_DRIVER -DSTM32F429xx -DUSE_STM324x9I_EVAL -DUSE_I2C1 -DUSE_SPIX -DUSE_USB_FS -DUSECB_SDOREQ -o ".\Objects\*.o" --omf_browse ".\Objects\*.crf" --depend ".\Objects\*.d"
1 ответ
Поскольку он работает без оптимизации, я бы посоветовал обмануть компилятор при включенной оптимизации. В частности, вы можете сделать что-то вроде этого:
if ( funcReturnsFalse() )
{
CallBackFcn( param1, param2 ); // Whatever your real callback is
}
Вы не хотите фактически делать вызов, потому что это меняет логику, с которой вы работаете, пока оптимизация выключена. И компилятор, вероятно, достаточно умен, чтобы понять что-то вроде if ( false ) ...
поэтому для вызова компилятора, вероятно, достаточно иметь вызов функции ( funcReturnsFalse()) CallBackFcn()
,
Я сталкивался с этой проблемой раньше (не с Micrium), и эта техника работала для меня.