blob: 3ae25fde43ab89f32624c1030104435b8f9af6ab [file] [log] [blame]
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.apache.avro.reflect;
import java.io.IOException;
import java.lang.reflect.Field;
import org.apache.avro.AvroRuntimeException;
import org.apache.avro.io.Decoder;
import org.apache.avro.io.Encoder;
import sun.misc.Unsafe;
@SuppressWarnings("restriction")
class FieldAccessUnsafe extends FieldAccess {
private static final Unsafe UNSAFE;
static {
try {
Field theUnsafe = Unsafe.class.getDeclaredField("theUnsafe");
theUnsafe.setAccessible(true);
UNSAFE = (Unsafe) theUnsafe.get(null);
// It seems not all Unsafe implementations implement the following method.
} catch (Exception e) {
throw new RuntimeException(e);
}
}
@Override
protected FieldAccessor getAccessor(Field field) {
AvroEncode enc = field.getAnnotation(AvroEncode.class);
if (enc != null)
try {
return new UnsafeCustomEncodedField(field, enc.using().newInstance() );
} catch (Exception e) {
throw new AvroRuntimeException("Could not instantiate custom Encoding");
}
Class<?> c = field.getType();
if (c == int.class)
return new UnsafeIntField(field);
else if (c == long.class)
return new UnsafeLongField(field);
else if (c == byte.class)
return new UnsafeByteField(field);
else if (c == float.class)
return new UnsafeFloatField(field);
else if (c == double.class)
return new UnsafeDoubleField(field);
else if (c == char.class)
return new UnsafeCharField(field);
else if (c == boolean.class)
return new UnsafeBooleanField(field);
else if (c == short.class)
return new UnsafeShortField(field);
else
return new UnsafeObjectField(field);
}
abstract static class UnsafeCachedField extends FieldAccessor {
protected final long offset;
protected Field field;
protected final boolean isStringable;
UnsafeCachedField(Field f) {
this.offset = UNSAFE.objectFieldOffset(f);
this.field = f;
this.isStringable = f.isAnnotationPresent(Stringable.class);
}
@Override
protected Field getField() {
return field;
}
@Override
protected boolean supportsIO() {
return true;
}
@Override
protected boolean isStringable() {
return isStringable;
}
}
final static class UnsafeIntField extends UnsafeCachedField {
UnsafeIntField(Field f) {
super(f);
}
@Override
protected void set(Object object, Object value) {
UNSAFE.putInt(object, offset, (Integer) value);
}
@Override
protected Object get(Object object) {
return UNSAFE.getInt(object, offset);
}
@Override
protected void read(Object object, Decoder in) throws IOException {
UNSAFE.putInt(object, offset, in.readInt());
}
@Override
protected void write(Object object, Encoder out) throws IOException {
out.writeInt(UNSAFE.getInt(object, offset));
}
}
final static class UnsafeFloatField extends UnsafeCachedField {
protected UnsafeFloatField(Field f) {
super(f);
}
@Override
protected void set(Object object, Object value) {
UNSAFE.putFloat(object, offset, (Float) value);
}
@Override
protected Object get(Object object) {
return UNSAFE.getFloat(object, offset);
}
@Override
protected void read(Object object, Decoder in) throws IOException {
UNSAFE.putFloat(object, offset, in.readFloat());
}
@Override
protected void write(Object object, Encoder out) throws IOException {
out.writeFloat(UNSAFE.getFloat(object, offset));
}
}
final static class UnsafeShortField extends UnsafeCachedField {
protected UnsafeShortField(Field f) {
super(f);
}
@Override
protected void set(Object object, Object value) {
UNSAFE.putShort(object, offset, (Short) value);
}
@Override
protected Object get(Object object) {
return UNSAFE.getShort(object, offset);
}
@Override
protected void read(Object object, Decoder in) throws IOException {
UNSAFE.putShort(object, offset, (short) in.readInt());
}
@Override
protected void write(Object object, Encoder out) throws IOException {
out.writeInt(UNSAFE.getShort(object, offset));
}
}
final static class UnsafeByteField extends UnsafeCachedField {
protected UnsafeByteField(Field f) {
super(f);
}
@Override
protected void set(Object object, Object value) {
UNSAFE.putByte(object, offset, (Byte) value);
}
@Override
protected Object get(Object object) {
return UNSAFE.getByte(object, offset);
}
@Override
protected void read(Object object, Decoder in) throws IOException {
UNSAFE.putByte(object, offset, (byte) in.readInt());
}
@Override
protected void write(Object object, Encoder out) throws IOException {
out.writeInt(UNSAFE.getByte(object, offset));
}
}
final static class UnsafeBooleanField extends UnsafeCachedField {
protected UnsafeBooleanField(Field f) {
super(f);
}
@Override
protected void set(Object object, Object value) {
UNSAFE.putBoolean(object, offset, (Boolean) value);
}
@Override
protected Object get(Object object) {
return UNSAFE.getBoolean(object, offset);
}
@Override
protected void read(Object object, Decoder in) throws IOException {
UNSAFE.putBoolean(object, offset, in.readBoolean());
}
@Override
protected void write(Object object, Encoder out) throws IOException {
out.writeBoolean(UNSAFE.getBoolean(object, offset));
}
}
final static class UnsafeCharField extends UnsafeCachedField {
protected UnsafeCharField(Field f) {
super(f);
}
@Override
protected void set(Object object, Object value) {
UNSAFE.putChar(object, offset, (Character) value);
}
@Override
protected Object get(Object object) {
return UNSAFE.getChar(object, offset);
}
@Override
protected void read(Object object, Decoder in) throws IOException {
UNSAFE.putChar(object, offset, (char) in.readInt());
}
@Override
protected void write(Object object, Encoder out) throws IOException {
out.writeInt(UNSAFE.getChar(object, offset));
}
}
final static class UnsafeLongField extends UnsafeCachedField {
protected UnsafeLongField(Field f) {
super(f);
}
@Override
protected void set(Object object, Object value) {
UNSAFE.putLong(object, offset, (Long) value);
}
@Override
protected Object get(Object object) {
return UNSAFE.getLong(object, offset);
}
@Override
protected void read(Object object, Decoder in) throws IOException {
UNSAFE.putLong(object, offset, in.readLong());
}
@Override
protected void write(Object object, Encoder out) throws IOException {
out.writeLong(UNSAFE.getLong(object, offset));
}
}
final static class UnsafeDoubleField extends UnsafeCachedField {
protected UnsafeDoubleField(Field f) {
super(f);
}
@Override
protected void set(Object object, Object value) {
UNSAFE.putDouble(object, offset, (Double) value);
}
@Override
protected Object get(Object object) {
return UNSAFE.getDouble(object, offset);
}
@Override
protected void read(Object object, Decoder in) throws IOException {
UNSAFE.putDouble(object, offset, in.readDouble());
}
@Override
protected void write(Object object, Encoder out) throws IOException {
out.writeDouble(UNSAFE.getDouble(object, offset));
}
}
final static class UnsafeObjectField extends UnsafeCachedField {
protected UnsafeObjectField(Field f) {
super(f);
}
@Override
protected void set(Object object, Object value) {
UNSAFE.putObject(object, offset, value);
}
@Override
protected Object get(Object object) {
return UNSAFE.getObject(object, offset);
}
@Override
protected boolean supportsIO() {
return false;
}
}
final static class UnsafeCustomEncodedField extends UnsafeCachedField {
private CustomEncoding<?> encoding;
UnsafeCustomEncodedField(Field f, CustomEncoding<?> encoding) {
super(f);
this.encoding = encoding;
}
@Override
protected Object get(Object object) throws IllegalAccessException {
return UNSAFE.getObject(object, offset);
}
@Override
protected void set(Object object, Object value) throws IllegalAccessException, IOException {
UNSAFE.putObject(object, offset, value);
}
@Override
protected void read(Object object, Decoder in) throws IOException {
UNSAFE.putObject(object, offset, encoding.read(in));
}
@Override
protected void write(Object object, Encoder out) throws IOException {
encoding.write(UNSAFE.getObject(object, offset), out);
}
protected boolean isCustomEncoded() {
return true;
}
}
}