1 /*!
2 * \file main.c
3 *
4 * \brief FUOTA interop tests - test 01
5 *
6 * \copyright Revised BSD License, see section \ref LICENSE.
7 *
8 * \code
9 * ______ _
10 * / _____) _ | |
11 * ( (____ _____ ____ _| |_ _____ ____| |__
12 * \____ \| ___ | (_ _) ___ |/ ___) _ \
13 * _____) ) ____| | | || |_| ____( (___| | | |
14 * (______/|_____)_|_|_| \__)_____)\____)_| |_|
15 * (C)2013-2018 Semtech
16 *
17 * \endcode
18 *
19 * \author Miguel Luis ( Semtech )
20 */
21
22 /*! \file fuota-test-01/NucleoL476/main.c */
23
24 #include <stdio.h>
25 #include "../firmwareVersion.h"
26 #include "../../common/githubVersion.h"
27 #include "utilities.h"
28 #include "board.h"
29 #include "gpio.h"
30 #include "uart.h"
31 #include "RegionCommon.h"
32
33 #include "cli.h"
34 #include "Commissioning.h"
35 #include "LmHandler.h"
36 #include "LmhpCompliance.h"
37 #include "LmhpClockSync.h"
38 #include "LmhpRemoteMcastSetup.h"
39 #include "LmhpFragmentation.h"
40 #include "LmHandlerMsgDisplay.h"
41
42 #ifndef ACTIVE_REGION
43
44 #warning "No active region defined, LORAMAC_REGION_EU868 will be used as default."
45
46 #define ACTIVE_REGION LORAMAC_REGION_EU868
47
48 #endif
49
50 /*!
51 * LoRaWAN default end-device class
52 */
53 #define LORAWAN_DEFAULT_CLASS CLASS_A
54
55 /*!
56 * Defines the application data transmission duty cycle. 40s, value in [ms].
57 */
58 #define APP_TX_DUTYCYCLE 40000
59
60 /*!
61 * Defines a random delay for application data transmission duty cycle. 5s,
62 * value in [ms].
63 */
64 #define APP_TX_DUTYCYCLE_RND 5000
65
66 /*!
67 * LoRaWAN Adaptive Data Rate
68 *
69 * \remark Please note that when ADR is enabled the end-device should be static
70 */
71 #define LORAWAN_ADR_STATE LORAMAC_HANDLER_ADR_ON
72
73 /*!
74 * Default datarate
75 *
76 * \remark Please note that LORAWAN_DEFAULT_DATARATE is used only when ADR is disabled
77 */
78 #define LORAWAN_DEFAULT_DATARATE DR_3
79
80 /*!
81 * LoRaWAN confirmed messages
82 */
83 #define LORAWAN_DEFAULT_CONFIRMED_MSG_STATE LORAMAC_HANDLER_UNCONFIRMED_MSG
84
85 /*!
86 * User application data buffer size
87 */
88 #define LORAWAN_APP_DATA_BUFFER_MAX_SIZE 242
89
90 /*!
91 * LoRaWAN ETSI duty cycle control enable/disable
92 *
93 * \remark Please note that ETSI mandates duty cycled transmissions. Use only for test purposes
94 */
95 #define LORAWAN_DUTYCYCLE_ON true
96
97 /*!
98 *
99 */
100 typedef enum
101 {
102 LORAMAC_HANDLER_TX_ON_TIMER,
103 LORAMAC_HANDLER_TX_ON_EVENT,
104 }LmHandlerTxEvents_t;
105
106 /*!
107 * User application data
108 */
109 static uint8_t AppDataBuffer[LORAWAN_APP_DATA_BUFFER_MAX_SIZE];
110
111 /*!
112 * Timer to handle the application data transmission duty cycle
113 */
114 static TimerEvent_t TxTimer;
115
116 /*!
117 * Timer to handle the state of LED1
118 */
119 static TimerEvent_t Led1Timer;
120
121 /*!
122 * Timer to handle the state of LED2
123 */
124 static TimerEvent_t Led2Timer;
125
126 /*!
127 * Timer to handle the state of LED beacon indicator
128 */
129 static TimerEvent_t LedBeaconTimer;
130
131 static void OnMacProcessNotify( void );
132 static void OnNvmDataChange( LmHandlerNvmContextStates_t state, uint16_t size );
133 static void OnNetworkParametersChange( CommissioningParams_t* params );
134 static void OnMacMcpsRequest( LoRaMacStatus_t status, McpsReq_t *mcpsReq, TimerTime_t nextTxIn );
135 static void OnMacMlmeRequest( LoRaMacStatus_t status, MlmeReq_t *mlmeReq, TimerTime_t nextTxIn );
136 static void OnJoinRequest( LmHandlerJoinParams_t* params );
137 static void OnTxData( LmHandlerTxParams_t* params );
138 static void OnRxData( LmHandlerAppData_t* appData, LmHandlerRxParams_t* params );
139 static void OnClassChange( DeviceClass_t deviceClass );
140 static void OnBeaconStatusChange( LoRaMacHandlerBeaconParams_t* params );
141 #if( LMH_SYS_TIME_UPDATE_NEW_API == 1 )
142 static void OnSysTimeUpdate( bool isSynchronized, int32_t timeCorrection );
143 #else
144 static void OnSysTimeUpdate( void );
145 #endif
146 #if( FRAG_DECODER_FILE_HANDLING_NEW_API == 1 )
147 static int8_t FragDecoderWrite( uint32_t addr, uint8_t *data, uint32_t size );
148 static int8_t FragDecoderRead( uint32_t addr, uint8_t *data, uint32_t size );
149 #endif
150 static void OnFragProgress( uint16_t fragCounter, uint16_t fragNb, uint8_t fragSize, uint16_t fragNbLost );
151 #if( FRAG_DECODER_FILE_HANDLING_NEW_API == 1 )
152 static void OnFragDone( int32_t status, uint32_t size );
153 #else
154 static void OnFragDone( int32_t status, uint8_t *file, uint32_t size );
155 #endif
156 static void StartTxProcess( LmHandlerTxEvents_t txEvent );
157 static void UplinkProcess( void );
158
159 static void OnTxPeriodicityChanged( uint32_t periodicity );
160 static void OnTxFrameCtrlChanged( LmHandlerMsgTypes_t isTxConfirmed );
161 static void OnPingSlotPeriodicityChanged( uint8_t pingSlotPeriodicity );
162
163 /*!
164 * Function executed on TxTimer event
165 */
166 static void OnTxTimerEvent( void* context );
167
168 /*!
169 * Function executed on Led 1 Timeout event
170 */
171 static void OnLed1TimerEvent( void* context );
172
173 /*!
174 * Function executed on Led 2 Timeout event
175 */
176 static void OnLed2TimerEvent( void* context );
177
178 /*!
179 * \brief Function executed on Beacon timer Timeout event
180 */
181 static void OnLedBeaconTimerEvent( void* context );
182
183 static LmHandlerCallbacks_t LmHandlerCallbacks =
184 {
185 .GetBatteryLevel = BoardGetBatteryLevel,
186 .GetTemperature = NULL,
187 .GetRandomSeed = BoardGetRandomSeed,
188 .OnMacProcess = OnMacProcessNotify,
189 .OnNvmDataChange = OnNvmDataChange,
190 .OnNetworkParametersChange = OnNetworkParametersChange,
191 .OnMacMcpsRequest = OnMacMcpsRequest,
192 .OnMacMlmeRequest = OnMacMlmeRequest,
193 .OnJoinRequest = OnJoinRequest,
194 .OnTxData = OnTxData,
195 .OnRxData = OnRxData,
196 .OnClassChange= OnClassChange,
197 .OnBeaconStatusChange = OnBeaconStatusChange,
198 .OnSysTimeUpdate = OnSysTimeUpdate,
199 };
200
201 static LmHandlerParams_t LmHandlerParams =
202 {
203 .Region = ACTIVE_REGION,
204 .AdrEnable = LORAWAN_ADR_STATE,
205 .IsTxConfirmed = LORAWAN_DEFAULT_CONFIRMED_MSG_STATE,
206 .TxDatarate = LORAWAN_DEFAULT_DATARATE,
207 .PublicNetworkEnable = LORAWAN_PUBLIC_NETWORK,
208 .DutyCycleEnabled = LORAWAN_DUTYCYCLE_ON,
209 .DataBufferMaxSize = LORAWAN_APP_DATA_BUFFER_MAX_SIZE,
210 .DataBuffer = AppDataBuffer,
211 .PingSlotPeriodicity = REGION_COMMON_DEFAULT_PING_SLOT_PERIODICITY,
212 };
213
214 static LmhpComplianceParams_t LmhpComplianceParams =
215 {
216 .FwVersion.Value = FIRMWARE_VERSION,
217 .OnTxPeriodicityChanged = OnTxPeriodicityChanged,
218 .OnTxFrameCtrlChanged = OnTxFrameCtrlChanged,
219 .OnPingSlotPeriodicityChanged = OnPingSlotPeriodicityChanged,
220 };
221
222 /*!
223 * Defines the maximum size for the buffer receiving the fragmentation result.
224 *
225 * \remark By default FragDecoder.h defines:
226 * \ref FRAG_MAX_NB 21
227 * \ref FRAG_MAX_SIZE 50
228 *
229 * FileSize = FRAG_MAX_NB * FRAG_MAX_SIZE
230 *
231 * If bigger file size is to be received or is fragmented differently
232 * one must update those parameters.
233 */
234 #define UNFRAGMENTED_DATA_SIZE ( 21 * 50 )
235
236 /*
237 * Un-fragmented data storage.
238 */
239 static uint8_t UnfragmentedData[UNFRAGMENTED_DATA_SIZE];
240
241 static LmhpFragmentationParams_t FragmentationParams =
242 {
243 #if( FRAG_DECODER_FILE_HANDLING_NEW_API == 1 )
244 .DecoderCallbacks =
245 {
246 .FragDecoderWrite = FragDecoderWrite,
247 .FragDecoderRead = FragDecoderRead,
248 },
249 #else
250 .Buffer = UnfragmentedData,
251 .BufferSize = UNFRAGMENTED_DATA_SIZE,
252 #endif
253 .OnProgress = OnFragProgress,
254 .OnDone = OnFragDone
255 };
256
257 /*!
258 * Indicates if LoRaMacProcess call is pending.
259 *
260 * \warning If variable is equal to 0 then the MCU can be set in low power mode
261 */
262 static volatile uint8_t IsMacProcessPending = 0;
263
264 static volatile uint8_t IsTxFramePending = 0;
265
266 static volatile uint32_t TxPeriodicity = 0;
267
268 /*
269 * Indicates if the system time has been synchronized
270 */
271 static volatile bool IsClockSynched = false;
272
273 /*
274 * MC Session Started
275 */
276 static volatile bool IsMcSessionStarted = false;
277
278 /*
279 * Indicates if the file transfer is done
280 */
281 static volatile bool IsFileTransferDone = false;
282
283 /*
284 * Received file computed CRC32
285 */
286 static volatile uint32_t FileRxCrc = 0;
287
288 /*!
289 * LED GPIO pins objects
290 */
291 extern Gpio_t Led1; // Tx
292 extern Gpio_t Led2; // Rx
293
294 /*!
295 * UART object used for command line interface handling
296 */
297 extern Uart_t Uart2;
298
299 /*!
300 * Main application entry point.
301 */
main(void)302 int main( void )
303 {
304 BoardInitMcu( );
305 BoardInitPeriph( );
306
307 TimerInit( &Led1Timer, OnLed1TimerEvent );
308 TimerSetValue( &Led1Timer, 25 );
309
310 TimerInit( &Led2Timer, OnLed2TimerEvent );
311 TimerSetValue( &Led2Timer, 100 );
312
313 TimerInit( &LedBeaconTimer, OnLedBeaconTimerEvent );
314 TimerSetValue( &LedBeaconTimer, 5000 );
315
316 // Initialize transmission periodicity variable
317 TxPeriodicity = APP_TX_DUTYCYCLE + randr( -APP_TX_DUTYCYCLE_RND, APP_TX_DUTYCYCLE_RND );
318
319 const Version_t appVersion = { .Value = FIRMWARE_VERSION };
320 const Version_t gitHubVersion = { .Value = GITHUB_VERSION };
321 DisplayAppInfo( "fuota-test-01",
322 &appVersion,
323 &gitHubVersion );
324
325 if ( LmHandlerInit( &LmHandlerCallbacks, &LmHandlerParams ) != LORAMAC_HANDLER_SUCCESS )
326 {
327 printf( "LoRaMac wasn't properly initialized\n" );
328 // Fatal error, endless loop.
329 while ( 1 )
330 {
331 }
332 }
333
334 // Set system maximum tolerated rx error in milliseconds
335 LmHandlerSetSystemMaxRxError( 20 );
336
337 // The LoRa-Alliance Compliance protocol package should always be
338 // initialized and activated.
339 LmHandlerPackageRegister( PACKAGE_ID_COMPLIANCE, &LmhpComplianceParams );
340 LmHandlerPackageRegister( PACKAGE_ID_CLOCK_SYNC, NULL );
341 LmHandlerPackageRegister( PACKAGE_ID_REMOTE_MCAST_SETUP, NULL );
342 LmHandlerPackageRegister( PACKAGE_ID_FRAGMENTATION, &FragmentationParams );
343
344 IsClockSynched = false;
345 IsFileTransferDone = false;
346
347 LmHandlerJoin( );
348
349 StartTxProcess( LORAMAC_HANDLER_TX_ON_TIMER );
350
351 while( 1 )
352 {
353 // Process characters sent over the command line interface
354 CliProcess( &Uart2 );
355
356 // Processes the LoRaMac events
357 LmHandlerProcess( );
358
359 // Process application uplinks management
360 UplinkProcess( );
361
362 CRITICAL_SECTION_BEGIN( );
363 if( IsMacProcessPending == 1 )
364 {
365 // Clear flag and prevent MCU to go into low power modes.
366 IsMacProcessPending = 0;
367 }
368 else
369 {
370 // The MCU wakes up through events
371 BoardLowPowerHandler( );
372 }
373 CRITICAL_SECTION_END( );
374 }
375 }
376
OnMacProcessNotify(void)377 static void OnMacProcessNotify( void )
378 {
379 IsMacProcessPending = 1;
380 }
381
OnNvmDataChange(LmHandlerNvmContextStates_t state,uint16_t size)382 static void OnNvmDataChange( LmHandlerNvmContextStates_t state, uint16_t size )
383 {
384 DisplayNvmDataChange( state, size );
385 }
386
OnNetworkParametersChange(CommissioningParams_t * params)387 static void OnNetworkParametersChange( CommissioningParams_t* params )
388 {
389 DisplayNetworkParametersUpdate( params );
390 }
391
OnMacMcpsRequest(LoRaMacStatus_t status,McpsReq_t * mcpsReq,TimerTime_t nextTxIn)392 static void OnMacMcpsRequest( LoRaMacStatus_t status, McpsReq_t *mcpsReq, TimerTime_t nextTxIn )
393 {
394 DisplayMacMcpsRequestUpdate( status, mcpsReq, nextTxIn );
395 }
396
OnMacMlmeRequest(LoRaMacStatus_t status,MlmeReq_t * mlmeReq,TimerTime_t nextTxIn)397 static void OnMacMlmeRequest( LoRaMacStatus_t status, MlmeReq_t *mlmeReq, TimerTime_t nextTxIn )
398 {
399 DisplayMacMlmeRequestUpdate( status, mlmeReq, nextTxIn );
400 }
401
OnJoinRequest(LmHandlerJoinParams_t * params)402 static void OnJoinRequest( LmHandlerJoinParams_t* params )
403 {
404 DisplayJoinRequestUpdate( params );
405 if( params->Status == LORAMAC_HANDLER_ERROR )
406 {
407 LmHandlerJoin( );
408 }
409 else
410 {
411 LmHandlerRequestClass( LORAWAN_DEFAULT_CLASS );
412 }
413 }
414
OnTxData(LmHandlerTxParams_t * params)415 static void OnTxData( LmHandlerTxParams_t* params )
416 {
417 DisplayTxUpdate( params );
418 }
419
OnRxData(LmHandlerAppData_t * appData,LmHandlerRxParams_t * params)420 static void OnRxData( LmHandlerAppData_t* appData, LmHandlerRxParams_t* params )
421 {
422 DisplayRxUpdate( appData, params );
423 }
424
OnClassChange(DeviceClass_t deviceClass)425 static void OnClassChange( DeviceClass_t deviceClass )
426 {
427 DisplayClassUpdate( deviceClass );
428
429 switch( deviceClass )
430 {
431 default:
432 case CLASS_A:
433 {
434 IsMcSessionStarted = false;
435 break;
436 }
437 case CLASS_B:
438 {
439 // Inform the server as soon as possible that the end-device has switched to ClassB
440 LmHandlerAppData_t appData =
441 {
442 .Buffer = NULL,
443 .BufferSize = 0,
444 .Port = 0,
445 };
446 LmHandlerSend( &appData, LORAMAC_HANDLER_UNCONFIRMED_MSG );
447 IsMcSessionStarted = true;
448 break;
449 }
450 case CLASS_C:
451 {
452 IsMcSessionStarted = true;
453 // Switch LED 2 ON
454 GpioWrite( &Led2, 1 );
455 break;
456 }
457 }
458 }
459
OnBeaconStatusChange(LoRaMacHandlerBeaconParams_t * params)460 static void OnBeaconStatusChange( LoRaMacHandlerBeaconParams_t* params )
461 {
462 switch( params->State )
463 {
464 case LORAMAC_HANDLER_BEACON_RX:
465 {
466 TimerStart( &LedBeaconTimer );
467 break;
468 }
469 case LORAMAC_HANDLER_BEACON_LOST:
470 case LORAMAC_HANDLER_BEACON_NRX:
471 {
472 TimerStop( &LedBeaconTimer );
473 break;
474 }
475 default:
476 {
477 break;
478 }
479 }
480
481 DisplayBeaconUpdate( params );
482 }
483
484 #if( LMH_SYS_TIME_UPDATE_NEW_API == 1 )
OnSysTimeUpdate(bool isSynchronized,int32_t timeCorrection)485 static void OnSysTimeUpdate( bool isSynchronized, int32_t timeCorrection )
486 {
487 IsClockSynched = isSynchronized;
488 }
489 #else
OnSysTimeUpdate(void)490 static void OnSysTimeUpdate( void )
491 {
492 IsClockSynched = true;
493 }
494 #endif
495
496 #if( FRAG_DECODER_FILE_HANDLING_NEW_API == 1 )
FragDecoderWrite(uint32_t addr,uint8_t * data,uint32_t size)497 static int8_t FragDecoderWrite( uint32_t addr, uint8_t *data, uint32_t size )
498 {
499 if( size >= UNFRAGMENTED_DATA_SIZE )
500 {
501 return -1; // Fail
502 }
503 for(uint32_t i = 0; i < size; i++ )
504 {
505 UnfragmentedData[addr + i] = data[i];
506 }
507 return 0; // Success
508 }
509
FragDecoderRead(uint32_t addr,uint8_t * data,uint32_t size)510 static int8_t FragDecoderRead( uint32_t addr, uint8_t *data, uint32_t size )
511 {
512 if( size >= UNFRAGMENTED_DATA_SIZE )
513 {
514 return -1; // Fail
515 }
516 for(uint32_t i = 0; i < size; i++ )
517 {
518 data[i] = UnfragmentedData[addr + i];
519 }
520 return 0; // Success
521 }
522 #endif
523
OnFragProgress(uint16_t fragCounter,uint16_t fragNb,uint8_t fragSize,uint16_t fragNbLost)524 static void OnFragProgress( uint16_t fragCounter, uint16_t fragNb, uint8_t fragSize, uint16_t fragNbLost )
525 {
526 // Switch LED 2 OFF for each received downlink
527 GpioWrite( &Led2, 0 );
528 TimerStart( &Led2Timer );
529
530 printf( "\n###### =========== FRAG_DECODER ============ ######\n" );
531 printf( "###### PROGRESS ######\n");
532 printf( "###### ===================================== ######\n");
533 printf( "RECEIVED : %5d / %5d Fragments\n", fragCounter, fragNb );
534 printf( " %5d / %5d Bytes\n", fragCounter * fragSize, fragNb * fragSize );
535 printf( "LOST : %7d Fragments\n\n", fragNbLost );
536 }
537
538 #if( FRAG_DECODER_FILE_HANDLING_NEW_API == 1 )
OnFragDone(int32_t status,uint32_t size)539 static void OnFragDone( int32_t status, uint32_t size )
540 {
541 FileRxCrc = Crc32( UnfragmentedData, size );
542 IsFileTransferDone = true;
543 // Switch LED 2 OFF
544 GpioWrite( &Led2, 0 );
545
546 printf( "\n###### =========== FRAG_DECODER ============ ######\n" );
547 printf( "###### FINISHED ######\n");
548 printf( "###### ===================================== ######\n");
549 printf( "STATUS : %ld\n", status );
550 printf( "CRC : %08lX\n\n", FileRxCrc );
551 }
552 #else
OnFragDone(int32_t status,uint8_t * file,uint32_t size)553 static void OnFragDone( int32_t status, uint8_t *file, uint32_t size )
554 {
555 FileRxCrc = Crc32( file, size );
556 IsFileTransferDone = true;
557 // Switch LED 2 OFF
558 GpioWrite( &Led2, 0 );
559
560 printf( "\n###### =========== FRAG_DECODER ============ ######\n" );
561 printf( "###### FINISHED ######\n");
562 printf( "###### ===================================== ######\n");
563 printf( "STATUS : %ld\n", status );
564 printf( "CRC : %08lX\n\n", FileRxCrc );
565 }
566 #endif
567
StartTxProcess(LmHandlerTxEvents_t txEvent)568 static void StartTxProcess( LmHandlerTxEvents_t txEvent )
569 {
570 switch( txEvent )
571 {
572 default:
573 // Intentional fall through
574 case LORAMAC_HANDLER_TX_ON_TIMER:
575 {
576 // Schedule 1st packet transmission
577 TimerInit( &TxTimer, OnTxTimerEvent );
578 TimerSetValue( &TxTimer, TxPeriodicity );
579 OnTxTimerEvent( NULL );
580 }
581 break;
582 case LORAMAC_HANDLER_TX_ON_EVENT:
583 {
584 }
585 break;
586 }
587 }
588
UplinkProcess(void)589 static void UplinkProcess( void )
590 {
591 LmHandlerErrorStatus_t status = LORAMAC_HANDLER_ERROR;
592
593 if( LmHandlerIsBusy( ) == true )
594 {
595 return;
596 }
597
598 uint8_t isPending = 0;
599 CRITICAL_SECTION_BEGIN( );
600 isPending = IsTxFramePending;
601 IsTxFramePending = 0;
602 CRITICAL_SECTION_END( );
603 if( isPending == 1 )
604 {
605 if( IsMcSessionStarted == false )
606 {
607 if( IsFileTransferDone == false )
608 {
609 if( IsClockSynched == false )
610 {
611 status = LmhpClockSyncAppTimeReq( );
612 }
613 else
614 {
615 AppDataBuffer[0] = randr( 0, 255 );
616 // Send random packet
617 LmHandlerAppData_t appData =
618 {
619 .Buffer = AppDataBuffer,
620 .BufferSize = 1,
621 .Port = 1,
622 };
623 status = LmHandlerSend( &appData, LmHandlerParams.IsTxConfirmed );
624 }
625 }
626 else
627 {
628 AppDataBuffer[0] = 0x05; // FragDataBlockAuthReq
629 AppDataBuffer[1] = FileRxCrc & 0x000000FF;
630 AppDataBuffer[2] = ( FileRxCrc >> 8 ) & 0x000000FF;
631 AppDataBuffer[3] = ( FileRxCrc >> 16 ) & 0x000000FF;
632 AppDataBuffer[4] = ( FileRxCrc >> 24 ) & 0x000000FF;
633
634 // Send FragAuthReq
635 LmHandlerAppData_t appData =
636 {
637 .Buffer = AppDataBuffer,
638 .BufferSize = 5,
639 .Port = 201,
640 };
641 status = LmHandlerSend( &appData, LmHandlerParams.IsTxConfirmed );
642 }
643 if( status == LORAMAC_HANDLER_SUCCESS )
644 {
645 // Switch LED 1 ON
646 GpioWrite( &Led1, 1 );
647 TimerStart( &Led1Timer );
648 }
649 }
650 }
651 }
652
OnTxPeriodicityChanged(uint32_t periodicity)653 static void OnTxPeriodicityChanged( uint32_t periodicity )
654 {
655 TxPeriodicity = periodicity;
656
657 if( TxPeriodicity == 0 )
658 { // Revert to application default periodicity
659 TxPeriodicity = APP_TX_DUTYCYCLE + randr( -APP_TX_DUTYCYCLE_RND, APP_TX_DUTYCYCLE_RND );
660 }
661
662 // Update timer periodicity
663 TimerStop( &TxTimer );
664 TimerSetValue( &TxTimer, TxPeriodicity );
665 TimerStart( &TxTimer );
666 }
667
OnTxFrameCtrlChanged(LmHandlerMsgTypes_t isTxConfirmed)668 static void OnTxFrameCtrlChanged( LmHandlerMsgTypes_t isTxConfirmed )
669 {
670 LmHandlerParams.IsTxConfirmed = isTxConfirmed;
671 }
672
OnPingSlotPeriodicityChanged(uint8_t pingSlotPeriodicity)673 static void OnPingSlotPeriodicityChanged( uint8_t pingSlotPeriodicity )
674 {
675 LmHandlerParams.PingSlotPeriodicity = pingSlotPeriodicity;
676 }
677
678 /*!
679 * Function executed on TxTimer event
680 */
OnTxTimerEvent(void * context)681 static void OnTxTimerEvent( void* context )
682 {
683 TimerStop( &TxTimer );
684
685 IsTxFramePending = 1;
686
687 // Schedule next transmission
688 TimerSetValue( &TxTimer, TxPeriodicity );
689 TimerStart( &TxTimer );
690 }
691
692 /*!
693 * Function executed on Led 1 Timeout event
694 */
OnLed1TimerEvent(void * context)695 static void OnLed1TimerEvent( void* context )
696 {
697 TimerStop( &Led1Timer );
698 // Switch LED 1 OFF
699 GpioWrite( &Led1, 0 );
700 }
701
702 /*!
703 * Function executed on Led 2 Timeout event
704 */
OnLed2TimerEvent(void * context)705 static void OnLed2TimerEvent( void* context )
706 {
707 TimerStop( &Led2Timer );
708 // Switch LED 2 ON
709 GpioWrite( &Led2, 1 );
710 }
711
712 /*!
713 * \brief Function executed on Beacon timer Timeout event
714 */
OnLedBeaconTimerEvent(void * context)715 static void OnLedBeaconTimerEvent( void* context )
716 {
717 GpioWrite( &Led2, 1 );
718 TimerStart( &Led2Timer );
719
720 TimerStart( &LedBeaconTimer );
721 }
722