The goal of this software is to automatically generate C/C++ code which reads and writes GOOSE and Sampled Value packets. Any valid IEC 61850 Substation Configuration Description (SCD) file, describing GOOSE and/or SV communications, can be used as the input. The output code is lightweight and platform-independent, so it can run on a variety of devices, including low-cost microcontrollers. It\'s ideal for rapid-prototyping new protection and control systems that require communications. This mbed project is a simple example of this functionality. Other code: https://github.com/stevenblair/rapid61850 Project homepage: http://personal.strath.ac.uk/steven.m.blair/

gseDecodePacket.c

Committer:
sblair
Date:
2011-10-07
Revision:
1:9399d44c2b1a
Parent:
0:230c10b228ea

File content as of revision 1:9399d44c2b1a:

/**
 * Rapid-prototyping protection schemes with IEC 61850
 *
 * Copyright (c) 2011 Steven Blair
 * 
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version 2
 * of the License, or (at your option) any later version.

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

 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
 */

#include "gseDecodeBasic.h"
#include "ied.h"
#include "gseDecode.h"
#include "gsePacketData.h"
#include "decodePacket.h"
#include <stddef.h>


unsigned char    datasetName[65] = {0};        // maximum length of 65 bytes
CTYPE_INT16U    datasetNameLength = 0;


void gseDecodePDU(unsigned char *buf) {
    unsigned char    tag = (unsigned char) buf[0];    // assumes only one byte is used
    CTYPE_INT16U    lengthFieldSize = getLengthFieldSize((unsigned char) buf[1]);
    CTYPE_INT16U    lengthValue = decodeLength((unsigned char *) &buf[1]);
    CTYPE_INT16U    offsetForSequence = 1 + lengthFieldSize;
    CTYPE_INT16U    offsetForNonSequence = 1 + lengthFieldSize + lengthValue;

    //printf("tag: %x, lengthFieldSize: %i, lengthValue: %i, offset: %i\n", tag, lengthFieldSize, lengthValue, offsetForNonSequence);

    switch (tag) {
    case 0x61:
        gseDecodePDU(&buf[offsetForSequence]);
        break;
    case 0x82:
        // save dataset name
        memcpy(datasetName, &buf[offsetForSequence], lengthValue);
        datasetNameLength = lengthValue;
        //printf("%s, %d\n", datasetName, datasetNameLength);
        //fflush(stdout);

        gseDecodePDU(&buf[offsetForNonSequence]);
        break;
    case 0xAB:
        gseDecodeDataset(&buf[offsetForSequence], lengthValue, datasetName, datasetNameLength);
        return;
        //break;
    default:
        gseDecodePDU(&buf[offsetForNonSequence]);
        break;
    }
}

void gseDecode(unsigned char *buf, int len) {
    gseDecodePDU(&buf[26]);    // cuts out frame header (fixed size of 26 bytes before start of APDU)
}