Skip to content

Commit

Permalink
Browse files Browse the repository at this point in the history
* fixes nim-lang#13881
* documented changed requirements for system.onThreadDestruction
* destructors.rst: update the documentation
  • Loading branch information
Araq authored May 12, 2020
1 parent 4277ab4 commit 06dfd31
Show file tree
Hide file tree
Showing 9 changed files with 84 additions and 23 deletions.
3 changes: 3 additions & 0 deletions changelog.md
Original file line number Diff line number Diff line change
Expand Up @@ -89,6 +89,9 @@
- Fix a bug where calling `close` on io streams in osproc.startProcess was a noop and led to
hangs if a process had both reads from stdin and writes (eg to stdout).

- The callback that is passed to `system.onThreadDestruction` must now be `.raises: []`.


## Language changes
- In the newruntime it is now allowed to assign discriminator field without restrictions as long as case object doesn't have custom destructor. Discriminator value doesn't have to be a constant either. If you have custom destructor for case object and you do want to freely assign discriminator fields, it is recommended to refactor object into 2 objects like this:

Expand Down
1 change: 1 addition & 0 deletions compiler/condsyms.nim
Original file line number Diff line number Diff line change
Expand Up @@ -117,3 +117,4 @@ proc initDefines*(symbols: StringTableRef) =
defineSymbol("nimNewIntegerOps")
defineSymbol("nimHasInvariant")
defineSymbol("nimHasStacktraceMsgs")
defineSymbol("nimDoesntTrackDefects")
47 changes: 44 additions & 3 deletions doc/destructors.rst
Original file line number Diff line number Diff line change
Expand Up @@ -469,10 +469,51 @@ for expressions of type ``lent T`` or of type ``var T``.
echo t[0] # accessor does not copy the element!
The .cursor annotation
======================

Under the ``--gc:arc|orc`` modes Nim's `ref` type is implemented via the same runtime
"hooks" and thus via reference counting. This means that cyclic structures cannot be freed
immediately (``--gc:orc`` ships with a cycle collector). With the ``.cursor`` annotation
one can break up cycles declaratively:

.. code-block:: nim
type
Node = ref object
left: Node # owning ref
right {.cursor.}: Node # non-owning ref
But please notice that this is not C++'s weak_ptr, it means the right field is not
involved in the reference counting, it is a raw pointer without runtime checks.

Automatic reference counting also has the disadvantage that it introduces overhead
when iterating over linked structures. The ``.cursor`` annotation can also be used
to avoid this overhead:

.. code-block:: nim
var it {.cursor.} = listRoot
while it != nil:
use(it)
it = it.next
In fact, ``.cursor`` more generally prevents object construction/destruction pairs
and so can also be useful in other contexts. The alternative solution would be to
use raw pointers (``ptr``) instead which is more cumbersome and also more dangerous
for Nim's evolution: Later on the compiler can try to prove ``.cursor`` annotations
to be safe, but for ``ptr`` the compiler has to remain silent about possible
problems.


Owned refs
==========

**Note**: The ``owned`` type constructor is only available with
the ``--newruntime`` compiler switch and is experimental.


Let ``W`` be an ``owned ref`` type. Conceptually its hooks look like:

.. code-block:: nim
Expand Down Expand Up @@ -568,14 +609,14 @@ used to specialize the object traversal in order to avoid deep recursions:
type Node = ref object
x, y: int32
left, right: owned Node
left, right: Node
type Tree = object
root: owned Node
root: Node
proc `=destroy`(t: var Tree) {.nodestroy.} =
# use an explicit stack so that we do not get stack overflows:
var s: seq[owned Node] = @[t.root]
var s: seq[Node] = @[t.root]
while s.len > 0:
let x = s.pop
if x.left != nil: s.add(x.left)
Expand Down
4 changes: 2 additions & 2 deletions lib/system.nim
Original file line number Diff line number Diff line change
Expand Up @@ -692,7 +692,7 @@ proc len*(x: string): int {.magic: "LengthStr", noSideEffect.}
proc len*(x: cstring): int {.magic: "LengthStr", noSideEffect.}
## Returns the length of a compatible string. This is sometimes
## an O(n) operation.
##
##
## **Note:** On the JS backend this currently counts UTF-16 code points
## instead of bytes at runtime (not at compile time). For now, if you
## need the byte length of the UTF-8 encoding, convert to string with
Expand Down Expand Up @@ -2093,7 +2093,7 @@ when not defined(js):

