1f591db6e7
* Added a MS/TP zero-config (automatically choose an unused MAC address) using an algorithm that starts with MAC=64 and waits for a random number of PFM (minimum of 8 plus modulo 64) before attempting to choose a MAC sequentially from 64..127. The confirmation uses a 128-bit UUID with the MSTP Test Request frame. The modifications are in src/bacnet/datalink/mstp.c and src/bacnet/datalink/dlmstp.c modules enabling any device to use zero-config if enabled. A working demonstration is in the ports/stm32f4xx for the NUCLEO board. Complete unit testing is included. Options include lurking forever (wait for a router or another master node before joining) or lurking for a minimum time (enables self forming automatic MAC addressing device nodes).
141 lines
5.0 KiB
C
141 lines
5.0 KiB
C
/************************************************************************
|
|
*
|
|
* Copyright (C) 2011 Steve Karg <skarg@users.sourceforge.net>
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining
|
|
* a copy of this software and associated documentation files (the
|
|
* "Software"), to deal in the Software without restriction, including
|
|
* without limitation the rights to use, copy, modify, merge, publish,
|
|
* distribute, sublicense, and/or sell copies of the Software, and to
|
|
* permit persons to whom the Software is furnished to do so, subject to
|
|
* the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included
|
|
* in all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
|
* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
|
* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
|
|
* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
|
* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*
|
|
*************************************************************************/
|
|
#include <stdlib.h>
|
|
#include <stdbool.h>
|
|
#include <stdint.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include "stm32f4xx.h"
|
|
#include "stm32f4xx_pwr.h"
|
|
#include "stm32f4xx_rcc.h"
|
|
#include "stm32f4xx_rng.h"
|
|
#include "system_stm32f4xx.h"
|
|
#include "bacnet/basic/object/device.h"
|
|
#include "bacnet/basic/sys/mstimer.h"
|
|
#include "bacnet/basic/sys/ringbuf.h"
|
|
#include "bacnet/datalink/datalink.h"
|
|
#include "bacnet/datalink/dlmstp.h"
|
|
#include "bacnet/datalink/mstp.h"
|
|
#include "rs485.h"
|
|
#include "led.h"
|
|
#include "bacnet.h"
|
|
|
|
/* MS/TP port */
|
|
static struct mstp_port_struct_t MSTP_Port;
|
|
static struct dlmstp_rs485_driver RS485_Driver = {
|
|
.init = rs485_init,
|
|
.send = rs485_bytes_send,
|
|
.read = rs485_byte_available,
|
|
.transmitting = rs485_rts_enabled,
|
|
.baud_rate = rs485_baud_rate,
|
|
.baud_rate_set = rs485_baud_rate_set,
|
|
.silence_milliseconds = rs485_silence_milliseconds,
|
|
.silence_reset = rs485_silence_reset
|
|
};
|
|
static struct dlmstp_user_data_t MSTP_User_Data;
|
|
static uint8_t Input_Buffer[DLMSTP_MPDU_MAX];
|
|
static uint8_t Output_Buffer[DLMSTP_MPDU_MAX];
|
|
|
|
/**
|
|
* @brief Called from _write() function from printf and friends
|
|
* @param[in] ch Character to send
|
|
*/
|
|
int __io_putchar(int ch)
|
|
{
|
|
(void)ch;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @brief Main function
|
|
* @return 0 - never returns
|
|
*/
|
|
int main(void)
|
|
{
|
|
struct mstimer Blink_Timer;
|
|
|
|
/*At this stage the microcontroller clock setting is already configured,
|
|
this is done through SystemInit() function which is called from startup
|
|
file (startup_stm32f4xx.s) before to branch to application main.
|
|
To reconfigure the default setting of SystemInit() function, refer to
|
|
system_stm32f4xx.c file */
|
|
SystemCoreClockUpdate();
|
|
/* enable some clocks - USART and GPIO clocks are enabled in our drivers */
|
|
RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR, ENABLE);
|
|
RCC_APB2PeriphClockCmd(RCC_APB2Periph_SYSCFG, ENABLE);
|
|
/* initialize hardware layer */
|
|
mstimer_init();
|
|
led_init();
|
|
rs485_init();
|
|
mstimer_set(&Blink_Timer, 500);
|
|
/* initialize MSTP datalink layer */
|
|
MSTP_Port.Nmax_info_frames = DLMSTP_MAX_INFO_FRAMES;
|
|
MSTP_Port.Nmax_master = DLMSTP_MAX_MASTER;
|
|
MSTP_Port.InputBuffer = Input_Buffer;
|
|
MSTP_Port.InputBufferSize = sizeof(Input_Buffer);
|
|
MSTP_Port.OutputBuffer = Output_Buffer;
|
|
MSTP_Port.OutputBufferSize = sizeof(Output_Buffer);
|
|
/* user data */
|
|
MSTP_Port.ZeroConfigEnabled = true;
|
|
MSTP_Port.SlaveNodeEnabled = false;
|
|
MSTP_User_Data.RS485_Driver = &RS485_Driver;
|
|
MSTP_Port.UserData = &MSTP_User_Data;
|
|
dlmstp_init((char *)&MSTP_Port);
|
|
if (MSTP_Port.ZeroConfigEnabled) {
|
|
dlmstp_set_mac_address(255);
|
|
} else {
|
|
/* FIXME: get the address from hardware DIP or from EEPROM */
|
|
dlmstp_set_mac_address(1);
|
|
}
|
|
/* FIXME: get the baud rate from hardware DIP or from EEPROM */
|
|
dlmstp_set_baud_rate(DLMSTP_BAUD_RATE_DEFAULT);
|
|
/* initialize application layer*/
|
|
bacnet_init();
|
|
/* FIXME: get the device ID from EEPROM */
|
|
Device_Set_Object_Instance_Number(103);
|
|
/* seed stdlib rand() with device-id to get pweudo consisten
|
|
zero-config poll slot, or use hardware RNG to get a more random slot */
|
|
#ifdef BACNET_ZERO_CONFIG_RNG_HARDWARE
|
|
/* enable the random number generator hardware */
|
|
RCC_AHB2PeriphClockCmd(RCC_AHB2Periph_RNG, ENABLE);
|
|
RNG_Cmd(ENABLE);
|
|
while (RNG_GetFlagStatus(RNG_FLAG_DRDY) == RESET) {
|
|
/* wait for 32-bit random number to generate */
|
|
}
|
|
srand(RNG_GetRandomNumber());
|
|
#else
|
|
srand(Device_Object_Instance_Number());
|
|
#endif
|
|
for (;;) {
|
|
if (mstimer_expired(&Blink_Timer)) {
|
|
mstimer_reset(&Blink_Timer);
|
|
led_toggle(LED_LD3);
|
|
led_toggle(LED_RS485);
|
|
}
|
|
led_task();
|
|
bacnet_task();
|
|
}
|
|
}
|