Skip to content

Commit

Permalink
Fix TypeWithAnnotations.ToTypeWithState() for (untyped) null literal (#…
Browse files Browse the repository at this point in the history
  • Loading branch information
cston authored Jul 27, 2020
1 parent dee18a3 commit e128267
Show file tree
Hide file tree
Showing 3 changed files with 163 additions and 10 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -1542,7 +1542,7 @@ private bool ExactOrBoundsNullableInference(ExactOrBoundsKind kind, TypeWithAnno
return false;

// True if the type is nullable.
bool isNullableOnly(TypeWithAnnotations type)
static bool isNullableOnly(TypeWithAnnotations type)
=> type.NullableAnnotation.IsAnnotated();
}

Expand Down
9 changes: 4 additions & 5 deletions src/Compilers/CSharp/Portable/Symbols/TypeWithAnnotations.cs
Original file line number Diff line number Diff line change
Expand Up @@ -772,11 +772,6 @@ internal bool IsSameAs(TypeWithAnnotations other)
/// </summary>
internal TypeWithState ToTypeWithState()
{
if (Type is null)
{
return default;
}

// This operation reflects reading from an lvalue, which produces an rvalue.
// Reading from a variable of a type parameter (that could be substituted with a nullable type), but which
// cannot itself be annotated (because it isn't known to be a reference type), may yield a null value
Expand All @@ -785,6 +780,10 @@ internal TypeWithState ToTypeWithState()

static NullableFlowState getFlowState(TypeSymbol type, NullableAnnotation annotation)
{
if (type is null)
{
return annotation.IsAnnotated() ? NullableFlowState.MaybeNull : NullableFlowState.NotNull;
}
if (type.IsPossiblyNullableReferenceTypeTypeParameter())
{
return annotation switch { NullableAnnotation.Annotated => NullableFlowState.MaybeDefault, NullableAnnotation.NotAnnotated => NullableFlowState.MaybeNull, _ => NullableFlowState.NotNull };
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -51082,12 +51082,12 @@ static void G(string x)
new[] { source }, options: WithNonNullTypesTrue(),
parseOptions: TestOptions.Regular8);
comp.VerifyDiagnostics(
// (9,54): warning CS8603: Possible null reference return.
// (9,9): warning CS8602: Dereference of a possibly null reference.
// F(() => { if (x.Length > 0) return x; return null; }).ToString();
Diagnostic(ErrorCode.WRN_NullReferenceReturn, "null").WithLocation(9, 54),
// (10,45): warning CS8603: Possible null reference return.
Diagnostic(ErrorCode.WRN_NullReferenceReceiver, "F(() => { if (x.Length > 0) return x; return null; })").WithLocation(9, 9),
// (10,9): warning CS8602: Dereference of a possibly null reference.
// F(() => { if (x.Length == 0) return null; return x; }).ToString();
Diagnostic(ErrorCode.WRN_NullReferenceReturn, "null").WithLocation(10, 45));
Diagnostic(ErrorCode.WRN_NullReferenceReceiver, "F(() => { if (x.Length == 0) return null; return x; })").WithLocation(10, 9));
}

[Fact]
Expand Down Expand Up @@ -133331,6 +133331,160 @@ internal interface IB : IA { }
comp.VerifyEmitDiagnostics();
}

[Fact]
[WorkItem(46342, "https://github.com/dotnet/roslyn/issues/46342")]
public void LambdaNullReturn_01()
{
var source =
@"#nullable enable
using System;
class Program
{
static void F<T>(Func<T> f)
{
}
static void M<T>(bool b, T t) where T : class
{
F(() =>
{
if (b) return null;
return t;
});
}
}";
var comp = CreateCompilation(source);
comp.VerifyEmitDiagnostics();
}

[Fact]
[WorkItem(46342, "https://github.com/dotnet/roslyn/issues/46342")]
public void LambdaNullReturn_02()
{
var source =
@"#nullable enable
using System;
class Program
{
static void F<T>(Func<T> f)
{
}
static void M<T>(bool b, T t) where T : class?
{
F(() =>
{
if (b) return null;
return t;
});
}
}";
var comp = CreateCompilation(source, parseOptions: TestOptions.RegularPreview);
comp.VerifyEmitDiagnostics();
}

[Fact]
[WorkItem(46342, "https://github.com/dotnet/roslyn/issues/46342")]
public void LambdaNullReturn_03()
{
var source =
@"#nullable enable
using System;
class Program
{
static void F<T>(Func<T> f)
{
}
static void M<T>(T? t) where T : class
{
F(() =>
{
if (t is null) return null;
return t;
});
}
}";
var comp = CreateCompilation(source);
comp.VerifyEmitDiagnostics();
}

[Fact]
[WorkItem(46342, "https://github.com/dotnet/roslyn/issues/46342")]
public void LambdaNullReturn_04()
{
var source =
@"#nullable enable
using System;
class Program
{
static void F<T>(Func<T> f)
{
}
static void M<T>(T? t) where T : class?
{
F(() =>
{
if (t is null) return null;
return t;
});
}
}";
var comp = CreateCompilation(source, parseOptions: TestOptions.RegularPreview);
comp.VerifyEmitDiagnostics();
}

[Fact]
[WorkItem(46342, "https://github.com/dotnet/roslyn/issues/46342")]
public void LambdaNullReturn_05()
{
var source =
@"#nullable enable
using System;
using System.Threading.Tasks;
class Program
{
static void F<T>(Func<Task<T>> f)
{
}
static void M<T>(T? t) where T : class
{
F(async () =>
{
await Task.Yield();
if (t is null) return null;
return t;
});
}
}";
var comp = CreateCompilation(source);
comp.VerifyEmitDiagnostics();
}

[Fact]
[WorkItem(46342, "https://github.com/dotnet/roslyn/issues/46342")]
public void LambdaNullReturn_06()
{
var source =
@"#nullable enable
using System;
using System.Threading.Tasks;
class Program
{
static void F<T>(Func<Task<T>> f)
{
}
static void M<T>(T? t) where T : class?
{
F(async () =>
{
await Task.Yield();
if (t is null) return null;
return t;
});
}
}";
var comp = CreateCompilation(source, parseOptions: TestOptions.RegularPreview);
comp.VerifyEmitDiagnostics();
}

[Fact]
[WorkItem(45862, "https://github.com/dotnet/roslyn/issues/45862")]
public void Issue_45862()
Expand Down

0 comments on commit e128267

Please sign in to comment.