You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
200 lines
6.3 KiB
C
200 lines
6.3 KiB
C
/*
|
|
* server_example_control.c
|
|
*
|
|
* How to use the different control handlers (TBD)
|
|
*/
|
|
|
|
#include "iec61850_server.h"
|
|
#include "hal_thread.h"
|
|
#include <signal.h>
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
|
|
#include "static_model.h"
|
|
|
|
static int running = 0;
|
|
static IedServer iedServer = NULL;
|
|
|
|
void
|
|
sigint_handler(int signalId)
|
|
{
|
|
running = 0;
|
|
}
|
|
|
|
static CheckHandlerResult
|
|
checkHandler(ControlAction action, void* parameter, MmsValue* ctlVal, bool test, bool interlockCheck)
|
|
{
|
|
ClientConnection clientCon = ControlAction_getClientConnection(action);
|
|
|
|
if (clientCon) {
|
|
printf("Control from client %s\n", ClientConnection_getPeerAddress(clientCon));
|
|
}
|
|
else {
|
|
printf("clientCon == NULL\n");
|
|
}
|
|
|
|
if (ControlAction_isSelect(action))
|
|
printf("check handler called by select command!\n");
|
|
else
|
|
printf("check handler called by operate command!\n");
|
|
|
|
if (interlockCheck)
|
|
printf(" with interlock check bit set!\n");
|
|
|
|
printf(" ctlNum: %i\n", ControlAction_getCtlNum(action));
|
|
|
|
if (parameter == IEDMODEL_GenericIO_GGIO1_SPCSO1)
|
|
return CONTROL_ACCEPTED;
|
|
|
|
if (parameter == IEDMODEL_GenericIO_GGIO1_SPCSO2)
|
|
return CONTROL_ACCEPTED;
|
|
|
|
if (parameter == IEDMODEL_GenericIO_GGIO1_SPCSO3)
|
|
return CONTROL_ACCEPTED;
|
|
|
|
if (parameter == IEDMODEL_GenericIO_GGIO1_SPCSO4)
|
|
return CONTROL_ACCEPTED;
|
|
|
|
if (parameter == IEDMODEL_GenericIO_GGIO1_SPCSO9)
|
|
return CONTROL_ACCEPTED;
|
|
|
|
return CONTROL_OBJECT_UNDEFINED;
|
|
}
|
|
|
|
static ControlHandlerResult
|
|
controlHandlerForBinaryOutput(ControlAction action, void* parameter, MmsValue* value, bool test)
|
|
{
|
|
uint64_t timestamp = Hal_getTimeInMs();
|
|
|
|
printf("control handler called\n");
|
|
printf(" ctlNum: %i\n", ControlAction_getCtlNum(action));
|
|
|
|
ClientConnection clientCon = ControlAction_getClientConnection(action);
|
|
|
|
if (clientCon) {
|
|
printf("Control from client %s\n", ClientConnection_getPeerAddress(clientCon));
|
|
}
|
|
else {
|
|
printf("clientCon == NULL!\n");
|
|
}
|
|
|
|
if (parameter == IEDMODEL_GenericIO_GGIO1_SPCSO1) {
|
|
IedServer_updateUTCTimeAttributeValue(iedServer, IEDMODEL_GenericIO_GGIO1_SPCSO1_t, timestamp);
|
|
IedServer_updateAttributeValue(iedServer, IEDMODEL_GenericIO_GGIO1_SPCSO1_stVal, value);
|
|
}
|
|
|
|
else if (parameter == IEDMODEL_GenericIO_GGIO1_SPCSO2) {
|
|
IedServer_updateUTCTimeAttributeValue(iedServer, IEDMODEL_GenericIO_GGIO1_SPCSO2_t, timestamp);
|
|
IedServer_updateAttributeValue(iedServer, IEDMODEL_GenericIO_GGIO1_SPCSO2_stVal, value);
|
|
}
|
|
|
|
else if (parameter == IEDMODEL_GenericIO_GGIO1_SPCSO3) {
|
|
IedServer_updateUTCTimeAttributeValue(iedServer, IEDMODEL_GenericIO_GGIO1_SPCSO3_t, timestamp);
|
|
IedServer_updateAttributeValue(iedServer, IEDMODEL_GenericIO_GGIO1_SPCSO3_stVal, value);
|
|
}
|
|
|
|
else if (parameter == IEDMODEL_GenericIO_GGIO1_SPCSO4) {
|
|
IedServer_updateUTCTimeAttributeValue(iedServer, IEDMODEL_GenericIO_GGIO1_SPCSO4_t, timestamp);
|
|
IedServer_updateAttributeValue(iedServer, IEDMODEL_GenericIO_GGIO1_SPCSO4_stVal, value);
|
|
}
|
|
else
|
|
return CONTROL_RESULT_FAILED;
|
|
|
|
return CONTROL_RESULT_OK;
|
|
}
|
|
|
|
static MmsDataAccessError
|
|
writeAccessHandler (DataAttribute* dataAttribute, MmsValue* value, ClientConnection connection, void* parameter)
|
|
{
|
|
ControlModel ctlModelVal = (ControlModel) MmsValue_toInt32(value);
|
|
|
|
/* we only allow status-only and direct-operate */
|
|
if ((ctlModelVal == CONTROL_MODEL_STATUS_ONLY) || (ctlModelVal == CONTROL_MODEL_DIRECT_NORMAL))
|
|
{
|
|
IedServer_updateCtlModel(iedServer, IEDMODEL_GenericIO_GGIO1_SPCSO1, ctlModelVal);
|
|
|
|
printf("Changed GGIO1.SPCSI.ctlModel to %i\n", ctlModelVal);
|
|
|
|
return DATA_ACCESS_ERROR_SUCCESS;
|
|
}
|
|
else {
|
|
return DATA_ACCESS_ERROR_OBJECT_VALUE_INVALID;
|
|
}
|
|
}
|
|
|
|
int
|
|
main(int argc, char** argv)
|
|
{
|
|
iedServer = IedServer_create(&iedModel);
|
|
int tcpPort = 102;
|
|
|
|
if (argc > 1) {
|
|
tcpPort = atoi(argv[1]);
|
|
}
|
|
|
|
IedServer_setControlHandler(iedServer, IEDMODEL_GenericIO_GGIO1_SPCSO1,
|
|
(ControlHandler) controlHandlerForBinaryOutput,
|
|
IEDMODEL_GenericIO_GGIO1_SPCSO1);
|
|
|
|
/*
|
|
* For SPCSO1 we want the user be able to change the control model by online service -
|
|
* so we install a write access handler to change the control model when the client
|
|
* writes to the "ctlModel" attribute.
|
|
*/
|
|
IedServer_handleWriteAccess(iedServer, IEDMODEL_GenericIO_GGIO1_SPCSO1_ctlModel, writeAccessHandler, NULL);
|
|
|
|
|
|
IedServer_setControlHandler(iedServer, IEDMODEL_GenericIO_GGIO1_SPCSO2,
|
|
(ControlHandler) controlHandlerForBinaryOutput,
|
|
IEDMODEL_GenericIO_GGIO1_SPCSO2);
|
|
|
|
/* this is optional - performs operative checks */
|
|
IedServer_setPerformCheckHandler(iedServer, IEDMODEL_GenericIO_GGIO1_SPCSO2, checkHandler,
|
|
IEDMODEL_GenericIO_GGIO1_SPCSO2);
|
|
|
|
IedServer_setControlHandler(iedServer, IEDMODEL_GenericIO_GGIO1_SPCSO3,
|
|
(ControlHandler) controlHandlerForBinaryOutput,
|
|
IEDMODEL_GenericIO_GGIO1_SPCSO3);
|
|
|
|
IedServer_setControlHandler(iedServer, IEDMODEL_GenericIO_GGIO1_SPCSO4,
|
|
(ControlHandler) controlHandlerForBinaryOutput,
|
|
IEDMODEL_GenericIO_GGIO1_SPCSO4);
|
|
|
|
/* this is optional - performs operative checks */
|
|
IedServer_setPerformCheckHandler(iedServer, IEDMODEL_GenericIO_GGIO1_SPCSO4, checkHandler,
|
|
IEDMODEL_GenericIO_GGIO1_SPCSO4);
|
|
|
|
|
|
IedServer_setControlHandler(iedServer, IEDMODEL_GenericIO_GGIO1_SPCSO9,
|
|
(ControlHandler) controlHandlerForBinaryOutput,
|
|
IEDMODEL_GenericIO_GGIO1_SPCSO9);
|
|
|
|
/* this is optional - performs operative checks */
|
|
IedServer_setPerformCheckHandler(iedServer, IEDMODEL_GenericIO_GGIO1_SPCSO9, checkHandler,
|
|
IEDMODEL_GenericIO_GGIO1_SPCSO9);
|
|
|
|
/* MMS server will be instructed to start listening to client connections. */
|
|
IedServer_start(iedServer, tcpPort);
|
|
|
|
if (!IedServer_isRunning(iedServer)) {
|
|
printf("Starting server failed! Exit.\n");
|
|
IedServer_destroy(iedServer);
|
|
exit(-1);
|
|
}
|
|
|
|
running = 1;
|
|
|
|
signal(SIGINT, sigint_handler);
|
|
|
|
while (running) {
|
|
Thread_sleep(1);
|
|
}
|
|
|
|
/* stop MMS server - close TCP server socket and all client sockets */
|
|
IedServer_stop(iedServer);
|
|
|
|
/* Cleanup - free all resources */
|
|
IedServer_destroy(iedServer);
|
|
return 0;
|
|
} /* main() */
|