/*
 * This file is part of the L2J Mobius project.
 * 
 * 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 3 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, see .
 */
package com.l2jmobius.util;
import java.util.Iterator;
import java.util.NoSuchElementException;
import com.l2jmobius.Config;
import com.l2jmobius.gameserver.model.L2Object;
/**
 * This class is a highly optimized hashtable, where keys are integers. The main goal of this class is to allow concurent read/iterate and write access to this table, plus minimal used memory. This class uses plain array as the table of values, and keys are used to get position in the table. If the
 * position is already busy, we iterate to the next position, unil we find the needed element or null. To iterate over the table (read access) we may simply iterate throgh table array. In case we remove an element from the table, we check - if the next position is null, we reset table's slot to
 * null, otherwice we assign it to a dummy value
 * @author mkizub
 * @param  type of values stored in this hashtable
 */
public final class L2ObjectHashMapextends L2ObjectMap
{
	private final boolean TRACE = false;
	private final boolean DEBUG = false;
	
	private final static int[] primes =
	{
		5,
		7,
		11,
		17,
		23,
		29,
		37,
		47,
		59,
		71,
		89,
		107,
		131,
		163,
		197,
		239,
		293,
		353,
		431,
		521,
		631,
		761,
		919,
		1103,
		1327,
		1597,
		1931,
		2333,
		2801,
		3371,
		4049,
		4861,
		5839,
		7013,
		8419,
		10103,
		12143,
		14591,
		17519,
		21023,
		25229,
		30293,
		36353,
		43627,
		52361,
		62851,
		75431,
		90523,
		108631,
		130363,
		156437,
		187751,
		225307,
		270371,
		324449,
		389357,
		467237,
		560689,
		672827,
		807403,
		968897,
		1162687,
		1395263,
		1674319,
		2009191,
		2411033,
		2893249,
		3471899,
		4166287,
		4999559,
		5999471,
		7199369
	};
	
	private T[] table;
	private int[] keys;
	private int count;
	
	private static int getPrime(int min)
	{
		for (final int prime : primes)
		{
			if (prime >= min)
			{
				return prime;
			}
		}
		throw new OutOfMemoryError();
	}
	
	@SuppressWarnings("unchecked")
	public L2ObjectHashMap()
	{
		final int size = primes[0];
		table = (T[]) new L2Object[size];
		keys = new int[size];
		if (DEBUG)
		{
			check();
		}
	}
	
	/*
	 * (non-Javadoc)
	 * @see com.l2jmobius.util.L2ObjectMap#size()
	 */
	@Override
	public int size()
	{
		return count;
	}
	
	/*
	 * (non-Javadoc)
	 * @see com.l2jmobius.util.L2ObjectMap#isEmpty()
	 */
	@Override
	public boolean isEmpty()
	{
		return count == 0;
	}
	
	/*
	 * (non-Javadoc)
	 * @see com.l2jmobius.util.L2ObjectMap#clear()
	 */
	@Override
	@SuppressWarnings("unchecked")
	public synchronized void clear()
	{
		final int size = primes[0];
		table = (T[]) new L2Object[size];
		keys = new int[size];
		count = 0;
		if (DEBUG)
		{
			check();
		}
	}
	
	private void check()
	{
		if (DEBUG)
		{
			int cnt = 0;
			for (int i = 0; i < table.length; i++)
			{
				final L2Object obj = table[i];
				if (obj == null)
				{
					assert (keys[i] == 0) || (keys[i] == 0x80000000);
				}
				else
				{
					cnt++;
					assert obj.getObjectId() == (keys[i] & 0x7FFFFFFF);
				}
			}
			assert cnt == count;
		}
	}
	
	/*
	 * (non-Javadoc)
	 * @see com.l2jmobius.util.L2ObjectMap#put(T)
	 */
	@Override
	public synchronized void put(T obj)
	{
		if (count >= (table.length / 2))
		{
			expand();
		}
		final int hashcode = obj.getObjectId();
		if (Config.ASSERT)
		{
			assert hashcode > 0;
		}
		int seed = hashcode;
		final int incr = 1 + (((seed >> 5) + 1) % (table.length - 1));
		int ntry = 0;
		int slot = -1; // keep last found slot
		do
		{
			final int pos = (seed % table.length) & 0x7FFFFFFF;
			if (table[pos] == null)
			{
				if (slot < 0)
				{
					slot = pos;
				}
				if (keys[pos] >= 0)
				{
					// found an empty slot without previous collisions,
					// but use previously found slot
					keys[slot] = hashcode;
					table[slot] = obj;
					count++;
					if (TRACE)
					{
						System.err.println("ht: put obj id=" + hashcode + " at slot=" + slot);
					}
					if (DEBUG)
					{
						check();
					}
					return;
				}
			}
			else
			{
				// check if we are adding the same object
				if (table[pos] == obj)
				{
					return;
				}
				// this should never happen
				if (Config.ASSERT)
				{
					assert obj.getObjectId() != table[pos].getObjectId();
				}
				// if there was no collisions at this slot, and we found a free
				// slot previously - use found slot
				if ((slot >= 0) && (keys[pos] > 0))
				{
					keys[slot] |= hashcode; // preserve collision bit
					table[slot] = obj;
					count++;
					if (TRACE)
					{
						System.err.println("ht: put obj id=" + hashcode + " at slot=" + slot);
					}
					if (DEBUG)
					{
						check();
					}
					return;
				}
			}
			
			// set collision bit
			keys[pos] |= 0x80000000;
			// calculate next slot
			seed += incr;
		}
		while (++ntry < table.length);
		if (DEBUG)
		{
			check();
		}
		throw new IllegalStateException();
	}
	
