// Copyright (c) 2008 The Board of Trustees of The Leland Stanford Junior University
// Copyright (c) 2011, 2012 Open Networking Foundation
// Copyright (c) 2012, 2013 Big Switch Networks, Inc.
// This library was generated by the LoxiGen Compiler.
// See the file LICENSE.txt which should have been included in the source distribution

// Automatically generated by LOXI from template const_serializer.java
// Do not modify

package org.projectfloodlight.openflow.protocol.ver10;

import org.projectfloodlight.openflow.protocol.*;
import org.projectfloodlight.openflow.protocol.action.*;
import org.projectfloodlight.openflow.protocol.actionid.*;
import org.projectfloodlight.openflow.protocol.bsntlv.*;
import org.projectfloodlight.openflow.protocol.errormsg.*;
import org.projectfloodlight.openflow.protocol.meterband.*;
import org.projectfloodlight.openflow.protocol.instruction.*;
import org.projectfloodlight.openflow.protocol.instructionid.*;
import org.projectfloodlight.openflow.protocol.match.*;
import org.projectfloodlight.openflow.protocol.oxm.*;
import org.projectfloodlight.openflow.protocol.queueprop.*;
import org.projectfloodlight.openflow.types.*;
import org.projectfloodlight.openflow.util.*;
import org.projectfloodlight.openflow.exceptions.*;
import org.projectfloodlight.openflow.protocol.OFFlowModFailedCode;
import org.jboss.netty.buffer.ChannelBuffer;
import com.google.common.hash.PrimitiveSink;

public class OFFlowModFailedCodeSerializerVer10 {

    public final static short ALL_TABLES_FULL_VAL = (short) 0x0;
    public final static short OVERLAP_VAL = (short) 0x1;
    public final static short EPERM_VAL = (short) 0x2;
    public final static short BAD_EMERG_TIMEOUT_VAL = (short) 0x3;
    public final static short BAD_COMMAND_VAL = (short) 0x4;
    public final static short UNSUPPORTED_VAL = (short) 0x5;

    public static OFFlowModFailedCode readFrom(ChannelBuffer bb) throws OFParseError {
        try {
            return ofWireValue(bb.readShort());
        } catch (IllegalArgumentException e) {
            throw new OFParseError(e);
        }
    }

    public static void writeTo(ChannelBuffer bb, OFFlowModFailedCode e) {
        bb.writeShort(toWireValue(e));
    }

    public static void putTo(OFFlowModFailedCode e, PrimitiveSink sink) {
        sink.putShort(toWireValue(e));
    }

    public static OFFlowModFailedCode ofWireValue(short val) {
        switch(val) {
            case ALL_TABLES_FULL_VAL:
                return OFFlowModFailedCode.ALL_TABLES_FULL;
            case OVERLAP_VAL:
                return OFFlowModFailedCode.OVERLAP;
            case EPERM_VAL:
                return OFFlowModFailedCode.EPERM;
            case BAD_EMERG_TIMEOUT_VAL:
                return OFFlowModFailedCode.BAD_EMERG_TIMEOUT;
            case BAD_COMMAND_VAL:
                return OFFlowModFailedCode.BAD_COMMAND;
            case UNSUPPORTED_VAL:
                return OFFlowModFailedCode.UNSUPPORTED;
            default:
                throw new IllegalArgumentException("Illegal wire value for type OFFlowModFailedCode in version 1.0: " + val);
        }
    }


    public static short toWireValue(OFFlowModFailedCode e) {
        switch(e) {
            case ALL_TABLES_FULL:
                return ALL_TABLES_FULL_VAL;
            case OVERLAP:
                return OVERLAP_VAL;
            case EPERM:
                return EPERM_VAL;
            case BAD_EMERG_TIMEOUT:
                return BAD_EMERG_TIMEOUT_VAL;
            case BAD_COMMAND:
                return BAD_COMMAND_VAL;
            case UNSUPPORTED:
                return UNSUPPORTED_VAL;
            default:
                throw new IllegalArgumentException("Illegal enum value for type OFFlowModFailedCode in version 1.0: " + e);
        }
    }

}
