org.jruby.ext.ffi.Pointer Maven / Gradle / Ivy
package org.jruby.ext.ffi;
import java.nio.ByteOrder;
import org.jruby.Ruby;
import org.jruby.RubyClass;
import org.jruby.RubyFixnum;
import org.jruby.RubyModule;
import org.jruby.RubyString;
import org.jruby.anno.JRubyClass;
import org.jruby.anno.JRubyMethod;
import org.jruby.runtime.ObjectAllocator;
import org.jruby.runtime.ThreadContext;
import org.jruby.runtime.builtin.IRubyObject;
import static org.jruby.runtime.Visibility.*;
/**
* C memory pointer operations.
*
* This is an abstract class that defines Pointer operations
*
*/
@JRubyClass(name="FFI::Pointer", parent=AbstractMemory.ABSTRACT_MEMORY_RUBY_CLASS)
public class Pointer extends AbstractMemory {
public static RubyClass createPointerClass(Ruby runtime, RubyModule module) {
RubyClass result = module.defineClassUnder("Pointer",
module.getClass(AbstractMemory.ABSTRACT_MEMORY_RUBY_CLASS),
PointerAllocator.INSTANCE);
result.defineAnnotatedMethods(Pointer.class);
result.defineAnnotatedConstants(Pointer.class);
module.defineClassUnder("NullPointerError", runtime.getRuntimeError(),
runtime.getRuntimeError().getAllocator());
// Add Pointer::NULL as a constant
result.setConstant("NULL", new Pointer(runtime, result, new NullMemoryIO(runtime)));
return result;
}
private static final class PointerAllocator implements ObjectAllocator {
static final ObjectAllocator INSTANCE = new PointerAllocator();
public IRubyObject allocate(Ruby runtime, RubyClass klazz) {
return new Pointer(runtime, klazz);
}
}
public static final Pointer getNull(Ruby runtime) {
return (Pointer) runtime.getFFI().pointerClass.getConstant("NULL");
}
Pointer(Ruby runtime, RubyClass klazz) {
super(runtime, klazz, runtime.getFFI().getNullMemoryIO(), 0);
}
public Pointer(Ruby runtime, DirectMemoryIO io) {
this(runtime, getPointerClass(runtime), io);
}
public Pointer(Ruby runtime, DirectMemoryIO io, long size, int typeSize) {
this(runtime, getPointerClass(runtime), io, size, typeSize);
}
protected Pointer(Ruby runtime, RubyClass klass, DirectMemoryIO io) {
super(runtime, klass, io, Long.MAX_VALUE);
}
protected Pointer(Ruby runtime, RubyClass klass, DirectMemoryIO io, long size) {
super(runtime, klass, io, size);
}
protected Pointer(Ruby runtime, RubyClass klass, DirectMemoryIO io, long size, int typeSize) {
super(runtime, klass, io, size, typeSize);
}
public static final RubyClass getPointerClass(Ruby runtime) {
return runtime.getFFI().pointerClass;
}
public final AbstractMemory order(Ruby runtime, ByteOrder order) {
return new Pointer(runtime,
order.equals(getMemoryIO().order()) ? (DirectMemoryIO) getMemoryIO() : new SwappedMemoryIO(runtime, getMemoryIO()),
size, typeSize);
}
@JRubyMethod(name = { "initialize" }, visibility = PRIVATE)
public IRubyObject initialize(ThreadContext context, IRubyObject address) {
io = address instanceof Pointer
? ((Pointer) address).getMemoryIO()
: Factory.getInstance().wrapDirectMemory(context.runtime, RubyFixnum.num2long(address));
size = Long.MAX_VALUE;
typeSize = 1;
return this;
}
@JRubyMethod(name = { "initialize" }, visibility = PRIVATE)
public IRubyObject initialize(ThreadContext context, IRubyObject type, IRubyObject address) {
io = address instanceof Pointer
? ((Pointer) address).getMemoryIO()
: Factory.getInstance().wrapDirectMemory(context.runtime, RubyFixnum.num2long(address));
size = Long.MAX_VALUE;
typeSize = calculateTypeSize(context, type);
return this;
}
/**
*
*/
@JRubyMethod(required = 1, visibility=PRIVATE)
public IRubyObject initialize_copy(ThreadContext context, IRubyObject other) {
if (this == other) {
return this;
}
Pointer orig = (Pointer) other;
this.typeSize = orig.typeSize;
this.size = orig.size;
setMemoryIO(orig.getMemoryIO().dup());
return this;
}
/**
* Tests if this Pointer represents the C NULL value.
*
* @return true if the address is NULL.
*/
@JRubyMethod(name = "null?")
public IRubyObject null_p(ThreadContext context) {
return context.runtime.newBoolean(getMemoryIO().isNull());
}
@Override
@JRubyMethod(name = { "to_s", "inspect" }, optional = 1)
public IRubyObject to_s(ThreadContext context, IRubyObject[] args) {
String s = size != Long.MAX_VALUE
? String.format("#<%s address=0x%x size=%s>", getMetaClass().getName(), getAddress(), size)
: String.format("#<%s address=0x%x>", getMetaClass().getName(), getAddress());
return RubyString.newString(context.runtime, s);
}
@JRubyMethod(name = { "address", "to_i" })
public IRubyObject address(ThreadContext context) {
return context.runtime.newFixnum(getAddress());
}
/**
* Gets the native memory address of this pointer.
*
* @return A long containing the native memory address.
*/
public final long getAddress() {
return ((DirectMemoryIO) getMemoryIO()).getAddress();
}
@JRubyMethod(name = "==", required = 1)
public IRubyObject op_equal(ThreadContext context, IRubyObject obj) {
return context.runtime.newBoolean(this == obj
|| getAddress() == 0L && obj.isNil()
|| (obj instanceof Pointer && ((Pointer) obj).getAddress() == getAddress()));
}
@Override
protected AbstractMemory slice(Ruby runtime, long offset) {
return new Pointer(runtime, getPointerClass(runtime),
(DirectMemoryIO) getMemoryIO().slice(offset),
size == Long.MAX_VALUE ? Long.MAX_VALUE : size - offset, typeSize);
}
@Override
protected AbstractMemory slice(Ruby runtime, long offset, long size) {
return new Pointer(runtime, getPointerClass(runtime),
(DirectMemoryIO) getMemoryIO().slice(offset, size), size, typeSize);
}
protected Pointer getPointer(Ruby runtime, long offset) {
return new Pointer(runtime, getPointerClass(runtime), getMemoryIO().getMemoryIO(offset), Long.MAX_VALUE);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy