mirror of https://github.com/icsharpcode/ILSpy.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
183 lines
5.0 KiB
183 lines
5.0 KiB
// Copyright (c) 2017 Daniel Grunwald |
|
// |
|
// 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; |
|
|
|
namespace ICSharpCode.Decompiler.Tests.TestCases.Correctness |
|
{ |
|
class NullableTests |
|
{ |
|
static void Main() |
|
{ |
|
AvoidLifting(); |
|
BitNot(); |
|
FieldAccessOrderOfEvaluation(null); |
|
FieldAccessOrderOfEvaluationWithStruct(null); |
|
ArrayAccessOrderOfEvaluation(); |
|
} |
|
|
|
struct SomeStruct |
|
{ |
|
public int IntField; |
|
} |
|
|
|
static void AvoidLifting() |
|
{ |
|
Console.WriteLine("MayThrow:"); |
|
Console.WriteLine(MayThrow(10, 2, 3)); |
|
Console.WriteLine(MayThrow(10, 0, null)); |
|
|
|
Console.WriteLine("NotUsingAllInputs:"); |
|
Console.WriteLine(NotUsingAllInputs(5, 3)); |
|
Console.WriteLine(NotUsingAllInputs(5, null)); |
|
|
|
Console.WriteLine("UsingUntestedValue:"); |
|
Console.WriteLine(UsingUntestedValue(5, 3)); |
|
Console.WriteLine(UsingUntestedValue(5, null)); |
|
} |
|
|
|
static int? MayThrow(int? a, int? b, int? c) |
|
{ |
|
// cannot be lifted without changing the exception behavior |
|
return a.HasValue & b.HasValue & c.HasValue ? a.GetValueOrDefault() / b.GetValueOrDefault() + c.GetValueOrDefault() : default(int?); |
|
} |
|
|
|
static int? NotUsingAllInputs(int? a, int? b) |
|
{ |
|
// cannot be lifted because the value differs if b == null |
|
return a.HasValue & b.HasValue ? a.GetValueOrDefault() + a.GetValueOrDefault() : default(int?); |
|
} |
|
|
|
static int? UsingUntestedValue(int? a, int? b) |
|
{ |
|
// cannot be lifted because the value differs if b == null |
|
return a.HasValue ? a.GetValueOrDefault() + b.GetValueOrDefault() : default(int?); |
|
} |
|
|
|
static void BitNot() |
|
{ |
|
UInt32? value = 0; |
|
Assert(~value == UInt32.MaxValue); |
|
UInt64? value2 = 0; |
|
Assert(~value2 == UInt64.MaxValue); |
|
UInt16? value3 = 0; |
|
Assert((UInt16)~value3 == (UInt16)UInt16.MaxValue); |
|
UInt32 value4 = 0; |
|
Assert(~value4 == UInt32.MaxValue); |
|
UInt64 value5 = 0; |
|
Assert(~value5 == UInt64.MaxValue); |
|
UInt16 value6 = 0; |
|
Assert((UInt16)~value6 == UInt16.MaxValue); |
|
} |
|
|
|
static void Assert(bool b) |
|
{ |
|
if (!b) |
|
throw new InvalidOperationException(); |
|
} |
|
|
|
|
|
static T GetValue<T>() |
|
{ |
|
Console.WriteLine("GetValue"); |
|
return default(T); |
|
} |
|
|
|
int intField; |
|
|
|
static void FieldAccessOrderOfEvaluation(NullableTests c) |
|
{ |
|
Console.WriteLine("GetInt, then NRE:"); |
|
try { |
|
c.intField = GetValue<int>(); |
|
} catch (Exception ex) { |
|
Console.WriteLine(ex.Message); |
|
} |
|
Console.WriteLine("NRE before GetInt:"); |
|
try { |
|
#if CS60 |
|
ref int i = ref c.intField; |
|
i = GetValue<int>(); |
|
#endif |
|
} catch (Exception ex) { |
|
Console.WriteLine(ex.Message); |
|
} |
|
} |
|
|
|
SomeStruct structField; |
|
|
|
static void FieldAccessOrderOfEvaluationWithStruct(NullableTests c) |
|
{ |
|
Console.WriteLine("GetInt, then NRE (with struct):"); |
|
try { |
|
c.structField.IntField = GetValue<int>(); |
|
} catch (Exception ex) { |
|
Console.WriteLine(ex.Message); |
|
} |
|
Console.WriteLine("NRE before GetInt (with struct):"); |
|
try { |
|
#if CS60 |
|
ref SomeStruct s = ref c.structField; |
|
s.IntField = GetValue<int>(); |
|
#endif |
|
} catch (Exception ex) { |
|
Console.WriteLine(ex.Message); |
|
} |
|
} |
|
|
|
static T[] GetArray<T>() |
|
{ |
|
Console.WriteLine("GetArray"); |
|
return null; |
|
} |
|
|
|
static int GetIndex() |
|
{ |
|
Console.WriteLine("GetIndex"); |
|
return 0; |
|
} |
|
|
|
static void ArrayAccessOrderOfEvaluation() |
|
{ |
|
Console.WriteLine("GetArray direct:"); |
|
try { |
|
GetArray<int>()[GetIndex()] = GetValue<int>(); |
|
} catch (Exception ex) { |
|
Console.WriteLine(ex.Message); |
|
} |
|
Console.WriteLine("GetArray with ref:"); |
|
try { |
|
#if CS60 |
|
ref int elem = ref GetArray<int>()[GetIndex()]; |
|
elem = GetValue<int>(); |
|
#endif |
|
} catch (Exception ex) { |
|
Console.WriteLine(ex.Message); |
|
} |
|
Console.WriteLine("GetArray direct with value-type:"); |
|
try { |
|
// This line is mis-compiled by legacy csc: |
|
// with the legacy compiler the NRE is thrown before the GetValue call; |
|
// with Roslyn the NRE is thrown after the GetValue call. |
|
GetArray<TimeSpan>()[GetIndex()] = GetValue<TimeSpan>(); |
|
} catch (Exception ex) { |
|
Console.WriteLine(ex.Message); |
|
} |
|
} |
|
} |
|
}
|
|
|