examples/CANOpenShell/CANOpenShell.c
author greg
Wed, 15 Jul 2009 17:14:44 +0200
changeset 558 bcf69ff203de
parent 555 ee24dcbd3e64
child 559 8543e7edae4a
permissions -rwxr-xr-x
Add build command for master OD and slave OD
/*
This file is part of CanFestival, a library implementing CanOpen Stack.

Copyright (C): Edouard TISSERANT and Francis DUPIN

See COPYING file for copyrights details.

This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version.

This library 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
Lesser General Public License for more details.

You should have received a copy of the GNU Lesser General Public
License along with this library; if not, write to the Free Software
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
*/


#if defined(WIN32) && !defined(__CYGWIN__)
	#include <windows.h>
	#define CLEARSCREEN "cls"
	#define SLEEP(time) Sleep(time * 1000)
#else
	#include <unistd.h>
	#include <stdio.h>
	#include <string.h>
	#include <stdlib.h>
	#include <signal.h>
	#define CLEARSCREEN "clear"
	#define SLEEP(time) sleep(time)
#endif

//****************************************************************************
// INCLUDES
#include "canfestival.h"
#include "CANOpenShell.h"
#include "CANOpenShellMasterOD.h"
#include "CANOpenShellSlaveOD.h"

//****************************************************************************
// DEFINES
#define MAX_NODES 127
#define cst_str4(c1, c2, c3, c4) ((((unsigned int)0 | \
                                    (char)c4 << 8) | \
                                   (char)c3) << 8 | \
                                  (char)c2) << 8 | \
                                 (char)c1

#define INIT_ERR 2
#define QUIT 1

//****************************************************************************
// GLOBALS
char BoardBusName[11];
char BoardBaudRate[5];
s_BOARD Board = {BoardBusName, BoardBaudRate};
CO_Data* CANOpenShellOD_Data;
int init_step = 0;
char LibraryPath[512];

/*****************************************************************************/
#if !defined(WIN32) || defined(__CYGWIN__)
void catch_signal(int sig)
{
	signal(SIGTERM, catch_signal);
	signal(SIGINT, catch_signal);
	printf("Got Signal %d\n",sig);
}
#endif

/* Sleep for n seconds */
void SleepFunction(int second)
{
	SLEEP(second);
}

/* Ask a slave node to go in operational mode */
void StartNode(CO_Data* d, UNS8 nodeid)
{
	EnterMutex();
	masterSendNMTstateChange(d, nodeid, NMT_Start_Node);
	LeaveMutex();
}

/* Ask a slave node to go in pre-operational mode */
void StopNode(CO_Data* d, UNS8 nodeid)
{
	EnterMutex();
	masterSendNMTstateChange(d, nodeid, NMT_Stop_Node);
	LeaveMutex();
}

/* Ask a slave node to reset */
void ResetNode(CO_Data* d, UNS8 nodeid)
{
	EnterMutex();
	masterSendNMTstateChange(d, nodeid, NMT_Reset_Node);
	LeaveMutex();
}

/* Reset all nodes on the network and print message when boot-up*/
void DiscoverNodes(CO_Data* d)
{
	printf("Wait for Slave nodes bootup...\n\n");
	ResetNode(CANOpenShellOD_Data, 0x0);
}

/* Callback function that check the read SDO demand */
void CheckReadInfoSDO(CO_Data* d, UNS8 nodeid)
{
	UNS32 abortCode;
	UNS32 data;
	UNS32 size=64;

	if(getReadResultNetworkDict(d, nodeid, &data, &size, &abortCode) != SDO_FINISHED)
		printf("Master : Failed in getting information for slave %2.2x, AbortCode :%4.4x \n", nodeid, abortCode);

	/* Finalise last SDO transfer with this node */
	closeSDOtransfer(CANOpenShellOD_Data, nodeid, SDO_CLIENT);

	/* Display data received */
	switch(init_step)
	{
		case 1:
				printf("Device type     : %x\n", data);
				break;
		case 2:
				printf("Vendor ID       : %x\n", data);
				break;
		case 3:
				printf("Product Code    : %x\n", data);
				break;
		case 4:
				printf("Revision Number : %x\n", data);
				break;
	}
	GetSlaveNodeInfo(d, nodeid);
}