	/*
	 * (non-Javadoc)
	 * @see com.l2jmobius.util.L2ObjectMap#remove(T)
	 */
	@Override
	public synchronized void remove(T obj)
	{
		final int hashcode = obj.getObjectId();
		if (Config.ASSERT)
		{
			assert hashcode > 0;
		}
		int seed = hashcode;
		final int incr = 1 + (((seed >> 5) + 1) % (table.length - 1));
		int ntry = 0;
		do
		{
			final int pos = (seed % table.length) & 0x7FFFFFFF;
			if (table[pos] == obj)
			{
				// found the object
				keys[pos] &= 0x80000000; // preserve collision bit
				table[pos] = null;
				count--;
				if (TRACE)
				{
					System.err.println("ht: remove obj id=" + hashcode + " from slot=" + pos);
				}
				if (DEBUG)
				{
					check();
				}
				return;
			}
			// check for collision (if we previously deleted element)
			if ((table[pos] == null) && (keys[pos] >= 0))
			{
				if (DEBUG)
				{
					check();
				}
				return; // throw new IllegalArgumentException();
			}
			// calculate next slot
			seed += incr;
		}
		while (++ntry < table.length);
		if (DEBUG)
		{
			check();
		}
		throw new IllegalStateException();
	}
	
	/*
	 * (non-Javadoc)
	 * @see com.l2jmobius.util.L2ObjectMap#get(int)
	 */
	@Override
	public T get(int id)
	{
		final int size = table.length;
		if (id <= 0)
		{
			return null;
		}
		if (size <= 11)
		{
			// for small tables linear check is fast
			for (int i = 0; i < size; i++)
			{
				if ((keys[i] & 0x7FFFFFFF) == id)
				{
					return table[i];
				}
			}
			return null;
		}
		int seed = id;
		final int incr = 1 + (((seed >> 5) + 1) % (size - 1));
		int ntry = 0;
		do
		{
			final int pos = (seed % size) & 0x7FFFFFFF;
			if ((keys[pos] & 0x7FFFFFFF) == id)
			{
				return table[pos];
			}
			// check for collision (if we previously deleted element)
			if ((table[pos] == null) && (keys[pos] >= 0))
			{
				return null;
			}
			// calculate next slot
			seed += incr;
		}
		while (++ntry < size);
		return null;
	}
	
	/*
	 * (non-Javadoc)
	 * @see com.l2jmobius.util.L2ObjectMap#contains(T)
	 */
	@Override
	public boolean contains(T obj)
	{
		return get(obj.getObjectId()) != null;
	}
	
	@SuppressWarnings("unchecked")
	private /* already synchronized in put() */ void expand()
	{
		final int newSize = getPrime(table.length + 1);
		final L2Object[] newTable = new L2Object[newSize];
		final int[] newKeys = new int[newSize];
		
		// over all old entries
		next_entry: for (int i = 0; i < table.length; i++)
		{
			final L2Object obj = table[i];
			if (obj == null)
			{
				continue;
			}
			final int hashcode = keys[i] & 0x7FFFFFFF;
			if (Config.ASSERT)
			{
				assert hashcode == obj.getObjectId();
			}
			int seed = hashcode;
			final int incr = 1 + (((seed >> 5) + 1) % (newSize - 1));
			int ntry = 0;
			do
			{
				final int pos = (seed % newSize) & 0x7FFFFFFF;
				if (newTable[pos] == null)
				{
					if (Config.ASSERT)
					{
						assert (newKeys[pos] == 0) && (hashcode != 0);
					}
					// found an empty slot without previous collisions,
					// but use previously found slot
					newKeys[pos] = hashcode;
					newTable[pos] = obj;
					if (TRACE)
					{
						System.err.println("ht: move obj id=" + hashcode + " from slot=" + i + " to slot=" + pos);
					}
					continue next_entry;
				}
				// set collision bit
				newKeys[pos] |= 0x80000000;
				// calculate next slot
				seed += incr;
			}
			while (++ntry < newSize);
			throw new IllegalStateException();
		}
		table = (T[]) newTable;
		keys = newKeys;
		if (DEBUG)
		{
			check();
		}
	}
	
	/*
	 * (non-Javadoc)
	 * @see com.l2jmobius.util.L2ObjectMap#iterator()
	 */
	@Override
	public Iterator iterator()
	{
		return new Itr(table);
	}
	
	class Itr implements Iterator
	{
		private final T[] array;
		private int nextIdx;
		private T nextObj;
		private T lastRet;
		
		Itr(T[] pArray)
		{
			this.array = pArray;
			for (; nextIdx < array.length; nextIdx++)
			{
				nextObj = array[nextIdx];
				if (nextObj != null)
				{
					return;
				}
			}
		}
		
		@Override
		public boolean hasNext()
		{
			return nextObj != null;
		}
		
		@Override
		public T next()
		{
			if (nextObj == null)
			{
				throw new NoSuchElementException();
			}
			lastRet = nextObj;
			for (nextIdx++; nextIdx < array.length; nextIdx++)
			{
				nextObj = array[nextIdx];
				if (nextObj != null)
				{
					break;
				}
			}
			if (nextIdx >= array.length)
			{
				nextObj = null;
			}
			return lastRet;
		}
		
		@Override
		public void remove()
		{
			if (lastRet == null)
			{
				throw new IllegalStateException();
			}
			L2ObjectHashMap.this.remove(lastRet);
		}
	}
}