Note
Access to this page requires authorization. You can try signing in or changing directories.
Access to this page requires authorization. You can try changing directories.
Note
This article is a feature specification. The specification serves as the design document for the feature. It includes proposed specification changes, along with information needed during the design and development of the feature. These articles are published until the proposed spec changes are finalized and incorporated in the current ECMA specification.
There may be some discrepancies between the feature specification and the completed implementation. Those differences are captured in the pertinent language design meeting (LDM) notes.
You can learn more about the process for adopting feature speclets into the C# language standard in the article on the specifications.
Champion issue: https://github.com/dotnet/csharplang/issues/8635
Summary
Extend all properties to allow them to reference an automatically generated backing field using the new contextual keyword field. Properties may now also contain an accessor without a body alongside an accessor with a body.
Motivation
Auto properties only allow for directly setting or getting the backing field, giving some control only by placing access modifiers on the accessors. Sometimes there is a need to have additional control over what happens in one or both accessors, but this confronts users with the overhead of declaring a backing field. The backing field name must then be kept in sync with the property, and the backing field is scoped to the entire class which can result in accidental bypassing of the accessors from within the class.
There are several common scenarios. Within the getter, there is lazy initialization, or default values when the property has never been given. Within the setter, there is applying a constraint to ensure the validity of a value, or detecting and propagating updates such as by raising the INotifyPropertyChanged.PropertyChanged event.
In these cases by now you always have to create an instance field and write the whole property yourself. This not only adds a fair amount of code, but it also leaks the backing field into the rest of the type's scope, when it is often desirable to only have it be available to the bodies of the accessors.
Glossary
- Auto property: Short for "automatically implemented property" (§15.7.4). Accessors on an auto property have no body. The implementation and backing storage are both provided by the compiler. Auto properties have - { get; },- { get; set; }, or- { get; init; }.
- Auto accessor: Short for "automatically implemented accessor." This is an accessor that has no body. The implementation and backing storage are both provided by the compiler. - get;,- set;and- init;are auto accessors.
- Full accessor: This is an accessor that has a body. The implementation is not provided by the compiler, though the backing storage may still be (as in the example - set => field = value;).
- Field-backed property: This is either a property using the - fieldkeyword within an accessor body, or an auto property.
- Backing field: This is the variable denoted by the - fieldkeyword in a property's accessors, which is also implicitly read or written in automatically implemented accessors (- get;,- set;, or- init;).
Detailed design
For properties with an init accessor, everything that applies below to set would apply instead to the init accessor.
There are two syntax changes:
- There is a new contextual keyword, - field, which may be used within property accessor bodies to access a backing field for the property declaration (LDM decision).
- Properties may now mix and match auto accessors with full accessors (LDM decision). "Auto property" will continue to mean a property whose accessors have no bodies. None of the examples below will be considered auto properties. 
Examples:
{ get; set => Set(ref field, value); }
{ get => field ?? parent.AmbientValue; set; }
Both accessors may be full accessors with either one or both making use of field:
{ get => field; set => field = value; }
{ get => field; set => throw new InvalidOperationException(); }
{ get => overriddenValue; set => field = value; }
{
    get;
    set
    {
        if (field == value) return;
        field = value;
        OnXyzChanged();
    }
}
Expression-bodied properties and properties with only a get accessor may also use field:
public string LazilyComputed => field ??= Compute();
public string LazilyComputed { get => field ??= Compute(); }
Set-only properties may also use field:
{
    set
    {
        if (field == value) return;
        field = value;
        OnXyzChanged(new XyzEventArgs(value));
    }
}
Breaking changes
The existence of the field contextual keyword within property accessor bodies is a potentially breaking change.
Since field is a keyword and not an identifier, it can only be "shadowed" by an identifier using the normal keyword-escaping route: @field. All identifiers named field declared within property accessor bodies can safeguard against breaks when upgrading from C# versions prior to 14 by adding the initial @.
If a variable named field is declared in a property accessor, an error is reported.
In language version 14 or later, a warning is reported if a primary expression field refers to the backing field, but would have referred to a different symbol in an earlier language version.
Field-targeted attributes
As with auto properties, any property that uses a backing field in one of its accessors will be able to use field-targeted attributes:
[field: Xyz]
public string Name => field ??= Compute();
[field: Xyz]
public string Name { get => field; set => field = value; }
A field-targeted attribute will remain invalid unless an accessor uses a backing field:
// ❌ Error, will not compile
[field: Xyz]
public string Name => Compute();
Property initializers
Properties with initializers may use field. The backing field is directly initialized rather than the setter being called (LDM decision).
Calling a setter for an initializer is not an option; initializers are processed before calling base constructors, and it is illegal to call any instance method before the base constructor is called. This is also important for default initialization/definite assignment of structs.
This yields flexible control over initialization. If you want to initialize without calling the setter, you use a property initializer. If you want to initialize by calling the setter, you use assign the property an initial value in the constructor.
Here's an example of where this is useful. We believe the field keyword will find a lot of its use with view models because of the elegant solution it brings for the INotifyPropertyChanged pattern. View model property setters are likely to be databound to UI and likely to cause change tracking or trigger other behaviors. The following code needs to initialize the default value of IsActive without setting HasPendingChanges to true:
class SomeViewModel
{
    public bool HasPendingChanges { get; private set; }
    public bool IsActive { get; set => Set(ref field, value); } = true;
    private bool Set<T>(ref T location, T value)
    {
        if (EqualityComparer<T>.Default.Equals(location, value))
            return false;
        location = value;
        HasPendingChanges = true;
        return true;
    }
}
This difference in behavior between a property initializer and assigning from the constructor can also be seen with virtual auto properties in previous versions of the language:
using System;
// Nothing is printed; the property initializer is not
// equivalent to `this.IsActive = true`.
_ = new Derived();
class Base
{
    public virtual bool IsActive { get; set; } = true;
}
class Derived : Base
{
    public override bool IsActive
    {
        get => base.IsActive;
        set
        {
            base.IsActive = value;
            Console.WriteLine("This will not be reached");
        }
    }
}
Constructor assignment
As with auto properties, assignment in the constructor calls the (potentially virtual) setter if it exists, and if there is no setter it falls back to directly assigning to the backing field.
class C
{
    public C()
    {
        P1 = 1; // Assigns P1's backing field directly
        P2 = 2; // Assigns P2's backing field directly
        P3 = 3; // Calls P3's setter
        P4 = 4; // Calls P4's setter
    }
    public int P1 => field;
    public int P2 { get => field; }
    public int P4 { get => field; set => field = value; }
    public int P3 { get => field; set; }
}
Definite assignment in structs
Even though they can't be referenced in the constructor, backing fields denoted by the field keyword are subject to default-initialization and disabled-by-default warnings under the same conditions as any other struct fields (LDM decision 1, LDM decision 2).
For example (these diagnostics are silent by default):
public struct S
{
    public S()
    {
        // CS9020 The 'this' object is read before all of its fields have been assigned, causing preceding implicit
        // assignments of 'default' to non-explicitly assigned fields.
        _ = P1;
    }
    public int P1 { get => field; }
}
public struct S
{
    public S()
    {
        // CS9020 The 'this' object is read before all of its fields have been assigned, causing preceding implicit
        // assignments of 'default' to non-explicitly assigned fields.
        P2 = 5;
    }
    public int P2 { get => field; set => field = value; }
}
Ref-returning properties
Like with auto properties, the field keyword will not be available for use in ref-returning properties. Ref-returning properties cannot have set accessors, and without a set accessor, the get accessor and the property initializer would be the only things able to access the backing field. Absent use cases for this, now is not the time for ref-returning properties to start to be able to be written as auto properties.
Nullability
A principle of the the Nullable Reference Types feature was to understand existing idiomatic coding patterns in C# and to require as little ceremony as possible around those patterns. The field keyword proposal enables simple, idiomatic patterns to address widely asked-for scenarios, such as lazily initialized properties. It's important for the Nullable Reference Types to mesh well with these new coding patterns.
Goals:
- A reasonable level of null-safety should be ensured for various usage patterns of the - fieldkeyword feature.
- Patterns that use the - fieldkeyword should feel as though they've always been part of the language. Avoid making the user jump through hoops to enable Nullable Reference Types in code that is perfectly idiomatic for the- fieldkeyword feature.
One of the key scenarios is lazily initialized properties:
public class C
{
    public C() { } // It would be undesirable to warn about 'Prop' being uninitialized here
    string Prop => field ??= GetPropValue();
}
The following nullability rules will apply not just to properties that use the field keyword, but also to existing auto properties.
Nullability of the backing field
See Glossary for definitions of new terms.
The backing field has the same type as the property. However, its nullable annotation may differ from the property. To determine this nullable annotation, we introduce the concept of null-resilience. Null-resilience intuitively means that the property's get accessor preserves null-safety even when the field contains the default value for its type.
A field-backed property is determined to be null-resilient or not by performing a special nullable analysis of its get accessor. Note that this analysis is only performed when the property may be of reference type and it is not-annotated.
- Two separate nullable analysis passes are performed: one where the field's nullable annotation is not-annotated, and one where its nullable annotation is annotated. The nullable diagnostics resulting from each analysis are recorded.
- If there is a nullable diagnostic in the annotated pass, which was not present in the not-annotated pass, then the property is not null-resilient. Otherwise, it is null-resilient.
- The implementation can optimize by first performing the annotated pass. If this results in no diagnostics at all, then the property is null-resilient and the not-annotated pass can be skipped.
 
