Просмотр исходного кода

2010-04-15 Jérémie Laval <[email protected]>

In class/System.Core/:
   * System.Core_test.dll.sources: Add PLinq unit tests

In class/System.Core/Test/System.Linq/:
   * ParallelEnumerableTests.cs:
   * ParallelTestHelper.cs: Add PLinq unit tests

svn path=/trunk/mcs/; revision=155534
Jérémie Laval 15 лет назад
Родитель
Сommit
f035f2879f

+ 4 - 0
mcs/class/System.Core/ChangeLog

@@ -1,3 +1,7 @@
+2010-04-15  Jérémie Laval  <[email protected]>
+
+	* System.Core_test.dll.sources: Add PLinq unit tests
+
 2010-04-15  Jérémie Laval  <[email protected]>
 
 	* net_4_0_System.Core.dll.sources: add PLinq sources

+ 2 - 0
mcs/class/System.Core/System.Core_test.dll.sources

@@ -78,3 +78,5 @@ System.Security.Cryptography/SHA384CryptoServiceProviderTest.cs
 System.Security.Cryptography/SHA512CngTest.cs
 System.Security.Cryptography/SHA512CryptoServiceProviderTest.cs
 System.Runtime.CompilerServices/DynamicAttributeTest.cs
+System.Linq/ParallelEnumerableTests.cs
+System.Linq/ParallelTestHelper.cs

+ 5 - 0
mcs/class/System.Core/Test/System.Linq/ChangeLog

@@ -1,3 +1,8 @@
+2010-04-15  Jérémie Laval  <[email protected]>
+
+	* ParallelEnumerableTests.cs:
+	* ParallelTestHelper.cs: Add PLinq unit tests
+
 2010-03-24  Jb Evain  <[email protected]>
 
 	* EnumerableTest.cs: add test for #582502.

+ 743 - 0
mcs/class/System.Core/Test/System.Linq/ParallelEnumerableTests.cs

