src/lifegrd.c
changeset 0 4472ee7c6c3e
child 53 73bc47a0db55
equal deleted inserted replaced
-1:000000000000 0:4472ee7c6c3e
       
     1 /*
       
     2 This file is part of CanFestival, a library implementing CanOpen Stack. 
       
     3 
       
     4 Copyright (C): Edouard TISSERANT and Francis DUPIN
       
     5 
       
     6 See COPYING file for copyrights details.
       
     7 
       
     8 This library is free software; you can redistribute it and/or
       
     9 modify it under the terms of the GNU Lesser General Public
       
    10 License as published by the Free Software Foundation; either
       
    11 version 2.1 of the License, or (at your option) any later version.
       
    12 
       
    13 This library is distributed in the hope that it will be useful,
       
    14 but WITHOUT ANY WARRANTY; without even the implied warranty of
       
    15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
       
    16 Lesser General Public License for more details.
       
    17 
       
    18 You should have received a copy of the GNU Lesser General Public
       
    19 License along with this library; if not, write to the Free Software
       
    20 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
       
    21 */
       
    22 
       
    23 #include <data.h>
       
    24 #include "lifegrd.h"
       
    25 
       
    26 e_nodeState getNodeState (CO_Data* d, UNS8 nodeId)
       
    27 {
       
    28 	e_nodeState networkNodeState = d->NMTable[nodeId];
       
    29 	return networkNodeState;
       
    30 }
       
    31 
       
    32 // The Consumer Timer Callback
       
    33 void ConsumerHearbeatAlarm(CO_Data* d, UNS32 id)
       
    34 {
       
    35         //MSG_WAR(0x00, "ConsumerHearbeatAlarm", 0x00);
       
    36 	
       
    37 	// call heartbeat error with NodeId
       
    38 	(*d->heartbeatError)((UNS8)( ((d->ConsumerHeartbeatEntries[id]) & (UNS32)0x00FF0000) >> (UNS8)16 ));
       
    39 }
       
    40 
       
    41 /* Retourne le node-id */
       
    42 void proceedNODE_GUARD(CO_Data* d, Message* m )
       
    43 {
       
    44   UNS8 nodeId = (UNS8) GET_NODE_ID((*m));
       
    45   
       
    46   if((m->rtr == 1) ) /* Notice that only the master can have sent this node guarding request */
       
    47   { // Receiving a NMT NodeGuarding (request of the state by the master)
       
    48     //  only answer to the NMT NodeGuarding request, the master is not checked (not implemented)
       
    49     if (nodeId == *d->bDeviceNodeId )
       
    50     {
       
    51       Message msg;
       
    52       msg.cob_id.w = *d->bDeviceNodeId + 0x700;
       
    53       msg.len = (UNS8)0x01;
       
    54       msg.rtr = 0;
       
    55       msg.data[0] = d->nodeState; 
       
    56       if (d->toggle)
       
    57       {
       
    58         msg.data[0] |= 0x80 ;
       
    59         d->toggle = 0 ;
       
    60       }
       
    61       else
       
    62         d->toggle = 1 ; 
       
    63       // send the nodeguard response.
       
    64       MSG_WAR(0x3130, "Sending NMT Nodeguard to master, state: ", d->nodeState);
       
    65       (*d->canSend)(&msg );
       
    66     }  
       
    67 
       
    68   }else{ // Not a request CAN  
       
    69 
       
    70     MSG_WAR(0x3110, "Received NMT nodeId : ", nodeId);
       
    71     /* the slave's state receievd is stored in the NMTable */
       
    72       // The state is stored on 7 bit
       
    73     d->NMTable[nodeId] = (e_nodeState) ((*m).data[0] & 0x7F) ;
       
    74     
       
    75     /* Boot-Up frame reception */
       
    76     if ( d->NMTable[nodeId] == Initialisation)
       
    77       {
       
    78         // The device send the boot-up message (Initialisation)
       
    79         // to indicate the master that it is entered in pre_operational mode
       
    80         // Because the  device enter automaticaly in pre_operational mode,
       
    81         // the pre_operational mode is stored 
       
    82 //        NMTable[bus_id][nodeId] = Pre_operational;
       
    83         MSG_WAR(0x3100, "The NMT is a bootup from node : ", nodeId);
       
    84       }
       
    85       
       
    86     if( d->NMTable[nodeId] != Unknown_state ) {
       
    87         UNS8 index, ConsummerHeartBeat_nodeId ;
       
    88         for( index = (UNS8)0x00; index < *d->ConsumerHeartbeatCount; index++ )
       
    89         {
       
    90             ConsummerHeartBeat_nodeId = (UNS8)( ((d->ConsumerHeartbeatEntries[index]) & (UNS32)0x00FF0000) >> (UNS8)16 );
       
    91             if ( nodeId == ConsummerHeartBeat_nodeId )
       
    92             {
       
    93                 TIMEVAL time = ( (d->ConsumerHeartbeatEntries[index]) & (UNS32)0x0000FFFF ) ;
       
    94             	// Renew alarm for next heartbeat.
       
    95             	DelAlarm(d->ConsumerHeartBeatTimers[index]);
       
    96             	d->ConsumerHeartBeatTimers[index] = SetAlarm(d, index, &ConsumerHearbeatAlarm, MS_TO_TIMEVAL(time), 0);
       
    97             }
       
    98         }
       
    99     }
       
   100   }
       
   101 }
       
   102 
       
   103 // The Consumer Timer Callback
       
   104 void ProducerHearbeatAlarm(CO_Data* d, UNS32 id)
       
   105 {
       
   106 	if(*d->ProducerHeartBeatTime)
       
   107 	{
       
   108 		Message msg;
       
   109 		// Time expired, the heartbeat must be sent immediately
       
   110 		// generate the correct node-id: this is done by the offset 1792
       
   111 		// (decimal) and additionaly
       
   112 		// the node-id of this device.
       
   113 		msg.cob_id.w = *d->bDeviceNodeId + 0x700;
       
   114 		msg.len = (UNS8)0x01;
       
   115 		msg.rtr = 0;
       
   116 		msg.data[0] = d->nodeState; // No toggle for heartbeat !
       
   117 		// send the heartbeat
       
   118       		MSG_WAR(0x3130, "Producing heartbeat: ", d->nodeState);
       
   119       		(*d->canSend)(&msg );
       
   120 	}else{
       
   121 		d->ProducerHeartBeatTimer = DelAlarm(d->ProducerHeartBeatTimer);
       
   122 	}
       
   123 }
       
   124 
       
   125 
       
   126 void heartbeatInit(CO_Data* d)
       
   127 {
       
   128     UNS8 index; // Index to scan the table of heartbeat consumers
       
   129 
       
   130     d->toggle = 0;
       
   131 
       
   132     for( index = (UNS8)0x00; index < *d->ConsumerHeartbeatCount; index++ )
       
   133     {
       
   134         TIMEVAL time = (UNS16) ( (d->ConsumerHeartbeatEntries[index]) & (UNS32)0x0000FFFF ) ;
       
   135         //MSG_WAR(0x3121, "should_time : ", should_time ) ;
       
   136         if ( time )
       
   137         {
       
   138             	d->ConsumerHeartBeatTimers[index] = SetAlarm(d, index, &ConsumerHearbeatAlarm, MS_TO_TIMEVAL(time), 0);
       
   139         }
       
   140     }
       
   141 
       
   142     if ( *d->ProducerHeartBeatTime )
       
   143     {
       
   144     	TIMEVAL time = *d->ProducerHeartBeatTime;
       
   145     	d->ProducerHeartBeatTimer = SetAlarm(d, 0, &ProducerHearbeatAlarm, MS_TO_TIMEVAL(time), MS_TO_TIMEVAL(time));
       
   146     }
       
   147 }
       
   148 
       
   149 
       
   150 void heartbeatStop(CO_Data* d)
       
   151 {
       
   152     UNS8 index;
       
   153     for( index = (UNS8)0x00; index < *d->ConsumerHeartbeatCount; index++ )
       
   154     {
       
   155         d->ConsumerHeartBeatTimers[index + 1] = DelAlarm(d->ConsumerHeartBeatTimers[index + 1]);;
       
   156     }
       
   157 
       
   158     d->ProducerHeartBeatTimer = DelAlarm(d->ProducerHeartBeatTimer);;
       
   159 }
       
   160