/* Retrieve node informations located at index 0x1000 (Device Type) and 0x1018 (Identity) */
void GetSlaveNodeInfo(CO_Data* d, UNS8 nodeid)
{
		switch(++init_step)
		{
			case 1: /* Get device type */
				printf("##################################\n");
				printf("#### Informations for node %x ####\n", nodeid);
				printf("##################################\n");
				readNetworkDictCallback(d, nodeid, 0x1000, 0x00, 0, CheckReadInfoSDO);
				break;

			case 2: /* Get Vendor ID */
				readNetworkDictCallback(d, nodeid, 0x1018, 0x01, 0, CheckReadInfoSDO);
				break;

			case 3: /* Get Product Code */
				readNetworkDictCallback(d, nodeid, 0x1018, 0x02, 0, CheckReadInfoSDO);
				break;

			case 4: /* Get Revision Number */
				readNetworkDictCallback(d, nodeid, 0x1018, 0x03, 0, CheckReadInfoSDO);
				break;

			case 5: /* Print node info */
				init_step = 0;
		}
}

/* Callback function that check the read SDO demand */
void CheckReadSDO(CO_Data* d, UNS8 nodeid)
{
	UNS32 abortCode;
	UNS32 data;
	UNS32 size=64;

	if(getReadResultNetworkDict(d, nodeid, &data, &size, &abortCode) != SDO_FINISHED)
		printf("\nResult : Failed in getting information for slave %2.2x, AbortCode :%4.4x \n", nodeid, abortCode);
	else
		printf("\nResult : = %x\n", data);

	/* Finalise last SDO transfer with this node */
	closeSDOtransfer(CANOpenShellOD_Data, nodeid, SDO_CLIENT);
}

/* Read a slave node object dictionary entry */
void ReadDeviceEntry(CO_Data* d, char* sdo)
{
	int nodeid;
	int index;
	int subindex;
	int size;
	int datatype = 0;

	sscanf(sdo, "%2x%4x%2x", &nodeid, &index, &subindex);

	printf("##################################\n");
	printf("#### Read SDO                 ####\n");
	printf("##################################\n");
	printf("NodeId   : %2.2x\n", nodeid);
	printf("Index    : %4.4x\n", index);
	printf("SubIndex : %2.2x\n", subindex);

	readNetworkDictCallback(d, (UNS8)nodeid, (UNS16)index, (UNS8)subindex, (UNS8)datatype, CheckReadSDO);
}

/* Callback function that check the write SDO demand */
void CheckWriteSDO(CO_Data* d, UNS8 nodeid)
{
	UNS32 abortCode;

	if(getWriteResultNetworkDict(d, nodeid, &abortCode) != SDO_FINISHED)
		printf("\nResult : Failed in getting information for slave %2.2x, AbortCode :%4.4x \n", nodeid, abortCode);
	else
		printf("\nSend data OK\n");

	/* Finalise last SDO transfer with this node */
	closeSDOtransfer(CANOpenShellOD_Data, nodeid, SDO_CLIENT);
}

/* Write a slave node object dictionnary entry */
void WriteDeviceEntry(CO_Data* d, char* sdo)
{
	int nodeid;
	int index;
	int subindex;
	int size;
	int data;

	sscanf(sdo, "%2x%4x%2x%2x%4x", &nodeid , &index, &subindex, &size, &data);

	printf("##################################\n");
	printf("#### Write SDO                ####\n");
	printf("##################################\n");
	printf("NodeId   : %2.2x\n", nodeid);
	printf("Index    : %4.4x\n", index);
	printf("SubIndex : %2.2x\n", subindex);
	printf("Size     : %2.2x\n", size);
	printf("Data     : %x\n", data);

	writeNetworkDictCallBackAI(d, nodeid, index, subindex, size, 0, &data, CheckWriteSDO, 1);
}

