257 lines
6.6 KiB
C++
257 lines
6.6 KiB
C++
/* * $Id$
|
|
*
|
|
* L O S A L A M O S
|
|
* Los Alamos National Laboratory
|
|
* Los Alamos, New Mexico 87545
|
|
*
|
|
* Copyright, 1986, The Regents of the University of California.
|
|
*
|
|
* Author: Jeff Hill
|
|
*/
|
|
|
|
#include "locationException.h"
|
|
|
|
/* Allocate storage for global variables in this module */
|
|
#define CA_GLBLSOURCE
|
|
#include "iocinf.h"
|
|
#include "addrList.h"
|
|
|
|
/*
|
|
* localHostName()
|
|
*
|
|
* o Indicates failure by setting ptr to nill
|
|
*
|
|
* o Calls non posix gethostbyname() so that we get DNS style names
|
|
* (gethostbyname() should be available with most BSD sock libs)
|
|
*
|
|
* vxWorks user will need to configure a DNS format name for the
|
|
* host name if they wish to be cnsistent with UNIX and VMS hosts.
|
|
*
|
|
* this needs to attempt to determine if the process is a remote
|
|
* login - hard to do under UNIX
|
|
*/
|
|
char *localHostName ()
|
|
{
|
|
int size;
|
|
int status;
|
|
char pName[MAXHOSTNAMELEN];
|
|
char *pTmp;
|
|
|
|
status = gethostname ( pName, sizeof (pName) );
|
|
if(status){
|
|
return NULL;
|
|
}
|
|
|
|
size = strlen (pName)+1;
|
|
pTmp = (char *) malloc (size);
|
|
if (!pTmp) {
|
|
return pTmp;
|
|
}
|
|
|
|
strncpy ( pTmp, pName, size-1 );
|
|
pTmp[size-1] = '\0';
|
|
|
|
return pTmp;
|
|
}
|
|
|
|
|
|
/*
|
|
* getToken()
|
|
*/
|
|
static char *getToken (const char **ppString, char *pBuf, unsigned bufSIze)
|
|
{
|
|
const char *pToken;
|
|
unsigned i;
|
|
|
|
pToken = *ppString;
|
|
while(isspace(*pToken)&&*pToken){
|
|
pToken++;
|
|
}
|
|
|
|
for (i=0u; i<bufSIze; i++) {
|
|
if ( isspace (pToken[i]) || pToken[i]=='\0') {
|
|
pBuf[i] = '\0';
|
|
break;
|
|
}
|
|
pBuf[i] = pToken[i];
|
|
}
|
|
|
|
*ppString = &pToken[i];
|
|
|
|
if(*pToken){
|
|
return pBuf;
|
|
}
|
|
else{
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* addAddrToChannelAccessAddressList ()
|
|
*/
|
|
epicsShareFunc void epicsShareAPI addAddrToChannelAccessAddressList
|
|
(ELLLIST *pList, const ENV_PARAM *pEnv, unsigned short port)
|
|
{
|
|
osiSockAddrNode *pNewNode;
|
|
const char *pStr;
|
|
const char *pToken;
|
|
struct sockaddr_in addr;
|
|
char buf[32u]; /* large enough to hold an IP address */
|
|
int status;
|
|
|
|
pStr = envGetConfigParamPtr (pEnv);
|
|
if (!pStr) {
|
|
return;
|
|
}
|
|
|
|
while ( ( pToken = getToken (&pStr, buf, sizeof (buf) ) ) ) {
|
|
status = aToIPAddr ( pToken, port, &addr );
|
|
if (status<0) {
|
|
ca_printf ("%s: Parsing '%s'\n", __FILE__, pEnv->name);
|
|
ca_printf ("\tBad internet address or host name: '%s'\n", pToken);
|
|
continue;
|
|
}
|
|
|
|
pNewNode = (osiSockAddrNode *) calloc (1, sizeof(*pNewNode));
|
|
if (pNewNode==NULL) {
|
|
ca_printf ("addAddrToChannelAccessAddressList(): no memory available for configuration\n");
|
|
return;
|
|
}
|
|
|
|
pNewNode->addr.ia = addr;
|
|
|
|
/*
|
|
* LOCK applied externally
|
|
*/
|
|
ellAdd (pList, &pNewNode->node);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* setPortAndRemoveDuplicates ()
|
|
*/
|
|
epicsShareFunc void epicsShareAPI setPortAndRemoveDuplicates
|
|
(ELLLIST *pDestList, ELLLIST *pSrcList, unsigned short port)
|
|
{
|
|
osiSockAddrNode *pNode;
|
|
|
|
/*
|
|
* eliminate duplicates and set the port
|
|
*/
|
|
while ( (pNode = (osiSockAddrNode *) ellGet ( pSrcList ) ) ) {
|
|
osiSockAddrNode *pTmpNode;
|
|
|
|
if ( pNode->addr.sa.sa_family == AF_INET ) {
|
|
/*
|
|
* set the correct destination port
|
|
*/
|
|
pNode->addr.ia.sin_port = htons (port);
|
|
|
|
pTmpNode = (osiSockAddrNode *) ellFirst (pDestList);
|
|
while ( pTmpNode ) {
|
|
if (pTmpNode->addr.sa.sa_family == AF_INET) {
|
|
if (pNode->addr.ia.sin_addr.s_addr == pTmpNode->addr.ia.sin_addr.s_addr &&
|
|
pNode->addr.ia.sin_port == pTmpNode->addr.ia.sin_port) {
|
|
char buf[64];
|
|
ipAddrToA (&pNode->addr.ia, buf, sizeof(buf));
|
|
ca_printf ("Warning: Duplicate EPICS CA Address list entry \"%s\" discarded\n", buf);
|
|
free (pNode);
|
|
pNode = NULL;
|
|
break;
|
|
}
|
|
}
|
|
pTmpNode = (osiSockAddrNode *) ellNext (&pNode->node);
|
|
}
|
|
if (pNode) {
|
|
ellAdd (pDestList, &pNode->node);
|
|
}
|
|
}
|
|
else {
|
|
ellAdd (pDestList, &pNode->node);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* configureChannelAccessAddressList ()
|
|
*/
|
|
epicsShareFunc void epicsShareAPI configureChannelAccessAddressList
|
|
(ELLLIST *pList, SOCKET sock, unsigned short port)
|
|
{
|
|
ELLLIST tmpList;
|
|
char *pstr;
|
|
char yesno[32u];
|
|
int yes;
|
|
|
|
/*
|
|
* dont load the list twice
|
|
*/
|
|
assert ( ellCount(pList) == 0 );
|
|
|
|
ellInit ( &tmpList );
|
|
|
|
/*
|
|
* Check to see if the user has disabled
|
|
* initializing the search b-cast list
|
|
* from the interfaces found.
|
|
*/
|
|
yes = TRUE;
|
|
pstr = envGetConfigParam ( &EPICS_CA_AUTO_ADDR_LIST,
|
|
sizeof (yesno), yesno );
|
|
if (pstr) {
|
|
if ( strstr ( pstr, "no" ) || strstr ( pstr, "NO" ) ) {
|
|
yes = FALSE;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* LOCK is for piiu->destAddr list
|
|
* (lock outside because this is used by the server also)
|
|
*/
|
|
if (yes) {
|
|
osiSockAddr addr;
|
|
addr.ia.sin_family = AF_UNSPEC;
|
|
osiSockDiscoverBroadcastAddresses ( &tmpList, sock, &addr );
|
|
if ( ellCount ( &tmpList ) == 0 ) {
|
|
osiSockAddrNode *pNewNode;
|
|
pNewNode = (osiSockAddrNode *) calloc ( 1, sizeof (*pNewNode) );
|
|
if ( pNewNode ) {
|
|
/*
|
|
* if no interfaces found then look for local channels
|
|
* with the loop back interface
|
|
*/
|
|
pNewNode->addr.ia.sin_family = AF_INET;
|
|
pNewNode->addr.ia.sin_addr.s_addr = htonl ( INADDR_LOOPBACK );
|
|
ellAdd ( &tmpList, &pNewNode->node );
|
|
}
|
|
else {
|
|
errlogPrintf ( "configureChannelAccessAddressList(): no memory available for configuration\n" );
|
|
}
|
|
}
|
|
}
|
|
|
|
addAddrToChannelAccessAddressList ( &tmpList, &EPICS_CA_ADDR_LIST, port );
|
|
|
|
setPortAndRemoveDuplicates (pList, &tmpList, port);
|
|
}
|
|
|
|
|
|
/*
|
|
* printChannelAccessAddressList ()
|
|
*/
|
|
epicsShareFunc void epicsShareAPI printChannelAccessAddressList (const ELLLIST *pList)
|
|
{
|
|
osiSockAddrNode *pNode;
|
|
|
|
printf ("Channel Access Address List\n");
|
|
pNode = (osiSockAddrNode *) ellFirst (pList);
|
|
while (pNode) {
|
|
char buf[64];
|
|
ipAddrToA (&pNode->addr.ia, buf, sizeof(buf));
|
|
printf ("%s\n", buf);
|
|
pNode = (osiSockAddrNode *) ellNext (&pNode->node);
|
|
}
|
|
}
|