Copied from https://github.com/dotnet/csharplang/blob/ef449a65b4c46e124b604213f11073b23675a93e/proposals/csharp-7.2/span-safety.md And then edited to use the terminilogy from dotnet/csharpstandard#795
The main reason for the additional safety rules when dealing with types like Span<T> and ReadOnlySpan<T> is that such types must be confined to the execution stack.
There are two reasons why Span<T> and similar types must be a stack-only types.
Span<T>is semantically a struct containing a reference and a range -(ref T data, int length). Regardless of actual implementation, writes to such struct would not be atomic. Concurrent "tearing" of such struct would lead to the possibility oflengthnot matching thedata, causing out-of-range accesses and type-safety violations, which ultimately could result in GC heap corruption in seemingly "safe" code.- Some implementations of
Span<T>literally contain a managed pointer in one of its fields. Managed pointers are not supported as fields of heap objects and code that manages to put a managed pointer on the GC heap typically crashes at JIT time.
All the above problems would be alleviated if instances of Span<T> are constrained to exist only on the execution stack.
An additional problem arises due to composition. It would be generally desirable to build more complex data types that would embed Span<T> and ReadOnlySpan<T> instances. Such composite types would have to be structs and would share all the hazards and requirements of Span<T>. As a result the safety rules described here should be viewed as applicable to the whole range of ref-like types.
The draft language specification is intended to ensure that values of a ref-like type occur only on the stack.
ref-like structs are explicitly marked in the source code using ref modifier:
ref struct TwoSpans<T>
{
// can have ref-like instance fields
public Span<T> first;
public Span<T> second;
}
// error: arrays of ref-like types are not allowed.
TwoSpans<T>[] arr = null;Designating a struct as ref-like will allow the struct to have ref-like instance fields and will also make all the requirements of ref-like types applicable to the struct.
Ref-like structs will be marked with System.Runtime.CompilerServices.IsRefLikeAttribute attribute.
The attribute will be added to common base libraries such as mscorlib. In a case if the attribute is not available, compiler will generate an internal one similarly to other embedded-on-demand attributes such as IsReadOnlyAttribute.
An additional measure will be taken to prevent the use of ref-like structs in compilers not familiar with the safety rules (this includes C# compilers prior to the one in which this feature is implemented).
Having no other good alternatives that work in old compilers without servicing, an Obsolete attribute with a known string will be added to all ref-like structs. Compilers that know how to use ref-like types will ignore this particular form of Obsolete.
A typical metadata representation:
[IsRefLike]
[Obsolete("Types with embedded references are not supported in this version of your compiler.")]
public struct TwoSpans<T>
{
// . . . .
}NOTE: it is not the goal to make it so that any use of ref-like types on old compilers fails 100%. That is hard to achieve and is not strictly necessary. For example there would always be a way to get around the Obsolete using dynamic code or, for example, creating an array of ref-like types through reflection.
In particular, if user wants to actually put an Obsolete or Deprecated attribute on a ref-like type, we will have no choice other than not emitting the predefined one since Obsolete attribute cannot be applied more than once..
SpanLikeType M1(ref SpanLikeType x, Span<byte> y)
{
// this is all valid, unconcerned with stack-referring stuff
var local = new SpanLikeType(y);
x = local;
return x;
}
void Test1(ref SpanLikeType param1, Span<byte> param2)
{
Span<byte> stackReferring1 = stackalloc byte[10];
var stackReferring2 = new SpanLikeType(stackReferring1);
// this is allowed
stackReferring2 = M1(ref stackReferring2, stackReferring1);
// this is NOT allowed
stackReferring2 = M1(ref param1, stackReferring1);
// this is NOT allowed
param1 = M1(ref stackReferring2, stackReferring1);
// this is NOT allowed
param2 = stackReferring1.Slice(10);
// this is allowed
param1 = new SpanLikeType(param2);
// this is allowed
stackReferring2 = param1;
}
ref SpanLikeType M2(ref SpanLikeType x)
{
return ref x;
}
ref SpanLikeType Test2(ref SpanLikeType param1, Span<byte> param2)
{
Span<byte> stackReferring1 = stackalloc byte[10];
var stackReferring2 = new SpanLikeType(stackReferring1);
ref var stackReferring3 = M2(ref stackReferring2);
// this is allowed
stackReferring3 = M1(ref stackReferring2, stackReferring1);
// this is allowed
M2(ref stackReferring3) = stackReferring2;
// this is NOT allowed
M1(ref param1) = stackReferring2;
// this is NOT allowed
param1 = stackReferring3;
// this is NOT allowed
return ref stackReferring3;
// this is allowed
return ref param1;
}Below we describe a set of safety rules for ref-like types (ref structs) to ensure that values of these types occur only on the stack. A different, simpler set of safety rules would be possible if locals cannot be passed by reference. This specification would also permit the safe reassignment of ref locals.
We associate with each expression at compile-time the concept of what scope that expression is permitted to escape to, "safe-context". Similarly, for each lvalue we maintain a concept of what scope a reference to it is permitted to escape to, "ref-safe-context". For a given lvalue expression, these may be different.
These are analogous to the "safe to return" of the ref locals feature, but it is more fine-grained. Where the "safe-to-return" of an expression records only whether (or not) it may escape the enclosing method as a whole, the safe-context records which scope it may escape to (which scope it may not escape beyond). The basic safety mechanism is enforced as follows. Given an assignment from an expression E1 with a safe-context scope S1, to an (lvalue) expression E2 with safe-context scope S2, it is an error if S2 is a wider scope than S1. By construction, the two scopes S1 and S2 are in a nesting relationship, because a legal expression is always safe-to-return from some scope enclosing the expression.
For the time being it is sufficient, for the purpose of the analysis, to support just two scopes - external to the method, and top-level scope of the method. That is because ref-like values with inner scopes cannot be created and ref locals do not support re-assignment. The rules, however, can support more than two scope levels.
The precise rules for computing the safe-to-return status of an expression, and the rules governing the legality of expressions, follow.
The ref-safe-context is a scope, enclosing an lvalue expression, to which it is safe for a ref to the lvalue to escape to. If that scope is the entire method, we say that a ref to the lvalue is safe to return from the method.
The ref-safe-context scope for an lvalue expression can never be to a greater scope than the safe-context for the same value. That means when the spec limits the safe-context of a value it is implicitly also limiting the ref-safe-context as well. However ref-safe-context scope can be to a smaller scope than safe-context. Consider that non-ref locals have safe-context scope outside method but ref-safe-context inside the method.
The safe-context is a scope, enclosing an expression, to which it is safe for the value to escape to. If that scope is the entire method, we say that the value is safe to return from the method.
An expression whose type is not a ref struct type is always safe-to-return from the entire enclosing method. Otherwise we refer to the rules below.
An lvalue designating a formal parameter is ref-safe-context (by reference) as follows:
- If the parameter is a
ref,out, orinparameter, it is ref-safe-context from the entire method (e.g. by areturn refstatement); otherwise - If the parameter is the
thisparameter of a struct type, it is ref-safe-context to the top-level scope of the method (but not from the entire method itself); Sample - Otherwise the parameter is a value parameter, and it is ref-safe-context to the top-level scope of the method (but not from the method itself).
An expression that is an rvalue designating the use of a formal parameter is safe-context (by value) from the entire method (e.g. by a return statement). This applies to the this parameter as well.
An lvalue designating a local variable is ref-safe-context (by reference) as follows:
- If the variable is a
refvariable, then its ref-safe-context is taken from the ref-safe-context of its initializing expression; otherwise - The variable is ref-safe-context the scope in which it was declared.
An expression that is an rvalue designating the use of a local variable is safe-context (by value) as follows:
- But the general rule above, a local whose type is not a
ref structtype is safe-to-return from the entire enclosing method. - If the variable is an iteration variable of a
foreachloop, then the variable's safe-context scope is the same as the safe-context of theforeachloop's expression. - A local of
ref structtype and uninitialized at the point of declaration is safe-to-return from the entire enclosing method. - Otherwise the variable's type is a
ref structtype, and the variable's declaration requires an initializer. The variable's safe-context scope is the same as the safe-context of its initializer.
An lvalue designating a reference to a field, e.F, is ref-safe-context (by reference) as follows:
- If
eis of a reference type, it is ref-safe-context from the entire method; otherwise - If
eis of a value type, its ref-safe-context is taken from the ref-safe-context ofe.
An rvalue designating a reference to a field, e.F, has a safe-context scope that is the same as the safe-context of e.
The application of a user-defined operator is treated as a method invocation.
For an operator that yields an rvalue, such as e1 + e2 or c ? e1 : e2, the safe-context of the result is the narrowest scope among the safe-context of the operands of the operator. As a consequence, for a unary operator that yields an rvalue, such as +e, the safe-context of the result is the safe-context of the operand.
For an operator that yields an lvalue, such as c ? ref e1 : ref e2
- the ref-safe-context of the result is the narrowest scope among the ref-safe-context of the operands of the operator.
- the safe-context of the operands must agree, and that is the safe-context of the resulting lvalue.
An lvalue resulting from a ref-returning method invocation e1.M(e2, ...) is ref-safe-context the smallest of the following scopes:
- The entire enclosing method
- the ref-safe-context of all
refandoutargument expressions (excluding the receiver) - For each
inparameter of the method, if there is a corresponding expression that is an lvalue, its ref-safe-context, otherwise the nearest enclosing scope - the safe-context of all argument expressions (including the receiver)
Note: the last bullet is necessary to handle code such as
var sp = new Span(...) return ref sp[0];or
return ref M(sp, 0);
An rvalue resulting from a method invocation e1.M(e2, ...) is safe-context from the smallest of the following scopes:
- The entire enclosing method
- the safe-context of all argument expressions (including the receiver)
An rvalue is ref-safe-context from the nearest enclosing scope. This occurs for example in an invocation such as M(ref d.Length) where d is of type dynamic. It is also consistent with (and perhaps subsumes) our handling of arguments corresponding to in parameters.
A property invocation (either get or set) it treated as a method invocation of the underlying method by the above rules.
A stackalloc expression is an rvalue that is safe-context to the top-level scope of the method (but not from the entire method itself).
A new expression that invokes a constructor obeys the same rules as a method invocation that is considered to return the type being constructed.
In addition safe-context is no wider than the smallest of the safe-context of all arguments/operands of the object initializer expressions, recursively, if initializer is present.
The language relies on Span<T> not having a constructor of the following form:
void Example(ref int x)
{
// Create a span of length one
var span = new Span<int>(ref x);
}Such a constructor makes Span<T> which are used as fields indistinguishable from a ref field. The safety rules described in this document
depend on ref fields not being a valid construct in C# or .NET.
A default expression is safe-context from the entire enclosing method.
We wish to ensure that no ref local variable, and no variable of ref struct type, refers to stack memory or variables that are no longer alive. We therefore have the following language constraints:
-
Neither a ref parameter, nor a ref local, nor a parameter or local of a
ref structtype can be lifted into a lambda or local function. -
Neither a ref parameter nor a parameter of a
ref structtype may be an argument on an iterator method or anasyncmethod. -
Neither a ref local, nor a local of a
ref structtype may be in scope at the point of ayield returnstatement or anawaitexpression. -
A
ref structtype may not be used as a type argument, or as an element type in a tuple type. -
A
ref structtype may not be the declared type of a field, except that it may be the declared type of an instance field of anotherref struct. -
A
ref structtype may not be the element type of an array. -
A value of a
ref structtype may not be boxed:- There is no conversion from a
ref structtype to the typeobjector the typeSystem.ValueType. - A
ref structtype may not be declared to implement any interface - No instance method declared in
objector inSystem.ValueTypebut not overridden in aref structtype may be called with a receiver of thatref structtype. - No instance method of a
ref structtype may be captured by method conversion to a delegate type.
- There is no conversion from a
-
For a ref reassignment
e1 = ref e2, the ref-safe-context ofe2must be at least as wide a scope as the ref-safe-context ofe1. -
For a ref return statement
return ref e1, the ref-safe-context ofe1must be ref-safe-context from the entire method. (TODO: Do we also need a rule thate1must be safe-context from the entire method, or is that redundant?) -
For a return statement
return e1, the safe-context ofe1must be safe-context from the entire method. -
For an assignment
e1 = e2, if the type ofe1is aref structtype, then the safe-context ofe2must be at least as wide a scope as the safe-context ofe1. -
For a method invocation if there is a
reforoutargument of aref structtype (including the receiver unless the type isreadonly), with safe-context E1, then no argument (including the receiver) may have a narrower safe-context than E1. Sample -
A local function or anonymous function may not refer to a local or parameter of
ref structtype declared in an enclosing scope.
Open Issue: We need some rule that permits us to produce an error when needing to spill a stack value of a
ref structtype at an await expression, for example in the codeFoo(new Span<int>(...), await e2);
These explanations and samples help explain why many of the safety rules above exist
When invoking a method where there is an out or ref parameter that is a ref struct then all of the ref struct parameters need to have the same lifetime. This is necessary because C# must make all of its decisions around lifetime safety based on the information available in the signature of the method and the lifetime of the values at the call site.
When there are ref parameters that are ref struct then there is the potential that they could swap around their contents. Hence at the call site we must ensure all of these potential swaps are compatible. If the language didn't enforce that then it will allow for bad code like the following.
void M1(ref Span<int> s1)
{
Span<int> s2 = stackalloc int[1];
Swap(ref s1, ref s2);
}
void Swap(ref Span<int> x, ref Span<int> y)
{
// This will effectively assign the stackalloc to the s1 parameter and allow it
// to escape to the caller of M1
ref x = ref y;
}This analysis of ref parameters includes the receiver in instance methods. This is necessary because it can be used to store values passed in as parameters, just as a ref parameter could`. This means with mismatched lifetimes you could create a type safety hole in the following way:
ref struct S
{
public Span<int> Span;
public void Set(Span<int> span)
{
Span = span;
}
}
void Broken(ref S s)
{
Span<int> span = stackalloc int[1];
// The result of a stackalloc is now stored in s.Span and escaped to the caller
// of Broken
s.Set(span);
}For the purpose of this analysis the receiver is considered an in, not a ref, if the type is a readonly struct. In that case the receiver cannot be used to store values from other parameters, it is effectively an in parameter for analysis purposes. Hence the same example above is legal when S is readonly because the span cannot be stored anywhere.
When it comes to span safety rules, the this value in an instance member is modeled as a parameter to the member. Now for a struct the type of this is actually ref S where in a class it's simply S (for members of a class / struct named S).
Yet this has different escaping rules than other ref parameters. Specifically it is not ref-safe-context while other parameters are:
ref struct S
{
int Field;
// Illegal because `this` isn't safe to escape as ref
ref int Get() => ref Field;
// Legal
ref int GetParam(ref int p) => ref p;
}The reason for this restriction actually has little to do with struct member invocation. There are some rules that need to be worked out with respect to member invocation on struct members where the receiver is an rvalue. But that is very approachable.
The reason for this restriction is actually about interface invocation. Specifically it comes down to whether or not the following sample should or should not compile;
interface I1
{
ref int Get();
}
ref int Use<T>(T p)
where T : I1
{
return ref p.Get();
}Consider the case where T is instantiated as a struct. If the this parameter is ref-safe-context then the return of p.Get could point to the stack (specifically it could be a field inside of the instantiated type of T). That means the language could not allow this sample to compile as it could be returning a ref to a stack location. On the other hand if this is not ref-safe-context then p.Get cannot refer to the stack and hence it's safe to return.
This is why the escapability of this in a struct is really all about interfaces. It can absolutely be made to work but it has a trade off. The design eventually came down in favor of making interfaces more flexible.
There is potential for us to relax this in the future though.
Though not legal today there are cases where creating a length one Span<T> instance over a value would be beneficial:
void RefExample()
{
int x = ...;
// Today creating a length one Span<int> requires a stackalloc and a new
// local
Span<int> span1 = stackalloc [] { x };
Use(span1);
x = span1[0];
// Simpler to just allow length one span
var span2 = new Span<int>(ref x);
Use(span2);
}This feature gets more compelling if we lift the restrictions on fixed sized buffers as it would
allow for Span<T> instances of even greater length.
If there is ever a need to go down this path then the language could accommodate this by ensuring such Span<T> instances
were downward facing only. That is they were only ever safe-context to the scope in which they were created. This ensure
the language never had to consider a ref value escaping a method via a ref struct return or field of ref struct. This
would likely also require further changes to recognize such constructors as capturing a ref parameter in this way though.