void CANOpenShellOD_post_SlaveBootup(CO_Data* d, UNS8 nodeid)
{
	printf("Slave %x boot up\n", nodeid);
}

/***************************  CALLBACK FUNCTIONS  *****************************************/
void CANOpenShellOD_initialisation(CO_Data* d)
{
	//printf("Master_initialisation\n");
}

void CANOpenShellOD_preOperational(CO_Data* d)
{
	//printf("Master_preOperational\n");
}

void CANOpenShellOD_operational(CO_Data* d)
{
	//printf("Master_operational\n");
}

void CANOpenShellOD_stopped(CO_Data* d)
{
	//printf("Master_stopped\n");
}

void CANOpenShellOD_post_sync(CO_Data* d)
{
	//printf("Master_post_sync\n");
}

void CANOpenShellOD_post_TPDO(CO_Data* d)
{
	//printf("Master_post_TPDO\n");
}

/***************************  MASTER INITIALISATION **********************************/
void Init(CO_Data* d, UNS32 nodeid)
{
	if(Board.baudrate)
	{
		/* Defining the node Id */
		setNodeId(CANOpenShellOD_Data, nodeid);

		/* Init */
		setState(CANOpenShellOD_Data, Initialisation);
	}
}

/***************************  MASTER CLEANUP  *****************************************/
void Exit(CO_Data* d, UNS32 nodeid)
{
	if(strcmp(Board.baudrate, "none"))
	{
		/* Reset all nodes on the network */
		masterSendNMTstateChange(CANOpenShellOD_Data, (UNS8)nodeid, NMT_Reset_Node);

		/* Stop master */
		setState(CANOpenShellOD_Data, Stopped);
	}
}

int ExtractNodeId(char *command) {
	int nodeid;
	sscanf(command, "%2x", &nodeid);
	return nodeid;
}

int NodeInit(CO_Data* d, int DeviceNodeID, int DeviceIsMaster)
{
	if(DeviceIsMaster)
	{
		CANOpenShellOD_Data = &CANOpenShellMasterOD_Data;
	}
	else
	{
		CANOpenShellOD_Data = &CANOpenShellSlaveOD_Data;
	}

	/* Load can library */
	LoadCanDriver(LibraryPath);

	/* Init stack timer */
	TimerInit();

	/* Define callback functions */
	CANOpenShellOD_Data->initialisation = CANOpenShellOD_initialisation;
	CANOpenShellOD_Data->preOperational = CANOpenShellOD_preOperational;
	CANOpenShellOD_Data->operational = CANOpenShellOD_operational;
	CANOpenShellOD_Data->stopped = CANOpenShellOD_stopped;
	CANOpenShellOD_Data->post_sync = CANOpenShellOD_post_sync;
	CANOpenShellOD_Data->post_TPDO = CANOpenShellOD_post_TPDO;
	CANOpenShellOD_Data->post_SlaveBootup=CANOpenShellOD_post_SlaveBootup;

	/* Open the Peak CANOpen device */
	if(!canOpen(&Board,CANOpenShellOD_Data)) return 1;

	/* Start Timer thread */
	StartTimerLoop(&Init);
	return 0;
}

void help_menu(void)
{
	printf("   MANDATORY COMMAND (must be the first command):\n");
	printf("      load#Can library path,channel,baudrate,device nodeid,device type (m for master, s for slave)\n");
	printf("\n");
	printf("   NETWORK: (if nodeid=0x00 : broadcast)\n");
	printf("     ssta#[nodeid] : Start a node\n");
	printf("     ssto#[nodeid] : Stop a node\n");
	printf("     srst#[nodeid] : Reset a node\n");
	printf("     scan : Reset all nodes and print message when bootup\n");
	printf("     wait#[seconds] : Sleep for n seconds\n");
	printf("\n");
	printf("   SDO: (size parameter : nb BYTES)\n");
	printf("     info#[nodeid]\n");
	printf("     rsdo#[nodeid][index][subindex] : read sdo\n");
	printf("     wsdo#[nodeid][index][subindex][size][data] : write sdo\n");
	printf("\n");
	printf("     help : Display this menu\n");
	printf("     quit : Quit application\n");
	printf("\n");
	printf("\n");
}