@@ -0,0 +1,743 @@
+// ParallelEnumerableTests.cs
+//
+// Copyright (c) 2008 Jérémie "Garuma" Laval
+//
+// Based on Enumerable test suite by Jb Evain ([email protected])
+//
+// Permission is hereby granted, free of charge, to any person obtaining a copy
+// of this software and associated documentation files (the "Software"), to deal
+// in the Software without restriction, including without limitation the rights
+// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+// copies of the Software, and to permit persons to whom the Software is
+// furnished to do so, subject to the following conditions:
+//
+// The above copyright notice and this permission notice shall be included in
+// all copies or substantial portions of the Software.
+//
+// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
+// THE SOFTWARE.
+//
+//
+
+using System;
+using System.Threading;
+using System.Linq;
+
+using System.Collections;
+using System.Collections.Generic;
+
+using NUnit.Framework;
+
+namespace MonoTests.System.Linq.Parallel
+{
+	[TestFixtureAttribute]
+	public class ParallelEnumerableTests
+	{
+		IEnumerable<int> baseEnumerable;
+		
+		[SetUpAttribute]
+		public void Setup ()
+		{
+			baseEnumerable = Enumerable.Range(1, 10000);
+		}
+		
+		void AreEquivalent (IEnumerable<int> syncEnumerable, IEnumerable<int> asyncEnumerable, int count)
+		{
+			int[] sync  = Enumerable.ToArray(syncEnumerable);
+			int[] async = Enumerable.ToArray(asyncEnumerable);
+			
+			// This is not AreEquals because ParallelQuery is non-deterministic (IParallelOrderedEnumerable is)
+			// thus the order of the initial Enumerable might not be preserved
+			CollectionAssert.AreEquivalent(sync, async, "#" + count);
+		}
+		
+		void AreEquivalent<T> (IEnumerable<T> syncEnumerable, IEnumerable<T> asyncEnumerable, int count)
+		{
+			T[] sync  = Enumerable.ToArray(syncEnumerable);
+			T[] async = Enumerable.ToArray(asyncEnumerable);
+			
+			// This is not AreEquals because ParallelQuery is non-deterministic (IParallelOrderedEnumerable is)
+			// thus the order of the initial Enumerable might not be preserved
+			CollectionAssert.AreEquivalent(sync, async, "#" + count);
+		}
+		
+		static void AssertAreSame<T> (IEnumerable<T> expected, IEnumerable<T> actual)
+		{
+			if (expected == null) {
+				Assert.IsNull (actual);
+				return;
+			}
+
+			Assert.IsNotNull (actual);
+
+			IEnumerator<T> ee = expected.GetEnumerator ();
+			IEnumerator<T> ea = actual.GetEnumerator ();
+
+			while (ee.MoveNext ()) {
+				Assert.IsTrue (ea.MoveNext (), "'" + ee.Current + "' expected.");
+				Assert.AreEqual (ee.Current, ea.Current);
+			}
+
+			if (ea.MoveNext ())
+				Assert.Fail ("Unexpected element: " + ea.Current);
+		}
+		
+		public static void AssertException<T> (Action action) where T : Exception
+		{
+			try {
+				action ();
+			}
+			catch (T) {
+				return;
+			}
+			Assert.Fail ("Expected: " + typeof (T).Name);
+		}
+
+		static void AssertAreSame<K, V> (K expectedKey, IEnumerable<V> expectedValues, IGrouping<K, V> actual)
+		{
+			if (expectedValues == null) {
+				Assert.IsNull (actual);
+				return;
+			}
+
+			Assert.IsNotNull (actual);
+
+			Assert.AreEqual (expectedKey, actual.Key);
+
+			var ee = expectedValues.GetEnumerator ();
+			var ea = actual.GetEnumerator ();
+
+			while (ee.MoveNext ()) {
+				Assert.IsTrue (ea.MoveNext (), "'" + ee.Current + "' expected.");
+				Assert.AreEqual (ee.Current, ea.Current);
+			}
+
+			if (ea.MoveNext ())
+				Assert.Fail ("Unexpected element: " + ee.Current);
+		}
+
+		static void AssertAreSame<K, V> (IDictionary<K, IEnumerable<V>> expected, IEnumerable<IGrouping<K, V>> actual)
+		{
+			if (expected == null) {
+				Assert.IsNull (actual);
+				return;
+			}
+
+			Assert.IsNotNull (actual);
+
+			var ee = expected.GetEnumerator ();
+			var ea = actual.GetEnumerator ();
+
+			while (ee.MoveNext ()) {
+				Assert.IsTrue (ea.MoveNext (), "'" + ee.Current.Key + "' expected.");
+				AssertAreSame (ee.Current.Key, ee.Current.Value, ea.Current);
+			}
+
+			if (ea.MoveNext ())
+				Assert.Fail ("Unexpected element: " + ee.Current.Key);
+		}
+
+		static void AssertAreSame<K, V> (IDictionary<K, IEnumerable<V>> expected, ILookup<K, V> actual)
+		{
+			if (expected == null) {
+				Assert.IsNull (actual);
+				return;
+			}
+
+			Assert.IsNotNull (actual);
+
+			var ee = expected.GetEnumerator ();
+			var ea = actual.GetEnumerator ();
+
+			while (ee.MoveNext ()) {
+				Assert.IsTrue (ea.MoveNext (), "'" + ee.Current.Key + "' expected.");
+				AssertAreSame (ee.Current.Key, ee.Current.Value, ea.Current);
+			}
+
+			if (ea.MoveNext ())
+				Assert.Fail ("Unexpected element: " + ee.Current.Key);
+		}
+
+		static void AssertAreSame<K, V> (IDictionary<K, V> expected, IDictionary<K, V> actual)
+		{
+			if (expected == null) {
+				Assert.IsNull (actual);
+				return;
+			}
+
+			Assert.IsNotNull (actual);
+
+			var ee = expected.GetEnumerator ();
+			var ea = actual.GetEnumerator ();
+
+			while (ee.MoveNext ()) {
+				Assert.IsTrue (ea.MoveNext (), "'" + ee.Current.Key + ", " + ee.Current.Value + "' expected.");
+				Assert.AreEqual (ee.Current.Key, ea.Current.Key);
+				Assert.AreEqual (ee.Current.Value, ea.Current.Value);
+			}
+
+			if (ea.MoveNext ())
+				Assert.Fail ("Unexpected element: " + ee.Current.Key + ", " + ee.Current.Value);
+		}
+
+		[Test]
+		public void SelectTestCase ()
+		{
+			ParallelTestHelper.Repeat (() => {
+				IEnumerable<int> sync  = baseEnumerable.Select (i => i * i);
+				IEnumerable<int> async = baseEnumerable.AsParallel ().Select (i => i * i);
+				
+				AreEquivalent(sync, async, 1);
+			});
+		}
+			
+		[Test]
+		public void WhereTestCase ()
+		{
+			ParallelTestHelper.Repeat (() => {
+				IEnumerable<int> sync  = baseEnumerable.Where(i => i % 2 == 0);
+				IEnumerable<int> async = baseEnumerable.AsParallel().Where(i => i % 2 == 0);
+				
+				AreEquivalent(sync, async, 1);
+			});
+		}
+		
+		[Test]
+		public void CountTestCase ()
+		{
+			ParallelTestHelper.Repeat (() => {
+				int sync  = baseEnumerable.Count();
+				int async = baseEnumerable.AsParallel().Count();
+				
+				Assert.AreEqual(sync, async, "#1");
+			});
+		}
+		
+		[Test]
+		public void AggregateTestCase ()
+		{
+			ParallelTestHelper.Repeat (() => {
+				ParallelQuery<int> range = ParallelEnumerable.Repeat (5, 2643);
+				double average = range.Aggregate(() => new double[2],
+				                                 (acc, elem) => { acc[0] += elem; acc[1]++; return acc; },
+				(acc1, acc2) => { acc1[0] += acc2[0]; acc1[1] += acc2[1]; return acc1; },
+				acc => acc[0] / acc[1]);
+				
+				Assert.AreEqual(5.0, average, "#1");
+			});
+		}
+		
+		[Test]
+		public void TestSimpleExcept ()
+		{
+			ParallelTestHelper.Repeat (() => {
+				int [] first = {0, 1, 2, 3, 4, 5};
+				int [] second = {2, 4, 6};
+				int [] result = {0, 1, 3, 5};
+	
+				AreEquivalent (result, first.AsParallel ().Except (second.AsParallel ()), 1);
+			});
+		}
+
+		[Test]
+		public void TestSimpleIntersect ()
+		{
+			ParallelTestHelper.Repeat (() => {
+				int [] first = {0, 1, 2, 3, 4, 5};
+				int [] second = {2, 4, 6};
+				int [] result = {2, 4};
+	
+				AreEquivalent (result, first.AsParallel ().Intersect (second.AsParallel ()), 1);
+			});
+		}
+
+		[Test]
+		public void TestSimpleUnion ()
+		{
+			ParallelTestHelper.Repeat (() => {
+				int [] first = {0, 1, 2, 3, 4, 5};
+				int [] second = {2, 4, 6};
+				int [] result = {0, 1, 2, 3, 4, 5, 6};
+				
+				AreEquivalent (result, first.AsParallel ().Union (second.AsParallel ()), 1);
+			});
+		}
+		
+		class Foo {}
+		class Bar : Foo {}
+
+		[Test]
+		public void TestCast ()
+		{
+			Bar a = new Bar ();
+			Bar b = new Bar ();
+			Bar c = new Bar ();
+
+			Foo [] foos = new Foo [] {a, b, c};
+			Bar [] result = new Bar [] {a, b, c};
+
+			AreEquivalent (result, foos.AsParallel ().Cast<Bar> (), 1);
+		}
+		
+		[Test]
+		public void TestSkip ()
+		{
+			int [] data = {0, 1, 2, 3, 4, 5};
+			int [] result = {3, 4, 5};
+
+			AssertAreSame (result, data.AsParallel ().AsOrdered ().Skip (3));
+		}
+
+		/*[Test]
+		public void TestSkipWhile ()
+		{
+			int [] data = {0, 1, 2, 3, 4, 5};
+			int [] result = {3, 4, 5};
+
+			AssertAreSame (result, data.AsParallel ().AsOrdered ().SkipWhile (i => i < 3));
+		}
+
+		[Test]
+		public void TestTake ()
+		{
+			int [] data = {0, 1, 2, 3, 4, 5};
+			int [] result = {0, 1, 2};
+
+			AssertAreSame (result, data.AsParallel ().AsOrdered ().Take (3));
+		}
+
+		[Test]
+		public void TestTakeWhile ()
+		{
+			int [] data = {0, 1, 2, 3, 4, 5};
+			int [] result = {0, 1, 2};
+
+			AssertAreSame (result, data.AsParallel ().AsOrdered ().TakeWhile (i => i < 3));
+		}*/
+		
+//		[Test]
+//		public void TestLast ()
+//		{
+//			int [] data = {1, 2, 3};
+//
+//			Assert.AreEqual (3, data.AsParallel ().Last ());
+//		}
+//
+//		[Test]
+//		public void TestLastOrDefault ()
+//		{
+//			int [] data = {};
+//
+//			Assert.AreEqual (default (int), data.AsParallel ().LastOrDefault ());
+//		}
+//
+//		[Test]
+//		public void TestFirst ()
+//		{
+//			int [] data = {1, 2, 3};
+//
+//			Assert.AreEqual (1, data.AsParallel ().First ());
+//		}
+//
+//		[Test]
+//		public void TestFirstOrDefault ()
+//		{
+//			int [] data = {};
+//
+//			Assert.AreEqual (default (int), data.AsParallel ().FirstOrDefault ());
+//		}
+		
+		[Test]
+		public void TestReverse ()
+		{
+			int [] data = {0, 1, 2, 3, 4};
+			int [] result = {4, 3, 2, 1, 0};
+
+			AssertAreSame (result, data.AsParallel ().AsOrdered ().Reverse ());
+			AssertAreSame (result, ParallelEnumerable.Range (0, 5).AsOrdered ().Reverse ());
+		}
+		
+		[Test]
+		public void TestOrderBy ()
+		{
+			ParallelTestHelper.Repeat (() => {
+				int [] array = { 14, 53, 3, 9, 11, 14, 5, 32, 2 };
+				
+				var q = array.AsParallel ().OrderBy ((i) => i);
+				AssertIsOrdered (q, array.Length);
+			});
+		}
+
+		class Baz {
+			string name;
+			int age;
+
+			public string Name
+			{
+				get {
+					if (string.IsNullOrEmpty (name))
+						return Age.ToString ();
+
+					return name + " (" + Age + ")";
+				}
+			}
+
+			public int Age
+			{
+				get { return age + 1; }
+			}
+
+			public Baz (string name, int age)
+			{
+				this.name = name;
+				this.age = age;
+			}
+
+			public override int GetHashCode ()
+			{
+				return this.Age ^ this.Name.GetHashCode ();
+			}
+
+			public override bool Equals (object obj)
+			{
+				Baz b = obj as Baz;
+				if (b == null)
+					return false;
+
+				return b.Age == this.Age && b.Name == this.Name;
+			}
+
+			public override string ToString ()
+			{
+				return this.Name;
+			}
+		}
+
+		static IEnumerable<Baz> CreateBazCollection ()
+		{
+			return new [] {
+				new Baz ("jb", 25),
+				new Baz ("ana", 20),
+				new Baz ("reg", 28),
+				new Baz ("ro", 25),
+				new Baz ("jb", 7),
+			};
+		}
+
+		[Test]
+		public void TestOrderByAgeAscendingTheByNameDescending ()
+		{
+			ParallelTestHelper.Repeat (() => {
+				var q = from b in CreateBazCollection ().AsParallel()
+						orderby b.Age ascending, b.Name descending
+						select b;
+				//var q = CreateBazCollection ().AsParallel ().OrderBy ((b) => b.Age).ThenByDescending ((b) => b.Name);
+	
+				var expected = new [] {
+					new Baz ("jb", 7),
+					new Baz ("ana", 20),
+					new Baz ("ro", 25),
+					new Baz ("jb", 25),
+					new Baz ("reg", 28),
+				};
+				
+				foreach (Baz b in q) {
+					Console.Write(b.Name + ", " + b.Age + "; ");
+				}
+	
+				AssertAreSame (expected, q);
+			});
+		}
+
+		class Data {
+			public int ID { get; set; }
+			public string Name { get; set; }
+
+			public override string ToString ()
+			{
+				return ID + " " + Name;
+			}
+		}
+
+		IEnumerable<Data> CreateData ()
+		{
+			return new [] {
+				new Data { ID = 10, Name = "bcd" },
+				new Data { ID = 20, Name = "Abcd" },
+				new Data { ID = 20, Name = "Ab" },
+				new Data { ID = 10, Name = "Zyx" },
+			};
+		}
+
+		[Test]
+		public void TestOrderByIdDescendingThenByNameAscending ()
+		{
+			ParallelTestHelper.Repeat (() => {
+				var q = from d in CreateData ().AsParallel()
+					orderby d.ID descending, d.Name ascending
+						select d;
+				
+				var list = new List<Data> (q);
+				
+				Assert.AreEqual ("Ab", list [0].Name);
+				Assert.AreEqual ("Abcd", list [1].Name);
+				Assert.AreEqual ("bcd", list [2].Name);
+				Assert.AreEqual ("Zyx", list [3].Name);
+			});
+		}
+
+		static void AssertIsOrdered (IEnumerable<int> e, int count)
+		{
+			int f = int.MinValue;
+			int c = 0;
+			
+			foreach (int i in e) {
+				Assert.IsTrue (f <= i, string.Format ("{0} <= {1}", f, i));
+				f = i;
+				c++;
+			}
+			
+			Assert.AreEqual (count, c);
+		}
+		
+		/*
+		[TestAttribute, Ignore]
+		public void ElementAtTestCase()
+		{
+			ParallelTestHelper.Repeat (() => {
+				Assert.AreEqual(1, baseEnumerable.ElementAt(0), "#1");
+				Assert.AreEqual(51, baseEnumerable.ElementAt(50), "#2");
+				Assert.AreEqual(489, baseEnumerable.ElementAt(488), "#3");
+			});
+		}
+		
+		[TestAttribute, Ignore]
+		public void TakeTestCase()
+		{
+			ParallelTestHelper.Repeat (() => {
+				ParallelQuery<int> async = baseEnumerable.AsParallel().Take(2000);
+				IEnumerable<int> sync = baseEnumerable.Take(2000);
+				
+				AreEquivalent(sync, async, 1);
+				
+				async = baseEnumerable.AsParallel().Take(100);
+				sync = baseEnumerable.Take(100);
+			
+				AreEquivalent(sync, async, 2);
+			});
+		}
+		
+		[TestAttribute, Ignore]
+		public void SkipTestCase()
+		{
+			ParallelTestHelper.Repeat (() => {
+				ParallelQuery<int> async = baseEnumerable.AsParallel().AsOrdered().Skip(2000);
+				IEnumerable<int> sync = baseEnumerable.Skip(2000);
+				
+				AreEquivalent(sync, async, 1);
+				
+				async = baseEnumerable.AsParallel().Skip(100);
+				sync = baseEnumerable.Skip(100);
+				
+				Assert.AreEqual(sync.Count(), async.Count(), "#2");
+			});
+		}
+
+		[TestAttribute, Ignore]
+		public void ZipTestCase()
+		{
+			ParallelTestHelper.Repeat (() => {
+				ParallelQuery<int> async1 = ParallelEnumerable.Range(0, 10000);
+				ParallelQuery<int> async2 = ParallelEnumerable.Repeat(1, 10000).Zip(async1, (e1, e2) => e1 + e2);
+				
+				int[] expected = Enumerable.Range (1, 10000).ToArray ();
+				CollectionAssert.AreEquivalent(expected, Enumerable.ToArray (async2), "#1");
+			});
+		}
+		*/
+		[Test]
+		public void RangeTestCase ()
+		{
+			ParallelTestHelper.Repeat (() => {
+				IEnumerable<int> sync  = Enumerable.Range(1, 1000);
+				IEnumerable<int> async = ParallelEnumerable.Range(1, 1000);
+				
+				AreEquivalent (sync, async, 1);
+			});
+		}
+		
+		[Test]
+		public void RepeatTestCase ()
+		{
+			ParallelTestHelper.Repeat (() => {
+				IEnumerable<int> sync  = Enumerable.Repeat(1, 1000);
+				IEnumerable<int> async = ParallelEnumerable.Repeat(1, 1000);
+				
+				AreEquivalent (sync, async, 1);
+			});
+		}
+		
+		[Test]
+		public void TestSum ()
+		{
+			int [] data = {1, 2, 3, 4};
+
+			Assert.AreEqual (10, data.AsParallel().Sum ());
+		}
+
+		[Test]
+		public void SumOnEmpty ()
+		{
+			int [] data = {};
+
+			Assert.AreEqual (0, data.AsParallel().Sum ());
+		}
+
+		[Test]
+		public void TestMax ()
+		{
+			int [] data = {1, 3, 5, 2};
+
+			Assert.AreEqual (5, data.AsParallel().Max ());
+		}
+
+		[Test]
+		public void TestMin ()
+		{
+			int [] data = {3, 5, 2, 6, 1, 7};
+
+			Assert.AreEqual (1, data.AsParallel().Min ());
+		}
+		
+		[Test]
+		public void TestToListOrdered ()
+		{
+			int [] data = { 2, 3, 5 };
+
+			var list = data.AsParallel().AsOrdered().ToList ();
+
+			AssertAreSame (data, list);
+			AssertIsOrdered (list, data.Length);
+
+			Assert.AreEqual (typeof (List<int>), list.GetType ());
+		}
+
+		[Test]
+		public void TestToArrayOrdered ()
+		{
+			ICollection<int> coll = new List<int> ();
+			coll.Add (0);
+			coll.Add (1);
+			coll.Add (2);
+
+			int [] result = {0, 1, 2};
+
+			var array = coll.AsParallel().AsOrdered().ToArray ();
+
+			AssertAreSame (result, array);
+			AssertIsOrdered (array, result.Length);
+
+			Assert.AreEqual (typeof (int []), array.GetType ());
+		}
+
+		[Test]
+		public void TestToList ()
+		{
+			int [] data = {3, 5, 2};
+
+			var list = data.AsParallel().ToList ();
+
+			CollectionAssert.AreEquivalent (data, list);
+
+			Assert.AreEqual (typeof (List<int>), list.GetType ());
+		}
+
+		[Test]
+		public void TestToArray ()
+		{
+			ICollection<int> coll = new List<int> ();
+			coll.Add (0);
+			coll.Add (1);
+			coll.Add (2);
+
+			int [] result = {0, 1, 2};
+
+			var array = coll.AsParallel().ToArray ();
+
+			CollectionAssert.AreEquivalent (result, array);
+
+			Assert.AreEqual (typeof (int []), array.GetType ());
+		}
+		
+		
+		[Test]
+		public void TestAverageOnInt32 ()
+		{
+			Assert.AreEqual (23.25, (new int [] { 24, 7, 28, 34 }).Average ());
+		}
+
+		[Test]
+		public void TestAverageOnInt64 ()
+		{
+			Assert.AreEqual (23.25, (new long [] { 24, 7, 28, 34 }).Average ());
+		}
+		
+		/*
+		[Test]
+		public void AnyArgumentNullTest ()
+		{
+			string [] data = { "2", "1", "5", "3", "4" };
+
+
+			// Any<TSource> ()
+			AssertException<ArgumentNullException> (delegate () { ((IEnumerable<string>) null).AsParallel ().Any (); });
+
+			// Any<TSource> (Func<TSource, bool>)
+			AssertException<ArgumentNullException> (delegate () { ((IEnumerable<string>) null).AsParallel ().Any (x => true); });
+			AssertException<ArgumentNullException> (delegate () { data.AsParallel ().Any ((Func<string, bool>) null); });
+		}*/
+
+		[Test]
+		public void AnyTest ()
+		{
+			int [] data = { 5, 2, 3, 1, 6 };
+			int [] empty = { };
+
+
+			// Any<TSource> ()
+			Assert.IsTrue (data.AsParallel ().Any ());
+			Assert.IsFalse (empty.AsParallel ().Any ());
+
+			// Any<TSource> (Func<TSource, bool>)
+			Assert.IsTrue (data.AsParallel ().Any (x => x == 5));
+			Assert.IsFalse (data.AsParallel ().Any (x => x == 9));
+			Assert.IsFalse (empty.AsParallel ().Any (x => true));
+		}
+
+		/*
+		[Test]
+		public void AllArgumentNullTest ()
+		{
+			string [] data = { "2", "1", "5", "3", "4" };
+
+			AssertException<ArgumentNullException> (delegate () { ((IEnumerable<string>) null).AsParallel ().All (x => true); });
+			AssertException<ArgumentNullException> (delegate () { data.AsParallel ().All ((Func<string, bool>) null); });
+		}*/
+
+		[Test]
+		public void AllTest ()
+		{
+			int [] data = { 5, 2, 3, 1, 6 };
+			int [] empty = { };
+
+			Assert.IsTrue (data.AsParallel ().All (x => true));
+			Assert.IsFalse (data.AsParallel ().All (x => x != 1));
+			Assert.IsTrue (empty.AsParallel ().All (x => false));
+		}
+	}
+}

