1 /*!
2 * \file main.c
3 *
4 * \brief Performs a periodic uplink
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 periodic-uplink/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 "CayenneLpp.h"
38 #include "LmHandlerMsgDisplay.h"
39
40 #ifndef ACTIVE_REGION
41
42 #warning "No active region defined, LORAMAC_REGION_EU868 will be used as default."
43
44 #define ACTIVE_REGION LORAMAC_REGION_EU868
45
46 #endif
47
48 /*!
49 * LoRaWAN default end-device class
50 */
51 #ifndef LORAWAN_DEFAULT_CLASS
52 #define LORAWAN_DEFAULT_CLASS CLASS_A
53 #endif
54
55 /*!
56 * Defines the application data transmission duty cycle. 5s, value in [ms].
57 */
58 #define APP_TX_DUTYCYCLE 5000
59
60 /*!
61 * Defines a random delay for application data transmission duty cycle. 1s,
62 * value in [ms].
63 */
64 #define APP_TX_DUTYCYCLE_RND 1000
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_0
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 * LoRaWAN application port
99 * @remark The allowed port range is from 1 up to 223. Other values are reserved.
100 */
101 #define LORAWAN_APP_PORT 2
102
103 /*!
104 *
105 */
106 typedef enum
107 {
108 LORAMAC_HANDLER_TX_ON_TIMER,
109 LORAMAC_HANDLER_TX_ON_EVENT,
110 }LmHandlerTxEvents_t;
111
112 /*!
113 * User application data
114 */
115 static uint8_t AppDataBuffer[LORAWAN_APP_DATA_BUFFER_MAX_SIZE];
116
117 /*!
118 * User application data structure
119 */
120 static LmHandlerAppData_t AppData =
121 {
122 .Buffer = AppDataBuffer,
123 .BufferSize = 0,
124 .Port = 0,
125 };
126
127 /*!
128 * Specifies the state of the application LED
129 */
130 static bool AppLedStateOn = false;
131
132 /*!
133 * Timer to handle the application data transmission duty cycle
134 */
135 static TimerEvent_t TxTimer;
136
137 /*!
138 * Timer to handle the state of LED1
139 */
140 static TimerEvent_t Led1Timer;
141
142 /*!
143 * Timer to handle the state of LED2
144 */
145 static TimerEvent_t Led2Timer;
146
147 /*!
148 * Timer to handle the state of LED beacon indicator
149 */
150 static TimerEvent_t LedBeaconTimer;
151
152 static void OnMacProcessNotify( void );
153 static void OnNvmDataChange( LmHandlerNvmContextStates_t state, uint16_t size );
154 static void OnNetworkParametersChange( CommissioningParams_t* params );
155 static void OnMacMcpsRequest( LoRaMacStatus_t status, McpsReq_t *mcpsReq, TimerTime_t nextTxIn );
156 static void OnMacMlmeRequest( LoRaMacStatus_t status, MlmeReq_t *mlmeReq, TimerTime_t nextTxIn );
157 static void OnJoinRequest( LmHandlerJoinParams_t* params );
158 static void OnTxData( LmHandlerTxParams_t* params );
159 static void OnRxData( LmHandlerAppData_t* appData, LmHandlerRxParams_t* params );
160 static void OnClassChange( DeviceClass_t deviceClass );
161 static void OnBeaconStatusChange( LoRaMacHandlerBeaconParams_t* params );
162 #if( LMH_SYS_TIME_UPDATE_NEW_API == 1 )
163 static void OnSysTimeUpdate( bool isSynchronized, int32_t timeCorrection );
164 #else
165 static void OnSysTimeUpdate( void );
166 #endif
167 static void PrepareTxFrame( void );
168 static void StartTxProcess( LmHandlerTxEvents_t txEvent );
169 static void UplinkProcess( void );
170
171 static void OnTxPeriodicityChanged( uint32_t periodicity );
172 static void OnTxFrameCtrlChanged( LmHandlerMsgTypes_t isTxConfirmed );
173 static void OnPingSlotPeriodicityChanged( uint8_t pingSlotPeriodicity );
174
175 /*!
176 * Function executed on TxTimer event
177 */
178 static void OnTxTimerEvent( void* context );
179
180 /*!
181 * Function executed on Led 1 Timeout event
182 */
183 static void OnLed1TimerEvent( void* context );
184
185 /*!
186 * Function executed on Led 2 Timeout event
187 */
188 static void OnLed2TimerEvent( void* context );
189
190 /*!
191 * \brief Function executed on Beacon timer Timeout event
192 */
193 static void OnLedBeaconTimerEvent( void* context );
194
195 static LmHandlerCallbacks_t LmHandlerCallbacks =
196 {
197 .GetBatteryLevel = BoardGetBatteryLevel,
198 .GetTemperature = NULL,
199 .GetRandomSeed = BoardGetRandomSeed,
200 .OnMacProcess = OnMacProcessNotify,
201 .OnNvmDataChange = OnNvmDataChange,
202 .OnNetworkParametersChange = OnNetworkParametersChange,
203 .OnMacMcpsRequest = OnMacMcpsRequest,
204 .OnMacMlmeRequest = OnMacMlmeRequest,
205 .OnJoinRequest = OnJoinRequest,
206 .OnTxData = OnTxData,
207 .OnRxData = OnRxData,
208 .OnClassChange= OnClassChange,
209 .OnBeaconStatusChange = OnBeaconStatusChange,
210 .OnSysTimeUpdate = OnSysTimeUpdate,
211 };
212
213 static LmHandlerParams_t LmHandlerParams =
214 {
215 .Region = ACTIVE_REGION,
216 .AdrEnable = LORAWAN_ADR_STATE,
217 .IsTxConfirmed = LORAWAN_DEFAULT_CONFIRMED_MSG_STATE,
218 .TxDatarate = LORAWAN_DEFAULT_DATARATE,
219 .PublicNetworkEnable = LORAWAN_PUBLIC_NETWORK,
220 .DutyCycleEnabled = LORAWAN_DUTYCYCLE_ON,
221 .DataBufferMaxSize = LORAWAN_APP_DATA_BUFFER_MAX_SIZE,
222 .DataBuffer = AppDataBuffer,
223 .PingSlotPeriodicity = REGION_COMMON_DEFAULT_PING_SLOT_PERIODICITY,
224 };
225
226 static LmhpComplianceParams_t LmhpComplianceParams =
227 {
228 .FwVersion.Value = FIRMWARE_VERSION,
229 .OnTxPeriodicityChanged = OnTxPeriodicityChanged,
230 .OnTxFrameCtrlChanged = OnTxFrameCtrlChanged,
231 .OnPingSlotPeriodicityChanged = OnPingSlotPeriodicityChanged,
232 };
233
234 /*!
235 * Indicates if LoRaMacProcess call is pending.
236 *
237 * \warning If variable is equal to 0 then the MCU can be set in low power mode
238 */
239 static volatile uint8_t IsMacProcessPending = 0;
240
241 static volatile uint8_t IsTxFramePending = 0;
242
243 static volatile uint32_t TxPeriodicity = 0;
244
245 /*!
246 * LED GPIO pins objects
247 */
248 extern Gpio_t Led1; // Tx
249 extern Gpio_t Led2; // Rx
250
251 /*!
252 * UART object used for command line interface handling
253 */
254 extern Uart_t Uart2;
255
256 /*!
257 * Main application entry point.
258 */
main(void)259 int main( void )
260 {
261 BoardInitMcu( );
262 BoardInitPeriph( );
263
264 TimerInit( &Led1Timer, OnLed1TimerEvent );
265 TimerSetValue( &Led1Timer, 25 );
266
267 TimerInit( &Led2Timer, OnLed2TimerEvent );
268 TimerSetValue( &Led2Timer, 25 );
269
270 TimerInit( &LedBeaconTimer, OnLedBeaconTimerEvent );
271 TimerSetValue( &LedBeaconTimer, 5000 );
272
273 // Initialize transmission periodicity variable
274 TxPeriodicity = APP_TX_DUTYCYCLE + randr( -APP_TX_DUTYCYCLE_RND, APP_TX_DUTYCYCLE_RND );
275
276 const Version_t appVersion = { .Value = FIRMWARE_VERSION };
277 const Version_t gitHubVersion = { .Value = GITHUB_VERSION };
278 DisplayAppInfo( "periodic-uplink-lpp",
279 &appVersion,
280 &gitHubVersion );
281
282 if ( LmHandlerInit( &LmHandlerCallbacks, &LmHandlerParams ) != LORAMAC_HANDLER_SUCCESS )
283 {
284 printf( "LoRaMac wasn't properly initialized\n" );
285 // Fatal error, endless loop.
286 while ( 1 )
287 {
288 }
289 }
290
291 // Set system maximum tolerated rx error in milliseconds
292 LmHandlerSetSystemMaxRxError( 20 );
293
294 // The LoRa-Alliance Compliance protocol package should always be
295 // initialized and activated.
296 LmHandlerPackageRegister( PACKAGE_ID_COMPLIANCE, &LmhpComplianceParams );
297
298 LmHandlerJoin( );
299
300 StartTxProcess( LORAMAC_HANDLER_TX_ON_TIMER );
301
302 while( 1 )
303 {
304 // Process characters sent over the command line interface
305 CliProcess( &Uart2 );
306
307 // Processes the LoRaMac events
308 LmHandlerProcess( );
309
310 // Process application uplinks management
311 UplinkProcess( );
312
313 CRITICAL_SECTION_BEGIN( );
314 if( IsMacProcessPending == 1 )
315 {
316 // Clear flag and prevent MCU to go into low power modes.
317 IsMacProcessPending = 0;
318 }
319 else
320 {
321 // The MCU wakes up through events
322 BoardLowPowerHandler( );
323 }
324 CRITICAL_SECTION_END( );
325 }
326 }
327
OnMacProcessNotify(void)328 static void OnMacProcessNotify( void )
329 {
330 IsMacProcessPending = 1;
331 }
332
OnNvmDataChange(LmHandlerNvmContextStates_t state,uint16_t size)333 static void OnNvmDataChange( LmHandlerNvmContextStates_t state, uint16_t size )
334 {
335 DisplayNvmDataChange( state, size );
336 }
337
OnNetworkParametersChange(CommissioningParams_t * params)338 static void OnNetworkParametersChange( CommissioningParams_t* params )
339 {
340 DisplayNetworkParametersUpdate( params );
341 }
342
OnMacMcpsRequest(LoRaMacStatus_t status,McpsReq_t * mcpsReq,TimerTime_t nextTxIn)343 static void OnMacMcpsRequest( LoRaMacStatus_t status, McpsReq_t *mcpsReq, TimerTime_t nextTxIn )
344 {
345 DisplayMacMcpsRequestUpdate( status, mcpsReq, nextTxIn );
346 }
347
OnMacMlmeRequest(LoRaMacStatus_t status,MlmeReq_t * mlmeReq,TimerTime_t nextTxIn)348 static void OnMacMlmeRequest( LoRaMacStatus_t status, MlmeReq_t *mlmeReq, TimerTime_t nextTxIn )
349 {
350 DisplayMacMlmeRequestUpdate( status, mlmeReq, nextTxIn );
351 }
352
OnJoinRequest(LmHandlerJoinParams_t * params)353 static void OnJoinRequest( LmHandlerJoinParams_t* params )
354 {
355 DisplayJoinRequestUpdate( params );
356 if( params->Status == LORAMAC_HANDLER_ERROR )
357 {
358 LmHandlerJoin( );
359 }
360 else
361 {
362 LmHandlerRequestClass( LORAWAN_DEFAULT_CLASS );
363 }
364 }
365
OnTxData(LmHandlerTxParams_t * params)366 static void OnTxData( LmHandlerTxParams_t* params )
367 {
368 DisplayTxUpdate( params );
369 }
370
OnRxData(LmHandlerAppData_t * appData,LmHandlerRxParams_t * params)371 static void OnRxData( LmHandlerAppData_t* appData, LmHandlerRxParams_t* params )
372 {
373 DisplayRxUpdate( appData, params );
374
375 switch( appData->Port )
376 {
377 case 1: // The application LED can be controlled on port 1 or 2
378 case LORAWAN_APP_PORT:
379 {
380 AppLedStateOn = appData->Buffer[0] & 0x01;
381 }
382 break;
383 default:
384 break;
385 }
386
387 // Switch LED 2 ON for each received downlink
388 GpioWrite( &Led2, 1 );
389 TimerStart( &Led2Timer );
390 }
391
OnClassChange(DeviceClass_t deviceClass)392 static void OnClassChange( DeviceClass_t deviceClass )
393 {
394 DisplayClassUpdate( deviceClass );
395
396 // Inform the server as soon as possible that the end-device has switched to ClassB
397 LmHandlerAppData_t appData =
398 {
399 .Buffer = NULL,
400 .BufferSize = 0,
401 .Port = 0,
402 };
403 LmHandlerSend( &appData, LORAMAC_HANDLER_UNCONFIRMED_MSG );
404 }
405
OnBeaconStatusChange(LoRaMacHandlerBeaconParams_t * params)406 static void OnBeaconStatusChange( LoRaMacHandlerBeaconParams_t* params )
407 {
408 switch( params->State )
409 {
410 case LORAMAC_HANDLER_BEACON_RX:
411 {
412 TimerStart( &LedBeaconTimer );
413 break;
414 }
415 case LORAMAC_HANDLER_BEACON_LOST:
416 case LORAMAC_HANDLER_BEACON_NRX:
417 {
418 TimerStop( &LedBeaconTimer );
419 break;
420 }
421 default:
422 {
423 break;
424 }
425 }
426
427 DisplayBeaconUpdate( params );
428 }
429
430 #if( LMH_SYS_TIME_UPDATE_NEW_API == 1 )
OnSysTimeUpdate(bool isSynchronized,int32_t timeCorrection)431 static void OnSysTimeUpdate( bool isSynchronized, int32_t timeCorrection )
432 {
433
434 }
435 #else
OnSysTimeUpdate(void)436 static void OnSysTimeUpdate( void )
437 {
438
439 }
440 #endif
441
442 /*!
443 * Prepares the payload of the frame and transmits it.
444 */
PrepareTxFrame(void)445 static void PrepareTxFrame( void )
446 {
447 if( LmHandlerIsBusy( ) == true )
448 {
449 return;
450 }
451
452 uint8_t channel = 0;
453
454 AppData.Port = LORAWAN_APP_PORT;
455
456 CayenneLppReset( );
457 CayenneLppAddDigitalInput( channel++, AppLedStateOn );
458 CayenneLppAddAnalogInput( channel++, BoardGetBatteryLevel( ) * 100 / 254 );
459
460 CayenneLppCopy( AppData.Buffer );
461 AppData.BufferSize = CayenneLppGetSize( );
462
463 if( LmHandlerSend( &AppData, LmHandlerParams.IsTxConfirmed ) == LORAMAC_HANDLER_SUCCESS )
464 {
465 // Switch LED 1 ON
466 GpioWrite( &Led1, 1 );
467 TimerStart( &Led1Timer );
468 }
469 }
470
StartTxProcess(LmHandlerTxEvents_t txEvent)471 static void StartTxProcess( LmHandlerTxEvents_t txEvent )
472 {
473 switch( txEvent )
474 {
475 default:
476 // Intentional fall through
477 case LORAMAC_HANDLER_TX_ON_TIMER:
478 {
479 // Schedule 1st packet transmission
480 TimerInit( &TxTimer, OnTxTimerEvent );
481 TimerSetValue( &TxTimer, TxPeriodicity );
482 OnTxTimerEvent( NULL );
483 }
484 break;
485 case LORAMAC_HANDLER_TX_ON_EVENT:
486 {
487 }
488 break;
489 }
490 }
491
UplinkProcess(void)492 static void UplinkProcess( void )
493 {
494 uint8_t isPending = 0;
495 CRITICAL_SECTION_BEGIN( );
496 isPending = IsTxFramePending;
497 IsTxFramePending = 0;
498 CRITICAL_SECTION_END( );
499 if( isPending == 1 )
500 {
501 PrepareTxFrame( );
502 }
503 }
504
OnTxPeriodicityChanged(uint32_t periodicity)505 static void OnTxPeriodicityChanged( uint32_t periodicity )
506 {
507 TxPeriodicity = periodicity;
508
509 if( TxPeriodicity == 0 )
510 { // Revert to application default periodicity
511 TxPeriodicity = APP_TX_DUTYCYCLE + randr( -APP_TX_DUTYCYCLE_RND, APP_TX_DUTYCYCLE_RND );
512 }
513
514 // Update timer periodicity
515 TimerStop( &TxTimer );
516 TimerSetValue( &TxTimer, TxPeriodicity );
517 TimerStart( &TxTimer );
518 }
519
OnTxFrameCtrlChanged(LmHandlerMsgTypes_t isTxConfirmed)520 static void OnTxFrameCtrlChanged( LmHandlerMsgTypes_t isTxConfirmed )
521 {
522 LmHandlerParams.IsTxConfirmed = isTxConfirmed;
523 }
524
OnPingSlotPeriodicityChanged(uint8_t pingSlotPeriodicity)525 static void OnPingSlotPeriodicityChanged( uint8_t pingSlotPeriodicity )
526 {
527 LmHandlerParams.PingSlotPeriodicity = pingSlotPeriodicity;
528 }
529
530 /*!
531 * Function executed on TxTimer event
532 */
OnTxTimerEvent(void * context)533 static void OnTxTimerEvent( void* context )
534 {
535 TimerStop( &TxTimer );
536
537 IsTxFramePending = 1;
538
539 // Schedule next transmission
540 TimerSetValue( &TxTimer, TxPeriodicity );
541 TimerStart( &TxTimer );
542 }
543
544 /*!
545 * Function executed on Led 1 Timeout event
546 */
OnLed1TimerEvent(void * context)547 static void OnLed1TimerEvent( void* context )
548 {
549 TimerStop( &Led1Timer );
550 // Switch LED 1 OFF
551 GpioWrite( &Led1, 0 );
552 }
553
554 /*!
555 * Function executed on Led 2 Timeout event
556 */
OnLed2TimerEvent(void * context)557 static void OnLed2TimerEvent( void* context )
558 {
559 TimerStop( &Led2Timer );
560 // Switch LED 2 OFF
561 GpioWrite( &Led2, 0 );
562 }
563
564 /*!
565 * \brief Function executed on Beacon timer Timeout event
566 */
OnLedBeaconTimerEvent(void * context)567 static void OnLedBeaconTimerEvent( void* context )
568 {
569 GpioWrite( &Led2, 1 );
570 TimerStart( &Led2Timer );
571
572 TimerStart( &LedBeaconTimer );
573 }
574