IC_104/lib60870-C/examples/cs101_slave/slave_example.c
George 1ddf693260 feat:
--READ smash_read_this.txt
2026-06-22 16:04:13 +05:30

421 lines
13 KiB
C

/*
* slave_example.c
*
* Example CS101 slave
*
*/
#include <signal.h>
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "cs101_slave.h"
#include "hal_serial.h"
#include "hal_thread.h"
#include "hal_time.h"
static bool running = true;
void
sigint_handler(int signalId)
{
running = false;
}
void
printCP56Time2a(CP56Time2a time)
{
printf("%02i:%02i:%02i %02i/%02i/%04i", CP56Time2a_getHour(time), CP56Time2a_getMinute(time),
CP56Time2a_getSecond(time), CP56Time2a_getDayOfMonth(time), CP56Time2a_getMonth(time),
CP56Time2a_getYear(time) + 2000);
}
/* Callback handler to log sent or received messages (optional) */
static void
rawMessageHandler(void* parameter, uint8_t* msg, int msgSize, bool sent)
{
if (sent)
{
printf("SEND: ");
}
else
{
printf("RCVD: ");
}
int i;
for (i = 0; i < msgSize; i++)
{
printf("%02x ", msg[i]);
}
printf("\n");
}
/* Callback handler that is called when a clock synchronization command is received */
static bool
clockSyncHandler(void* parameter, IMasterConnection connection, CS101_ASDU asdu, CP56Time2a newTime)
{
printf("Process time sync command with time ");
printCP56Time2a(newTime);
printf("\n");
return true;
}
static Semaphore gi_stateLock = NULL;
static int gi_state = 0; /* 0 = IDLE, 1 = IN_PROGRESS */
static IMasterConnection gi_connection = NULL;
static int gi_progress = 0;
static sCS101_StaticASDU gi_asdu;
static uint8_t gi_oa = 0;
static CS101_ASDU
getNextInterrogationASDUHandler(void* parameter, IMasterConnection connection)
{
CS101_ASDU retVal = NULL;
Semaphore_wait(gi_stateLock);
if (gi_state == 1)
{
CS101_AppLayerParameters alParams = IMasterConnection_getApplicationLayerParameters(connection);
if (gi_progress == 0)
{
CS101_ASDU newAsdu = CS101_ASDU_initializeStatic(&gi_asdu, alParams, false, CS101_COT_INTERROGATED_BY_STATION,
gi_oa, 1, false, false);
InformationObject io = (InformationObject) MeasuredValueScaled_create(NULL, 100, -1, IEC60870_QUALITY_GOOD);
CS101_ASDU_addInformationObject(newAsdu, io);
CS101_ASDU_addInformationObject(newAsdu, (InformationObject)
MeasuredValueScaled_create((MeasuredValueScaled) io, 101, 23, IEC60870_QUALITY_GOOD));
CS101_ASDU_addInformationObject(newAsdu, (InformationObject)
MeasuredValueScaled_create((MeasuredValueScaled) io, 102, 2300, IEC60870_QUALITY_GOOD));
InformationObject_destroy(io);
gi_progress++;
retVal = (CS101_ASDU)&gi_asdu;
}
else if (gi_progress == 1)
{
CS101_ASDU newAsdu = CS101_ASDU_initializeStatic(&gi_asdu, alParams, false, CS101_COT_INTERROGATED_BY_STATION,
gi_oa, 1, false, false);
InformationObject io = (InformationObject) SinglePointInformation_create(NULL, 104, true, IEC60870_QUALITY_GOOD);
CS101_ASDU_addInformationObject(newAsdu, io);
CS101_ASDU_addInformationObject(newAsdu, (InformationObject)
SinglePointInformation_create((SinglePointInformation) io, 105, false, IEC60870_QUALITY_GOOD));
InformationObject_destroy(io);
gi_progress++;
retVal = (CS101_ASDU)&gi_asdu;
}
else if (gi_progress == 2)
{
CS101_ASDU newAsdu = CS101_ASDU_initializeStatic(&gi_asdu, alParams, true, CS101_COT_INTERROGATED_BY_STATION,
gi_oa, 1, false, false);
InformationObject io = NULL;
CS101_ASDU_addInformationObject(newAsdu, io = (InformationObject) SinglePointInformation_create(NULL, 300, true, IEC60870_QUALITY_GOOD));
CS101_ASDU_addInformationObject(newAsdu, (InformationObject) SinglePointInformation_create((SinglePointInformation) io, 301, false, IEC60870_QUALITY_GOOD));
CS101_ASDU_addInformationObject(newAsdu, (InformationObject) SinglePointInformation_create((SinglePointInformation) io, 302, true, IEC60870_QUALITY_GOOD));
CS101_ASDU_addInformationObject(newAsdu, (InformationObject) SinglePointInformation_create((SinglePointInformation) io, 303, false, IEC60870_QUALITY_GOOD));
CS101_ASDU_addInformationObject(newAsdu, (InformationObject) SinglePointInformation_create((SinglePointInformation) io, 304, true, IEC60870_QUALITY_GOOD));
CS101_ASDU_addInformationObject(newAsdu, (InformationObject) SinglePointInformation_create((SinglePointInformation) io, 305, false, IEC60870_QUALITY_GOOD));
CS101_ASDU_addInformationObject(newAsdu, (InformationObject) SinglePointInformation_create((SinglePointInformation) io, 306, true, IEC60870_QUALITY_GOOD));
CS101_ASDU_addInformationObject(newAsdu, (InformationObject) SinglePointInformation_create((SinglePointInformation) io, 307, false, IEC60870_QUALITY_GOOD));
InformationObject_destroy(io);
gi_progress++;
retVal = (CS101_ASDU)&gi_asdu;
}
else if (gi_progress == 3)
{
CS101_ASDU newAsdu = CS101_ASDU_initializeStatic(&gi_asdu, alParams, true, CS101_COT_INTERROGATED_BY_STATION,
gi_oa, 1, false, false);
InformationObject io = (InformationObject) BitString32_create(NULL, 500, 0xaaaa);
CS101_ASDU_addInformationObject(newAsdu, io);
InformationObject_destroy(io);
gi_progress++;
retVal = (CS101_ASDU)&gi_asdu;
}
else if (gi_progress == 4)
{
/* send ACT_TERM */
CS101_ASDU newAsdu = CS101_ASDU_initializeStatic(&gi_asdu, alParams, false, CS101_COT_ACTIVATION_TERMINATION,
gi_oa, 1, false, false);
InformationObject io = (InformationObject) InterrogationCommand_create(NULL, 0, 20);
CS101_ASDU_addInformationObject(newAsdu, io);
InformationObject_destroy(io);
/* this one we do not return but use the sendASDU function to put it into the class 1 queue */
IMasterConnection_sendASDU(connection, newAsdu);
/* finished */
gi_state = 0;
gi_connection = NULL;
gi_progress = 0;
}
}
Semaphore_post(gi_stateLock);
return retVal;
}
/* Callback handler that is called when an interrogation command is received */
static bool
interrogationHandler(void* parameter, IMasterConnection connection, CS101_ASDU asdu, uint8_t qoi)
{
printf("Received interrogation for group %i\n", qoi);
int ca = CS101_ASDU_getCA(asdu);
if (ca == 1) /* only handle interrogation for CA(SDU) 1 */
{
if (qoi == 20) /* only handle station interrogation */
{
Semaphore_wait(gi_stateLock);
gi_state = 1;
gi_connection = connection;
gi_progress = 0;
gi_oa = (uint8_t)CS101_ASDU_getOA(asdu);
/* this will put the ACT-CON ASDU into the class 1 queue */
IMasterConnection_sendACT_CON(connection, asdu, false);
/* the remaining GI data has to be sent as class 2 data in the getNextInterrogationASDUHandler callback */
/* the final ACT-TERM message has to sent as class 1 data when the GI process is finished */
Semaphore_post(gi_stateLock);
}
else
{
IMasterConnection_sendACT_CON(connection, asdu, true);
}
}
else
{
/* send error response */
CS101_ASDU_setCOT(asdu, CS101_COT_UNKNOWN_CA);
CS101_ASDU_setNegative(asdu, true);
IMasterConnection_sendASDU(connection, asdu);
}
return true;
}
static bool
asduHandler(void* parameter, IMasterConnection connection, CS101_ASDU asdu)
{
if (CS101_ASDU_getTypeID(asdu) == C_SC_NA_1)
{
printf("received single command\n");
if (CS101_ASDU_getCOT(asdu) == CS101_COT_ACTIVATION)
{
InformationObject io = CS101_ASDU_getElement(asdu, 0);
if (io)
{
if (InformationObject_getObjectAddress(io) == 5000)
{
SingleCommand sc = (SingleCommand)io;
printf("IOA: %i switch to %i\n", InformationObject_getObjectAddress(io),
SingleCommand_getState(sc));
CS101_ASDU_setCOT(asdu, CS101_COT_ACTIVATION_CON);
}
else
CS101_ASDU_setCOT(asdu, CS101_COT_UNKNOWN_IOA);
InformationObject_destroy(io);
}
else
{
printf("ERROR: ASDU contains no information object!\n");
return true;
}
}
else
CS101_ASDU_setCOT(asdu, CS101_COT_UNKNOWN_COT);
IMasterConnection_sendASDU(connection, asdu);
return true;
}
return false;
}
static void
resetCUHandler(void* parameter)
{
printf("Received reset CU\n");
CS101_Slave_flushQueues((CS101_Slave)parameter);
}
static void
linkLayerStateChanged(void* parameter, int address, LinkLayerState state)
{
printf("Link layer state: ");
switch (state)
{
case LL_STATE_IDLE:
printf("IDLE\n");
break;
case LL_STATE_ERROR:
printf("ERROR\n");
break;
case LL_STATE_BUSY:
printf("BUSY\n");
break;
case LL_STATE_AVAILABLE:
printf("AVAILABLE\n");
break;
}
}
int
main(int argc, char** argv)
{
/* Add Ctrl-C handler */
signal(SIGINT, sigint_handler);
const char* serialPort = "/dev/ttyUSB0";
if (argc > 1)
serialPort = argv[1];
SerialPort port = SerialPort_create(serialPort, 9600, 8, 'E', 1);
if (!port)
{
fprintf(stderr, "Failed to create serial port\n");
return 1;
}
gi_stateLock = Semaphore_create(1);
/* create a new slave/server instance with default link layer and application layer parameters */
//CS101_Slave slave = CS101_Slave_create(port, NULL, NULL, IEC60870_LINK_LAYER_BALANCED);
CS101_Slave slave = CS101_Slave_create(port, NULL, NULL, IEC60870_LINK_LAYER_UNBALANCED);
CS101_Slave_setLinkLayerAddress(slave, 1);
CS101_Slave_setLinkLayerAddressOtherStation(slave, 1);
/* get the application layer parameters - we need them to create correct ASDUs */
CS101_AppLayerParameters alParameters = CS101_Slave_getAppLayerParameters(slave);
/* change default application layer parameters (optional) */
alParameters->sizeOfCA = 2;
alParameters->sizeOfIOA = 3;
alParameters->sizeOfCOT = 2;
LinkLayerParameters llParameters = CS101_Slave_getLinkLayerParameters(slave);
llParameters->timeoutForAck = 500;
llParameters->addressLength = 1;
/* set the callback handler for the clock synchronization command */
CS101_Slave_setClockSyncHandler(slave, clockSyncHandler, NULL);
/* set the callback handler for the interrogation command */
CS101_Slave_setInterrogationHandler(slave, interrogationHandler, NULL);
/* set handler for other message types */
CS101_Slave_setASDUHandler(slave, asduHandler, NULL);
/* set handler for reset CU (reset communication unit) message */
CS101_Slave_setResetCUHandler(slave, resetCUHandler, (void*)slave);
CS101_Slave_setGetNextInterrogationASDUHandler(slave, getNextInterrogationASDUHandler, slave);
/* set timeout for detecting connection loss */
CS101_Slave_setIdleTimeout(slave, 1500);
/* set handler for link layer state changes */
CS101_Slave_setLinkLayerStateChanged(slave, linkLayerStateChanged, NULL);
/* log messages */
CS101_Slave_setRawMessageHandler(slave, rawMessageHandler, NULL);
int16_t scaledValue = 0;
uint64_t lastMessageSent = 0;
if (!SerialPort_open(port))
{
fprintf(stderr, "Failed to open serial port\n");
return 1;
}
while (running)
{
/* has to be called periodically */
CS101_Slave_run(slave);
/* Enqueue a measurement every second */
if (Hal_getTimeInMs() > (lastMessageSent + 1000))
{
CS101_ASDU newAsdu = CS101_ASDU_create(alParameters, false, CS101_COT_PERIODIC, 0, 1, false, false);
InformationObject io =
(InformationObject)MeasuredValueScaled_create(NULL, 110, scaledValue, IEC60870_QUALITY_GOOD);
scaledValue++;
CS101_ASDU_addInformationObject(newAsdu, io);
InformationObject_destroy(io);
CS101_Slave_enqueueUserDataClass2(slave, newAsdu);
CS101_ASDU_destroy(newAsdu);
lastMessageSent = Hal_getTimeInMs();
}
Thread_sleep(1);
}
goto exit_program;
exit_program:
CS101_Slave_destroy(slave);
SerialPort_close(port);
SerialPort_destroy(port);
Semaphore_destroy(gi_stateLock);
return 0;
}