+ 90 - 0
mcs/class/System.Core/Test/System.Linq/ParallelTestHelper.cs

@@ -0,0 +1,90 @@
+#if NET_4_0
+// TestHelper.cs
+//
+// Copyright (c) 2008 Jérémie "Garuma" Laval
+//
+// Permission is hereby granted, free of charge, to any person obtaining a copy
+// of this software and associated documentation files (the "Software"), to deal
+// in the Software without restriction, including without limitation the rights
+// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+// copies of the Software, and to permit persons to whom the Software is
+// furnished to do so, subject to the following conditions:
+//
+// The above copyright notice and this permission notice shall be included in
+// all copies or substantial portions of the Software.
+//
+// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
+// THE SOFTWARE.
+//
+//
+
+using System;
+using System.Threading;
+using System.Collections.Concurrent;
+
+namespace MonoTests.System.Linq.Parallel
+{
+	public static class ParallelTestHelper
+	{
+		const int NumRun = 100;
+		
+		public static void Repeat (Action action)
+		{
+			Repeat (action, NumRun);
+		}
+		
+		public static void Repeat (Action action, int numRun)
+		{
+			for (int i = 0; i < numRun; i++) {
+			  //Console.WriteLine ("Run " + i.ToString ());
+				action ();
+			}
+		}
+		
+		public static void ParallelStressTest<TSource>(TSource obj, Action<TSource> action)
+		{
+			ParallelStressTest(obj, action, Environment.ProcessorCount + 2);
+		}
+		
+		public static void ParallelStressTest<TSource>(TSource obj, Action<TSource> action, int numThread)
+		{
+			Thread[] threads = new Thread[numThread];
+			for (int i = 0; i < numThread; i++) {
+				threads[i] = new Thread(new ThreadStart(delegate { action(obj); }));
+				threads[i].Start();
+			}
+			
+			// Wait for the completion
+			for (int i = 0; i < numThread; i++)
+				threads[i].Join();
+		}
+		
+		public static void ParallelAdder(IProducerConsumerCollection<int> collection, int numThread)
+		{
+			int startIndex = -10;
+			ParallelTestHelper.ParallelStressTest(collection, delegate (IProducerConsumerCollection<int> c) {
+				int start = Interlocked.Add(ref startIndex, 10);
+				for (int i = start; i < start + 10; i++) {
+					c.TryAdd(i);
+				}
+			}, numThread);
+		}
+		
+		public static void ParallelRemover(IProducerConsumerCollection<int> collection, int numThread, int times)
+		{
+			int t = -1;
+			ParallelTestHelper.ParallelStressTest(collection, delegate (IProducerConsumerCollection<int> c) {
+				int num = Interlocked.Increment(ref t);
+				int value;
+				if (num < times)
+					c.TryTake (out value);
+			}, numThread);
+		}
+	}
+}
+#endif