uk.ac.lancs.stopcock.netty.OpenFlowDecoder.java Source code

Java tutorial

Introduction

Here is the source code for uk.ac.lancs.stopcock.netty.OpenFlowDecoder.java

Source

/*
 * Copyright 2014 University of Lancaster
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package uk.ac.lancs.stopcock.netty;

import io.netty.buffer.ByteBuf;
import io.netty.channel.ChannelHandlerContext;
import io.netty.handler.codec.MessageToMessageDecoder;
import org.projectfloodlight.openflow.protocol.OFFactories;
import org.projectfloodlight.openflow.protocol.OFMessage;
import uk.ac.lancs.stopcock.openflow.Container;
import uk.ac.lancs.stopcock.openflow.Header;
import uk.ac.lancs.stopcock.openflow.Type;

import java.util.List;

/**
 * OpenFlowDecode is responsible for the first pass of decoding incoming OpenFlow packets and separating them into
 * the header and raw data. The pipeline will have already split up incoming data into individual packets.
 */
class OpenFlowDecoder extends MessageToMessageDecoder<ByteBuf> {
    @Override
    protected void decode(ChannelHandlerContext channelHandlerContext, ByteBuf byteBuf, List<Object> objects)
            throws Exception {
        /* Sanity check. */
        if (byteBuf.readableBytes() < 8) {
            throw new IllegalStateException();
        }

        /* Read OpenFlow Header. */
        short version = byteBuf.readUnsignedByte();
        short typeId = byteBuf.readUnsignedByte();
        int length = byteBuf.readUnsignedShort();
        long transactionId = byteBuf.readUnsignedInt();

        /* Sanity check that we have enough data. */
        if (byteBuf.readableBytes() < (length - 8)) {
            throw new IllegalStateException();
        }

        /* Copy the data portion of the OpenFlow packet and store it. */
        byteBuf.resetReaderIndex();
        byte[] originalData = byteBuf.readBytes(length).array();

        /* Construct the OpenFlow header and container for both it and data. */
        Header header = new Header(version, typeId, length, transactionId);
        Type type = Type.getById(typeId);

        /* Get full openflow packet for processing with openflowj */
        byteBuf.resetReaderIndex();

        /* Call openflowj using our Netty 3.9.X -> Netty 4.0.0 proxy object. */
        OFMessage message = OFFactories.getGenericReader().readFrom(new NettyCompatibilityChannelBuffer(byteBuf));

        /* Container object for header, raw data and openflowj message. */
        Container container = new Container(header, originalData, type, message);

        /* Add to the Netty pipeline. */
        objects.add(container);
    }
}