1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126 |
- #include "cmGlobal.h"
- #include "cmRpt.h"
- #include "cmErr.h"
- #include "cmCtx.h"
- #include "cmMem.h"
- #include "cmMallocDebug.h"
- #include "cmLinkedHeap.h"
- #include "cmUdpPort.h"
- #include "cmRtSysMsg.h"
- #include "cmRtNet.h"
- #include "cmTime.h"
- #include "cmText.h"
-
- // flags for cmRtNetNode_t.flags;
- enum
- {
- kLocalNodeNetFl = 0x01,
- kValidNodeNetFl = 0x02
- };
-
- // flags for cmRtNet_t.flags
- enum
- {
- kReportSyncNetFl = 0x01
- };
-
- struct cmRtNetNode_str;
-
- typedef struct cmRtNetEnd_str
- {
- cmChar_t* label;
- unsigned id;
- struct cmRtNetNode_str* np; // Owner node.
- struct cmRtNetEnd_str* link;
- } cmRtNetEnd_t;
-
- struct cmRtNet_str;
-
- typedef struct cmRtNetNode_str
- {
- unsigned rtSubIdx; // rtSubIdx of the sub-system which owns this node
- cmChar_t* label; // Node label.
- struct sockaddr_in sockaddr; // Socket address
- cmChar_t* addr; // IP Address (human readable)
- cmUdpPort_t port; // Socket port
- unsigned flags; // See kXXXNodeNetFl flags above.
- unsigned endPtIdx; // tracks the next endpoint to send during sync-mode
- unsigned endPtCnt; // local-node=actual cnt of endpt's remote-node:expected cnt of endpt's
- cmTimeSpec_t lastSendTime; // Time of last message sent
- cmRtNetEnd_t* ends; // End point list for this node
- struct cmRtNetNode_str* link;
- } cmRtNetNode_t;
-
- typedef struct cmRtNet_str
- {
- cmErr_t err; // Error state object
- unsigned flags; // See kXXXNetFl above.
- unsigned rtSubIdx; // rtSubIdx of the owning sub-system
- cmUdpH_t udpH; // UDP port handle
- cmUdpCallback_t cbFunc; // Client callback to receive incoming messages from network.
- void* cbArg; //
- cmRtNetNode_t* nodes; // Node list.
- cmRtNetNode_t* localNode; // Pointer to local node (which is also in node list)
- unsigned udpRecvBufByteCnt; // UDP port receive buffer size.
- unsigned udpTimeOutMs; // UDP time-out period
- cmChar_t* bcastAddr; // Network broadcast address
- } cmRtNet_t;
-
- cmRtNetH_t cmRtNetNullHandle = cmSTATIC_NULL_HANDLE;
- cmRtNetEndptH_t cmRtNetEndptNullHandle = cmSTATIC_NULL_HANDLE;
-
- cmRtNet_t* _cmRtNetHandleToPtr( cmRtNetH_t h )
- {
- cmRtNet_t* p = (cmRtNet_t*)h.h;
- assert( p != NULL );
- return p;
- }
-
- cmRtNetEnd_t* _cmRtNetEndptHandleToPtr( cmRtNetEndptH_t h )
- {
- cmRtNetEnd_t* p = (cmRtNetEnd_t*)h.h;
- assert( p != NULL );
- return p;
- }
-
- void _cmRtNetVRpt( cmRtNet_t* p, const cmChar_t* fmt, va_list vl )
- {
- if( cmIsFlag(p->flags,kReportSyncNetFl) )
- cmRptVPrintf(p->err.rpt,fmt,vl);
- }
-
- void _cmRtNetRpt( cmRtNet_t* p, const cmChar_t* fmt, ... )
- {
- va_list vl;
- va_start(vl,fmt);
- _cmRtNetVRpt(p,fmt,vl);
- va_end(vl);
- }
-
- cmRtNetNode_t* _cmRtNetFindNode( cmRtNet_t* p, const cmChar_t* label )
- {
- if( label == NULL )
- return NULL;
-
- cmRtNetNode_t* np = p->nodes;
- for(; np!=NULL; np=np->link)
- if( strcmp(label,np->label)==0)
- return np;
-
- return NULL;
- }
-
- cmRtNetNode_t* _cmRtNetFindNodeFromSockAddr( cmRtNet_t* p, const struct sockaddr_in* saddr )
- {
- if( saddr == NULL )
- return NULL;
-
- cmRtNetNode_t* np = p->nodes;
- for(; np!=NULL; np=np->link)
- if( np->sockaddr.sin_addr.s_addr == saddr->sin_addr.s_addr && np->sockaddr.sin_port == saddr->sin_port )
- return np;
-
- return NULL;
- }
-
- void _cmRtNetFreeNode( cmRtNetNode_t* np )
- {
- cmRtNetEnd_t* ep = np->ends;
- while( ep != NULL )
- {
- cmRtNetEnd_t* nep = ep->link;
- cmMemFree(ep->label);
- cmMemFree(ep);
- ep = nep;
- }
-
- cmMemFree(np->label);
- cmMemFree(np->addr);
- cmMemFree(np);
- }
-
- void _cmRtNetReleaseNodes( cmRtNet_t* p )
- {
- cmRtNetNode_t* np = p->nodes;
- while( np != NULL )
- {
- cmRtNetNode_t* nnp = np->link;
-
- _cmRtNetFreeNode(np);
-
- np = nnp;
- }
- p->nodes = NULL;
- p->localNode = NULL;
- }
-
- cmRtNetRC_t _cmRtNetReleaseNode( cmRtNet_t* p, cmRtNetNode_t* np )
- {
- cmRtNetNode_t* cnp = p->nodes;
- cmRtNetNode_t* pnp = NULL;
-
- while( cnp != NULL )
- {
- cmRtNetNode_t* nnp = cnp->link;
- if( np == cnp )
- {
- if( pnp == NULL )
- p->nodes = np->link;
- else
- pnp->link = np->link;
-
- _cmRtNetFreeNode(np);
-
- return kOkNetRC;
- }
-
- pnp = np;
- cnp = nnp;
- }
-
- assert(0);
- return cmErrMsg(&p->err,kNodeNotFoundNetRC,"Node to release not found.");
- }
-
- cmRtNetRC_t _cmRtNetCreateNode( cmRtNet_t* p, const cmChar_t* label, unsigned rtSubIdx, const cmChar_t* addr, cmUdpPort_t port, const struct sockaddr_in* saddr, unsigned flags, unsigned endPtCnt )
- {
- cmRtNetRC_t rc = kOkNetRC;
- cmRtNetNode_t* np;
-
- if( cmTextIsEmpty(label) )
- return cmErrMsg(&p->err,kInvalidLabelNetRC,"A null or blank node label was encountered.");
-
- if((np = _cmRtNetFindNode(p,label)) != NULL )
- return cmErrMsg(&p->err,kDuplLabelNetRC,"The node label '%s' is already in use.",cmStringNullGuard(label));
-
- np = cmMemAllocZ(cmRtNetNode_t,1);
- np->label = cmMemAllocStr(label);
-
- if( saddr != NULL )
- np->sockaddr = *saddr;
-
- np->rtSubIdx = rtSubIdx;
- np->addr = addr==NULL ? NULL : cmMemAllocStr(addr);
- np->port = port;
- np->flags = flags;
- np->endPtCnt = endPtCnt;
- np->link = p->nodes;
- p->nodes = np;
-
- return rc;
- }
-
- cmRtNetEnd_t* _cmRtNetFindNodeEnd(cmRtNetNode_t* np, const cmChar_t* endPtLabel )
- {
- cmRtNetEnd_t* ep = np->ends;
- for(; ep!=NULL; ep=ep->link)
- if( strcmp(ep->label,endPtLabel)==0 )
- return ep;
- return NULL;
- }
-
- cmRtNetEnd_t* _cmRtNetIndexToEndpoint( cmRtNet_t* p, cmRtNetNode_t* np, unsigned endIndex )
- {
- cmRtNetEnd_t* ep = np->ends;
- unsigned i;
- for(i=0; ep!=NULL; ep=ep->link)
- {
- if( i == endIndex )
- return ep;
- ++i;
- }
-
- return NULL;
- }
-
- cmRtNetRC_t _cmRtNetCreateEndpoint( cmRtNet_t* p, cmRtNetNode_t* np, const cmChar_t* endPtLabel, unsigned endPtId )
- {
- if( endPtLabel == NULL )
- return cmErrMsg(&p->err,kInvalidLabelNetRC,"A null or blank node label was encountered.");
-
- if( _cmRtNetFindNodeEnd( np, endPtLabel) != NULL)
- return cmErrMsg(&p->err,kDuplEndNetRC,"A duplicate endpoint ('%s') was encountered on node '%s'.",endPtLabel,np->label);
-
- cmRtNetRC_t rc = kOkNetRC;
- cmRtNetEnd_t* ep = cmMemAllocZ(cmRtNetEnd_t,1);
-
- ep->label = cmMemAllocStr(endPtLabel);
- ep->id = endPtId;
- ep->np = np;
- ep->link = np->ends;
- np->ends = ep;
-
- return rc;
- }
-
- unsigned _cmRtNetNodeEndpointCount( cmRtNetNode_t* np )
- {
- cmRtNetEnd_t* ep = np->ends;
- unsigned n = 0;
- for(; ep!=NULL; ep=ep->link)
- ++n;
- return n;
- }
-
- unsigned _cmRtNetSyncMsgSerialByteCount( const cmRtNetSyncMsg_t* m )
- { return sizeof(cmRtNetSyncMsg_t) + (m->label==NULL ? 1 : strlen(m->label) + 1); }
-
- cmRtNetRC_t _cmRtNetSerializeSyncMsg( cmRtNet_t* p, const cmRtNetSyncMsg_t* m, void* buf, unsigned n )
- {
- unsigned bn = _cmRtNetSyncMsgSerialByteCount(m);
- char* b = (char*)buf;
-
- if( bn > n )
- return cmErrMsg(&p->err,kBufToSmallNetRC,"Serialize buffer too small.");
-
- memcpy(b,m,sizeof(*m));
- strcpy(b + sizeof(*m),m->label==NULL ? "" : m->label);
- return kOkNetRC;
- }
-
- cmRtNetRC_t _cmRtNetDeserializeSyncMsg( const void* buf, unsigned n, cmRtNetSyncMsg_t* m )
- {
- assert( n > sizeof(*m));
- memcpy(m,buf,sizeof(*m));
- const cmRtNetSyncMsg_t* mp = (const cmRtNetSyncMsg_t*)buf;
- const cmChar_t* s = (const cmChar_t*)(mp+1);
- m->label = cmMemAllocStr(s);
- return kOkNetRC;
- }
-
- cmRtNetRC_t _cmRtNetSendSyncMsg( cmRtNet_t* p, cmRtNetNode_t* np, cmRtNetSelId_t selId, const cmChar_t* msgLabel, unsigned msgId, unsigned msgRtSubIdx )
- {
- cmRtNetSyncMsg_t m;
- cmRtNetRC_t rc = kOkNetRC;
- cmUdpRC_t udpRC = kOkUdpRC;
-
- m.hdr.rtSubIdx = cmInvalidIdx;
- m.hdr.selId = kNetSyncSelRtId;
- m.selId = selId;
- m.label = msgLabel;
- m.id = msgId;
- m.rtSubIdx = msgRtSubIdx;
-
- // determine size of msg to send
- unsigned n = _cmRtNetSyncMsgSerialByteCount(&m);
- cmChar_t buf[n];
-
- // serialize msg into buf[]
- if((rc = _cmRtNetSerializeSyncMsg(p,&m,buf,n)) != kOkNetRC )
- return rc;
-
- // send the msg
- if( selId == kHelloSelNetId )
- udpRC = cmUdpSend2(p->udpH, buf, n, p->bcastAddr, np->port );
- else
- udpRC = cmUdpSendTo(p->udpH, buf, n, &np->sockaddr );
-
- // check for send errors
- if( udpRC != kOkUdpRC )
- {
- rc = cmErrMsg(&p->err,kUdpPortFailNetRC,"Sync msg. send on UDP port failed.");
- }
- else
- {
- // record the last send time
- cmTimeGet(&np->lastSendTime);
- }
-
- return rc;
- }
-
- cmRtNetRC_t _cmRtNetFree( cmRtNet_t* p )
- {
- cmRtNetRC_t rc = kOkNetRC;
-
- if( cmUdpFree(&p->udpH) != kOkUdpRC )
- cmErrMsg(&p->err,kUdpPortFailNetRC,"UDP Port free failed.");
-
- _cmRtNetReleaseNodes(p);
-
- cmMemFree(p->bcastAddr);
-
- cmMemFree(p);
- return rc;
- }
-
- const cmRtNetNode_t* _cmRtNetIndexToRemoteNode( cmRtNet_t* p, unsigned idx )
- {
- const cmRtNetNode_t* np = p->nodes;
- unsigned i = 0;
-
- for(; np!=NULL; np=np->link)
- if( np != p->localNode )
- {
- if( i == idx )
- return np;
- ++i;
- }
-
- return NULL;
- }
-
- const cmRtNetEnd_t* _cmRtNetIndexToEndpt( const cmRtNetNode_t* np, unsigned endIdx )
- {
- const cmRtNetEnd_t* ep = np->ends;
- unsigned i = 0;
-
- for(; ep!=NULL; ep=ep->link,++i)
- if( i==endIdx )
- return ep;
-
- return NULL;
- }
-
- const cmRtNetEnd_t* _cmRtNetFindEndpt( cmRtNet_t* p, unsigned nodeIdx, unsigned epIdx )
- {
- const cmRtNetNode_t* np;
- const cmRtNetEnd_t* ep;
-
- if((np = _cmRtNetIndexToRemoteNode( p, nodeIdx )) == NULL )
- return NULL;
-
- if((ep = _cmRtNetIndexToEndpt( np, epIdx )) == NULL )
- return NULL;
-
- return ep;
- }
-
-
- const cmChar_t* cmRtNetSyncMsgLabel( const cmRtNetSyncMsg_t* m )
- {
- if( m->selId==kNodeSelNetId || m->selId==kEndpointSelNetId )
- return (const cmChar_t*)(m+1);
-
- return "";
- }
-
- cmRtNetRC_t cmRtNetAlloc( cmCtx_t* ctx, cmRtNetH_t* hp, unsigned rtSubIdx, cmUdpCallback_t cbFunc, void* cbArg )
- {
- cmRtNetRC_t rc;
- if((rc = cmRtNetFree(hp)) != kOkNetRC )
- return rc;
-
- cmRtNet_t* p = cmMemAllocZ(cmRtNet_t,1);
- cmErrSetup(&p->err,&ctx->rpt,"cmRtNet");
-
- // allocate the UDP port
- if(cmUdpAlloc(ctx,&p->udpH) != kOkUdpRC )
- {
- cmErrMsg(&p->err,kUdpPortFailNetRC,"UDP Port allocate failed.");
- goto errLabel;
- }
-
- p->rtSubIdx = rtSubIdx;
- p->udpTimeOutMs = 50;
- p->udpRecvBufByteCnt = 8192;
- p->cbFunc = cbFunc;
- p->cbArg = cbArg;
-
- hp->h = p;
-
- errLabel:
- if(rc != kOkNetRC )
- _cmRtNetFree(p);
-
- return rc;
- }
-
- cmRtNetRC_t cmRtNetFree( cmRtNetH_t* hp )
- {
- cmRtNetRC_t rc = kOkNetRC;
-
- if( hp==NULL || cmRtNetIsValid(*hp)==false )
- return rc;
-
- cmRtNet_t* p = _cmRtNetHandleToPtr(*hp);
-
- if((rc = _cmRtNetFree(p)) != kOkNetRC )
- return rc;
-
- hp->h = NULL;
-
- return rc;
- }
-
- const cmChar_t* cmRtNetLocalHostName( cmRtNetH_t h )
- {
- cmRtNet_t* p = _cmRtNetHandleToPtr(h);
- return cmUdpHostName(p->udpH);
- }
-
-
- bool cmRtNetIsValid( cmRtNetH_t h )
- { return h.h !=NULL; }
-
- cmUdpH_t cmRtNetUdpPortHandle( cmRtNetH_t h )
- {
- cmRtNet_t* p = _cmRtNetHandleToPtr(h);
- return p->udpH;
- }
-
- cmRtNetRC_t _cmRtNetSendEndpointReplyMsg( cmRtNet_t* p, cmRtNetNode_t* np, cmRtNetSelId_t srcSelId )
- {
- cmRtNetRC_t rc = kOkNetRC;
- cmRtNetEnd_t* ep = NULL;
- const cmChar_t* msgLabel = NULL;
- unsigned msgId = cmInvalidId;
- unsigned msgRtSubIdx = cmInvalidIdx;
- cmRtNetSelId_t selId = kEndpointSelNetId;
- const cmChar_t* rptLabel = "endpoint";
-
- if( np == NULL )
- return cmErrMsg(&p->err,kNodeNotFoundNetRC,"The net node associated with an endpoint reply was not found.");
-
- // if we got here by receiving a 'done' msg from the remote node ...
- if( srcSelId == kDoneSelNetId )
- {
- // ... then mark the remote node as having recieved all endpoints
- unsigned n;
- if((n = _cmRtNetNodeEndpointCount(np)) != np->endPtCnt )
- rc = cmErrMsg(&p->err,kNodeEndCntErrNetRC,"The node '%s' expected %i endpoints but received %i.",cmStringNullGuard(np->label),np->endPtCnt,n);
- else
- np->flags = cmSetFlag(np->flags,kValidNodeNetFl);
- }
-
- // attempt to get the next local endpoint to send ...
- if((ep = _cmRtNetIndexToEndpoint(p,p->localNode,np->endPtIdx)) != NULL )
- {
- msgLabel = ep->label; // ... send next local endpoint
- msgId = ep->id;
- msgRtSubIdx = ep->np->rtSubIdx;
- }
- else // .... all local endpoints have been sent
- {
- selId = kInvalidSelNetId;
- rptLabel = "done";
-
- // verify that no endpoints are available
- if( np->endPtIdx < p->localNode->endPtCnt )
- rc = cmErrMsg(&p->err,kSyncFailNetRC,"More endpoints are available to send but are not reachable.");
- else
- {
- // if the remote node still has endpts to send then continue
- // sending 'done' messages.
- if( np->endPtIdx==p->localNode->endPtCnt || srcSelId != kDoneSelNetId )
- selId = kDoneSelNetId;
- }
- }
-
- // selId is set to kInvalidSelNetId when we encounter the (stop) criteria
- if( selId != kInvalidSelNetId )
- {
- if((rc = _cmRtNetSendSyncMsg(p,np,selId,msgLabel,msgId,msgRtSubIdx)) != kOkNetRC )
- rc = cmErrMsg(&p->err,rc,"Send '%s' to %s:%s:%i failed.",rptLabel,cmStringNullGuard(np->label),cmStringNullGuard(np->addr),np->port);
- else
- _cmRtNetRpt(p,"Sent %s.\n",cmStringNullGuard(rptLabel));
-
- np->endPtIdx += 1;
- }
-
- return rc;
- }
-
- bool _cmRtNetIsSyncModeMsg( const void* data, unsigned dataByteCnt )
- {
- cmRtNetSyncMsg_t* m = (cmRtNetSyncMsg_t*)data;
- return dataByteCnt >= sizeof(cmRtNetSyncMsg_t) && m->hdr.selId == kNetSyncSelRtId;
- }
-
- // When the network message recieve function (See cmRtNetAlloc() 'cbFunc')
- // receives a message with the cmRtSysMsgHdr_t.selId == kNetSyncSelRtId
- // it should call this function to update the current sync state of the
- // cmRtNet.
- cmRtNetRC_t _cmRtNetSyncModeRecv( cmRtNet_t* p, const char* data, unsigned dataByteCnt, const struct sockaddr_in* fromAddr )
- {
- cmRtNetRC_t rc = kOkNetRC;
- cmRtNetSyncMsg_t m;
- m.label = NULL;
-
- assert( _cmRtNetIsSyncModeMsg(data,dataByteCnt));
-
- if( _cmRtNetDeserializeSyncMsg(data,dataByteCnt,&m) != kOkNetRC )
- {
- rc = cmErrMsg(&p->err,rc,"Net sync. receive failed due to deserialize fail.");
- goto errLabel;
- }
-
- _cmRtNetRpt(p,"recv from:%s\n",cmUdpAddrToString(p->udpH, fromAddr ));
-
-
- assert( m.hdr.selId == kNetSyncSelRtId );
-
- // attempt to locate the remote node which sent the msg
- cmRtNetNode_t* np = _cmRtNetFindNodeFromSockAddr(p,fromAddr);
-
-
- switch( m.selId )
- {
- case kHelloSelNetId:
- // if this is a response to a broadcast from the local node then ignore it
- if(m.label!=NULL && p->localNode->label!=NULL && (np = _cmRtNetFindNode(p,m.label)) != NULL && strcmp(p->localNode->label,m.label)==0 )
- {
- const cmChar_t* fromAddrStr = cmUdpAddrToString(p->udpH,fromAddr);
- const cmChar_t* localAddrStr = cmUdpAddrToString(p->udpH,cmUdpLocalAddr(p->udpH));
-
- if( fromAddrStr!=NULL && localAddrStr!=NULL && strcmp(fromAddrStr,localAddrStr)!=0)
- cmErrMsg(&p->err,kDuplLocalNetRC,"The node label '%s' appears to be duplicated at address %s and locally.",cmStringNullGuard(m.label),fromAddrStr);
-
- np->sockaddr = *fromAddr;
- goto errLabel;
- }
- // fall through
-
- case kNodeSelNetId:
- {
- // if the node already exists ...
- if( np != NULL )
- {
- // ... delete it because we are about to get new info. about it.
- if((rc = _cmRtNetReleaseNode(p,np )) != kOkNetRC )
- goto errLabel;
- }
-
- // create a node proxy to represent the remote node
- // (Note:m.id == remote node endpoint count (i.e. the count of endpoints expected for the remote node.))
- if(( rc = _cmRtNetCreateNode(p,m.label,m.rtSubIdx,NULL,0,fromAddr,0,m.id)) != kOkNetRC )
- goto errLabel;
-
- np = p->nodes; // newest node is always the first node
-
- // send response
- switch( m.selId )
- {
- case kHelloSelNetId:
- _cmRtNetRpt(p,"rcv hello\n"); // reply with local node
- rc = _cmRtNetSendSyncMsg( p, np, kNodeSelNetId, p->localNode->label, p->localNode->endPtCnt, p->localNode->rtSubIdx );
- break;
-
- case kNodeSelNetId:
- _cmRtNetRpt(p,"rcv node\n");
- _cmRtNetSendEndpointReplyMsg( p, np, m.selId ); // reply with first endpoint
- break;
-
- default:
- assert(0);
- }
-
- }
- break;
-
- case kDoneSelNetId:
- //case kEndpointAckSelNetId:
- rc = _cmRtNetSendEndpointReplyMsg(p,np,m.selId);
- break;
-
- case kEndpointSelNetId:
- {
- cmRtNetEnd_t* ep;
-
- // verify the remote node exists.
- if( np == NULL )
- {
- rc = cmErrMsg(&p->err,kNodeNotFoundNetRC,"The net node associated with an endpoint receive was not found.");
- goto errLabel;
- }
-
- // attempt to find the end point
- if((ep = _cmRtNetFindNodeEnd(np, m.label)) != NULL )
- ep->id = m.id; // the endpoint was found update the endPtId
- else
- {
- // create a local proxy for the endpoint
- if((rc = _cmRtNetCreateEndpoint(p,np,m.label,m.id)) != kOkNetRC )
- goto errLabel;
- }
-
- // reply with a local endpoint or 'done' msg
- rc = _cmRtNetSendEndpointReplyMsg( p, np, m.selId );
- }
- break;
-
- default:
- assert(0);
- break;
- }
-
- errLabel:
-
- cmMemFree((cmChar_t*)m.label);
- return rc;
- }
-
- unsigned _cmRtNetAddrToNodeIndex( cmRtNet_t* p, const struct sockaddr_in* addr )
- {
- unsigned i;
- cmRtNetNode_t* np = p->nodes;
- for(i=0; np!=NULL; np=np->link,++i)
- if( cmUdpAddrIsEqual( addr, &np->sockaddr ) )
- return i;
-
- return cmInvalidIdx;
- }
-
-
- // This is called in the context of cmRtNetReceive().
- void _cmRtNetRecv( void* cbArg, const char* data, unsigned dataByteCnt, const struct sockaddr_in* fromAddr )
- {
- cmRtNet_t* p = (cmRtNet_t*)cbArg;
-
- // if this is a sync msg - then handle it here
- if( _cmRtNetIsSyncModeMsg(data,dataByteCnt))
- _cmRtNetSyncModeRecv(p,data,dataByteCnt,fromAddr);
- else
- {
- // All non-sync messages arriving here are prefixed by a cmRtNetMsg_t header - fill in the source addr here.
-
- // NOTE: the source addr could be filled in by the sender but this would increase the size
- // of the msg. Instead we choose the more time consuming method of looking up the
- // soure node here - hmmmm????.
-
- cmRtNetMsg_t* hdr = (cmRtNetMsg_t*)(data);
- hdr->srcNodeIdx = _cmRtNetAddrToNodeIndex(p,fromAddr);
- }
-
- p->cbFunc(p->cbArg,data,dataByteCnt,fromAddr);
-
- }
-
- cmRtNetRC_t cmRtNetInitialize( cmRtNetH_t h, const cmChar_t* bcastAddr, const cmChar_t* nodeLabel, const cmChar_t* ipAddr, cmUdpPort_t port )
- {
- cmRtNet_t* p = _cmRtNetHandleToPtr(h);
- cmRtNetRC_t rc;
-
- // release the local node
- if((rc = cmRtNetFinalize(h)) != kOkNetRC )
- goto errLabel;
-
- if( cmTextIsEmpty(bcastAddr) )
- {
- rc = cmErrMsg(&p->err,kInvalidArgNetRC,"The 'broadcast address' is not valid.");
- goto errLabel;
- }
-
- // if this is the local node then initialze the local socket
- if( cmUdpInit(p->udpH,port,kNonBlockingUdpFl | kBroadcastUdpFl,_cmRtNetRecv,p,NULL,0,p->udpRecvBufByteCnt,p->udpTimeOutMs) != kOkUdpRC )
- {
- rc = cmErrMsg(&p->err,kUdpPortFailNetRC,"The UDP port initialization failed.");
- goto errLabel;
- }
-
- // create the local node
- if((rc = _cmRtNetCreateNode(p,nodeLabel, p->rtSubIdx, ipAddr, port, NULL, kLocalNodeNetFl, 0)) != kOkNetRC )
- goto errLabel;
-
- // the last created node is always the first node on the list
- p->localNode = p->nodes;
- p->bcastAddr = cmMemResizeStr(p->bcastAddr,bcastAddr);
-
- // begin listening on the local port
- if( cmUdpEnableListen(p->udpH, true ) != kOkUdpRC )
- {
- rc = cmErrMsg(&p->err,kUdpPortFailNetRC,"The UDP port failed to enter 'listen' mode.");
- goto errLabel;
- }
-
-
- errLabel:
- return rc;
- }
-
-
- bool cmRtNetIsInitialized( cmRtNetH_t h )
- {
- if( cmRtNetIsValid(h) == false )
- return false;
-
- cmRtNet_t* p = _cmRtNetHandleToPtr(h);
- return p->localNode != NULL && cmTextIsNotEmpty(p->bcastAddr);
- }
-
-
- cmRtNetRC_t cmRtNetRegisterEndPoint( cmRtNetH_t h, const cmChar_t* endPtLabel, unsigned endPtId )
- {
- cmRtNetRC_t rc = kOkNetRC;
- cmRtNet_t* p = _cmRtNetHandleToPtr(h);
-
- if( p->localNode == NULL )
- return cmErrMsg(&p->err,kLocalNodeNetRC,"Local endpoints may not be added if a local node has not been defined.");
-
- if((rc = _cmRtNetCreateEndpoint(p, p->localNode,endPtLabel,endPtId )) == kOkNetRC )
- p->localNode->endPtCnt += 1;
-
- return rc;
- }
-
- cmRtNetRC_t cmRtNetFinalize( cmRtNetH_t h )
- {
- cmRtNet_t* p = _cmRtNetHandleToPtr(h);
- _cmRtNetReleaseNodes(p);
-
- return kOkNetRC;
- }
-
- cmRtNetRC_t cmRtNetDoSync( cmRtNetH_t h )
- {
- cmRtNet_t* p = _cmRtNetHandleToPtr(h);
-
- // broadcast 'node' msg
- return _cmRtNetSendSyncMsg( p, p->localNode, kHelloSelNetId, p->localNode->label, p->localNode->endPtCnt, cmInvalidIdx );
- }
-
- cmRtNetRC_t cmRtNetReceive( cmRtNetH_t h )
- {
- cmRtNetRC_t rc = kOkNetRC;
- cmRtNet_t* p = _cmRtNetHandleToPtr(h);
-
- // Calling this function results in callbacks to _cmRtNetRecv() (above)
- if( cmUdpGetAvailData(p->udpH, NULL, NULL, NULL ) != kOkUdpRC )
- {
- cmErrMsg(&p->err,kUdpPortFailNetRC,"UDP port query failed.");
- goto errLabel;
- }
- errLabel:
- return rc;
- }
-
- unsigned cmRtNetAddrToNodeIndex( cmRtNetH_t h, const struct sockaddr_in* a )
- {
- cmRtNet_t* p = _cmRtNetHandleToPtr(h);
- return _cmRtNetAddrToNodeIndex(p,a);
- }
-
-
- cmRtNetRC_t cmRtNetEndpointHandle( cmRtNetH_t h, const cmChar_t* nodeLabel, const cmChar_t* endptLabel, cmRtNetEndptH_t* hp )
- {
- cmRtNetRC_t rc = kOkNetRC;
- cmRtNet_t* p = _cmRtNetHandleToPtr(h);
- cmRtNetNode_t* np;
- cmRtNetEnd_t* ep;
-
- if(( np = _cmRtNetFindNode(p,nodeLabel)) == NULL )
- return cmErrMsg(&p->err,kNodeNotFoundNetRC,"The node '%s' was not found.",cmStringNullGuard(nodeLabel));
-
-
- if(( ep = _cmRtNetFindNodeEnd(np, endptLabel )) == NULL )
- return cmErrMsg(&p->err,kEndNotFoundNetRC,"The endpoint '%s' on '%s' on node was not found.",cmStringNullGuard(endptLabel),cmStringNullGuard(nodeLabel));
-
- hp->h = ep;
-
- return rc;
- }
-
- bool cmRtNetEndpointIsValid( cmRtNetEndptH_t endPtH )
- { return endPtH.h != NULL; }
-
- unsigned cmRtNetEndpointId( cmRtNetEndptH_t endPtH )
- {
- if( !cmRtNetEndpointIsValid(endPtH) )
- return cmInvalidId;
-
- cmRtNetEnd_t* ep = _cmRtNetEndptHandleToPtr( endPtH );
- return ep->id;
- }
-
- const cmChar_t* cmRtNetEndpointLabel( cmRtNetEndptH_t endPtH )
- {
- if( !cmRtNetEndpointIsValid(endPtH) )
- return NULL;
-
- cmRtNetEnd_t* ep = _cmRtNetEndptHandleToPtr( endPtH );
- return ep->label;
- }
-
-
- cmRtNetRC_t _cmRtNetSend( cmRtNet_t* p, unsigned srcEndPtId, const cmRtNetEnd_t* ep, const void* msg, unsigned msgByteCnt )
- {
- cmRtNetRC_t rc = kOkNetRC;
-
- unsigned hN = sizeof(cmRtNetMsg_t);
- unsigned dN = hN + msgByteCnt;
- char data[ dN ];
-
- cmRtNetMsg_t* r = (cmRtNetMsg_t*)data;
- r->hdr.rtSubIdx = ep->np->rtSubIdx;
- r->hdr.selId = kMsgSelRtId;
- r->dstEndPtId = ep->id;
- r->srcEndPtId = srcEndPtId;
- memcpy(data+hN,msg,msgByteCnt);
-
- // ep->np->sockaddr identifies the node on the receiving cmRtNet.
- // cmRtNetMsg_t* r.endptId is then used by the receiving cmRtNet to indicate which endpoint on
- // the node the incoming message should be associated with.
-
- if( cmUdpSendTo(p->udpH, data, dN, &ep->np->sockaddr ) != kOkUdpRC )
- return cmErrMsg(&p->err,kUdpPortFailNetRC,"Send to node:%s endpt:%s failed.\n",cmStringNullGuard(ep->np->label),cmStringNullGuard(ep->label));
-
- return rc;
- }
-
- cmRtNetRC_t cmRtNetSend( cmRtNetH_t h, unsigned srcEndPtId, cmRtNetEndptH_t epH, const void* msg, unsigned msgByteCnt )
- {
- cmRtNet_t* p = _cmRtNetHandleToPtr(h);
- cmRtNetEnd_t* ep = _cmRtNetEndptHandleToPtr(epH);
-
- assert( ep != NULL );
- return _cmRtNetSend(p,srcEndPtId,ep,msg,msgByteCnt);
- }
-
-
- cmRtNetRC_t cmRtNetSendByLabels( cmRtNetH_t h, unsigned srcEndPtId, const cmChar_t* nodeLabel, const cmChar_t* endptLabel, const void* msg, unsigned msgByteCnt )
- {
- cmRtNetRC_t rc = kOkNetRC;
- cmRtNetEndptH_t epH = cmRtNetEndptNullHandle;
-
- if((rc = cmRtNetEndpointHandle(h,nodeLabel,endptLabel,&epH)) != kOkNetRC )
- return rc;
-
- return cmRtNetSend(h,srcEndPtId,epH,msg,msgByteCnt);
- }
-
- cmRtNetRC_t cmRtNetSendByIndex( cmRtNetH_t h, unsigned srcEndPtId, unsigned nodeIdx, unsigned endptIdx, const void* msg, unsigned msgByteCnt )
- {
- cmRtNet_t* p = _cmRtNetHandleToPtr(h);
-
- const cmRtNetEnd_t* ep;
-
- if((ep = _cmRtNetFindEndpt(p, nodeIdx, endptIdx )) == NULL )
- return cmErrMsg(&p->err,kEndNotFoundNetRC,"The endpoint at node index %i endpoint index %i was not found.",nodeIdx,endptIdx);
-
- return _cmRtNetSend( p, srcEndPtId, ep, msg, msgByteCnt );
- }
-
-
- bool cmRtNetReportSyncEnable( cmRtNetH_t h, bool enableFl )
- {
- cmRtNet_t* p = _cmRtNetHandleToPtr(h);
- bool fl = cmIsFlag(p->flags,kReportSyncNetFl);
- p->flags = cmEnaFlag(p->flags,kReportSyncNetFl,enableFl);
- return fl;
- }
-
- bool cmRtNetReportSyncIsEnabled( cmRtNetH_t h )
- {
- cmRtNet_t* p = _cmRtNetHandleToPtr(h);
- return cmIsFlag(p->flags,kReportSyncNetFl);
- }
-
- void cmRtNetReport( cmRtNetH_t h )
- {
- cmRtNet_t* p = _cmRtNetHandleToPtr(h);
- cmRpt_t* rpt = p->err.rpt;
-
-
- cmRtNetNode_t* np = p->nodes;
- for(; np!=NULL; np=np->link)
- {
- cmRptPrintf(rpt,"Node: %s ",np->label);
-
- if( np->addr != NULL )
- cmRptPrintf(rpt,"%s ",np->addr );
-
- if( cmIsFlag(np->flags,kLocalNodeNetFl) )
- cmRptPrintf(rpt,"LOCAL ");
-
- cmRptPrintf(rpt,"%s ",cmStringNullGuard(cmUdpAddrToString(p->udpH,&np->sockaddr)));
-
- if( np->port != kInvalidUdpPortNumber )
- cmRptPrintf(rpt,"%i ",np->port );
-
- cmRptPrintf(rpt,"\n");
-
- cmRtNetEnd_t* ep = np->ends;
- for(; ep!=NULL; ep=ep->link)
- {
- cmRptPrintf(rpt," endpt: %i %s\n",ep->id,cmStringNullGuard(ep->label ));
- }
- }
- }
-
- const cmChar_t* cmRtNetLocalNodeLabel( cmRtNetH_t h )
- {
- cmRtNet_t* p = _cmRtNetHandleToPtr( h );
- return p->localNode->label;
- }
-
- unsigned cmRtNetRemoteNodeCount( cmRtNetH_t h )
- {
- cmRtNet_t* p = _cmRtNetHandleToPtr( h );
- const cmRtNetNode_t* np = p->nodes;
- unsigned n = 0;
-
- for(; np!=NULL; np=np->link)
- if( np != p->localNode )
- ++n;
-
- return n;
- }
-
- const cmChar_t* cmRtNetRemoteNodeLabel( cmRtNetH_t h, unsigned idx )
- {
- cmRtNet_t* p = _cmRtNetHandleToPtr( h );
- const cmRtNetNode_t* np;
-
- if((np = _cmRtNetIndexToRemoteNode( p, idx )) == NULL )
- return NULL;
-
- return np->label;
- }
-
- unsigned cmRtNetRemoteNodeEndPointCount( cmRtNetH_t h, unsigned nodeIdx )
- {
- const cmRtNetNode_t* np;
- const cmRtNetEnd_t* ep;
- cmRtNet_t* p = _cmRtNetHandleToPtr( h );
- unsigned n = 0;
-
- if((np = _cmRtNetIndexToRemoteNode( p, nodeIdx )) == NULL )
- return 0;
-
- for(ep=np->ends; ep!=NULL; ep=ep->link)
- ++n;
-
- return n;
- }
-
- cmRtNetRC_t cmRtNetRemoteNodeEndPoint(
- cmRtNetH_t h,
- unsigned nodeIdx,
- unsigned epIdx,
- const cmChar_t** labelRef,
- unsigned* idRef,
- unsigned* rsiRef )
- {
- const cmRtNetEnd_t* ep;
- cmRtNet_t* p = _cmRtNetHandleToPtr( h );
-
- if(( ep = _cmRtNetFindEndpt(p, nodeIdx, epIdx )) == NULL )
- {
- *labelRef = NULL;
- *idRef = cmInvalidId;
- *rsiRef = cmInvalidIdx;
- return kEndNotFoundNetRC;
- }
-
- *labelRef = ep->label;
- *idRef = ep->id;
- *rsiRef = ep->np->rtSubIdx;
-
- return kOkNetRC;
- }
-
-
-
- //==========================================================================
- #include "cmThread.h"
-
-
- typedef struct
- {
- cmThreadH_t thH;
- cmRtNetH_t netH;
- unsigned msgVal;
- } _cmRtNetTest_t;
-
- // This function is called within the context of cmRtNetReceive().
- void _cmRtNetTestRecv( void* cbArg, const char* data, unsigned dataByteCnt, const struct sockaddr_in* fromAddr )
- {
- //_cmRtNetTest_t* p = (_cmRtNetTest_t*)cbArg;
-
- cmRtNetMsg_t* r = (cmRtNetMsg_t*)data;
- unsigned i = *(unsigned*)(data + sizeof(cmRtNetMsg_t));
- printf("rtSubIdx:%i endptId:%i %i\n",r->hdr.rtSubIdx,r->dstEndPtId,i);
-
- }
-
-
- bool _cmRtNetTestThreadFunc(void* param)
- {
- _cmRtNetTest_t* p = (_cmRtNetTest_t*)param;
-
- if( cmRtNetIsValid(p->netH) )
- {
- cmRtNetReceive(p->netH);
- }
-
- cmSleepMs(40);
-
- return true;
- }
-
- void cmRtNetTest( cmCtx_t* ctx, bool mstrFl )
- {
- char c;
- _cmRtNetTest_t t;
- const unsigned rtSubIdx = 0;
- cmUdpPort_t port = 5876;
- _cmRtNetTest_t* p = &t;
- cmRtNetRC_t rc = kOkNetRC;
- const cmChar_t* localHostStr = mstrFl ? "master" : "slave";
- const cmChar_t* localEndpStr = mstrFl ? "master_ep" : "slave_ep";
- const cmChar_t* remoteHostStr = !mstrFl ? "master" : "slave";
- const cmChar_t* remoteEndpStr = !mstrFl ? "master_ep" : "slave_ep";
- const cmChar_t* bcastAddr = "192.168.15.255";
- cmRtNetEndptH_t eH = cmRtNetEndptNullHandle;
- unsigned srcEndPtId = cmInvalidId;
-
- memset(&t,0,sizeof(t));
-
- if( cmThreadCreate(&p->thH,_cmRtNetTestThreadFunc,p,&ctx->rpt) != kOkThRC )
- goto errLabel;
-
- if((rc = cmRtNetAlloc(ctx,&p->netH,rtSubIdx,_cmRtNetTestRecv,p)) != kOkNetRC )
- goto errLabel;
-
- cmRtNetReportSyncEnable(p->netH,true); // enable sync. protocol reporting
-
- if((rc = cmRtNetInitialize(p->netH, bcastAddr, localHostStr, NULL, port )) != kOkNetRC)
- goto errLabel;
-
- if((rc = cmRtNetRegisterEndPoint(p->netH,localEndpStr, 0 )) != kOkNetRC )
- goto errLabel;
-
- if((rc = cmRtNetEndpointHandle(p->netH, localHostStr, localEndpStr, &eH )) != kOkNetRC )
- goto errLabel;
-
- if((srcEndPtId = cmRtNetEndpointId(eH)) == cmInvalidIdx )
- goto errLabel;
-
- if( cmThreadPause(p->thH,0) != kOkThRC )
- goto errLabel;
-
- cmRptPrintf(&ctx->rpt,"%s t=transmit s=sync r=report q=quit\n", localHostStr );
-
- while( (c=getchar()) != 'q' )
- {
- switch(c)
- {
- case 'r':
- cmRtNetReport(p->netH);
- break;
-
- case 's':
- cmRtNetDoSync(p->netH);
- break;
-
- case 't':
- {
- if( cmRtNetSendByLabels(p->netH, srcEndPtId, remoteHostStr, remoteEndpStr, &p->msgVal, sizeof(p->msgVal)) == kOkNetRC )
- p->msgVal += 1;
-
- }
- break;
- }
-
- }
-
- errLabel:
-
- cmThreadDestroy(&p->thH);
-
- cmRtNetFree(&p->netH);
-
-
- return;
-
- }
|