org.robovm.rt.bro.Struct Maven / Gradle / Ivy
/*
* Copyright (C) 2012 RoboVM AB
*
* 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 org.robovm.rt.bro;
import java.lang.reflect.Array;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import org.robovm.rt.VM;
import org.robovm.rt.bro.annotation.Marshaler;
import org.robovm.rt.bro.annotation.MarshalsArray;
import org.robovm.rt.bro.annotation.MarshalsPointer;
/**
*
* @version $Id$
*/
@Marshaler(Struct.Marshaler.class)
public abstract class Struct> extends NativeObject implements Iterable {
protected Struct() {
setHandle(VM.allocateMemory(_sizeOf()));
}
protected Struct(long handle) {
setHandle(handle);
}
public T copy() {
return copy(1);
}
@SuppressWarnings("unchecked")
public T copy(int n) {
T o = (T) allocate(getClass(), n);
VM.memcpy(o.getHandle(), getHandle(), _sizeOf() * n);
return o;
}
public T copyWithMalloc() {
return copyWithMalloc(1);
}
@SuppressWarnings("unchecked")
public T copyWithMalloc(int n) {
T o = (T) malloc(getClass(), n);
VM.memcpy(o.getHandle(), getHandle(), _sizeOf() * n);
return o;
}
@SuppressWarnings("unchecked")
public U as(Class type) {
if (Struct.class.isAssignableFrom(type)) {
@SuppressWarnings("rawtypes")
Class c = type;
return (U) Struct.toStruct(c, getHandle());
}
return super.as(type);
}
protected int _sizeOf() {
return 0;
}
public void clear() {
clear(1);
}
public void clear(int n) {
if (n < 1) {
throw new IllegalArgumentException("n < 1");
}
VM.memset(getHandle(), (byte) 0, _sizeOf() * n);
}
public void free() {
VM.free(getHandle());
}
@SuppressWarnings("unchecked")
public Iterator iterator() {
return new StructIterator((T) this);
}
@SuppressWarnings("unchecked")
public Iterator iterator(final int n) {
if (n < 1) {
throw new IllegalArgumentException("n < 1");
}
return new StructIterator((T) this, n);
}
@SuppressWarnings("unchecked")
protected T wrap(long address) {
return (T) toStruct(getClass(), address);
}
public T next() {
return next(1);
}
@SuppressWarnings("unchecked")
public T next(long delta) {
if (delta == 0) {
return (T) this;
}
return wrap(getHandle() + _sizeOf() * delta);
}
public T previous() {
return previous(1);
}
public T previous(long delta) {
return next(-delta);
}
@SuppressWarnings("unchecked")
public T[] toArray(int n) {
T[] array = (T[]) Array.newInstance(getClass(), n);
for (int i = 0; i < n; i++) {
array[i] = next(i);
}
return array;
}
/**
* Updates the memory starting at this {@link Struct} with the
* {@link Struct} starting at the specified instance.
*
* @param o the {@link Struct} to write to the address of this
* {@link Struct}.
* @throws NullPointerException if {@code o} is {@code null}.
* @throws IllegalArgumentException if the class of {@code o} is not the
* same as this {@link Struct}'s class.
*/
public void update(T o) {
update(o, 1);
}
/**
* Updates the memory starting at this {@link Struct} with the
* {@link Struct}s starting at the specified instance.
*
* @param o the first {@link Struct} to write to the address of this
* {@link Struct}.
* @param n the number of {@link Struct}s to write.
* @throws NullPointerException if {@code o} is {@code null}.
* @throws IllegalArgumentException if the class of {@code o} is not the
* same as this {@link Struct}'s class.
*/
public void update(T o, int n) {
if (o == null) {
throw new NullPointerException("o");
}
if (o.getClass() != this.getClass()) {
throw new IllegalArgumentException("Expected an instance of "
+ this.getClass().getName() + ". Actual type: "
+ o.getClass().getName());
}
if (n < 1) {
throw new IllegalArgumentException("n < 1");
}
VM.memcpy(getHandle(), o.getHandle(), _sizeOf() * n);
}
/**
* Updates the memory starting at this {@link Struct} with the
* {@link Struct}s in the specified array.
*
* @param array the array of {@link Struct}s to write.
* @throws NullPointerException if {@code array} or any of the objects in
* {@code array} are {@code null}.
* @throws IllegalArgumentException if the class of any of the objects in
* the array is not the same as this {@link Struct}'s class.
*/
public void update(T[] array) {
if (array == null) {
throw new NullPointerException("array");
}
Class> cls = this.getClass();
int len = array.length;
for (int i = 0; i < len; i++) {
T o = array[i];
if (o == null) {
throw new NullPointerException("null at index " + i);
}
if (o.getClass() != cls) {
throw new IllegalArgumentException("Expected an instance of "
+ cls.getName() + " at index " + i + ". Actual type: "
+ o.getClass().getName());
}
}
long dst = getHandle();
int size = _sizeOf();
for (int i = 0; i < len; i++) {
VM.memcpy(dst, array[i].getHandle(), size);
dst += size;
}
}
public List toList(int n) {
List l = new ArrayList(n);
for (int i = 0; i < n; i++) {
l.add(next(i));
}
return l;
}
public static int sizeOf() {
return 0;
}
public static int sizeOf(Struct> struct) {
return struct._sizeOf();
}
public static int offsetOf(int index) {
return 0;
}
public static > T allocate(Class cls) {
return allocate(cls, 1);
}
public static > T allocate(Class cls, int n) {
if (n < 1) {
throw new IllegalArgumentException("n < 1");
}
T o = VM.allocateObject(cls);
long handle = VM.allocateMemory(o._sizeOf() * n);
o.setHandle(handle);
return o;
}
public static > T malloc(Class cls) {
return malloc(cls, 1);
}
public static > T malloc(Class cls, int n) {
if (n < 1) {
throw new IllegalArgumentException("n < 1");
}
T o = VM.allocateObject(cls);
long handle = VM.malloc(o._sizeOf() * n);
o.setHandle(handle);
return o;
}
public static > T toStruct(Class cls, long handle) {
if (handle == 0L) {
return null;
}
T o = VM.allocateObject(cls);
o.setHandle(handle);
return o;
}
public static class Marshaler {
@MarshalsPointer
public static > T toObject(Class cls, long handle, long flags) {
return Struct.toStruct(cls, handle);
}
@MarshalsPointer
public static long toNative(Struct> o, long flags) {
if (o == null) {
return 0L;
}
return o.getHandle();
}
@MarshalsArray
public static > T toObject(Class cls, long handle, long flags, int d1) {
T s = Struct.toStruct(cls, handle);
return s;
}
@MarshalsArray
public static > void toNative(T o, long handle, long flags, int d1) {
if (o.getHandle() == handle) {
return;
}
VM.memcpy(handle, o.getHandle(), d1 * o._sizeOf());
}
@MarshalsArray
public static > T toObject(Class cls, long handle, long flags, int d1, int d2) {
T s = Struct.toStruct(cls, handle);
return s;
}
@MarshalsArray
public static > void toNative(T o, long handle, long flags, int d1, int d2) {
toNative(o, handle, flags, d1 * d2);
}
@MarshalsArray
public static > T toObject(Class cls, long handle, long flags, int d1, int d2, int d3) {
T s = Struct.toStruct(cls, handle);
return s;
}
@MarshalsArray
public static > void toNative(T o, long handle, long flags, int d1, int d2, int d3) {
toNative(o, handle, flags, d1 * d2 * d3);
}
private static void checkDimensions(Class> baseType, String format, int actual, int expected) {
if (actual != expected) {
String suffixActual = String.format(format, actual);
String suffixExpected = String.format(format, expected);
throw new IllegalArgumentException(
"Expected " + baseType.getName() + suffixExpected
+ ". Got " + baseType.getName() + suffixActual);
}
}
@MarshalsArray
@SuppressWarnings("unchecked")
public static > T[] array1DToObject(Class arrayClass, long handle, long flags, int d1) {
T s = Struct.toStruct((Class) arrayClass.getComponentType(), handle);
return s.toArray(d1);
}
@MarshalsArray
@SuppressWarnings("unchecked")
public static > void array1DToNative(T[] o, long handle, long flags, int d1) {
Class structClass = (Class) o.getClass().getComponentType();
checkDimensions(structClass, "[%d]", o.length, d1);
Struct s = Struct.toStruct((Class) structClass, handle);
s.update(o);
}
@MarshalsArray
@SuppressWarnings("unchecked")
public static > T[][] array2DToObject(Class arrayClass, long handle, long flags, int d1, int d2) {
Class structClass = (Class) arrayClass.getComponentType().getComponentType();
T[][] o = (T[][]) Array.newInstance(structClass, d1, d2);
T s = Struct.toStruct((Class) structClass, handle);
int len1 = o.length;
for (int i = 0; i < len1; i++) {
o[i] = s.toArray(d2);
s = s.next(d2);
}
return o;
}
@MarshalsArray
@SuppressWarnings("unchecked")
public static > void array2DToNative(T[][] o, long handle, long flags, int d1, int d2) {
Class structClass = (Class) o.getClass().getComponentType().getComponentType();
checkDimensions(structClass, "[%d][]", o.length, d1);
int len1 = o.length;
for (int i = 0; i < len1; i++) {
checkDimensions(structClass, "[][%d]", o[i].length, d2);
}
Struct s = Struct.toStruct((Class) structClass, handle);
for (int i = 0; i < len1; i++) {
s.update(o[i]);
s = s.next(d2);
}
}
@MarshalsArray
@SuppressWarnings("unchecked")
public static > T[][][] array3DToObject(Class arrayClass, long handle, long flags, int d1, int d2, int d3) {
Class structClass = (Class) arrayClass.getComponentType().getComponentType().getComponentType();
T[][][] o = (T[][][]) Array.newInstance(structClass, d1, d2, d3);
T s = Struct.toStruct((Class) structClass, handle);
int len1 = o.length;
for (int i = 0; i < len1; i++) {
int len2 = o[i].length;
for (int j = 0; j < len2; j++) {
o[i][j] = s.toArray(d3);
s = s.next(d3);
}
}
return o;
}
@MarshalsArray
@SuppressWarnings("unchecked")
public static > void array3DToNative(T[][][] o, long handle, long flags, int d1, int d2, int d3) {
Class structClass = (Class) o.getClass().getComponentType().getComponentType().getComponentType();
checkDimensions(structClass, "[%d][][]", o.length, d1);
int len1 = o.length;
for (int i = 0; i < len1; i++) {
T[][] p = o[i];
checkDimensions(structClass, "[][%d][]", p.length, d2);
int len2 = p.length;
for (int j = 0; j < len2; j++) {
checkDimensions(structClass, "[][][%d]", p[j].length, d3);
}
}
Struct s = Struct.toStruct((Class) structClass, handle);
for (int i = 0; i < len1; i++) {
T[][] p = o[i];
int len2 = p.length;
for (int j = 0; j < len2; j++) {
s.update(o[i][j]);
s = s.next(d3);
}
}
}
}
static class StructIterator> implements Iterator {
private T next;
private int n;
private int i = 0;
StructIterator(T first) {
this(first, -1);
}
StructIterator(T first, int n) {
this.next = first;
this.n = n;
}
@Override
public boolean hasNext() {
return n == -1 || i < n;
}
@Override
public T next() {
T o = next;
next = next.next();
i++;
return o;
}
@Override
public void remove() {
throw new UnsupportedOperationException();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy