mirror of
				https://github.com/stargieg/bacnet-stack
				synced 2025-10-26 23:35:52 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			195 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			195 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /**************************************************************************
 | |
| *
 | |
| * Copyright (C) 2008 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 <stddef.h>
 | |
| #include <stdint.h>
 | |
| #include <errno.h>
 | |
| #include <string.h>
 | |
| #include "config.h"
 | |
| #include "bacdef.h"
 | |
| #include "bacdcode.h"
 | |
| #include "address.h"
 | |
| #include "tsm.h"
 | |
| #include "dcc.h"
 | |
| #include "npdu.h"
 | |
| #include "apdu.h"
 | |
| #include "device.h"
 | |
| #include "datalink.h"
 | |
| #include "iam.h"
 | |
| /* some demo stuff needed */
 | |
| #include "handlers.h"
 | |
| #include "client.h"
 | |
| 
 | |
| /** @file s_iam.c  Send an I-Am message. */
 | |
| 
 | |
| /** Encode an I Am message to be broadcast.
 | |
|  * @param buffer [in,out] The buffer to use for building the message.
 | |
|  * @param dest [out] The destination address information.
 | |
|  * @param npdu_data [out] The NPDU structure describing the message.
 | |
|  * @return The length of the message in buffer[].
 | |
|  */
 | |
| int iam_encode_pdu(
 | |
|     uint8_t * buffer,
 | |
|     BACNET_ADDRESS * dest,
 | |
|     BACNET_NPDU_DATA * npdu_data)
 | |
| {
 | |
|     int len = 0;
 | |
|     int pdu_len = 0;
 | |
|     BACNET_ADDRESS my_address;
 | |
|     datalink_get_my_address(&my_address);
 | |
| 
 | |
|     datalink_get_broadcast_address(dest);
 | |
|     /* encode the NPDU portion of the packet */
 | |
|     npdu_encode_npdu_data(npdu_data, false, MESSAGE_PRIORITY_NORMAL);
 | |
|     pdu_len = npdu_encode_pdu(&buffer[0], dest, &my_address, npdu_data);
 | |
| 
 | |
|     /* encode the APDU portion of the packet */
 | |
|     len =
 | |
|         iam_encode_apdu(&buffer[pdu_len], Device_Object_Instance_Number(),
 | |
|         MAX_APDU, SEGMENTATION_NONE, Device_Vendor_Identifier());
 | |
|     pdu_len += len;
 | |
| 
 | |
|     return pdu_len;
 | |
| }
 | |
| 
 | |
| /** Broadcast an I Am message.
 | |
|  * @ingroup DMDDB
 | |
|  *
 | |
|  * @param buffer [in] The buffer to use for building and sending the message.
 | |
|  */
 | |
| void Send_I_Am(
 | |
|     uint8_t * buffer)
 | |