when hasAlloc:
when not defined(gcRegions) and not usesDestructors:
proc initGC() {.gcsafe.}
proc initGC() {.gcsafe, raises: [].}

proc initStackBottom() {.inline, compilerproc.} =
# WARNING: This is very fragile! An array size of 8 does not work on my
Expand Down
8 changes: 8 additions & 0 deletions lib/system/excpt.nim
Original file line number Diff line number Diff line change
Expand Up @@ -493,6 +493,14 @@ proc reraiseException() {.compilerRtl.} =
else:
raiseExceptionAux(currException)

proc threadTrouble() =
# also forward declared, it is 'raises: []' hence the try-except.
try:
if currException != nil: reportUnhandledError(currException)
except:
discard
quit 1

proc writeStackTrace() =
when hasSomeStackTrace:
var s = ""
Expand Down
7 changes: 4 additions & 3 deletions lib/system/gc.nim
Original file line number Diff line number Diff line change
Expand Up @@ -843,9 +843,10 @@ when not defined(useNimRtl):
proc GC_disable() =
inc(gch.recGcLock)
proc GC_enable() =
if gch.recGcLock <= 0:
raise newException(AssertionDefect,
"API usage error: GC_enable called but GC is already enabled")
when defined(nimDoesntTrackDefects):
if gch.recGcLock <= 0:
raise newException(AssertionDefect,
"API usage error: GC_enable called but GC is already enabled")
dec(gch.recGcLock)

proc GC_setStrategy(strategy: GC_Strategy) =
Expand Down
6 changes: 3 additions & 3 deletions lib/system/gc_interface.nim
Original file line number Diff line number Diff line change
Expand Up @@ -14,15 +14,15 @@ when hasAlloc:
gcOptimizeSpace ## optimize for memory footprint

when hasAlloc and not defined(js) and not usesDestructors:
proc GC_disable*() {.rtl, inl, benign.}
proc GC_disable*() {.rtl, inl, benign, raises: [].}
## Disables the GC. If called `n` times, `n` calls to `GC_enable`
## are needed to reactivate the GC.
##
## Note that in most circumstances one should only disable
## the mark and sweep phase with
## `GC_disableMarkAndSweep <#GC_disableMarkAndSweep>`_.

proc GC_enable*() {.rtl, inl, benign.}
proc GC_enable*() {.rtl, inl, benign, raises: [].}
## Enables the GC again.

proc GC_fullCollect*() {.rtl, benign.}
Expand Down Expand Up @@ -54,7 +54,7 @@ when hasAlloc and not defined(js) and not usesDestructors:
proc GC_unref*(x: string) {.magic: "GCunref", benign.}
## See the documentation of `GC_ref <#GC_ref,string>`_.

proc nimGC_setStackBottom*(theStackBottom: pointer) {.compilerRtl, noinline, benign.}
proc nimGC_setStackBottom*(theStackBottom: pointer) {.compilerRtl, noinline, benign, raises: [].}
## Expands operating GC stack range to `theStackBottom`. Does nothing
## if current stack bottom is already lower than `theStackBottom`.

Expand Down
7 changes: 4 additions & 3 deletions lib/system/gc_ms.nim
Original file line number Diff line number Diff line change
Expand Up @@ -495,9 +495,10 @@ when not defined(useNimRtl):
proc GC_disable() =
inc(gch.recGcLock)
proc GC_enable() =
if gch.recGcLock <= 0:
raise newException(AssertionDefect,
"API usage error: GC_enable called but GC is already enabled")
when defined(nimDoesntTrackDefects):
if gch.recGcLock <= 0:
raise newException(AssertionDefect,
"API usage error: GC_enable called but GC is already enabled")
dec(gch.recGcLock)

proc GC_setStrategy(strategy: GC_Strategy) = discard
Expand Down
24 changes: 15 additions & 9 deletions lib/system/threads.nim
Original file line number Diff line number Diff line change
Expand Up @@ -89,9 +89,9 @@ type
data: TArg

var
threadDestructionHandlers {.rtlThreadVar.}: seq[proc () {.closure, gcsafe.}]
threadDestructionHandlers {.rtlThreadVar.}: seq[proc () {.closure, gcsafe, raises: [].}]

proc onThreadDestruction*(handler: proc () {.closure, gcsafe.}) =
proc onThreadDestruction*(handler: proc () {.closure, gcsafe, raises: [].}) =
## Registers a *thread local* handler that is called at the thread's
## destruction.
##
Expand All @@ -105,7 +105,10 @@ template afterThreadRuns() =
threadDestructionHandlers[i]()

when not defined(boehmgc) and not hasSharedHeap and not defined(gogc) and not defined(gcRegions):
proc deallocOsPages() {.rtl.}
proc deallocOsPages() {.rtl, raises: [].}

proc threadTrouble() {.raises: [].}
## defined in system/excpt.nim

when defined(boehmgc):
type GCStackBaseProc = proc(sb: pointer, t: pointer) {.noconv.}
Expand All @@ -116,19 +119,21 @@ when defined(boehmgc):
proc boehmGC_unregister_my_thread()
{.importc: "GC_unregister_my_thread", boehmGC.}

proc threadProcWrapDispatch[TArg](sb: pointer, thrd: pointer) {.noconv.} =
proc threadProcWrapDispatch[TArg](sb: pointer, thrd: pointer) {.noconv, raises: [].} =
boehmGC_register_my_thread(sb)
try:
let thrd = cast[ptr Thread[TArg]](thrd)
when TArg is void:
thrd.dataFn()
else:
thrd.dataFn(thrd.data)
except:
threadTrouble()
finally:
afterThreadRuns()
boehmGC_unregister_my_thread()
else:
proc threadProcWrapDispatch[TArg](thrd: ptr Thread[TArg]) =
proc threadProcWrapDispatch[TArg](thrd: ptr Thread[TArg]) {.raises: [].} =
try:
when TArg is void:
thrd.dataFn()
Expand All @@ -139,21 +144,22 @@ else:
var x: TArg
deepCopy(x, thrd.data)
thrd.dataFn(x)
except:
threadTrouble()
finally:
afterThreadRuns()

proc threadProcWrapStackFrame[TArg](thrd: ptr Thread[TArg]) =
proc threadProcWrapStackFrame[TArg](thrd: ptr Thread[TArg]) {.raises: [].} =
when defined(boehmgc):
boehmGC_call_with_stack_base(threadProcWrapDispatch[TArg], thrd)
elif not defined(nogc) and not defined(gogc) and not defined(gcRegions) and not usesDestructors:
var p {.volatile.}: proc(a: ptr Thread[TArg]) {.nimcall, gcsafe.} =
threadProcWrapDispatch[TArg]
var p {.volatile.}: pointer
# init the GC for refc/markandsweep
nimGC_setStackBottom(addr(p))
initGC()
when declared(threadType):
threadType = ThreadType.NimThread
p(thrd)
threadProcWrapDispatch[TArg](thrd)
when declared(deallocOsPages): deallocOsPages()
else:
threadProcWrapDispatch(thrd)
Expand Down

0 comments on commit 06dfd31

Please sign in to comment.