using System;
using System.Collections.Generic;

// Generics mutate tests

class Disposable<T> : IDisposable
{
	public void Dispose ()
	{
	}
}

interface IFoo<TOne, TTwo>
{
}

class CA<T>
{
	public struct Nested
	{
		public static readonly T Value;
		public readonly T Value2;
	}
}

class Test
{
	static Func<T[]> For<T> (List<T> list)
	{
		T[] t = new T[2];
		return () => {
			for (int i = 0; i < t.Length; ++i) {
				t[i] = list[i];
			}

			return t;
		};
	}

	static Func<T> Throw<T> (T t)
	{
		T l = t;
		return () => {
			throw new ApplicationException (l.ToString ());
		};
	}

	static Func<Type> TypeOf<T> (T t)
	{
		T l = t;
		return () => {
			l = t;
			var e = typeof (Disposable<T>);
			e = typeof (Disposable<>);
			e = typeof (IFoo<,>);
			return typeof (T);
		};
	}

	static Func<T> Do<T> (T t)
	{
		T l = t;
		return () => {
			T t2;
			do {
				t2 = l;
			} while (default (T) != null);

			return t2;
		};
	}

	static Func<T> Lock<T> (T t)
	{
		T l = t;
		return () => {
			lock (l.GetType ()) {
				l = default (T);
				return l;
			}
		};
	}

	static Func<T> Catch<T> (T t)
	{
		T l = t;
		return () => {
			try {
				throw new ApplicationException (l.ToString ());
			} catch {
				return l;
			}
		};
	}

	static Func<T> Catch_2<T> (T t) where T : Exception
	{
		T l = t;
		return () => {
			try {
				throw new NotSupportedException ();
			} catch (T e) {
				return l;
			} catch {
				throw new ApplicationException ("Should not be reached");
			}
		};
	}

	static Func<T> Finally<T> (T t)
	{
		T l = t;
		return () => {
			try {
				l = Lock (l) ();
			} finally {
				l = default (T);
			}

			return l;
		};
	}

	static Func<T> Using<T> (T t)
	{
		T l = t;
		using (var d = new Disposable<T> ()) {
			return () => {
				return l;
			};
		}
	}

	static Func<T> Switch<T> (T t)
	{
		T l = t;
		int? i = 0;
		return () => {
			switch (i) {
			default: return l;
			}
		};
	}

	static Func<List<T>> ForForeach<T> (T[] t)
	{
		return () => {
			foreach (T e in t)
				return new List<T> () { e };

			throw new ApplicationException ();
		};
	}

	public void ArrayMutate<T> (T[] array)
	{
		int r = 4;
		Action<int> anonMeth = delegate (int slc) {
			long[] idx = new long[] { 0, 0 };
			for (int i = 0; i < r; i++) {
				idx[0] = i;
			}
		};
	}

	static Func<T[][]> ArrayMultiMutate<T> (T[][] array)
	{
		return () => {
			for (int i = 0; i < 3; i++) {
				array[i][i] = default (T);
			}

			return array;
		};
	}

	static Func<int> ArrayMultiMutate<T> (T[,] array)
	{
		return () => {
			return array[0, 0].GetHashCode ();
		};
	}

	static Func<T[]> NestedTypeMutate<T> () where T : IEquatable<T>
	{
		var local = new CA<T>.Nested ();
		return () => {
			return new[] { CA<T>.Nested.Value, local.Value2 };
		};
	}

	public static int Main ()
	{
		if (For (new List<int> { 5, 10 }) ()[1] != 10)
			return 1;

		var t = Throw (5);
		try {
			t ();
			return 2;
		} catch (ApplicationException) {
		}

		var t3 = Do ("rr");
		if (t3 () != "rr")
			return 3;

		var t4 = Lock ('f');
		if (t4 () != '\0')
			return 4;

		var t5 = Catch (3);
		if (t5 () != 3)
			return 5;

		var ex = new NotSupportedException ();
		var t5_2 = Catch_2 (ex);
		if (t5_2 () != ex)
			return 52;

		var t6 = Finally (5);
		if (t6 () != 0)
			return 6;

		var t7 = Using (1.1);
		if (t7 () != 1.1)
			return 7;

		var t8 = Switch (55);
		if (t8 () != 55)
			return 8;

		var t9 = ForForeach (new[] { 4, 1 });
		if (t9 ()[0] != 4)
			return 9;

		var t10 = ArrayMultiMutate (new string[][] { new string[] { "a", "b", "c" }, new string[] { "1", "2", "3" }, new string[] { "A", "B", "C" } });
		if (t10 ()[2][2] != null)
			return 10;

		var array = new short[,] { { 10, 20 } };
		var t10a = ArrayMultiMutate (array);
		if (t10a () != array[0, 0].GetHashCode ())
			return 100;

		var t11 = TypeOf ("b");
		if (t11 () != typeof (string))
			return 11;

		var t12 = NestedTypeMutate<ulong> () ();
		if (t12[0] != 0 || t12[1] != 0)
			return 12;

		Console.WriteLine ("OK");
		return 0;
	}
}