/*
* Copyright (C) 2008 The Guava Authors
 *
 * Licensed 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 com.google.common.collect;

import static com.google.common.collect.CollectPreconditions.checkEntryNotNull;

import javax.annotation.Nullable;

import com.google.common.annotations.GwtCompatible;
import com.google.common.collect.ImmutableMapEntry.TerminalEntry;

/**
 * Implementation of {@link ImmutableMap} with two or more entries.
 *
 * @author Jesse Wilson
 * @author Kevin Bourrillion
 * @author Gregory Kick
 */
@GwtCompatible(serializable = true, emulated = true)
final class RegularImmutableMap<K, V> extends ImmutableMap<K, V> {

	// entries in insertion order
	private final transient ImmutableMapEntry<K, V>[] entries;
	// array of linked lists of entries
	private final transient ImmutableMapEntry<K, V>[] table;
	// 'and' with an int to get a table index
	private final transient int mask;

	RegularImmutableMap(TerminalEntry<?, ?>... theEntries) {
		this(theEntries.length, theEntries);
	}

	/**
	 * Constructor for RegularImmutableMap that takes as input an array of
	 * {@code TerminalEntry} entries. Assumes that these entries have already been
	 * checked for null.
	 * 
	 * <p>
	 * This allows reuse of the entry objects from the array in the actual
	 * implementation.
	 */
	RegularImmutableMap(int size, TerminalEntry<?, ?>[] theEntries) {
		entries = createEntryArray(size);
		int tableSize = Hashing.closedTableSize(size, MAX_LOAD_FACTOR);
		table = createEntryArray(tableSize);
		mask = tableSize - 1;
		for (int entryIndex = 0; entryIndex < size; entryIndex++) {
			@SuppressWarnings("unchecked")
			TerminalEntry<K, V> entry = (TerminalEntry<K, V>) theEntries[entryIndex];
			K key = entry.getKey();
			int tableIndex = Hashing.smear(key.hashCode()) & mask;
			@Nullable
			ImmutableMapEntry<K, V> existing = table[tableIndex];
			// prepend, not append, so the entries can be immutable
			ImmutableMapEntry<K, V> newEntry = (existing == null) ? entry
					: new NonTerminalMapEntry<K, V>(entry, existing);
			table[tableIndex] = newEntry;
			entries[entryIndex] = newEntry;
			checkNoConflictInBucket(key, newEntry, existing);
		}
	}

	/**
	 * Constructor for RegularImmutableMap that makes no assumptions about the input
	 * entries.
	 */
	RegularImmutableMap(Entry<?, ?>[] theEntries) {
		int size = theEntries.length;
		entries = createEntryArray(size);
		int tableSize = Hashing.closedTableSize(size, MAX_LOAD_FACTOR);
		table = createEntryArray(tableSize);
		mask = tableSize - 1;
		for (int entryIndex = 0; entryIndex < size; entryIndex++) {
			@SuppressWarnings("unchecked") // all our callers carefully put in only Entry<K, V>s
			Entry<K, V> entry = (Entry<K, V>) theEntries[entryIndex];
			K key = entry.getKey();
			V value = entry.getValue();
			checkEntryNotNull(key, value);
			int tableIndex = Hashing.smear(key.hashCode()) & mask;
			@Nullable
			ImmutableMapEntry<K, V> existing = table[tableIndex];
			// prepend, not append, so the entries can be immutable
			ImmutableMapEntry<K, V> newEntry = (existing == null) ? new TerminalEntry<K, V>(key, value)
					: new NonTerminalMapEntry<K, V>(key, value, existing);
			table[tableIndex] = newEntry;
			entries[entryIndex] = newEntry;
			checkNoConflictInBucket(key, newEntry, existing);
		}
	}

	private void checkNoConflictInBucket(K key, ImmutableMapEntry<K, V> entry, ImmutableMapEntry<K, V> bucketHead) {
		for (; bucketHead != null; bucketHead = bucketHead.getNextInKeyBucket()) {
			checkNoConflict(!key.equals(bucketHead.getKey()), "key", entry, bucketHead);
		}
	}

	private static final class NonTerminalMapEntry<K, V> extends ImmutableMapEntry<K, V> {
		private final ImmutableMapEntry<K, V> nextInKeyBucket;

		NonTerminalMapEntry(K key, V value, ImmutableMapEntry<K, V> nextInKeyBucket) {
			super(key, value);
			this.nextInKeyBucket = nextInKeyBucket;
		}

		NonTerminalMapEntry(ImmutableMapEntry<K, V> contents, ImmutableMapEntry<K, V> nextInKeyBucket) {
			super(contents);
			this.nextInKeyBucket = nextInKeyBucket;
		}

		@Override
		ImmutableMapEntry<K, V> getNextInKeyBucket() {
			return nextInKeyBucket;
		}

		@Override
		@Nullable
		ImmutableMapEntry<K, V> getNextInValueBucket() {
			return null;
		}

	}

	/**
	 * Closed addressing tends to perform well even with high load factors. Being
	 * conservative here ensures that the table is still likely to be relatively
	 * sparse (hence it misses fast) while saving space.
	 */
	private static final double MAX_LOAD_FACTOR = 1.2;

	/**
	 * Creates an {@code ImmutableMapEntry} array to hold parameterized entries. The
	 * result must never be upcast back to ImmutableMapEntry[] (or Object[], etc.),
	 * or allowed to escape the class.
	 */
	@SuppressWarnings("unchecked") // Safe as long as the javadocs are followed
	private ImmutableMapEntry<K, V>[] createEntryArray(int size) {
		return new ImmutableMapEntry[size];
	}

	@Override
	public V get(@Nullable Object key) {
		if (key == null) {
			return null;
		}
		int index = Hashing.smear(key.hashCode()) & mask;
		for (ImmutableMapEntry<K, V> entry = table[index]; entry != null; entry = entry.getNextInKeyBucket()) {
			K candidateKey = entry.getKey();

			/*
			 * Assume that equals uses the == optimization when appropriate, and that it
			 * would check hash codes as an optimization when appropriate. If we did these
			 * things, it would just make things worse for the most performance-conscious
			 * users.
			 */
			if (key.equals(candidateKey)) {
				return entry.getValue();
			}
		}
		return null;
	}

	@Override
	public int size() {
		return entries.length;
	}

	@Override
	boolean isPartialView() {
		return false;
	}

	@Override
	ImmutableSet<Entry<K, V>> createEntrySet() {
		return new EntrySet();
	}

	@SuppressWarnings("serial") // uses writeReplace(), not default serialization
	private class EntrySet extends ImmutableMapEntrySet<K, V> {
		@Override
		ImmutableMap<K, V> map() {
			return RegularImmutableMap.this;
		}

		@Override
		public UnmodifiableIterator<Entry<K, V>> iterator() {
			return asList().iterator();
		}

		@Override
		ImmutableList<Entry<K, V>> createAsList() {
			return new RegularImmutableAsList<Entry<K, V>>(this, entries);
		}
	}

	// This class is never actually serialized directly, but we have to make the
	// warning go away (and suppressing would suppress for all nested classes too)
	private static final long serialVersionUID = 0;
}