ExpressionTest_AddChecked.cs 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. // Permission is hereby granted, free of charge, to any person obtaining
  2. // a copy of this software and associated documentation files (the
  3. // "Software"), to deal in the Software without restriction, including
  4. // without limitation the rights to use, copy, modify, merge, publish,
  5. // distribute, sublicense, and/or sell copies of the Software, and to
  6. // permit persons to whom the Software is furnished to do so, subject to
  7. // the following conditions:
  8. //
  9. // The above copyright notice and this permission notice shall be
  10. // included in all copies or substantial portions of the Software.
  11. //
  12. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  13. // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  14. // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  15. // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
  16. // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
  17. // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
  18. //
  19. // Authors:
  20. // Federico Di Gregorio <[email protected]>
  21. using System;
  22. using System.Reflection;
  23. using System.Linq;
  24. using System.Linq.Expressions;
  25. using NUnit.Framework;
  26. namespace MonoTests.System.Linq.Expressions
  27. {
  28. [TestFixture]
  29. public class ExpressionTest_AddChecked
  30. {
  31. [Test]
  32. [ExpectedException (typeof (ArgumentNullException))]
  33. public void Arg1Null ()
  34. {
  35. Expression.AddChecked (null, Expression.Constant(1));
  36. }
  37. [Test]
  38. [ExpectedException (typeof (ArgumentNullException))]
  39. public void Arg2Null ()
  40. {
  41. Expression.AddChecked (Expression.Constant (1), null);
  42. }
  43. [Test]
  44. [ExpectedException (typeof (InvalidOperationException))]
  45. public void ArgTypesDifferent ()
  46. {
  47. Expression.AddChecked (Expression.Constant (1), Expression.Constant (2.0));
  48. }
  49. [Test]
  50. [ExpectedException (typeof (InvalidOperationException))]
  51. public void NoOperatorClass ()
  52. {
  53. Expression.AddChecked (Expression.Constant (new NoOpClass ()), Expression.Constant (new NoOpClass ()));
  54. }
  55. [Test]
  56. [ExpectedException (typeof (InvalidOperationException))]
  57. public void Boolean ()
  58. {
  59. Expression.AddChecked (Expression.Constant (true), Expression.Constant (false));
  60. }
  61. [Test]
  62. public void Numeric ()
  63. {
  64. BinaryExpression expr = Expression.AddChecked (Expression.Constant (1), Expression.Constant (2));
  65. Assert.AreEqual (ExpressionType.AddChecked, expr.NodeType, "AddChecked#01");
  66. Assert.AreEqual (typeof (int), expr.Type, "AddChecked#02");
  67. Assert.IsNull (expr.Method, "AddChecked#03");
  68. Assert.AreEqual ("(1 + 2)", expr.ToString(), "AddChecked#15");
  69. }
  70. [Test]
  71. public void Nullable ()
  72. {
  73. int? a = 1;
  74. int? b = 2;
  75. BinaryExpression expr = Expression.AddChecked (Expression.Constant (a), Expression.Constant (b));
  76. Assert.AreEqual (ExpressionType.AddChecked, expr.NodeType, "AddChecked#04");
  77. Assert.AreEqual (typeof (int), expr.Type, "AddChecked#05");
  78. Assert.IsNull (expr.Method, null, "AddChecked#06");
  79. Assert.AreEqual ("(1 + 2)", expr.ToString(), "AddChecked#16");
  80. }
  81. [Test]
  82. public void UserDefinedClass ()
  83. {
  84. // We can use the simplest version of GetMethod because we already know only one
  85. // exists in the very simple class we're using for the tests.
  86. MethodInfo mi = typeof (OpClass).GetMethod ("op_Addition");
  87. BinaryExpression expr = Expression.AddChecked (Expression.Constant (new OpClass ()), Expression.Constant (new OpClass ()));
  88. Assert.AreEqual (ExpressionType.AddChecked, expr.NodeType, "AddChecked#07");
  89. Assert.AreEqual (typeof (OpClass), expr.Type, "AddChecked#08");
  90. Assert.AreEqual (mi, expr.Method, "AddChecked#09");
  91. Assert.AreEqual ("op_Addition", expr.Method.Name, "AddChecked#10");
  92. Assert.AreEqual ("(value(MonoTests.System.Linq.Expressions.OpClass) + value(MonoTests.System.Linq.Expressions.OpClass))",
  93. expr.ToString(), "AddChecked#17");
  94. }
  95. [Test]
  96. public void UserDefinedStruct ()
  97. {
  98. // We can use the simplest version of GetMethod because we already know only one
  99. // exists in the very simple class we're using for the tests.
  100. MethodInfo mi = typeof (OpStruct).GetMethod ("op_Addition");
  101. BinaryExpression expr = Expression.AddChecked (Expression.Constant (new OpStruct ()), Expression.Constant (new OpStruct ()));
  102. Assert.AreEqual (ExpressionType.AddChecked, expr.NodeType, "AddChecked#11");
  103. Assert.AreEqual (typeof (OpStruct), expr.Type, "AddChecked#12");
  104. Assert.AreEqual (mi, expr.Method, "AddChecked#13");
  105. Assert.AreEqual ("op_Addition", expr.Method.Name, "AddChecked#14");
  106. Assert.AreEqual ("(value(MonoTests.System.Linq.Expressions.OpStruct) + value(MonoTests.System.Linq.Expressions.OpStruct))",
  107. expr.ToString(), "AddChecked#18");
  108. }
  109. //
  110. // This method makes sure that compiling an AddChecked on two values
  111. // throws an OverflowException, if it doesnt, it fails
  112. //
  113. static void MustOverflow<T> (T v1, T v2)
  114. {
  115. Expression<Func<T>> l = Expression.Lambda<Func<T>> (
  116. Expression.AddChecked (Expression.Constant (v1), Expression.Constant (v2)));
  117. Func<T> del = l.Compile ();
  118. T res = default (T);
  119. try {
  120. res = del ();
  121. } catch (OverflowException){
  122. // OK
  123. return;
  124. }
  125. throw new Exception (String.Format ("AddChecked on {2} should have thrown an exception with values {0} {1}, result was: {3}",
  126. v1, v2, v1.GetType (), res));
  127. }
  128. //
  129. // This routine should execute the code, but not throw an
  130. // overflow exception
  131. //
  132. static void MustNotOverflow<T> (T v1, T v2)
  133. {
  134. Expression<Func<T>> l = Expression.Lambda<Func<T>> (
  135. Expression.AddChecked (Expression.Constant (v1), Expression.Constant (v2)));
  136. Func<T> del = l.Compile ();
  137. del ();
  138. }
  139. //
  140. // SubtractChecked is not defined for small types (byte, sbyte)
  141. //
  142. static void InvalidOperation<T> (T v1, T v2)
  143. {
  144. try {
  145. Expression.Lambda<Func<T>> (
  146. Expression.AddChecked (Expression.Constant (v1), Expression.Constant (v2)));
  147. } catch (InvalidOperationException){
  148. // OK
  149. return;
  150. }
  151. throw new Exception (String.Format ("AddChecked should have thrown for the creation of a tree with {0} operands", v1.GetType ()));
  152. }
  153. [Test]
  154. public void TestOverflows ()
  155. {
  156. // These should overflow, check the various types and codepaths
  157. // in BinaryExpression:
  158. MustOverflow<int> (Int32.MaxValue, 1);
  159. MustOverflow<int> (Int32.MinValue, -11);
  160. MustOverflow<long> (Int64.MaxValue, 1);
  161. MustOverflow<long> (Int64.MinValue, -1);
  162. // unsigned values use Add_Ovf_Un, check that too:
  163. MustOverflow<ulong> (UInt64.MaxValue, 1);
  164. MustOverflow<uint> (UInt32.MaxValue, 1);
  165. }
  166. //
  167. // These should not overflow
  168. //
  169. [Test]
  170. public void TestNoOverflow ()
  171. {
  172. // Simple stuff
  173. MustNotOverflow<int> (10, 20);
  174. // These are invalid:
  175. InvalidOperation<byte> (Byte.MaxValue, 2);
  176. InvalidOperation<sbyte> (SByte.MaxValue, 2);
  177. // Doubles, floats, do not overflow
  178. MustNotOverflow<float> (Single.MaxValue, 1);
  179. MustNotOverflow<double> (Double.MaxValue, 1);
  180. }
  181. }
  182. }