/*
 * Copyright (c) 2000, 2003, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * This code 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
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

package com.sun.corba.se.impl.dynamicany;

import org.omg.CORBA.TypeCode;
import org.omg.CORBA.TCKind;
import org.omg.CORBA.Any;
import org.omg.CORBA.TypeCodePackage.BadKind;
import org.omg.CORBA.TypeCodePackage.Bounds;
import org.omg.CORBA.portable.InputStream;
import org.omg.DynamicAny.*;
import org.omg.DynamicAny.DynAnyPackage.TypeMismatch;
import org.omg.DynamicAny.DynAnyPackage.InvalidValue;
import org.omg.DynamicAny.DynAnyFactoryPackage.InconsistentTypeCode;

import com.sun.corba.se.spi.orb.ORB ;
import com.sun.corba.se.spi.logging.CORBALogDomains ;
import com.sun.corba.se.impl.logging.ORBUtilSystemException ;

public class DynUnionImpl extends DynAnyConstructedImpl implements DynUnion
{
    //
    // Instance variables
    //

    DynAny discriminator = null;
    // index either points to the discriminator or the named member is it exists.
    // The currently active member, which is of the same type as the discriminator.
    DynAny currentMember = null;
    int currentMemberIndex = NO_INDEX;

    //
    // Constructors
    //

    private DynUnionImpl() {
        this(null, (Any)null, false);
    }

    protected DynUnionImpl(ORB orb, Any any, boolean copyValue) {
        // We can be sure that typeCode is of kind tk_union
        super(orb, any, copyValue);
    }

    protected DynUnionImpl(ORB orb, TypeCode typeCode) {
        // We can be sure that typeCode is of kind tk_union
        super(orb, typeCode);
    }

    protected boolean initializeComponentsFromAny() {
        try {
            InputStream input = any.create_input_stream();
            Any discriminatorAny = DynAnyUtil.extractAnyFromStream(discriminatorType(), input, orb);
            discriminator = DynAnyUtil.createMostDerivedDynAny(discriminatorAny, orb, false);
            currentMemberIndex = currentUnionMemberIndex(discriminatorAny);
            Any memberAny = DynAnyUtil.extractAnyFromStream(memberType(currentMemberIndex), input, orb);
            currentMember = DynAnyUtil.createMostDerivedDynAny(memberAny, orb, false);
            components = new DynAny[] {discriminator, currentMember};
        } catch (InconsistentTypeCode ictc) { // impossible
        }
        return true;
    }

    // Sets the current position to zero.
    // The discriminator value is set to a value consistent with the first named member
    // of the union. That member is activated and (recursively) initialized to its default value.
    protected boolean initializeComponentsFromTypeCode() {
        //System.out.println(this + " initializeComponentsFromTypeCode");
        try {
            // We can be sure that memberCount() > 0 according to the IDL language spec
            discriminator = DynAnyUtil.createMostDerivedDynAny(memberLabel(0), orb, false);
            index = 0;
            currentMemberIndex = 0;
            currentMember = DynAnyUtil.createMostDerivedDynAny(memberType(0), orb);
            components = new DynAny[] {discriminator, currentMember};
        } catch (InconsistentTypeCode ictc) { // impossible
        }
        return true;
    }

    //
    // Convenience methods
    //

    private TypeCode discriminatorType() {
        TypeCode discriminatorType = null;
        try {
            discriminatorType = any.type().discriminator_type();
        } catch (BadKind bad) {
        }
        return discriminatorType;
    }

    private int memberCount() {
        int memberCount = 0;
        try {
            memberCount = any.type().member_count();
        } catch (BadKind bad) {
        }
        return memberCount;
    }

    private Any memberLabel(int i) {
        Any memberLabel = null;
        try {
            memberLabel = any.type().member_label(i);
        } catch (BadKind bad) {
        } catch (Bounds bounds) {
        }
        return memberLabel;
    }

    private TypeCode memberType(int i) {
        TypeCode memberType = null;
        try {
            memberType = any.type().member_type(i);
        } catch (BadKind bad) {
        } catch (Bounds bounds) {
        }
        return memberType;
    }

    private String memberName(int i) {
        String memberName = null;
        try {
            memberName = any.type().member_name(i);
        } catch (BadKind bad) {
        } catch (Bounds bounds) {
        }
        return memberName;
    }

    private int defaultIndex() {
        int defaultIndex = -1;
        try {
            defaultIndex = any.type().default_index();
        } catch (BadKind bad) {
        }
        return defaultIndex;
    }

    private int currentUnionMemberIndex(Any discriminatorValue) {
        int memberCount = memberCount();
        Any memberLabel;
        for (int i=0; i<memberCount; i++) {
            memberLabel = memberLabel(i);
            if (memberLabel.equal(discriminatorValue)) {
                return i;
            }
        }
        if (defaultIndex() != -1) {
            return defaultIndex();
        }
        return NO_INDEX;
    }

    protected void clearData() {
        super.clearData();
        discriminator = null;
        // Necessary to guarantee OBJECT_NOT_EXIST in member()
        currentMember.destroy();
        currentMember = null;
        currentMemberIndex = NO_INDEX;
    }

    //
    // DynAny interface methods
    //

    // _REVISIT_ More efficient copy operation

    //
    // DynUnion interface methods
    //

    
Returns the current discriminator value.
/** * Returns the current discriminator value. */
public org.omg.DynamicAny.DynAny get_discriminator () { if (status == STATUS_DESTROYED) { throw wrapper.dynAnyDestroyed() ; } return (checkInitComponents() ? discriminator : null); } // Sets the discriminator of the DynUnion to the specified value. // If the TypeCode of the parameter is not equivalent // to the TypeCode of the unions discriminator, the operation raises TypeMismatch. // // Setting the discriminator to a value that is consistent with the currently // active union member does not affect the currently active member. // Setting the discriminator to a value that is inconsistent with the currently // active member deactivates the member and activates the member that is consistent // with the new discriminator value (if there is a member for that value) // by initializing the member to its default value. // // If the discriminator value indicates a non-existent union member // this operation sets the current position to 0 // (has_no_active_member returns true in this case). // Otherwise the current position is set to 1 (has_no_active_member returns false and // component_count returns 2 in this case). public void set_discriminator (org.omg.DynamicAny.DynAny newDiscriminator) throws org.omg.DynamicAny.DynAnyPackage.TypeMismatch { if (status == STATUS_DESTROYED) { throw wrapper.dynAnyDestroyed() ; } if ( ! newDiscriminator.type().equal(discriminatorType())) { throw new TypeMismatch(); } newDiscriminator = DynAnyUtil.convertToNative(newDiscriminator, orb); Any newDiscriminatorAny = getAny(newDiscriminator); int newCurrentMemberIndex = currentUnionMemberIndex(newDiscriminatorAny); if (newCurrentMemberIndex == NO_INDEX) { clearData(); index = 0; } else { // _REVISIT_ Could possibly optimize here if we don't need to initialize components checkInitComponents(); if (currentMemberIndex == NO_INDEX || newCurrentMemberIndex != currentMemberIndex) { clearData(); index = 1; currentMemberIndex = newCurrentMemberIndex; try { currentMember = DynAnyUtil.createMostDerivedDynAny(memberType(currentMemberIndex), orb); } catch (InconsistentTypeCode ictc) {} discriminator = newDiscriminator; components = new DynAny[] { discriminator, currentMember }; representations = REPRESENTATION_COMPONENTS; } } } // Sets the discriminator to a value that is consistent with the value // of the default case of a union; it sets the current position to // zero and causes component_count to return 2. // Calling set_to_default_member on a union that does not have an explicit // default case raises TypeMismatch. public void set_to_default_member () throws org.omg.DynamicAny.DynAnyPackage.TypeMismatch { if (status == STATUS_DESTROYED) { throw wrapper.dynAnyDestroyed() ; } int defaultIndex = defaultIndex(); if (defaultIndex == -1) { throw new TypeMismatch(); } try { clearData(); index = 1; currentMemberIndex = defaultIndex; currentMember = DynAnyUtil.createMostDerivedDynAny(memberType(defaultIndex), orb); components = new DynAny[] {discriminator, currentMember}; Any discriminatorAny = orb.create_any(); discriminatorAny.insert_octet((byte)0); discriminator = DynAnyUtil.createMostDerivedDynAny(discriminatorAny, orb, false); representations = REPRESENTATION_COMPONENTS; } catch (InconsistentTypeCode ictc) {} } // Sets the discriminator to a value that does not correspond // to any of the unions case labels. // It sets the current position to zero and causes component_count to return 1. // Calling set_to_no_active_member on a union that has an explicit default case // or on a union that uses the entire range of discriminator values // for explicit case labels raises TypeMismatch. public void set_to_no_active_member () throws org.omg.DynamicAny.DynAnyPackage.TypeMismatch { if (status == STATUS_DESTROYED) { throw wrapper.dynAnyDestroyed() ; } // _REVISIT_ How does one check for "entire range of discriminator values"? if (defaultIndex() != -1) { throw new TypeMismatch(); } checkInitComponents(); Any discriminatorAny = getAny(discriminator); // erase the discriminators value so that it does not correspond // to any of the unions case labels discriminatorAny.type(discriminatorAny.type()); index = 0; currentMemberIndex = NO_INDEX; // Necessary to guarantee OBJECT_NOT_EXIST in member() currentMember.destroy(); currentMember = null; components[0] = discriminator; representations = REPRESENTATION_COMPONENTS; } // Returns true if the union has no active member // (that is, the unions value consists solely of its discriminator because the // discriminator has a value that is not listed as an explicit case label). // Calling this operation on a union that has a default case returns false. // Calling this operation on a union that uses the entire range of discriminator // values for explicit case labels returns false. public boolean has_no_active_member () { if (status == STATUS_DESTROYED) { throw wrapper.dynAnyDestroyed() ; } // _REVISIT_ How does one check for "entire range of discriminator values"? if (defaultIndex() != -1) { return false; } checkInitComponents(); return (checkInitComponents() ? (currentMemberIndex == NO_INDEX) : false); } public org.omg.CORBA.TCKind discriminator_kind () { if (status == STATUS_DESTROYED) { throw wrapper.dynAnyDestroyed() ; } return discriminatorType().kind(); } // Returns the currently active member. // If the union has no active member, the operation raises InvalidValue. // Note that the returned reference remains valid only for as long // as the currently active member does not change. // Using the returned reference beyond the life time // of the currently active member raises OBJECT_NOT_EXIST. public org.omg.DynamicAny.DynAny member () throws org.omg.DynamicAny.DynAnyPackage.InvalidValue { if (status == STATUS_DESTROYED) { throw wrapper.dynAnyDestroyed() ; } if ( ! checkInitComponents() || currentMemberIndex == NO_INDEX) throw new InvalidValue(); return currentMember; } // Returns the name of the currently active member. // If the unions TypeCode does not contain a member name for the currently active member, // the operation returns an empty string. // Calling member_name on a union without an active member raises InvalidValue. public String member_name () throws org.omg.DynamicAny.DynAnyPackage.InvalidValue { if (status == STATUS_DESTROYED) { throw wrapper.dynAnyDestroyed() ; } if ( ! checkInitComponents() || currentMemberIndex == NO_INDEX) throw new InvalidValue(); String memberName = memberName(currentMemberIndex); return (memberName == null ? "" : memberName); } // Returns the TCKind value of the TypeCode of the currently active member. // If the union has no active member, the operation raises InvalidValue. public org.omg.CORBA.TCKind member_kind () throws org.omg.DynamicAny.DynAnyPackage.InvalidValue { if (status == STATUS_DESTROYED) { throw wrapper.dynAnyDestroyed() ; } if ( ! checkInitComponents() || currentMemberIndex == NO_INDEX) throw new InvalidValue(); return memberType(currentMemberIndex).kind(); } }