mirror of
https://github.com/nim-lang/Nim.git
synced 2026-01-02 11:12:37 +00:00
* Nil checking * Enable current older not nil checking again, run new checking only under flag, skip our test * Enable tests, work on try/except and bugs, fix notnil tests * Enable strictNotNil tests (currently with lowercase category) and add some expected output * Work on try/except/finally: still some things unclear and a lot of code can raise out of try * Fix the notnil build by going back to the old version of a test which I shouldn't have changed * Fix test : use action compile * Work on mutation and aliasing: not finished * Render var parititions graph, try to understand it, fix a nilcheck if bug * Rebase, progress on working with partitions * Improve time logic * Fix some bugs, use graph indices instead of symbol in nil map * Fix bugs, test simpler ident aliasing for now, support two mutation levels * Support ContentMutation and ReAssignment: for now just detect possible re assignment for var parameters of calls * Enable several simple passing tests * Cleanup a bit, fix condition/branch infix-related bug * Remove some files, address some comments by Araq * Use internalError and no quit for now * Separate tests with expected warnings and with expected ok, fix a bug with if with a single branch related to copyMap * Fix new data structures, bugs: make tests pass, disable some for now * Work on fixing errors with non-sym nodes, aliasing: tests fail * Work on alias support: simple set-based logic, todo more tests and ref sets? * Use ref sets: TODO can we think of handle seq-s similar to varpartitions' Araq ones * Handle defers in one place, stop raising in reverse to make an async test compile with strictNotNil, add a commented out test * Dot expressions: call/reassignment. Other refactorings and distinct, SeqOfDistinct support. Checkout an older varpartitions * Work on field tracking * Backup : trying to fix bugs when running some stdlib stuff for running an async test * Start a section about strict not nil checking in experimental manual * Fix experimental strict not nil manual section and move it to another file based on Araq feedback * Fix unstructured flow and double warning problems, fix manual, cleanup * Fix if/elif/else : take in account structure according to Araq feedback * Refactor a bit * Work on bracket expr support, re-enable tests, clarify in manual/tests/implementation static index support for now * Work on compiling stdlib and compiler with strictNotNil * Small fixes to the manual for strictNotNil * Fix idgen for strict check nil rebase * Enable some simple tests, remove old stuff, comment out code/print * Copy the original varpartitions source instead of my changes * Remove some files
383 lines
8.2 KiB
Nim
383 lines
8.2 KiB
Nim
discard """
|
|
cmd: "nim check $file"
|
|
action: "reject"
|
|
"""
|
|
|
|
import tables
|
|
|
|
{.experimental: "strictNotNil".}
|
|
|
|
type
|
|
Nilable* = ref object
|
|
a*: int
|
|
field*: Nilable
|
|
|
|
NonNilable* = Nilable not nil
|
|
|
|
Nilable2* = nil NonNilable
|
|
|
|
|
|
# proc `[]`(a: Nilable, b: int): Nilable =
|
|
# nil
|
|
|
|
|
|
# Nilable tests
|
|
|
|
|
|
|
|
# test deref
|
|
proc testDeref(a: Nilable) =
|
|
echo a.a > 0 #[tt.Warning
|
|
^ can't deref a, it might be nil
|
|
]#
|
|
|
|
|
|
|
|
# # # test if else
|
|
proc testIfElse(a: Nilable) =
|
|
if a.isNil:
|
|
echo a.a #[tt.Warning
|
|
^ can't deref a, it is nil
|
|
]#
|
|
else:
|
|
echo a.a # ok
|
|
|
|
proc testIfNoElse(a: Nilable) =
|
|
if a.isNil:
|
|
echo a.a #[tt.Warning
|
|
^ can't deref a, it is nil
|
|
]#
|
|
echo a.a #[tt.Warning
|
|
^ can't deref a, it might be nil
|
|
]#
|
|
|
|
proc testIfReturn(a: Nilable) =
|
|
if not a.isNil:
|
|
return
|
|
echo a.a #[tt.Warning
|
|
^ can't deref a, it is nil
|
|
]#
|
|
|
|
proc testIfBreak(a: seq[Nilable]) =
|
|
for b in a:
|
|
if not b.isNil:
|
|
break
|
|
echo b.a #[tt.Warning
|
|
^ can't deref b, it is nil
|
|
]#
|
|
|
|
proc testIfContinue(a: seq[Nilable]) =
|
|
for b in a:
|
|
if not b.isNil:
|
|
continue
|
|
echo b.a #[tt.Warning
|
|
^ can't deref b, it is nil
|
|
]#
|
|
|
|
proc testIfRaise(a: Nilable) =
|
|
if not a.isNil:
|
|
raise newException(ValueError, "")
|
|
echo a.a #[tt.Warning
|
|
^ can't deref a, it is nil
|
|
]#
|
|
|
|
proc testIfElif(a: Nilable) =
|
|
var c = 0
|
|
if c == 0:
|
|
echo a.a #[tt.Warning
|
|
^ can't deref a, it might be nil
|
|
]#
|
|
elif c == 1:
|
|
echo a.a #[tt.Warning
|
|
^ can't deref a, it might be nil
|
|
]#
|
|
elif not a.isNil:
|
|
echo a.a # ok
|
|
elif c == 2:
|
|
echo 0
|
|
else:
|
|
echo a.a #[tt.Warning
|
|
^ can't deref a, it is nil
|
|
]#
|
|
|
|
proc testAssignUnify(a: Nilable, b: int) =
|
|
var a2 = a
|
|
if b == 0:
|
|
a2 = Nilable()
|
|
echo a2.a #[tt.Warning
|
|
^ can't deref a2, it might be nil
|
|
]#
|
|
|
|
|
|
# # test assign in branch and unifiying that with the main block after end of branch
|
|
proc testAssignUnifyNil(a: Nilable, b: int) =
|
|
var a2 = a
|
|
if b == 0:
|
|
a2 = nil
|
|
echo a2.a #[tt.Warning
|
|
^ can't deref a2, it might be nil
|
|
]#
|
|
|
|
# test loop
|
|
proc testForLoop(a: Nilable) =
|
|
var b = Nilable()
|
|
for i in 0 .. 5:
|
|
echo b.a #[tt.Warning
|
|
^ can't deref b, it might be nil
|
|
]#
|
|
if i == 2:
|
|
b = a
|
|
echo b.a #[tt.Warning
|
|
^ can't deref b, it might be nil
|
|
]#
|
|
|
|
|
|
|
|
# # TODO implement this after discussion
|
|
# # proc testResultCompoundNonNilableElement(a: Nilable): (NonNilable, NonNilable) = #[t t.Warning
|
|
# # ^ result might be not initialized, so it or an element might be nil
|
|
# # ]#
|
|
# # if not a.isNil:
|
|
# # result[0] = a #[t t.Warning
|
|
# # ^ can't assign nilable to non nilable: it might be nil
|
|
# # #]
|
|
|
|
# # proc testNonNilDeref(a: NonNilable) =
|
|
# # echo a.a # ok
|
|
|
|
|
|
|
|
# # # not only calls: we can use partitions for dependencies for field aliases
|
|
# # # so we can detect on change what does this affect or was this mutated between us and the original field
|
|
|
|
# # proc testRootAliasField(a: Nilable) =
|
|
# # var aliasA = a
|
|
# # if not a.isNil and not a.field.isNil:
|
|
# # aliasA.field = nil
|
|
# # # a.field = nil
|
|
# # # aliasA = nil
|
|
# # echo a.field.a # [tt.Warning
|
|
# # ^ can't deref a.field, it might be nil
|
|
# # ]#
|
|
|
|
|
|
proc testAliasChanging(a: Nilable) =
|
|
var b = a
|
|
var aliasA = b
|
|
b = Nilable()
|
|
if not b.isNil:
|
|
echo aliasA.a #[tt.Warning
|
|
^ can't deref aliasA, it might be nil
|
|
]#
|
|
|
|
# # TODO
|
|
# # proc testAliasUnion(a: Nilable) =
|
|
# # var a2 = a
|
|
# # var b = a2
|
|
# # if a.isNil:
|
|
# # b = Nilable()
|
|
# # a2 = nil
|
|
# # else:
|
|
# # a2 = Nilable()
|
|
# # b = a2
|
|
# # if not b.isNil:
|
|
# # echo a2.a #[ tt.Warning
|
|
# # ^ can't deref a2, it might be nil
|
|
# # ]#
|
|
|
|
# # TODO after alias support
|
|
# #proc callVar(a: var Nilable) =
|
|
# # a.field = nil
|
|
|
|
|
|
# # TODO ptr support
|
|
# # proc testPtrAlias(a: Nilable) =
|
|
# # # pointer to a: hm.
|
|
# # # alias to a?
|
|
# # var ptrA = a.unsafeAddr # {0, 1}
|
|
# # if not a.isNil: # {0, 1}
|
|
# # ptrA[] = nil # {0, 1} 0: MaybeNil 1: MaybeNil
|
|
# # echo a.a #[ tt.Warning
|
|
# # ^ can't deref a, it might be nil
|
|
# # ]#
|
|
|
|
# # TODO field stuff
|
|
# # currently it just doesnt support dot, so accidentally it shows a warning but because that
|
|
# # not alias i think
|
|
# # proc testFieldAlias(a: Nilable) =
|
|
# # var b = a # {0, 1} {2}
|
|
# # if not a.isNil and not a.field.isNil: # {0, 1} {2}
|
|
# # callVar(b) # {0, 1} {2} 0: Safe 1: Safe
|
|
# # echo a.field.a #[ tt.Warning
|
|
# # ^ can't deref a.field, it might be nil
|
|
# # ]#
|
|
# #
|
|
# # proc testUniqueHashTree(a: Nilable): Nilable =
|
|
# # # TODO what would be a clash
|
|
# # var field = 0
|
|
# # if not a.isNil and not a.field.isNil:
|
|
# # # echo a.field.a
|
|
# # echo a[field].a
|
|
# # result = Nilable()
|
|
|
|
# # proc testSeparateShadowingResult(a: Nilable): Nilable =
|
|
# # result = Nilable()
|
|
# # if not a.isNil:
|
|
# # var result: Nilable = nil
|
|
# # echo result.a
|
|
|
|
|
|
proc testCStringDeref(a: cstring) =
|
|
echo a[0] #[tt.Warning
|
|
^ can't deref a, it might be nil
|
|
]#
|
|
|
|
|
|
proc testNilablePtr(a: ptr int) =
|
|
if not a.isNil:
|
|
echo a[] # ok
|
|
echo a[] #[tt.Warning
|
|
^ can't deref a, it might be nil
|
|
]#
|
|
|
|
# # proc testNonNilPtr(a: ptr int not nil) =
|
|
# # echo a[] # ok
|
|
|
|
proc raiseCall: NonNilable = #[tt.Warning
|
|
^ return value is nil
|
|
]#
|
|
raise newException(ValueError, "raise for test")
|
|
|
|
# proc testTryCatch(a: Nilable) =
|
|
# var other = a
|
|
# try:
|
|
# other = raiseCall()
|
|
# except:
|
|
# discard
|
|
# echo other.a #[ tt.Warning
|
|
# ^ can't deref other, it might be nil
|
|
# ]#
|
|
|
|
# # proc testTryCatchDetectNoRaise(a: Nilable) =
|
|
# # var other = Nilable()
|
|
# # try:
|
|
# # other = nil
|
|
# # other = a
|
|
# # other = Nilable()
|
|
# # except:
|
|
# # other = nil
|
|
# # echo other.a # ok
|
|
|
|
# # proc testTryCatchDetectFinally =
|
|
# # var other = Nilable()
|
|
# # try:
|
|
# # other = nil
|
|
# # other = Nilable()
|
|
# # except:
|
|
# # other = Nilable()
|
|
# # finally:
|
|
# # other = nil
|
|
# # echo other.a # can't deref other: it is nil
|
|
|
|
# # proc testTryCatchDetectNilableWithRaise(b: bool) =
|
|
# # var other = Nilable()
|
|
# # try:
|
|
# # if b:
|
|
# # other = nil
|
|
# # else:
|
|
# # other = Nilable()
|
|
# # var other2 = raiseCall()
|
|
# # except:
|
|
# # echo other.a # ok
|
|
|
|
# # echo other.a # can't deref a: it might be nil
|
|
|
|
# # proc testRaise(a: Nilable) =
|
|
# # if a.isNil:
|
|
# # raise newException(ValueError, "a == nil")
|
|
# # echo a.a # ok
|
|
|
|
|
|
# # proc testBlockScope(a: Nilable) =
|
|
# # var other = a
|
|
# # block:
|
|
# # var other = Nilable()
|
|
# # echo other.a # ok
|
|
# # echo other.a # can't deref other: it might be nil
|
|
|
|
# # ok we can't really get the nil value from here, so should be ok
|
|
# # proc testDirectRaiseCall: NonNilable =
|
|
# # var a = raiseCall()
|
|
# # result = NonNilable()
|
|
|
|
# # proc testStmtList =
|
|
# # var a = Nilable()
|
|
# # block:
|
|
# # a = nil
|
|
# # a = Nilable()
|
|
# # echo a.a # ok
|
|
|
|
proc callChange(a: Nilable) =
|
|
if not a.isNil:
|
|
a.field = nil
|
|
|
|
proc testCallChangeField =
|
|
var a = Nilable()
|
|
a.field = Nilable()
|
|
callChange(a)
|
|
echo a.field.a #[ tt.Warning
|
|
^ can't deref a.field, it might be nil
|
|
]#
|
|
|
|
proc testReassignVarWithField =
|
|
var a = Nilable()
|
|
a.field = Nilable()
|
|
echo a.field.a # ok
|
|
a = Nilable()
|
|
echo a.field.a #[ tt.Warning
|
|
^ can't deref a.field, it might be nil
|
|
]#
|
|
|
|
|
|
proc testItemDeref(a: var seq[Nilable]) =
|
|
echo a[0].a #[tt.Warning
|
|
^ can't deref a[0], it might be nil
|
|
]#
|
|
a[0] = Nilable() # good: now .. if we dont track, how do we know
|
|
echo a[0].a # ok
|
|
echo a[1].a #[tt.Warning
|
|
^ can't deref a[1], it might be nil
|
|
]#
|
|
var b = 1
|
|
if a[b].isNil:
|
|
echo a[1].a #[tt.Warning
|
|
^ can't deref a[1], it might be nil
|
|
]#
|
|
var c = 0
|
|
echo a[c].a #[tt.Warning
|
|
^ can't deref a[c], it might be nil
|
|
]#
|
|
|
|
# known false positive
|
|
if not a[b].isNil:
|
|
echo a[b].a #[tt.Warning
|
|
^ can't deref a[b], it might be nil
|
|
]#
|
|
|
|
const c = 0
|
|
if a[c].isNil:
|
|
echo a[0].a #[tt.Warning
|
|
^ can't deref a[0], it is nil
|
|
]#
|
|
a[c] = Nilable()
|
|
echo a[0].a # ok
|
|
|
|
|
|
|
|
# # # proc test10(a: Nilable) =
|
|
# # # if not a.isNil and not a.b.isNil:
|
|
# # # c_memset(globalA.addr, 0, globalA.sizeOf.csize_t)
|
|
# # # globalA = nil
|
|
# # # echo a.a # can't deref a: it might be nil
|
|
|