244 lines
6.4 KiB
Java
244 lines
6.4 KiB
Java
/*
|
|
* ProtocolLib - Bukkit server library that allows access to the Minecraft protocol.
|
|
* Copyright (C) 2012 Kristian S. Stangeland
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it under the terms of the
|
|
* GNU General Public License as published by the Free Software Foundation; either version 2 of
|
|
* the License, or (at your option) any later version.
|
|
*
|
|
* This program 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 for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along with this program;
|
|
* if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
|
|
* 02111-1307 USA
|
|
*/
|
|
|
|
package com.comphenix.protocol.reflect;
|
|
|
|
import java.lang.reflect.Field;
|
|
|
|
import com.comphenix.protocol.ProtocolLogger;
|
|
import com.comphenix.protocol.reflect.accessors.Accessors;
|
|
import com.comphenix.protocol.reflect.accessors.FieldAccessor;
|
|
import com.google.common.base.Objects;
|
|
|
|
/**
|
|
* Represents a field that will revert to its original state when this class is garbaged collected.
|
|
*
|
|
* @author Kristian
|
|
*/
|
|
public class VolatileField {
|
|
private FieldAccessor accessor;
|
|
private Object container;
|
|
|
|
// The current and previous values
|
|
private Object previous;
|
|
private Object current;
|
|
|
|
// Whether or not we must reset or load
|
|
private boolean previousLoaded;
|
|
private boolean currentSet;
|
|
|
|
// Whether or not to break access restrictions
|
|
private boolean forceAccess;
|
|
|
|
/**
|
|
* Initializes a volatile field with an associated object.
|
|
* @param field - the field.
|
|
* @param container - the object this field belongs to.
|
|
*/
|
|
public VolatileField(Field field, Object container) {
|
|
this.accessor = Accessors.getFieldAccessor(field);
|
|
this.container = container;
|
|
}
|
|
|
|
/**
|
|
* Initializes a volatile field with an associated object.
|
|
* @param field - the field.
|
|
* @param container - the object this field belongs to.
|
|
* @param forceAccess - whether or not to override any scope restrictions.
|
|
*/
|
|
public VolatileField(Field field, Object container, boolean forceAccess) {
|
|
this.accessor = Accessors.getFieldAccessor(field, true);
|
|
this.container = container;
|
|
this.forceAccess = forceAccess;
|
|
}
|
|
|
|
/**
|
|
* Initializes a volatile field with the given accessor and associated object.
|
|
* @param accessor - the field accessor.
|
|
* @param container - the object this field belongs to.
|
|
*/
|
|
public VolatileField(FieldAccessor accessor, Object container) {
|
|
this.accessor = accessor;
|
|
this.container = container;
|
|
}
|
|
|
|
/**
|
|
* Retrieves the current field.
|
|
* @return The stored field.
|
|
*/
|
|
public Field getField() {
|
|
return accessor.getField();
|
|
}
|
|
|
|
/**
|
|
* Retrieves the object the field is stored.
|
|
* @return The reference object.
|
|
*/
|
|
public Object getContainer() {
|
|
return container;
|
|
}
|
|
|
|
/**
|
|
* Retrieves whether or not not to override any scope restrictions.
|
|
* @return TRUE if we override scope, FALSE otherwise.
|
|
*/
|
|
public boolean isForceAccess() {
|
|
return forceAccess;
|
|
}
|
|
|
|
/**
|
|
* Sets whether or not not to override any scope restrictions.
|
|
* @param forceAccess - TRUE if we override scope, FALSE otherwise.
|
|
*/
|
|
public void setForceAccess(boolean forceAccess) {
|
|
this.forceAccess = forceAccess;
|
|
}
|
|
|
|
/**
|
|
* Retrieves the current field value.
|
|
* @return The current field value.
|
|
*/
|
|
public Object getValue() {
|
|
// Retrieve the correct value
|
|
if (!currentSet) {
|
|
ensureLoaded();
|
|
return previous;
|
|
} else {
|
|
return current;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Retrieves the field value before the previous setValue(), unless saveValue() has been called.
|
|
* @return Previous value.
|
|
*/
|
|
public Object getOldValue() {
|
|
ensureLoaded();
|
|
return previous;
|
|
}
|
|
|
|
/**
|
|
* Sets the current value. This will be reverted unless saveValue() is called.
|
|
* @param newValue - new field value.
|
|
*/
|
|
public void setValue(Object newValue) {
|
|
// Remember to safe the previous value
|
|
ensureLoaded();
|
|
|
|
writeFieldValue(newValue);
|
|
current = newValue;
|
|
currentSet = true;
|
|
}
|
|
|
|
/**
|
|
* Reapply the current changed value.
|
|
* <p>
|
|
* Also refresh the previously set value.
|
|
*/
|
|
public void refreshValue() {
|
|
Object fieldValue = readFieldValue();
|
|
|
|
if (currentSet) {
|
|
// If they differ, we need to set them again
|
|
if (!Objects.equal(current, fieldValue)) {
|
|
previous = readFieldValue();
|
|
previousLoaded = true;
|
|
writeFieldValue(current);
|
|
}
|
|
} else if (previousLoaded) {
|
|
// Update that too
|
|
previous = fieldValue;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Ensure that the current value is still set after this class has been garbaged collected.
|
|
*/
|
|
public void saveValue() {
|
|
previous = current;
|
|
currentSet = false;
|
|
}
|
|
|
|
/**
|
|
* Revert to the previously set value.
|
|
*/
|
|
public void revertValue() {
|
|
// Reset value if it hasn't been changed by anyone else
|
|
if (currentSet) {
|
|
if (getValue() == current) {
|
|
setValue(previous);
|
|
currentSet = false;
|
|
} else {
|
|
// This can be a bad sign
|
|
ProtocolLogger.log("Unable to switch {0} to {1}. Expected {2}, but got {3}.", getField().toGenericString(), previous, current, getValue());
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Retrieve a synchronized version of the current field.
|
|
* @return A synchronized volatile field.
|
|
*/
|
|
public VolatileField toSynchronized() {
|
|
return new VolatileField(Accessors.getSynchronized(accessor), container);
|
|
}
|
|
|
|
/**
|
|
* Determine whether or not we'll need to revert the value.
|
|
* @return True if it is set, false if not.
|
|
*/
|
|
public boolean isCurrentSet() {
|
|
return currentSet;
|
|
}
|
|
|
|
private void ensureLoaded() {
|
|
// Load the value if we haven't already
|
|
if (!previousLoaded) {
|
|
previous = readFieldValue();
|
|
previousLoaded = true;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Read the content of the underlying field.
|
|
* @return The field value.
|
|
*/
|
|
private Object readFieldValue() {
|
|
return accessor.get(container);
|
|
}
|
|
|
|
/**
|
|
* Write the given value to the underlying field.
|
|
* @param newValue - the new value.
|
|
*/
|
|
private void writeFieldValue(Object newValue) {
|
|
accessor.set(container, newValue);
|
|
}
|
|
|
|
@Override
|
|
protected void finalize() throws Throwable {
|
|
revertValue();
|
|
}
|
|
|
|
@Override
|
|
public String toString() {
|
|
return "VolatileField [accessor=" + accessor + ", container=" + container + ", previous="
|
|
+ previous + ", current=" + current + ", previousLoaded=" + previousLoaded
|
|
+ ", currentSet=" + currentSet + ", forceAccess=" + forceAccess + "]";
|
|
}
|
|
}
|