doc: describe upcoming automatic lock feature (#5795)
parent
8df8866c5a
commit
3c3a91697e
|
@ -7,7 +7,11 @@ for the current state of V***
|
||||||
## Table of Contents
|
## Table of Contents
|
||||||
|
|
||||||
* [Concurrency](#concurrency)
|
* [Concurrency](#concurrency)
|
||||||
* [Variable Declarations](#concurrency-variable-declarations)
|
* [Variable Declarations](#variable-declarations)
|
||||||
|
* [Strengths](#strengths)
|
||||||
|
* [Weaknesses](#weaknesses)
|
||||||
|
* [Compatibility](#compatibility)
|
||||||
|
* [Automatic Lock](#automatic-lock)
|
||||||
|
|
||||||
## Concurrency
|
## Concurrency
|
||||||
|
|
||||||
|
@ -111,3 +115,78 @@ counting. Once the counter reaches 0 the object is automatically freed.
|
||||||
<sup>3</sup> Since an `atomic` variable is only a few bytes in size
|
<sup>3</sup> Since an `atomic` variable is only a few bytes in size
|
||||||
allocation would be an unnecessary overhead. Instead the compiler
|
allocation would be an unnecessary overhead. Instead the compiler
|
||||||
creates a global.
|
creates a global.
|
||||||
|
|
||||||
|
### Compatibility
|
||||||
|
Outside of `lock`/`rlock` blocks function arguments must in general
|
||||||
|
match - with the familiar exception that objects declared `mut` can be
|
||||||
|
used to call functions expecting immutable arguments:
|
||||||
|
|
||||||
|
```v
|
||||||
|
fn f(x St) {...}
|
||||||
|
fn g(mut x St) {...}
|
||||||
|
fn h(shared x St) {...}
|
||||||
|
fn i(atomic x u64) {...}
|
||||||
|
|
||||||
|
a := St{...}
|
||||||
|
f(a)
|
||||||
|
|
||||||
|
mut b := &St{...} // reference since transferred to coroutine
|
||||||
|
f(b)
|
||||||
|
go g(mut b)
|
||||||
|
// `b` should not be accessed here any more
|
||||||
|
|
||||||
|
shared c := &St{...}
|
||||||
|
h(shared c)
|
||||||
|
|
||||||
|
atomic d &u64
|
||||||
|
i(atomic d)
|
||||||
|
```
|
||||||
|
|
||||||
|
Inside a `lock c {...}` block `c` behaves like a `mut`,
|
||||||
|
inside an `rlock c {...}` block like an immutable:
|
||||||
|
```v
|
||||||
|
shared c := &St{...}
|
||||||
|
lock c {
|
||||||
|
g(mut c)
|
||||||
|
f(c)
|
||||||
|
// call to h() not allowed inside `lock` block
|
||||||
|
// since h() will lock `c` itself
|
||||||
|
}
|
||||||
|
rlock c {
|
||||||
|
f(c)
|
||||||
|
// call to g() or h() not allowed
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Automatic Lock
|
||||||
|
In general the compiler will generate an error message when a `shared`
|
||||||
|
object is accessed outside of any corresponding `lock`/`rlock`
|
||||||
|
block. However in simple and obvious cases the necessary lock/unlock
|
||||||
|
can be generated automatically for `array`/`map` operations:
|
||||||
|
|
||||||
|
```v
|
||||||
|
shared a []int{...}
|
||||||
|
go h2(shared a)
|
||||||
|
a << 3
|
||||||
|
// keep in mind that `h2()` could change `a` between these statements
|
||||||
|
a << 4
|
||||||
|
x := a[1] // not necessarily `4`
|
||||||
|
|
||||||
|
shared b map[string]int
|
||||||
|
go h3(shared b)
|
||||||
|
b['apple'] = 3
|
||||||
|
c['plume'] = 7
|
||||||
|
y := b['apple'] // not necesarily `3`
|
||||||
|
|
||||||
|
// iteration over elements
|
||||||
|
for k, v in b {
|
||||||
|
// concurrently changed k/v pairs may or my not be included
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
This is handy, but since other coroutines might access the `array`/`map`
|
||||||
|
concurrently between the automatically locked statements, the results
|
||||||
|
are sometimes surprising. Each statement should be seen as a single
|
||||||
|
transaction that is unrelated to the previous or following
|
||||||
|
statement. Therefore - but also for performance reasons - it's often
|
||||||
|
better to group consecutive coherent statements in an explicit `lock` block.
|
||||||
|
|
Loading…
Reference in New Issue