mirror of
https://github.com/ViaVersion/ViaVersion.git
synced 2024-09-29 15:07:29 +02:00
Fix #450, ProtocolSupport concurrency issues, also make listwrapper sync'd
This commit is contained in:
parent
559674bf58
commit
17f3d13267
@ -27,6 +27,7 @@ import us.myles.ViaVersion.handlers.ViaVersionInitializer;
|
|||||||
import us.myles.ViaVersion.protocols.base.ProtocolInfo;
|
import us.myles.ViaVersion.protocols.base.ProtocolInfo;
|
||||||
import us.myles.ViaVersion.update.UpdateListener;
|
import us.myles.ViaVersion.update.UpdateListener;
|
||||||
import us.myles.ViaVersion.update.UpdateUtil;
|
import us.myles.ViaVersion.update.UpdateUtil;
|
||||||
|
import us.myles.ViaVersion.util.ConcurrentList;
|
||||||
import us.myles.ViaVersion.util.ListWrapper;
|
import us.myles.ViaVersion.util.ListWrapper;
|
||||||
import us.myles.ViaVersion.util.ProtocolSupportUtil;
|
import us.myles.ViaVersion.util.ProtocolSupportUtil;
|
||||||
import us.myles.ViaVersion.util.ReflectionUtil;
|
import us.myles.ViaVersion.util.ReflectionUtil;
|
||||||
@ -53,6 +54,20 @@ public class ViaVersionPlugin extends JavaPlugin implements ViaVersionAPI {
|
|||||||
@Getter
|
@Getter
|
||||||
private ViaConfig conf;
|
private ViaConfig conf;
|
||||||
|
|
||||||
|
public ViaVersionPlugin() {
|
||||||
|
// Check if we're using protocol support too
|
||||||
|
protocolSupport = Bukkit.getPluginManager().getPlugin("ProtocolSupport") != null;
|
||||||
|
|
||||||
|
if (protocolSupport) {
|
||||||
|
getLogger().info("Patching to prevent concurrency issues...");
|
||||||
|
try {
|
||||||
|
patchLists();
|
||||||
|
} catch (Exception e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onLoad() {
|
public void onLoad() {
|
||||||
// Config magic
|
// Config magic
|
||||||
@ -87,9 +102,6 @@ public class ViaVersionPlugin extends JavaPlugin implements ViaVersionAPI {
|
|||||||
compatSpigotBuild = false;
|
compatSpigotBuild = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if we're using protocol support too
|
|
||||||
protocolSupport = Bukkit.getPluginManager().getPlugin("ProtocolSupport") != null;
|
|
||||||
|
|
||||||
// Generate classes needed (only works if it's compat or ps)
|
// Generate classes needed (only works if it's compat or ps)
|
||||||
ClassGenerator.generate();
|
ClassGenerator.generate();
|
||||||
lateBind = !isBinded();
|
lateBind = !isBinded();
|
||||||
@ -248,6 +260,27 @@ public class ViaVersionPlugin extends JavaPlugin implements ViaVersionAPI {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public void patchLists() throws Exception {
|
||||||
|
Object connection = getServerConnection();
|
||||||
|
if (connection == null) {
|
||||||
|
getLogger().warning("We failed to find the ServerConnection? :( What server are you running?");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
for (Field field : connection.getClass().getDeclaredFields()) {
|
||||||
|
field.setAccessible(true);
|
||||||
|
final Object value = field.get(connection);
|
||||||
|
if (value instanceof List) {
|
||||||
|
if (!(value instanceof ConcurrentList)) {
|
||||||
|
ConcurrentList list = new ConcurrentList();
|
||||||
|
list.addAll((List) value);
|
||||||
|
field.set(connection, list);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
public boolean isBinded() {
|
public boolean isBinded() {
|
||||||
try {
|
try {
|
||||||
Object connection = getServerConnection();
|
Object connection = getServerConnection();
|
||||||
|
268
src/main/java/us/myles/ViaVersion/util/ConcurrentList.java
Normal file
268
src/main/java/us/myles/ViaVersion/util/ConcurrentList.java
Normal file
@ -0,0 +1,268 @@
|
|||||||
|
package us.myles.ViaVersion.util;
|
||||||
|
|
||||||
|
import lombok.SneakyThrows;
|
||||||
|
|
||||||
|
import java.lang.reflect.Field;
|
||||||
|
import java.util.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by wea_ondara licensed under MIT
|
||||||
|
* Same license as in LICENSE
|
||||||
|
*
|
||||||
|
* Taken from:
|
||||||
|
* https://github.com/weaondara/BungeePerms/blob/master/src/main/java/net/alpenblock/bungeeperms/util/ConcurrentList.java
|
||||||
|
* @param <E> List Type
|
||||||
|
*/
|
||||||
|
public class ConcurrentList<E> extends ArrayList<E> {
|
||||||
|
|
||||||
|
private final Object lock = new Object();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean add(E e) {
|
||||||
|
synchronized (lock) {
|
||||||
|
return super.add(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void add(int index, E element) {
|
||||||
|
synchronized (lock) {
|
||||||
|
super.add(index, element);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean addAll(Collection<? extends E> c) {
|
||||||
|
synchronized (lock) {
|
||||||
|
return super.addAll(c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean addAll(int index, Collection<? extends E> c) {
|
||||||
|
synchronized (lock) {
|
||||||
|
return super.addAll(index, c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void clear() {
|
||||||
|
synchronized (lock) {
|
||||||
|
super.clear();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
@SneakyThrows
|
||||||
|
public Object clone() {
|
||||||
|
synchronized (lock) {
|
||||||
|
ConcurrentList<E> clist = (ConcurrentList<E>) super.clone();
|
||||||
|
clist.modCount = 0;
|
||||||
|
Field f = ArrayList.class.getDeclaredField("elementData");
|
||||||
|
f.setAccessible(true);
|
||||||
|
f.set(clist, Arrays.copyOf((Object[]) f.get(this), this.size()));
|
||||||
|
|
||||||
|
return clist;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean contains(Object o) {
|
||||||
|
synchronized (lock) {
|
||||||
|
return super.contains(o);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void ensureCapacity(int minCapacity) {
|
||||||
|
synchronized (lock) {
|
||||||
|
super.ensureCapacity(minCapacity);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public E get(int index) {
|
||||||
|
synchronized (lock) {
|
||||||
|
return super.get(index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int indexOf(Object o) {
|
||||||
|
synchronized (lock) {
|
||||||
|
return super.indexOf(o);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int lastIndexOf(Object o) {
|
||||||
|
synchronized (lock) {
|
||||||
|
return super.lastIndexOf(o);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public E remove(int index) {
|
||||||
|
synchronized (lock) {
|
||||||
|
return super.remove(index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean remove(Object o) {
|
||||||
|
synchronized (lock) {
|
||||||
|
return super.remove(o);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean removeAll(Collection<?> c) {
|
||||||
|
synchronized (lock) {
|
||||||
|
return super.removeAll(c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean retainAll(Collection<?> c) {
|
||||||
|
synchronized (lock) {
|
||||||
|
return super.retainAll(c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public E set(int index, E element) {
|
||||||
|
synchronized (lock) {
|
||||||
|
return super.set(index, element);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<E> subList(int fromIndex, int toIndex) {
|
||||||
|
synchronized (lock) {
|
||||||
|
return super.subList(fromIndex, toIndex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Object[] toArray() {
|
||||||
|
synchronized (lock) {
|
||||||
|
return super.toArray();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public <T> T[] toArray(T[] a) {
|
||||||
|
synchronized (lock) {
|
||||||
|
return super.toArray(a);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void trimToSize() {
|
||||||
|
synchronized (lock) {
|
||||||
|
super.trimToSize();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ListIterator<E> listIterator() {
|
||||||
|
return new ListItr(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Iterator<E> iterator() {
|
||||||
|
return new Itr();
|
||||||
|
}
|
||||||
|
|
||||||
|
private class Itr implements Iterator<E> {
|
||||||
|
|
||||||
|
protected int cursor;
|
||||||
|
protected int lastRet;
|
||||||
|
final ConcurrentList l;
|
||||||
|
|
||||||
|
public Itr() {
|
||||||
|
cursor = 0;
|
||||||
|
lastRet = -1;
|
||||||
|
l = (ConcurrentList) ConcurrentList.this.clone();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean hasNext() {
|
||||||
|
return cursor < l.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public E next() {
|
||||||
|
int i = cursor;
|
||||||
|
if (i >= l.size()) {
|
||||||
|
throw new NoSuchElementException();
|
||||||
|
}
|
||||||
|
cursor = i + 1;
|
||||||
|
return (E) l.get(lastRet = i);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void remove() {
|
||||||
|
if (lastRet < 0) {
|
||||||
|
throw new IllegalStateException();
|
||||||
|
}
|
||||||
|
|
||||||
|
l.remove(lastRet);
|
||||||
|
ConcurrentList.this.remove(lastRet);
|
||||||
|
cursor = lastRet;
|
||||||
|
lastRet = -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class ListItr extends Itr implements ListIterator<E> {
|
||||||
|
|
||||||
|
ListItr(int index) {
|
||||||
|
super();
|
||||||
|
cursor = index;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean hasPrevious() {
|
||||||
|
return cursor > 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int nextIndex() {
|
||||||
|
return cursor;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int previousIndex() {
|
||||||
|
return cursor - 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public E previous() {
|
||||||
|
int i = cursor - 1;
|
||||||
|
if (i < 0) {
|
||||||
|
throw new NoSuchElementException();
|
||||||
|
}
|
||||||
|
cursor = i;
|
||||||
|
return (E) l.get(lastRet = i);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void set(E e) {
|
||||||
|
if (lastRet < 0) {
|
||||||
|
throw new IllegalStateException();
|
||||||
|
}
|
||||||
|
|
||||||
|
l.set(lastRet, e);
|
||||||
|
ConcurrentList.this.set(lastRet, e);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void add(E e) {
|
||||||
|
int i = cursor;
|
||||||
|
l.add(i, e);
|
||||||
|
ConcurrentList.this.add(i, e);
|
||||||
|
cursor = i + 1;
|
||||||
|
lastRet = -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,11 +1,9 @@
|
|||||||
package us.myles.ViaVersion.util;
|
package us.myles.ViaVersion.util;
|
||||||
|
|
||||||
import java.util.Collection;
|
import java.util.*;
|
||||||
import java.util.Iterator;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.ListIterator;
|
|
||||||
|
|
||||||
public abstract class ListWrapper implements List {
|
public abstract class ListWrapper implements List {
|
||||||
|
public final Object lock = new Object();
|
||||||
private final List list;
|
private final List list;
|
||||||
|
|
||||||
public ListWrapper(List inputList) {
|
public ListWrapper(List inputList) {
|
||||||
@ -20,123 +18,169 @@ public abstract class ListWrapper implements List {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized int size() {
|
public synchronized int size() {
|
||||||
|
synchronized (lock) {
|
||||||
return this.list.size();
|
return this.list.size();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized boolean isEmpty() {
|
public synchronized boolean isEmpty() {
|
||||||
|
synchronized (lock) {
|
||||||
return this.list.isEmpty();
|
return this.list.isEmpty();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized boolean contains(Object o) {
|
public synchronized boolean contains(Object o) {
|
||||||
|
synchronized (lock) {
|
||||||
return this.list.contains(o);
|
return this.list.contains(o);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized Iterator iterator() {
|
public synchronized Iterator iterator() {
|
||||||
return this.list.iterator();
|
synchronized (lock) {
|
||||||
|
return listIterator();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized Object[] toArray() {
|
public synchronized Object[] toArray() {
|
||||||
|
synchronized (lock) {
|
||||||
return this.list.toArray();
|
return this.list.toArray();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized boolean add(Object o) {
|
public synchronized boolean add(Object o) {
|
||||||
|
synchronized (lock) {
|
||||||
handleAdd(o);
|
handleAdd(o);
|
||||||
return this.list.add(o);
|
return this.list.add(o);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized boolean remove(Object o) {
|
public synchronized boolean remove(Object o) {
|
||||||
|
synchronized (lock) {
|
||||||
return this.list.remove(o);
|
return this.list.remove(o);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized boolean addAll(Collection c) {
|
public synchronized boolean addAll(Collection c) {
|
||||||
|
synchronized (lock) {
|
||||||
for (Object o : c) {
|
for (Object o : c) {
|
||||||
handleAdd(o);
|
handleAdd(o);
|
||||||
}
|
}
|
||||||
return this.list.addAll(c);
|
return this.list.addAll(c);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized boolean addAll(int index, Collection c) {
|
public synchronized boolean addAll(int index, Collection c) {
|
||||||
|
synchronized (lock) {
|
||||||
for (Object o : c) {
|
for (Object o : c) {
|
||||||
handleAdd(o);
|
handleAdd(o);
|
||||||
}
|
}
|
||||||
return this.list.addAll(index, c);
|
return this.list.addAll(index, c);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized void clear() {
|
public synchronized void clear() {
|
||||||
|
synchronized (lock) {
|
||||||
this.list.clear();
|
this.list.clear();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized Object get(int index) {
|
public synchronized Object get(int index) {
|
||||||
|
synchronized (lock) {
|
||||||
return this.list.get(index);
|
return this.list.get(index);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized Object set(int index, Object element) {
|
public synchronized Object set(int index, Object element) {
|
||||||
|
synchronized (lock) {
|
||||||
return this.list.set(index, element);
|
return this.list.set(index, element);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized void add(int index, Object element) {
|
public synchronized void add(int index, Object element) {
|
||||||
|
synchronized (lock) {
|
||||||
this.list.add(index, element);
|
this.list.add(index, element);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized Object remove(int index) {
|
public synchronized Object remove(int index) {
|
||||||
|
synchronized (lock) {
|
||||||
return this.list.remove(index);
|
return this.list.remove(index);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized int indexOf(Object o) {
|
public synchronized int indexOf(Object o) {
|
||||||
|
synchronized (lock) {
|
||||||
return this.list.indexOf(o);
|
return this.list.indexOf(o);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized int lastIndexOf(Object o) {
|
public synchronized int lastIndexOf(Object o) {
|
||||||
|
synchronized (lock) {
|
||||||
return this.list.lastIndexOf(o);
|
return this.list.lastIndexOf(o);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized ListIterator listIterator() {
|
public synchronized ListIterator listIterator() {
|
||||||
|
synchronized (lock) {
|
||||||
return this.list.listIterator();
|
return this.list.listIterator();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized ListIterator listIterator(int index) {
|
public synchronized ListIterator listIterator(int index) {
|
||||||
|
synchronized (lock) {
|
||||||
return this.list.listIterator(index);
|
return this.list.listIterator(index);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized List subList(int fromIndex, int toIndex) {
|
public synchronized List subList(int fromIndex, int toIndex) {
|
||||||
|
synchronized (lock) {
|
||||||
return this.list.subList(fromIndex, toIndex);
|
return this.list.subList(fromIndex, toIndex);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized boolean retainAll(Collection c) {
|
public synchronized boolean retainAll(Collection c) {
|
||||||
|
synchronized (lock) {
|
||||||
return this.list.retainAll(c);
|
return this.list.retainAll(c);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized boolean removeAll(Collection c) {
|
public synchronized boolean removeAll(Collection c) {
|
||||||
|
synchronized (lock) {
|
||||||
return this.list.removeAll(c);
|
return this.list.removeAll(c);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized boolean containsAll(Collection c) {
|
public synchronized boolean containsAll(Collection c) {
|
||||||
|
synchronized (lock) {
|
||||||
return this.list.containsAll(c);
|
return this.list.containsAll(c);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public synchronized Object[] toArray(Object[] a) {
|
public synchronized Object[] toArray(Object[] a) {
|
||||||
|
synchronized (lock) {
|
||||||
return this.list.toArray(a);
|
return this.list.toArray(a);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user