- If the property does not have a get accessor, it is (vacuously) null-resilient.
- If the get accessor is auto-implemented, the property is not null-resilient.
Null-forgiving (!) operators, directives like #nullable disable and #pragma disable warning, and conventional project-level settings like <NoWarn>, are all respected when deciding if a nullable analysis has diagnostics. DiagnosticSuppressors are ignored when deciding if a nullable analysis has diagnostics.
The nullable annotation of the backing field is determined as follows:
- If the associated property's nullable annotation is annotated or oblivious, then the nullable annotation is the same as the nullable annotation of the associated property.
- If the associated property's nullable annotation is not-annotated, then:
- If the property is null-resilient, the nullable annotation is annotated.
- If the property is not null-resilient, the nullable annotation is not-annotated.
 
Constructor analysis
Currently, an auto property is treated very similarly to an ordinary field in nullable constructor analysis. We extend this treatment to field-backed properties, by treating every field-backed property as a proxy to its backing field.
We update the following spec language from the previous proposed approach to accomplish this (new language in bold):
At each explicit or implicit 'return' in a constructor, we give a warning for each member whose flow state is incompatible with its annotations and nullability attributes. If the member is a field-backed property, the nullable annotation of the backing field is used for this check. Otherwise, the nullable annotation of the member itself is used. A reasonable proxy for this is: if assigning the member to itself at the return point would produce a nullability warning, then a nullability warning will be produced at the return point.
Note
A key trait of a null-resilient property is that it returns a value with "correct" nullability even when the backing field's value is default. Given this, we could consider not even tracking a flow state for such properties. However, this seems like a larger change in nullable analysis of properties, which we don't have any motivating scenario for changing at this time.
This is essentially a constrained interprocedural analysis. We anticipate that in order to analyze a constructor, it will be necessary to do binding and "null-resilience" analysis on all potentially applicable get accessors in the same type, which use the field contextual keyword and have not-annotated nullability. We speculate that this is not prohibitively expensive because getter bodies are usually not very complex, and that the "null-resilience" analysis only needs to be performed once regardless of how many constructors are in the type.
Setter analysis
For simplicity, we use the terms "setter" and "set accessor" to refer to either a set or init accessor.
There is a need to check that setters of non-null-resilient field-backed properties actually initialize the backing field.
class C
{
    string Prop
    {
        get => field;
        // getter is not null-resilient, so `field` is not-annotated.
        // We should warn here that `field` may be null when exiting.
        set { }
    }
    public C()
    {
        Prop = "a"; // ok
    }
    public static void Main()
    {
        new C().Prop.ToString(); // no warning, NRE at runtime
    }
}
The initial flow state of the backing field in the setter of a field-backed property is determined as follows:
- If the property has an initializer, then the initial flow state is the same as the flow state after visiting the initializer.
- Otherwise, the initial flow state is the same as the flow state given by field = default;.
At each explicit or implicit 'return' in the setter, a warning is reported if the flow state of the backing field is incompatible with its annotations and nullability attributes.
Remarks
This formulation is intentionally similar to ordinary fields in constructors. Essentially, because only the property accessors can actually refer to the backing field, the setter is treated as a "mini-constructor" for the backing field.
Much like with ordinary fields, we usually know the property was initialized in the constructor because it was set, but not necessarily. Simply returning within a branch where Prop != null was true is also good enough for our constructor analysis, since we understand that untracked mechanisms may have been used to set the property.
Alternatives were considered; see the Nullability alternatives section.
nameof
In places where field is a keyword, nameof(field) will fail to compile (LDM decision), like nameof(nint). It is not like nameof(value), which is the thing to use when property setters throw ArgumentException as some do in the .NET core libraries. In contrast, nameof(field) has no expected use cases.
Overrides
Overriding properties may use field. Such usages of field refer to the backing field for the overriding property, separate from the backing field of the base property if it has one. There is no ABI for exposing the backing field of a base property to overriding classes since this would break encapsulation.
Like with auto properties, properties which use the field keyword and override a base property must override all accessors (LDM decision).
Captures
field should be able to be captured in local functions and lambdas, and references to field from inside local functions and lambdas are allowed even if there are no other references (LDM decision 1, LDM decision 2):
public class C
{
    public static int P
    {
        get
        {
            Func<int> f = static () => field;
            return f();
        }
    }
}
Field usage warnings
When the field keyword is used in an accessor, the compiler's existing analysis of unassigned or unread fields will include that field.
- CS0414: The backing field for property 'Xyz' is assigned but its value is never used
- CS0649: The backing field for property 'Xyz' is never assigned to, and will always have its default value
Specification changes
Syntax
When compiling with language version 14 or higher, field is considered a keyword when used as a primary expression (LDM decision) in the following locations (LDM decision):
- In method bodies of get,set, andinitaccessors in properties but not indexers
- In attributes applied to those accessors
- In nested lambda expressions and local functions, and in LINQ expressions in those accessors
In all other cases, including when compiling with language version 12 or lower, field is considered an identifier.
primary_no_array_creation_expression
    : literal
