Description
Now that the swap operator has been removed, I do not believe the
restriction against mutating LV
is needed, and in fact it prevents
some useful patterns. For example, the following function will
fail to compile:
fn mut_shift_ref<'a,T>(x: &mut &'a mut [T]) -> &'a mut T {
// `mut_split` will restrict mutation against *x:
let (head, tail) = (*x).mut_split(1);
// Hence mutating `*x` yields an error here:
*x = tail;
&mut head[0]
}
Note that this function -- which adjusts the slice *x
in place so
that it no longer contains the head element and then returns a
pointer to that element separately -- is perfectly valid. It is
currently implemented using unsafe code. I believe that now that
the swap operator is removed from the language, we could liberalize
the rules and make this function be accepted normally. The idea
would be to have the assignment to *x
kill the loans of *x
and
its subpaths -- after all, those subpaths are no longer accessible
through *x
, since it has been overwritten with a new value. Thus
those subpaths are only accessible through prior existing borrows
of *x
, if any. The danger of the swap operator was that it
allowed *x
to be mutated without making the subpaths of *x
inaccessible: worse, they became accessible through a new path (I
suppose that we could have supported a swap operator, too, if needed, by moving the loans over to the new path).