| {
 | |
|     int pdu_len = 0;
 | |
|     BACNET_ADDRESS dest;
 | |
|     int bytes_sent = 0;
 | |
|     BACNET_NPDU_DATA npdu_data;
 | |
| 
 | |
| #if 0
 | |
|     /* note: there is discussion in the BACnet committee
 | |
|        that we should allow a device to reply with I-Am
 | |
|        so that dynamic binding always work.  If the DCC
 | |
|        initiator loses the MAC address and routing info,
 | |
|        they can never re-enable DCC because they can't
 | |
|        find the device with WhoIs/I-Am */
 | |
|     /* are we are forbidden to send? */
 | |
|     if (!dcc_communication_enabled())
 | |
|         return 0;
 | |
| #endif
 | |
| 
 | |
|     /* encode the data */
 | |
|     pdu_len = iam_encode_pdu(buffer, &dest, &npdu_data);
 | |
|     /* send data */
 | |
|     bytes_sent = datalink_send_pdu(&dest, &npdu_data, &buffer[0], pdu_len);
 | |
| 
 | |
| #if PRINT_ENABLED
 | |
|     if (bytes_sent <= 0) {
 | |
|         fprintf(stderr, "Failed to Send I-Am Reply (%s)!\n", strerror(errno));
 | |
|     }
 | |
| #else
 | |
|     bytes_sent = bytes_sent;
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /** Encode an I Am message to be unicast directly back to the src.
 | |
|  *
 | |
|  * @param buffer [in,out] The buffer to use for building the message.
 | |
|  * @param src [in] The source address information. If the src address is not
 | |
|  *                 given, the dest address will be a broadcast address.
 | |
|  * @param dest [out] The destination address information.
 | |
|  * @param npdu_data [out] The NPDU structure describing the message.
 | |
|  * @return The length of the message in buffer[].
 | |
|  */
 | |
| int iam_unicast_encode_pdu(
 | |
|     uint8_t * buffer,
 | |
|     BACNET_ADDRESS * src,
 | |
|     BACNET_ADDRESS * dest,
 | |
|     BACNET_NPDU_DATA * npdu_data)
 | |
| {
 | |
|     int npdu_len = 0;
 | |
|     int apdu_len = 0;
 | |
|     int pdu_len = 0;
 | |
|     BACNET_ADDRESS my_address;
 | |
|     /* The destination will be the same as the src, so copy it over. */
 | |
|     bacnet_address_copy(dest, src);
 | |
|     /* dest->net = 0; - no, must direct back to src->net to meet BTL tests */
 | |
| 
 | |
|     datalink_get_my_address(&my_address);
 | |
|     /* encode the NPDU portion of the packet */
 | |
|     npdu_encode_npdu_data(npdu_data, false, MESSAGE_PRIORITY_NORMAL);
 | |
|     npdu_len = npdu_encode_pdu(&buffer[0], dest, &my_address, npdu_data);
 | |
|     /* encode the APDU portion of the packet */
 | |
|     apdu_len =
 | |
|         iam_encode_apdu(&buffer[npdu_len], Device_Object_Instance_Number(),
 | |
|         MAX_APDU, SEGMENTATION_NONE, Device_Vendor_Identifier());
 | |
|     pdu_len = npdu_len + apdu_len;
 | |
| 
 | |
|     return pdu_len;
 | |
| }
 | |
| 
 | |
| /** Send an I-Am message by unicasting directly back to the src.
 | |
|  * @ingroup DMDDB
 | |
|  * @note As of Addendum 135-2008q-1, unicast responses are allowed;
 | |
|  *  in modern firewalled corporate networks, this may be the
 | |
|  *  only type of response that will reach the source on
 | |
|  *  another subnet (without using the BBMD).  <br>
 | |
|  *  However, some BACnet routers may not correctly handle this message.
 | |
|  *
 | |
|  * @param buffer [in] The buffer to use for building and sending the message.
 | |
|  * @param src [in] The source address information from service handler.
 | |
|  */
 | |
| void Send_I_Am_Unicast(
 | |
|     uint8_t * buffer,
 | |
|     BACNET_ADDRESS * src)
 | |
| {
 | |
|     int pdu_len = 0;
 | |
|     BACNET_ADDRESS dest;
 | |
|     int bytes_sent = 0;
 | |
|     BACNET_NPDU_DATA npdu_data;
 | |
| 
 | |
| #if 0
 | |
|     /* note: there is discussion in the BACnet committee
 | |
|        that we should allow a device to reply with I-Am
 | |
|        so that dynamic binding always work.  If the DCC
 | |
|        initiator loses the MAC address and routing info,
 | |
|        they can never re-enable DCC because they can't
 | |
|        find the device with WhoIs/I-Am */
 | |
|     /* are we are forbidden to send? */
 | |
|     if (!dcc_communication_enabled())
 | |
|         return 0;
 | |
| #endif
 | |
| 
 | |
|     /* encode the data */
 | |
|     pdu_len = iam_unicast_encode_pdu(buffer, src, &dest, &npdu_data);
 | |
|     /* send data */
 | |
|     bytes_sent = datalink_send_pdu(&dest, &npdu_data, &buffer[0], pdu_len);
 | |
| 
 | |
| #if PRINT_ENABLED
 | |
|     if (bytes_sent <= 0)
 | |
|         fprintf(stderr, "Failed to Send I-Am Reply (%s)!\n", strerror(errno));
 | |
| #else
 | |
|     bytes_sent = bytes_sent;
 | |
| #endif
 | |
| }
 | 