+   | 'field'
    | interpolated_string_expression
    | ...
    ;
Properties
§15.7.1 Properties - General
A property_initializer may only be given for
an automatically implemented property, anda property that has a backing field that will be emitted. The property_initializer causes the initialization of the underlying field of such properties with the value given by the expression.
§15.7.4 Automatically implemented properties
An automatically implemented property (or auto-property for short), is a non-abstract, non-extern, non-ref-valued property with
semicolon-only accessor bodies. Auto-properties shall have a get accessor and may optionally have a set accessor.either or both of:
- an accessor with a semicolon-only body
- usage of the
fieldcontextual keyword within the accessors or expression body of the propertyWhen a property is specified as an automatically implemented property, a hidden unnamed backing field is automatically available for the property
, and the accessors are implemented to read from and write to that backing field. For auto-properties, any semicolon-onlygetaccessor is implemented to read from, and any semicolon-onlysetaccessor to write to its backing field.
The hidden backing field is inaccessible, it can be read and written only through the automatically implemented property accessors, even within the containing type.The backing field can be referenced directly using thefieldkeyword within all accessors and within the property expression body. Because the field is unnamed, it cannot be used in anameofexpression.If the auto-property has
no set accessoronly a semicolon-only get accessor, the backing field is consideredreadonly(§15.5.3). Just like areadonlyfield, a read-only auto-property (without a set accessor or an init accessor) may also be assigned to in the body of a constructor of the enclosing class. Such an assignment assigns directly to theread-onlybacking field of the property.An auto-property is not allowed to only have a single semicolon-only
setaccessor without agetaccessor.An auto-property may optionally have a property_initializer, which is applied directly to the backing field as a variable_initializer (§17.7).
The following example:
// No 'field' symbol in scope.
public class Point
{
    public int X { get; set; }
    public int Y { get; set; }
}
is equivalent to the following declaration:
// No 'field' symbol in scope.
public class Point
{
    public int X { get { return field; } set { field = value; } }
    public int Y { get { return field; } set { field = value; } }
}
which is equivalent to:
// No 'field' symbol in scope.
public class Point
{
    private int __x;
    private int __y;
    public int X { get { return __x; } set { __x = value; } }
    public int Y { get { return __y; } set { __y = value; } }
}
The following example:
// No 'field' symbol in scope.
public class LazyInit
{
    public string Value => field ??= ComputeValue();
    private static string ComputeValue() { /*...*/ }
}
is equivalent to the following declaration:
// No 'field' symbol in scope.
public class Point
{
    private string __value;
    public string Value { get { return __value ??= ComputeValue(); } }
    private static string ComputeValue() { /*...*/ }
}
Alternatives
Nullability alternatives
In addition to the null-resilience approach outlined in the Nullability section, the working group suggested the following alternatives for the LDM's consideration:
Do nothing
We could introduce no special behavior at all here. In effect:
- Treat a field-backed property the same way auto-properties are treated today--must be initialized in constructor except when marked required, etc.
- No special treatment of the field variable when analyzing property accessors. It is simply a variable with the same type and nullability as the property.
Note that this would result in nuisance warnings for "lazy property" scenarios, in which case users would likely need to assign null! or similar to silence constructor warnings.
A "sub-alternative" we can consider is to also completely ignore properties using field keyword for nullable constructor analysis. In that case, there would be no warnings anywhere about the user needing to initialize anything, but also no nuisance for the user, regardless of what initialization pattern they may be using.
Because we are only planning to ship the field keyword feature under the Preview LangVersion in .NET 9, we expect to have some ability to change the nullable behavior for the feature in .NET 10. Therefore, we could consider adopting a "lower-cost" solution like this one in the short term, and growing up to one of the more complex solutions in the long term.
field-targeted nullability attributes
We could introduce the following defaults, achieving a reasonable level of null safety, without involving any interprocedural analysis at all:
- The fieldvariable always has the same nullable annotation as the property.
- Nullability attributes [field: MaybeNull, AllowNull]etc. can be used to customize the nullability of the backing field.
- field-backed properties are checked for initialization in constructors based on the field's nullable annotation and attributes.
- setters in field-backed properties check for initialization of fieldsimilarly to constructors.
This would mean the "little-l lazy scenario" would look like this instead:
class C
{
    public C() { } // no need to warn about initializing C.Prop, as the backing field is marked nullable using attributes.
    [field: AllowNull, MaybeNull]
    public string Prop => field ??= GetPropValue();
}
One reason we shied away from using nullability attributes here is that the ones we have are really oriented around describing inputs and outputs of signatures. They are cumbersome to use to describe the nullability of long-lived variables.
- In practice, [field: MaybeNull, AllowNull]is required to make the field behave "reasonably" as a nullable variable, which gives maybe-null initial flow state, and allows possible null values to be written to it. This feels cumbersome to ask users to do for relatively common "little-l lazy" scenarios.
- If we pursued this approach, we would consider adding a warning when [field: AllowNull]is used, suggesting to also addMaybeNull. This is because AllowNull by itself doesn't do what users need out of a nullable variable: it assumes the field is initially not-null when we never saw anything write to it yet.
- We could also consider adjusting the behavior of [field: MaybeNull]on thefieldkeyword, or even fields in general, to allow nulls to also be written to the variable, as ifAllowNullwere implicitly also present.
Infer the initial flow state of the field
Instead of defining the behavior in terms of the field's nullable annotation, we could just define the initial flow state of the field in the 'get' accessor. This would reflect the reality that it's ok to assign a possible null value to the field. The other forms of analysis we have would succeed in identifying any resulting null safety issues from doing that.
However, this came about in a phase of implementation where the churn of making this change was not thought to be worthwhile. This is something we could conceivably adjust under the hood in the future, with very few users perceiving any break--this alternative approach being generally more permissive than the one we actually implemented.
Answered LDM questions
Syntax locations for keywords
In accessors where field and value could bind to a synthesized backing field or an implicit setter parameter, in which syntax locations should the identifiers be considered keywords?
- always
- primary expressions only
- never
The first two cases are breaking changes.
If the identifiers are always considered keywords, that is a breaking change for the following for instance:
class MyClass
{
    private int field;
    public int P => this.field; // error: expected identifier
    private int value;
    public int Q
    {
        set { this.value = value; } // error: expected identifier
    }
}
If the identifiers are keywords when used as primary expressions only, the breaking change is smaller. The most common break may be unqualified use of an existing member named field.
class MyClass
{
    private int field;
    public int P => field; // binds to synthesized backing field rather than 'this.field'
}
There is also a break when field or value is redeclared in a nested function. This may be the only break for value for primary expressions.
class MyClass
{
    private IEnumerable<string> _fields;
    public bool HasNotNullField
    {
        get => _fields.Any(field => field is { }); // 'field' binds to synthesized backing field
    }
    public IEnumerable<string> Fields
    {
        get { return _fields; }
        set { _fields = value.Where(value => Filter(value)); } // 'value' binds to setter parameter
    }
}
If the identifiers are never considered keywords, the identifiers will only bind to a synthesized backing field or the implicit parameter when the identifiers do not bind to other members. There is no breaking change for this case.
Answer
field is a keyword in appropriate accessors when used as a primary expression only; value is never considered a keyword.
Scenarios similar to { set; }
{ set; } is currently disallowed and this makes sense: the field which this creates can never be read. There are now new ways to end up in a situation where the setter introduces a backing field that is never read, such as the expansion of { set; } into { set => field = value; }.
Which of these scenarios should be allowed to compile? Assume that the "field is never read" warning would apply just like with a manually declared field.
- { set; }- Disallowed today, continue disallowing
- { set => field = value; }
- { get => unrelated; set => field = value; }
- { get => unrelated; set; }
- 
{ set { if (field == value) return; field = value; SendEvent(nameof(Prop), value); } }
- 
{ get => unrelated; set { if (field == value) return; field = value; SendEvent(nameof(Prop), value); } }
Answer
Only disallow what is already disallowed today in auto properties, the bodyless set;.
field in event accessor
Should field be a keyword in an event accessor, and should the compiler generate a backing field?
class MyClass
{
    public event EventHandler E
    {
        add { field += value; }
        remove { field -= value; }
    }
}
Recommendation: field is not a keyword within an event accessor, and no backing field is generated.
Answer
Recommendation taken. field is not a keyword within an event accessor, and no backing field is generated.
Nullability of field
Should the proposed nullability of field be accepted? See the Nullability section, and the open question within.
Answer
General proposal is adopted. Specific behavior still needs more review.
field in property initializer
Should field be a keyword in a property initializer and bind to the backing field?
class A
{
    const int field = -1;
    object P1 { get; } = field; // bind to const (ok) or backing field (error)?
}
Are there useful scenarios for referencing the backing field in the initializer?
class B
{
    object P2 { get; } = (field = 2);        // error: initializer cannot reference instance member
    static object P3 { get; } = (field = 3); // ok, but useful?
}
In the example above, binding to the backing field should result in an error: "initializer cannot reference non-static field".
Answer
We will bind the initializer as in previous versions of C#. We won't put the backing field in scope, nor will we prevent referencing other members named field.
Interaction with partial properties
Initializers
When a partial property uses field, which parts should be allowed to have an initializer?
partial class C
{
    public partial int Prop { get; set; } = 1;
    public partial int Prop { get => field; set => field = value; } = 2;
}
- It seems clear that an error should occur when both parts have an initializer.
- We can think of use cases where either the definition or implementation part might want to set the initial value of the field.
- It seems like if we permit the initializer on the definition part, it is effectively forcing the implementer to use fieldin order for the program to be valid. Is that fine?
- We think it will be common for generators to use fieldwhenever a backing field of the same type is needed in the implementation. This is in part because generators often want to enable their users to use[field: ...]targeted attributes on the property definition part. Using thefieldkeyword saves the generator implementer the trouble of "forwarding" such attributes to some generated field and suppressing the warnings on the property. Those same generators are likely to also want to allow the user to specify an initial value for the field.
Recommendation: Permit an initializer on either part of a partial property when the implementation part uses field. Report an error if both parts have an initializer.
Answer
Recommendation accepted. Either declaring or implementing property locations can use an initializer, but not both at the same time.
Auto-accessors
As originally designed, partial property implementation must have bodies for all the accessors. However, recent iterations of the field keyword feature have included the notion of "auto-accessors". Should partial property implementations be able to use such accessors? If they are used exclusively, it will be indistinguishable from a defining declaration.
partial class C
{
    public partial int Prop0 { get; set; }
    public partial int Prop0 { get => field; set => field = value; } // this is equivalent to the two "semi-auto" forms below.
    public partial int Prop1 { get; set; }
    public partial int Prop1 { get => field; set; } // is this a valid implementation part?
    public partial int Prop2 { get; set; }
    public partial int Prop2 { get; set => field = value; } // what about this? will there be disagreement about which is the "best" style?
    public partial int Prop3 { get; }
    public partial int Prop3 { get => field; } // it will only be valid to use at most 1 auto-accessor, when a second accessor is manually implemented.
Recommendation: Disallow auto-accessors in partial property implementations, because the limitations around when they would be usable are more confusing to follow than the benefit of allowing them.
Answer
At least one implementing accessor must be manually implemented, but the other accessor can be automatically implemented.
Readonly field
When should the synthesized backing field be considered read-only?
struct S
{
    readonly object P0 { get => field; } = "";         // ok
    object P1          { get => field ??= ""; }        // ok
    readonly object P2 { get => field ??= ""; }        // error: 'field' is readonly
    readonly object P3 { get; set { _ = field; } }     // ok
    readonly object P4 { get; set { field = value; } } // error: 'field' is readonly
}
When the backing field is considered read-only, the field emitted to metadata is marked initonly, and an error is reported if field is modified other than in an initializer or constructor.
Recommendation: The synthesized backing field is read-only when the containing type is a struct and the property or containing type is declared readonly.
Answer
Recommendation is accepted.
Readonly context and set
Should a set accessor be allowed in a readonly context for a property that uses field?
readonly struct S1
{
    readonly object _p1;
    object P1 { get => _p1; set { } }   // ok
    object P2 { get; set; }             // error: auto-prop in readonly struct must be readonly
    object P3 { get => field; set { } } // ok?
}
struct S2
{
    readonly object _p1;
    readonly object P1 { get => _p1; set { } }   // ok
    readonly object P2 { get; set; }             // error: auto-prop with set marked readonly
    readonly object P3 { get => field; set { } } // ok?
}
Answer
There could be scenarios for this where you're implementing a set accessor on a readonly struct and either passing it through, or throwing. We will allow this.
[Conditional] code
Should the synthesized field be generated when field is used only in omitted calls to conditional methods?
For instance, should a backing field be generated for the following in a non-DEBUG build?
class C
{
    object P
    {
        get
        {
            Debug.Assert(field is null);
            return null;
        }
    }
}
For reference, fields for primary constructor parameters are generated in similar cases - see sharplab.io.
Recommendation: The backing field is generated when field is used only in omitted calls to conditional methods.
Answer
Conditional code can have effects on non-conditional code, such as Debug.Assert changing nullability. It would be strange if field didn't have similar impacts. It is also unlikely to come up in most code, so we'll do the simple thing and accept the recommendation.
Interface properties and auto-accessors
Is a combination of manually- and auto-implemented accessors recognized for an interface property where the auto-implemented accessor refers to a synthesized backing field?
For an instance property, an error will be reported that instance fields are not supported.
interface I
{
           object P1 { get; set; }                           // ok: not an implementation
           object P2 { get => field; set { field = value; }} // error: instance field
           object P3 { get; set { } } // error: instance field
    static object P4 { get; set { } } // ok: equivalent to { get => field; set { } }
}
Recommendation: Auto-accessors are recognized in interface properties, and the auto-accessors refer to a synthesized backing field. For an instance property, an error is reported that instance fields are not supported.
Answer
Standardizing around the instance field itself being the cause of the error is consistent with partial properties in classes, and we like that outcome. The recommendation is accepted.
C# feature specifications