All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.sun.jna.Union Maven / Gradle / Ivy

There is a newer version: 3.5.2
Show newest version
/* Copyright (c) 2007 Timothy Wall, All Rights Reserved
 * 
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 * 
 * This library 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
 * Lesser General Public License for more details.  
 */
package com.sun.jna;

import java.util.Iterator;

/** Represents a native union.  When writing to native memory, the field
 * corresponding to the type passed to {@link #setType} will be written
 * to native memory.  Upon reading from native memory, Structure, String,
 * or WString fields will not be initialized unless they are 
 * the current field as identified by a call to {@link #setType}.  The current
 * field is always unset by default to avoid accidentally attempting to read
 * a field that is not valid.  In the case of a String, for instance, an 
 * invalid pointer may result in a memory fault when attempting to initialize
 * the String. 
 */
public abstract class Union extends Structure {
    private StructField activeField;
    StructField biggestField;
    /** Create a Union whose size and alignment will be calculated 
     * automatically.
     */
    protected Union() { }
    /** Create a Union of the given size, using default alignment. */
    protected Union(Pointer p) {
        super(p);
    }
    /** Create a Union of the given size and alignment type. */
    protected Union(Pointer p, int alignType) {
        super(p, alignType);
    }
    /** Create a Union of the given size and alignment type. */
    protected Union(TypeMapper mapper) {
        super(mapper);
    }
    /** Create a Union of the given size and alignment type. */
    protected Union(Pointer p, int alignType, TypeMapper mapper) {
        super(p, alignType, mapper);
    }
    /** Indicates by type which field will be used to write to native memory.  
     * If there are multiple fields of the same type, use {@link
     * #setType(String)} instead with the field name.
     * @throws IllegalArgumentException if the type does not correspond to 
     * any declared union field.
     */
    public void setType(Class type) {
        ensureAllocated();
        for (Iterator i=fields().values().iterator();i.hasNext();) {
            StructField f = (StructField)i.next();
            if (f.type == type) {
                activeField = f;
                return;
            }
        }
        throw new IllegalArgumentException("No field of type " + type + " in " + this);
    }
    
    /**
     * Indicates which field will be used to write to native memory.
     * @throws IllegalArgumentException if the name does not correspond to
     * any declared union field.
     */
    public void setType(String fieldName) {
        ensureAllocated();
        StructField f = (StructField) fields().get(fieldName);
        if (f != null) {
            activeField = f;
        }
        else {
            throw new IllegalArgumentException("No field named " + fieldName
                                               + " in " + this);
        }
    }

    /** Force a read of the given field from native memory.
     * @return the new field value, after updating
     * @throws IllegalArgumentException if no field exists with the given name
     */
    public Object readField(String fieldName) {
        ensureAllocated();
        setType(fieldName);
        return super.readField(fieldName);
    }

    /** Write the given field value to native memory.
     * The given field will become the active one.
     * @throws IllegalArgumentException if no field exists with the given name
     */
    public void writeField(String fieldName) {
        ensureAllocated();
        setType(fieldName);
        super.writeField(fieldName);
    }

    /** Write the given field value to the field and native memory.
     * The given field will become the active one.
     * @throws IllegalArgumentException if no field exists with the given name
     */
    public void writeField(String fieldName, Object value) {
        ensureAllocated();
        setType(fieldName);
        super.writeField(fieldName, value);
    }

    /** Reads the Structure field of the given type from memory, sets it as
     * the active type and returns it.  Convenience method for
     * 

     * Union u;
     * Class type;
     * u.setType(type);
     * u.read();
     * value = u.field;
     * 
* @param type class type of the Structure field to read * @return the Structure field with the given type */ public Object getTypedValue(Class type) { ensureAllocated(); for (Iterator i=fields().values().iterator();i.hasNext();) { StructField f = (StructField)i.next(); if (f.type == type) { activeField = f; read(); return getField(activeField); } } throw new IllegalArgumentException("No field of type " + type + " in " + this); } /** Set the active type and its value. Convenience method for *

     * Union u;
     * Class type;
     * u.setType(type);
     * u.field = value;
     * 
* @param object instance of a class which is part of the union * @return this Union object */ public Object setTypedValue(Object object) { StructField f = findField(object.getClass()); if (f != null) { activeField = f; setField(f, object); return this; } throw new IllegalArgumentException("No field of type " + object.getClass() + " in " + this); } /** Returns the field in this union with the same type as type, * if any, null otherwise. * @param type type to search for * @return StructField of matching type */ private StructField findField(Class type) { ensureAllocated(); for (Iterator i=fields().values().iterator();i.hasNext();) { StructField f = (StructField)i.next(); if (f.type.isAssignableFrom(type)) { return f; } } return null; } /** Only the currently selected field will be written. */ void writeField(StructField field) { if (field == activeField) { super.writeField(field); } } /** Avoid reading pointer-based fields and structures unless explicitly * selected. Structures may contain pointer-based fields which can * crash the VM if not properly initialized. */ Object readField(StructField field) { if (field == activeField || (!Structure.class.isAssignableFrom(field.type) && !String.class.isAssignableFrom(field.type) && !WString.class.isAssignableFrom(field.type))) { return super.readField(field); } // Field not accessible // TODO: read structure, to the extent possible; need a "recursive" // flag to "read" return null; } /** Adjust the size to be the size of the largest element, and ensure * all fields begin at offset zero. */ int calculateSize(boolean force, boolean avoidFFIType) { int size = super.calculateSize(force, avoidFFIType); if (size != CALCULATE_SIZE) { int fsize = 0; for (Iterator i=fields().values().iterator();i.hasNext();) { StructField f = (StructField)i.next(); f.offset = 0; if (f.size > fsize // Prefer aggregate types to simple types, since they // will have more complex packing rules (some platforms // have specific methods for packing small structs into // registers, which may not match the packing of bytes // for a primitive type). || (f.size == fsize && Structure.class.isAssignableFrom(f.type))) { fsize = f.size; biggestField = f; } } size = calculateAlignedSize(fsize); if (size > 0) { // Update native FFI type information, if needed if (this instanceof ByValue && !avoidFFIType) { getTypeInfo(); } } } return size; } /** All fields are considered the "first" element. */ protected int getNativeAlignment(Class type, Object value, boolean isFirstElement) { return super.getNativeAlignment(type, value, true); } /** Avoid calculating type information until we know our biggest field. * Return type information for the largest field to ensure all available * bits are used. */ Pointer getTypeInfo() { if (biggestField == null) { // Not calculated yet return null; } return super.getTypeInfo(); } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy