196 lines
5.6 KiB
Markdown
196 lines
5.6 KiB
Markdown
# V Work In Progress
|
|
|
|
***This document describes features that are not implemented, yet.
|
|
Please refer to [docs.md](https://github.com/vlang/v/blob/master/doc/docs.md)
|
|
for the current state of V***
|
|
|
|
## Table of Contents
|
|
|
|
* [Concurrency](#concurrency)
|
|
* [Variable Declarations](#variable-declarations)
|
|
* [Strengths](#strengths)
|
|
* [Weaknesses](#weaknesses)
|
|
* [Compatibility](#compatibility)
|
|
* [Automatic Lock](#automatic-lock)
|
|
* [Channels](#channels)
|
|
|
|
## Concurrency
|
|
|
|
### Variable Declarations
|
|
|
|
Objects that are supposed to be used to exchange data between
|
|
coroutines have to be declared with special care. Exactly one of the following
|
|
4 kinds of declaration has to be chosen:
|
|
|
|
```v ignore
|
|
a := ...
|
|
mut b := ...
|
|
shared c := ...
|
|
atomic d := ...
|
|
```
|
|
|
|
- `a` is declared as *constant* that can be passed to
|
|
other coroutines and read without limitations. However
|
|
it cannot be changed.
|
|
- `b` can be accessed reading and writing but only from one
|
|
coroutine. That coroutine *owns* the object. A `mut` variable can
|
|
be passed to another coroutine (as receiver or function argument in
|
|
the `go` statement or via a channel) but then ownership is passed,
|
|
too, and only the other coroutine can access the object.<sup>1</sup>
|
|
- `c` can be passed to coroutines an accessed
|
|
*concurrently*.<sup>2</sup> In order to avoid data races it has to
|
|
be locked before access can occur and unlocked to allow access to
|
|
other coroutines. This is done by one the following block structures:
|
|
```v ignore
|
|
lock c {
|
|
// read, modify, write c
|
|
...
|
|
}
|
|
```
|
|
|
|
```v ignore
|
|
rlock c {
|
|
// read c
|
|
...
|
|
}
|
|
```
|
|
Several variables may be specified: `lock x, y, z { ... }`.
|
|
They are unlocked in the opposite order.
|
|
- `d` can be passed to coroutines and accessed *concurrently*,
|
|
too.<sup>3</sup> No lock is needed in this case, however
|
|
`atomic` variables can only be 32/64 bit integers (or pointers)
|
|
and access is limited to a small set of predefined idioms that have
|
|
native hardware support.
|
|
|
|
To help making the correct decision the following table summarizes the
|
|
different capabilities:
|
|
|
|
| | *default* | `mut` | `shared` | `atomic` |
|
|
| :--- | :---: | :---: | :---: | :---: |
|
|
| write access | | + | + | + |
|
|
| concurrent access | + | | + | + |
|
|
| performance | ++ | ++ | | + |
|
|
| sophisticated operations | + | + | + | |
|
|
| structured data types | + | + | + | |
|
|
|
|
### Strengths
|
|
**default**
|
|
- very fast
|
|
- unlimited access from different coroutines
|
|
- easy to handle
|
|
|
|
**`mut`**
|
|
- very fast
|
|
- easy to handle
|
|
|
|
**`shared`**
|
|
- concurrent access from different coroutines
|
|
- data type may be complex structure
|
|
- sophisticated access possible (several statements within one `lock`
|
|
block)
|
|
|
|
**`atomic`**
|
|
- concurrent access from different coroutines
|
|
- reasonably fast
|
|
|
|
### Weaknesses
|
|
**default**
|
|
- read only
|
|
|
|
**`mut`**
|
|
- access only from one coroutine at a time
|
|
|
|
**`shared`**
|
|
- lock/unlock are slow
|
|
- moderately difficult to handle (needs `lock` block)
|
|
|
|
**`atomic`**
|
|
- limited to single (max. 64 bit) integers (and pointers)
|
|
- only a small set of predefined operations possible
|
|
- very difficult to handle correctly
|
|
|
|
<sup>1</sup> The owning coroutine will also free the memory space used
|
|
for the object when it is no longer needed.
|
|
<sup>2</sup> For `shared` objects the compiler adds code for reference
|
|
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
|
|
allocation would be an unnecessary overhead. Instead the compiler
|
|
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 ignore
|
|
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{...}
|
|
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 ignore
|
|
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 ignore
|
|
shared a := []int{cap: 5}
|
|
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.
|
|
|
|
### Channels
|