mirror of
https://github.com/stargieg/bacnet-stack
synced 2025-10-26 23:35:52 +08:00
276 lines
9.3 KiB
C
276 lines
9.3 KiB
C
/*####COPYRIGHTBEGIN####
|
|
-------------------------------------------
|
|
Copyright (C) 2005 Steve Karg
|
|
|
|
This program is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public License
|
|
as published by the Free Software Foundation; either version 2
|
|
of the License, or (at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to:
|
|
The Free Software Foundation, Inc.
|
|
59 Temple Place - Suite 330
|
|
Boston, MA 02111-1307, USA.
|
|
|
|
As a special exception, if other files instantiate templates or
|
|
use macros or inline functions from this file, or you compile
|
|
this file and link it with other works to produce a work based
|
|
on this file, this file does not by itself cause the resulting
|
|
work to be covered by the GNU General Public License. However
|
|
the source code for this file must still be made available in
|
|
accordance with section (3) of the GNU General Public License.
|
|
|
|
This exception does not invalidate any other reasons why a work
|
|
based on this file might be covered by the GNU General Public
|
|
License.
|
|
-------------------------------------------
|
|
####COPYRIGHTEND####*/
|
|
|
|
#include <stdint.h> /* for standard integer types uint8_t etc. */
|
|
#include <stdbool.h> /* for the standard bool type. */
|
|
#include "bacdcode.h"
|
|
#include "bip.h"
|
|
|
|
static int interface = SOCKET_ERROR; /* SOCKET_ERROR means no open interface */
|
|
|
|
/*-----------------------------------*/
|
|
static void Error(
|
|
const char *Msg)
|
|
{
|
|
int Code = WSAGetLastError();
|
|
#ifdef HOST
|
|
printf("%s, error code: %i\n", Msg, Code);
|
|
#else
|
|
printf("%s, error code: %s\n", Msg, xn_geterror_string(Code));
|
|
#endif
|
|
exit(1);
|
|
}
|
|
|
|
#ifndef HOST
|
|
/*-----------------------------------*/
|
|
void InterfaceCleanup(
|
|
void)
|
|
{
|
|
if (interface != SOCKET_ERROR) {
|
|
xn_interface_close(interface);
|
|
interface = SOCKET_ERROR;
|
|
#if DEVICE_ID == PRISM_PCMCIA_DEVICE
|
|
RTPCShutDown();
|
|
#endif
|
|
}
|
|
}
|
|
|
|
static void NetInitialize(
|
|
void)
|
|
/* initialize the TCP/IP stack */
|
|
{
|
|
int Result;
|
|
|
|
#ifndef HOST
|
|
if (!RTKDebugVersion()) /* switch of all diagnostics and error messages of RTIP-32 */
|
|
xn_callbacks()->cb_wr_screen_string_fnc = NULL;
|
|
|
|
#ifdef RTUSB_VER
|
|
RTURegisterCallback(USBAX172); /* ax172 and ax772 drivers */
|
|
RTURegisterCallback(USBAX772);
|
|
RTURegisterCallback(USBKeyboard); /* support USB keyboards */
|
|
FindUSBControllers(); /* install USB host controllers */
|
|
Sleep(2000); /* give the USB stack time to enumerate devices */
|
|
#endif
|
|
|
|
#ifdef DHCP
|
|
XN_REGISTER_DHCP_CLI() /* and optionally the DHCP client */
|
|
#endif
|
|
Result = xn_rtip_init(); /* Initialize the RTIP stack */
|
|
if (Result != 0)
|
|
Error("xn_rtip_init failed");
|
|
|
|
atexit(InterfaceCleanup); /* make sure the driver is shut down properly */
|
|
RTCallDebugger(RT_DBG_CALLRESET, (DWORD) exit, 0); /* even if we get restarted by the debugger */
|
|
|
|
Result = BIND_DRIVER(MINOR_0); /* tell RTIP what Ethernet driver we want (see netcfg.h) */
|
|
if (Result != 0)
|
|
Error("driver initialization failed");
|
|
|
|
#if DEVICE_ID == PRISM_PCMCIA_DEVICE
|
|
/* if this is a PCMCIA device, start the PCMCIA driver */
|
|
if (RTPCInit(-1, 0, 2, NULL) == 0)
|
|
Error("No PCMCIA controller found");
|
|
#endif
|
|
|
|
/* Open the interface */
|
|
interface =
|
|
xn_interface_open_config(DEVICE_ID, MINOR_0, ED_IO_ADD, ED_IRQ,
|
|
ED_MEM_ADD);
|
|
if (interface == SOCKET_ERROR)
|
|
Error("xn_interface_open_config failed");
|
|
else {
|
|
struct _iface_info ii;
|
|
#ifdef BACDL_ETHERNET
|
|
BACNET_ADDRESS my_address;
|
|
unsigned i;
|
|
#endif
|
|
xn_interface_info(interface, &ii);
|
|
printf
|
|
("Interface opened, MAC address: %02x-%02x-%02x-%02x-%02x-%02x\n",
|
|
ii.my_ethernet_address[0], ii.my_ethernet_address[1],
|
|
ii.my_ethernet_address[2], ii.my_ethernet_address[3],
|
|
ii.my_ethernet_address[4], ii.my_ethernet_address[5]);
|
|
#ifdef BACDL_ETHERNET
|
|
for (i = 0; i < 6; i++) {
|
|
my_address.mac[i] = ii.my_ethernet_address[i];
|
|
}
|
|
ethernet_set_my_address(&my_address);
|
|
#endif
|
|
}
|
|
|
|
#if DEVICE_ID == PRISM_PCMCIA_DEVICE || DEVICE_ID == PRISM_DEVICE
|
|
xn_wlan_setup(interface, /* iface_no: value returned by xn_interface_open_config() */
|
|
"network name", /* SSID : network name set in the access point */
|
|
"station name", /* Name : name of this node */
|
|
0, /* Channel : 0 for access points, 1..14 for ad-hoc */
|
|
0, /* KeyIndex: 0 .. 3 */
|
|
"12345", /* WEP Key : key to use (5 or 13 bytes) */
|
|
0); /* Flags : see manual and Wlanapi.h for details */
|
|
Sleep(1000); /* wireless devices need a little time before they can be used */
|
|
#endif /* WLAN device */
|
|
|
|
#if defined(AUTO_IP) /* use xn_autoip() to get an IP address */
|
|
Result = xn_autoip(interface, MinIP, MaxIP, NetMask, TargetIP);
|
|
if (Result == SOCKET_ERROR)
|
|
Error("xn_autoip failed");
|
|
else {
|
|
printf("Auto-assigned IP address %i.%i.%i.%i\n", TargetIP[0],
|
|
TargetIP[1], TargetIP[2], TargetIP[3]);
|
|
/* define default gateway and DNS server */
|
|
xn_rt_add(RT_DEFAULT, ip_ffaddr, DefaultGateway, 1, interface, RT_INF);
|
|
xn_set_server_list((DWORD *) DNSServer, 1);
|
|
}
|
|
#elif defined(DHCP) /* use DHCP */
|
|
{
|
|
DHCP_param param[] = { {SUBNET_MASK, 1}
|
|
, {DNS_OP, 1}
|
|
, {ROUTER_OPTION, 1}
|
|
};
|
|
DHCP_session DS;
|
|
DHCP_conf DC;
|
|
|
|
xn_init_dhcp_conf(&DC); /* load default DHCP options */
|
|
DC.plist = param; /* add MASK, DNS, and gateway options */
|
|
DC.plist_entries = sizeof(param) / sizeof(param[0]);
|
|
printf("Contacting DHCP server, please wait...\n");
|
|
Result = xn_dhcp(interface, &DS, &DC); /* contact DHCP server */
|
|
if (Result == SOCKET_ERROR)
|
|
Error("xn_dhcp failed");
|
|
memcpy(TargetIP, DS.client_ip, 4);
|
|
printf("My IP address is: %i.%i.%i.%i\n", TargetIP[0], TargetIP[1],
|
|
TargetIP[2], TargetIP[3]);
|
|
}
|
|
#else
|
|
/* Set the IP address and interface */
|
|
printf("Using static IP address %i.%i.%i.%i\n", TargetIP[0], TargetIP[1],
|
|
TargetIP[2], TargetIP[3]);
|
|
Result = xn_set_ip(interface, TargetIP, NetMask);
|
|
/* define default gateway and DNS server */
|
|
xn_rt_add(RT_DEFAULT, ip_ffaddr, DefaultGateway, 1, interface, RT_INF);
|
|
xn_set_server_list((DWORD *) DNSServer, 1);
|
|
#endif
|
|
|
|
#else /* HOST defined, run on Windows */
|
|
|
|
WSADATA wd;
|
|
Result = WSAStartup(0x0101, &wd);
|
|
|
|
#endif
|
|
|
|
if (Result != 0)
|
|
Error("TCP/IP stack initialization failed");
|
|
}
|
|
#endif
|
|
|
|
/******************************************************************
|
|
* DESCRIPTION: Converts the byte stored address to an inet address
|
|
* RETURN: none
|
|
* ALGORITHM: none
|
|
* NOTES: none
|
|
******************************************************************/
|
|
static void RTIP_To_Network_Address(
|
|
BYTE * octet_address,
|
|
struct in_addr *addr)
|
|
{
|
|
uint32_t ip_address = 0; /* for decoding the subnet mask */
|
|
|
|
decode_unsigned32(octet_address, &ip_address);
|
|
addr->s_addr = htonl(ip_address);
|
|
|
|
return;
|
|
}
|
|
|
|
static void set_broadcast_address(
|
|
uint32_t net_address)
|
|
{
|
|
long broadcast_address = 0;
|
|
long mask = 0;
|
|
|
|
/* Note: sometimes INADDR_BROADCAST does not let me get
|
|
any unicast messages. Not sure why... */
|
|
#if USE_INADDR
|
|
(void) net_address;
|
|
bip_set_broadcast_addr(INADDR_BROADCAST);
|
|
#else
|
|
if (IN_CLASSA(ntohl(net_address)))
|
|
broadcast_address =
|
|
(ntohl(net_address) & ~IN_CLASSA_HOST) | IN_CLASSA_HOST;
|
|
else if (IN_CLASSB(ntohl(net_address)))
|
|
broadcast_address =
|
|
(ntohl(net_address) & ~IN_CLASSB_HOST) | IN_CLASSB_HOST;
|
|
else if (IN_CLASSC(ntohl(net_address)))
|
|
broadcast_address =
|
|
(ntohl(net_address) & ~IN_CLASSC_HOST) | IN_CLASSC_HOST;
|
|
else if (IN_CLASSD(ntohl(net_address)))
|
|
broadcast_address =
|
|
(ntohl(net_address) & ~IN_CLASSD_HOST) | IN_CLASSD_HOST;
|
|
else
|
|
broadcast_address = INADDR_BROADCAST;
|
|
bip_set_broadcast_addr(htonl(broadcast_address));
|
|
#endif
|
|
}
|
|
|
|
bool bip_init(
|
|
char *ifname)
|
|
{
|
|
int rv = 0; /* return from socket lib calls */
|
|
struct sockaddr_in sin = { -1 };
|
|
int value = 1;
|
|
int sock_fd = -1;
|
|
struct in_addr my_addr;
|
|
|
|
(void) ifname;
|
|
|
|
NetInitialize();
|
|
|
|
RTIP_To_Network_Address(TargetIP, &my_addr);
|
|
bip_set_addr(my_addr.s_addr);
|
|
set_broadcast_address(my_addr.s_addr);
|
|
bip_set_port(htons((0xBAC0));
|
|
/* assumes that the driver has already been initialized */
|
|
sock_fd = socket(AF_INET, SOCK_DGRAM, IPROTO_UDP);
|
|
bip_set_socket(sock_fd);
|
|
if (sock_fd < 0)
|
|
return false;
|
|
/* bind the socket to the local port number and IP address */
|
|
sin.sin_family = AF_INET; sin.sin_addr.s_addr = htonl(INADDR_ANY);
|
|
sin.sin_port = bip_get_port(); memset(&(sin.sin_zero), '\0', 8);
|
|
rv =
|
|
bind(sock_fd, (const struct sockaddr *) &sin, sizeof(struct sockaddr));
|
|
if (rv < 0) {
|
|
close(sock_fd); bip_set_socket(-1); return false;}
|
|
|
|
return true;}
|