int ProcessCommand(char* command)
{
	int ret = 0;
	int sec = 0;
	int DeviceNodeID;
	int DeviceType;

	switch(cst_str4(command[0], command[1], command[2], command[3]))
	{
		case cst_str4('l', 'o', 'a', 'd') : /* Library Interface*/
					ret = sscanf(command, "load#%100[^,],%10[^,],%4[^,],%d,%d",
							LibraryPath,
							BoardBusName,
							BoardBaudRate,
							&DeviceNodeID,
							&DeviceType);

					if(ret == 5)
					{
						ret = NodeInit(CANOpenShellOD_Data, DeviceNodeID, DeviceType);
						return ret;
					}
					else{
						help_menu();
						exit(1);
					}
					break;
		case cst_str4('h', 'e', 'l', 'p') : /* Display Help*/
					help_menu();
					break;
		case cst_str4('s', 's', 't', 'a') : /* Slave Start*/
					StartNode(CANOpenShellOD_Data, ExtractNodeId(command + 5));
					break;
		case cst_str4('s', 's', 't', 'o') : /* Slave Stop */
					StopNode(CANOpenShellOD_Data, ExtractNodeId(command + 5));
					break;
		case cst_str4('s', 'r', 's', 't') : /* Slave Reset */
					ResetNode(CANOpenShellOD_Data, ExtractNodeId(command + 5));
					break;
		case cst_str4('i', 'n', 'f', 'o') : /* Retrieve node informations */
					GetSlaveNodeInfo(CANOpenShellOD_Data, ExtractNodeId(command + 5));
					break;
		case cst_str4('r', 's', 'd', 'o') : /* Read device entry */
					ReadDeviceEntry(CANOpenShellOD_Data, command + 5);
					break;
		case cst_str4('w', 's', 'd', 'o') : /* Write device entry */
					WriteDeviceEntry(CANOpenShellOD_Data, command + 5);
					break;
		case cst_str4('s', 'c', 'a', 'n') : /* Display master node state */
					DiscoverNodes(CANOpenShellOD_Data);
					break;
		case cst_str4('w', 'a', 'i', 't') : /* Display master node state */
					ret = sscanf(command, "wait=%d", &sec);
					if(ret == 1) SleepFunction(sec);
					break;
		case cst_str4('q', 'u', 'i', 't') : /* Quit application */
					return QUIT;
		default :
					help_menu();
	}
	return 0;
}

/****************************************************************************/
/***************************  MAIN  *****************************************/
/****************************************************************************/

int main(int argc, char** argv)
{
	extern char *optarg;
	char command[20];
	char* res;
	int ret=0;
	int sysret=0;
	int i=0;

	/* Print help and exit immediatly*/
	if(argc < 1)
	{
		help_menu();
		exit(1);
	}

	/* Strip command-line*/
	for(i=1 ; i<argc ; i++)
	{
		if(ProcessCommand(argv[i]) == INIT_ERR) goto init_fail;
	}

	#if !defined(WIN32) || defined(__CYGWIN__)
		/* install signal handler for manual break */
		signal(SIGTERM, catch_signal);
		signal(SIGINT, catch_signal);
	#endif

	/* Enter in a loop to read stdin command until "quit" is called */
	while(ret != QUIT)
	{
		// wait on stdin for string command
		res = fgets(command, sizeof(command), stdin);
		sysret = system(CLEARSCREEN);
		ret = ProcessCommand(command);
		fflush(stdout);
	}

	printf("Finishing.\n");

	// Stop timer thread
	StopTimerLoop(&Exit);

init_fail:
	TimerCleanup();
	return 0;
}