mirror of
https://github.com/nim-lang/Nim.git
synced 2025-12-29 01:14:41 +00:00
1118 lines
37 KiB
Nim
1118 lines
37 KiB
Nim
#
|
|
#
|
|
# The Nim Compiler
|
|
# (c) Copyright 2015 Andreas Rumpf
|
|
#
|
|
# See the file "copying.txt", included in this
|
|
# distribution, for details about the copyright.
|
|
#
|
|
|
|
# included from cgen.nim
|
|
|
|
const
|
|
RangeExpandLimit = 256 # do not generate ranges
|
|
# over 'RangeExpandLimit' elements
|
|
stringCaseThreshold = 8
|
|
# above X strings a hash-switch for strings is generated
|
|
|
|
proc registerGcRoot(p: BProc, v: PSym) =
|
|
if gSelectedGC in {gcMarkAndSweep, gcGenerational} and
|
|
containsGarbageCollectedRef(v.loc.t):
|
|
# we register a specialized marked proc here; this has the advantage
|
|
# that it works out of the box for thread local storage then :-)
|
|
let prc = genTraverseProcForGlobal(p.module, v)
|
|
linefmt(p.module.initProc, cpsStmts,
|
|
"#nimRegisterGlobalMarker($1);$n", prc)
|
|
|
|
proc isAssignedImmediately(n: PNode): bool {.inline.} =
|
|
if n.kind == nkEmpty: return false
|
|
if isInvalidReturnType(n.typ):
|
|
# var v = f()
|
|
# is transformed into: var v; f(addr v)
|
|
# where 'f' **does not** initialize the result!
|
|
return false
|
|
result = true
|
|
|
|
proc genVarTuple(p: BProc, n: PNode) =
|
|
var tup, field: TLoc
|
|
if n.kind != nkVarTuple: internalError(n.info, "genVarTuple")
|
|
var L = sonsLen(n)
|
|
|
|
# if we have a something that's been captured, use the lowering instead:
|
|
var useLowering = false
|
|
for i in countup(0, L-3):
|
|
if n[i].kind != nkSym:
|
|
useLowering = true; break
|
|
if useLowering:
|
|
genStmts(p, lowerTupleUnpacking(n, p.prc))
|
|
return
|
|
genLineDir(p, n)
|
|
initLocExpr(p, n.sons[L-1], tup)
|
|
var t = tup.t.getUniqueType
|
|
for i in countup(0, L-3):
|
|
var v = n.sons[i].sym
|
|
if sfCompileTime in v.flags: continue
|
|
if sfGlobal in v.flags:
|
|
assignGlobalVar(p, v)
|
|
genObjectInit(p, cpsInit, v.typ, v.loc, true)
|
|
registerGcRoot(p, v)
|
|
else:
|
|
assignLocalVar(p, v)
|
|
initLocalVar(p, v, immediateAsgn=isAssignedImmediately(n[L-1]))
|
|
initLoc(field, locExpr, t.sons[i], tup.s)
|
|
if t.kind == tyTuple:
|
|
field.r = "$1.Field$2" % [rdLoc(tup), rope(i)]
|
|
else:
|
|
if t.n.sons[i].kind != nkSym: internalError(n.info, "genVarTuple")
|
|
field.r = "$1.$2" % [rdLoc(tup), mangleRecFieldName(t.n.sons[i].sym, t)]
|
|
putLocIntoDest(p, v.loc, field)
|
|
|
|
proc genDeref(p: BProc, e: PNode, d: var TLoc; enforceDeref=false)
|
|
|
|
proc loadInto(p: BProc, le, ri: PNode, a: var TLoc) {.inline.} =
|
|
if ri.kind in nkCallKinds and (ri.sons[0].kind != nkSym or
|
|
ri.sons[0].sym.magic == mNone):
|
|
genAsgnCall(p, le, ri, a)
|
|
elif ri.kind in {nkDerefExpr, nkHiddenDeref}:
|
|
# this is a hacky way to fix #1181 (tmissingderef)::
|
|
#
|
|
# var arr1 = cast[ptr array[4, int8]](addr foo)[]
|
|
#
|
|
# However, fixing this properly really requires modelling 'array' as
|
|
# a 'struct' in C to preserve dereferencing semantics completely. Not
|
|
# worth the effort until version 1.0 is out.
|
|
genDeref(p, ri, a, enforceDeref=true)
|
|
else:
|
|
expr(p, ri, a)
|
|
|
|
proc startBlock(p: BProc, start: FormatStr = "{$n",
|
|
args: varargs[Rope]): int {.discardable.} =
|
|
lineCg(p, cpsStmts, start, args)
|
|
inc(p.labels)
|
|
result = len(p.blocks)
|
|
setLen(p.blocks, result + 1)
|
|
p.blocks[result].id = p.labels
|
|
p.blocks[result].nestedTryStmts = p.nestedTryStmts.len.int16
|
|
p.blocks[result].nestedExceptStmts = p.inExceptBlock.int16
|
|
|
|
proc assignLabel(b: var TBlock): Rope {.inline.} =
|
|
b.label = "LA" & b.id.rope
|
|
result = b.label
|
|
|
|
proc blockBody(b: var TBlock): Rope =
|
|
result = b.sections[cpsLocals]
|
|
if b.frameLen > 0:
|
|
result.addf("FR.len+=$1;$n", [b.frameLen.rope])
|
|
result.add(b.sections[cpsInit])
|
|
result.add(b.sections[cpsStmts])
|
|
|
|
proc endBlock(p: BProc, blockEnd: Rope) =
|
|
let topBlock = p.blocks.len-1
|
|
# the block is merged into the parent block
|
|
add(p.blocks[topBlock-1].sections[cpsStmts], p.blocks[topBlock].blockBody)
|
|
setLen(p.blocks, topBlock)
|
|
# this is done after the block is popped so $n is
|
|
# properly indented when pretty printing is enabled
|
|
line(p, cpsStmts, blockEnd)
|
|
|
|
proc endBlock(p: BProc) =
|
|
let topBlock = p.blocks.len - 1
|
|
var blockEnd = if p.blocks[topBlock].label != nil:
|
|
rfmt(nil, "} $1: ;$n", p.blocks[topBlock].label)
|
|
else:
|
|
~"}$n"
|
|
let frameLen = p.blocks[topBlock].frameLen
|
|
if frameLen > 0:
|
|
blockEnd.addf("FR.len-=$1;$n", [frameLen.rope])
|
|
endBlock(p, blockEnd)
|
|
|
|
proc genSimpleBlock(p: BProc, stmts: PNode) {.inline.} =
|
|
startBlock(p)
|
|
genStmts(p, stmts)
|
|
endBlock(p)
|
|
|
|
proc exprBlock(p: BProc, n: PNode, d: var TLoc) =
|
|
startBlock(p)
|
|
expr(p, n, d)
|
|
endBlock(p)
|
|
|
|
template preserveBreakIdx(body: stmt): stmt {.immediate.} =
|
|
var oldBreakIdx = p.breakIdx
|
|
body
|
|
p.breakIdx = oldBreakIdx
|
|
|
|
proc genState(p: BProc, n: PNode) =
|
|
internalAssert n.len == 1 and n.sons[0].kind == nkIntLit
|
|
let idx = n.sons[0].intVal
|
|
linefmt(p, cpsStmts, "STATE$1: ;$n", idx.rope)
|
|
|
|
proc genGotoState(p: BProc, n: PNode) =
|
|
# we resist the temptation to translate it into duff's device as it later
|
|
# will be translated into computed gotos anyway for GCC at least:
|
|
# switch (x.state) {
|
|
# case 0: goto STATE0;
|
|
# ...
|
|
var a: TLoc
|
|
initLocExpr(p, n.sons[0], a)
|
|
lineF(p, cpsStmts, "switch ($1) {$n", [rdLoc(a)])
|
|
p.beforeRetNeeded = true
|
|
lineF(p, cpsStmts, "case -1: goto BeforeRet;$n", [])
|
|
for i in 0 .. lastOrd(n.sons[0].typ):
|
|
lineF(p, cpsStmts, "case $1: goto STATE$1;$n", [rope(i)])
|
|
lineF(p, cpsStmts, "}$n", [])
|
|
|
|
proc genBreakState(p: BProc, n: PNode) =
|
|
var a: TLoc
|
|
if n.sons[0].kind == nkClosure:
|
|
# XXX this produces quite inefficient code!
|
|
initLocExpr(p, n.sons[0].sons[1], a)
|
|
lineF(p, cpsStmts, "if (((NI*) $1)[0] < 0) break;$n", [rdLoc(a)])
|
|
else:
|
|
initLocExpr(p, n.sons[0], a)
|
|
# the environment is guaranteed to contain the 'state' field at offset 0:
|
|
lineF(p, cpsStmts, "if ((((NI*) $1.ClEnv)[0]) < 0) break;$n", [rdLoc(a)])
|
|
# lineF(p, cpsStmts, "if (($1) < 0) break;$n", [rdLoc(a)])
|
|
|
|
proc genVarPrototypeAux(m: BModule, sym: PSym)
|
|
|
|
proc genGotoVar(p: BProc; value: PNode) =
|
|
if value.kind notin {nkCharLit..nkUInt64Lit}:
|
|
localError(value.info, "'goto' target must be a literal value")
|
|
else:
|
|
lineF(p, cpsStmts, "goto NIMSTATE_$#;$n", [value.intVal.rope])
|
|
|
|
proc genSingleVar(p: BProc, a: PNode) =
|
|
var v = a.sons[0].sym
|
|
if {sfCompileTime, sfGoto} * v.flags != {}:
|
|
# translate 'var state {.goto.} = X' into 'goto LX':
|
|
if sfGoto in v.flags: genGotoVar(p, a.sons[2])
|
|
return
|
|
var targetProc = p
|
|
if sfGlobal in v.flags:
|
|
if v.flags * {sfImportc, sfExportc} == {sfImportc} and
|
|
a.sons[2].kind == nkEmpty and
|
|
v.loc.flags * {lfHeader, lfNoDecl} != {}:
|
|
return
|
|
if sfPure in v.flags:
|
|
# v.owner.kind != skModule:
|
|
targetProc = p.module.preInitProc
|
|
assignGlobalVar(targetProc, v)
|
|
# XXX: be careful here.
|
|
# Global variables should not be zeromem-ed within loops
|
|
# (see bug #20).
|
|
# That's why we are doing the construction inside the preInitProc.
|
|
# genObjectInit relies on the C runtime's guarantees that
|
|
# global variables will be initialized to zero.
|
|
genObjectInit(p.module.preInitProc, cpsInit, v.typ, v.loc, true)
|
|
# Alternative construction using default constructor (which may zeromem):
|
|
# if sfImportc notin v.flags: constructLoc(p.module.preInitProc, v.loc)
|
|
if sfExportc in v.flags and generatedHeader != nil:
|
|
genVarPrototypeAux(generatedHeader, v)
|
|
registerGcRoot(p, v)
|
|
else:
|
|
let value = a.sons[2]
|
|
let imm = isAssignedImmediately(value)
|
|
if imm and p.module.compileToCpp and p.splitDecls == 0 and
|
|
not containsHiddenPointer(v.typ):
|
|
# C++ really doesn't like things like 'Foo f; f = x' as that invokes a
|
|
# parameterless constructor followed by an assignment operator. So we
|
|
# generate better code here:
|
|
genLineDir(p, a)
|
|
let decl = localVarDecl(p, v)
|
|
var tmp: TLoc
|
|
if value.kind in nkCallKinds and value[0].kind == nkSym and
|
|
sfConstructor in value[0].sym.flags:
|
|
var params: Rope
|
|
let typ = skipTypes(value.sons[0].typ, abstractInst)
|
|
assert(typ.kind == tyProc)
|
|
for i in 1.. <value.len:
|
|
if params != nil: params.add(~", ")
|
|
assert(sonsLen(typ) == sonsLen(typ.n))
|
|
add(params, genOtherArg(p, value, i, typ))
|
|
lineF(p, cpsStmts, "$#($#);$n", [decl, params])
|
|
else:
|
|
initLocExprSingleUse(p, value, tmp)
|
|
lineF(p, cpsStmts, "$# = $#;$n", [decl, tmp.rdLoc])
|
|
return
|
|
assignLocalVar(p, v)
|
|
initLocalVar(p, v, imm)
|
|
|
|
if a.sons[2].kind != nkEmpty:
|
|
genLineDir(targetProc, a)
|
|
loadInto(targetProc, a.sons[0], a.sons[2], v.loc)
|
|
|
|
proc genClosureVar(p: BProc, a: PNode) =
|
|
var immediateAsgn = a.sons[2].kind != nkEmpty
|
|
if immediateAsgn:
|
|
var v: TLoc
|
|
initLocExpr(p, a.sons[0], v)
|
|
genLineDir(p, a)
|
|
loadInto(p, a.sons[0], a.sons[2], v)
|
|
|
|
proc genVarStmt(p: BProc, n: PNode) =
|
|
for i in countup(0, sonsLen(n) - 1):
|
|
var a = n.sons[i]
|
|
if a.kind == nkCommentStmt: continue
|
|
if a.kind == nkIdentDefs:
|
|
# can be a lifted var nowadays ...
|
|
if a.sons[0].kind == nkSym:
|
|
genSingleVar(p, a)
|
|
else:
|
|
genClosureVar(p, a)
|
|
else:
|
|
genVarTuple(p, a)
|
|
|
|
proc genConstStmt(p: BProc, t: PNode) =
|
|
for i in countup(0, sonsLen(t) - 1):
|
|
var it = t.sons[i]
|
|
if it.kind == nkCommentStmt: continue
|
|
if it.kind != nkConstDef: internalError(t.info, "genConstStmt")
|
|
var c = it.sons[0].sym
|
|
if c.typ.containsCompileTimeOnly: continue
|
|
if sfFakeConst in c.flags:
|
|
genSingleVar(p, it)
|
|
elif c.typ.kind in ConstantDataTypes and lfNoDecl notin c.loc.flags and
|
|
c.ast.len != 0:
|
|
if not emitLazily(c): requestConstImpl(p, c)
|
|
|
|
proc genIf(p: BProc, n: PNode, d: var TLoc) =
|
|
#
|
|
# { if (!expr1) goto L1;
|
|
# thenPart }
|
|
# goto LEnd
|
|
# L1:
|
|
# { if (!expr2) goto L2;
|
|
# thenPart2 }
|
|
# goto LEnd
|
|
# L2:
|
|
# { elsePart }
|
|
# Lend:
|
|
var
|
|
a: TLoc
|
|
lelse: TLabel
|
|
if not isEmptyType(n.typ) and d.k == locNone:
|
|
getTemp(p, n.typ, d)
|
|
genLineDir(p, n)
|
|
let lend = getLabel(p)
|
|
for i in countup(0, sonsLen(n) - 1):
|
|
let it = n.sons[i]
|
|
if it.len == 2:
|
|
when newScopeForIf: startBlock(p)
|
|
initLocExprSingleUse(p, it.sons[0], a)
|
|
lelse = getLabel(p)
|
|
inc(p.labels)
|
|
lineF(p, cpsStmts, "if (!$1) goto $2;$n",
|
|
[rdLoc(a), lelse])
|
|
when not newScopeForIf: startBlock(p)
|
|
if p.module.compileToCpp:
|
|
# avoid "jump to label crosses initialization" error:
|
|
add(p.s(cpsStmts), "{")
|
|
expr(p, it.sons[1], d)
|
|
add(p.s(cpsStmts), "}")
|
|
else:
|
|
expr(p, it.sons[1], d)
|
|
endBlock(p)
|
|
if sonsLen(n) > 1:
|
|
lineF(p, cpsStmts, "goto $1;$n", [lend])
|
|
fixLabel(p, lelse)
|
|
elif it.len == 1:
|
|
startBlock(p)
|
|
expr(p, it.sons[0], d)
|
|
endBlock(p)
|
|
else: internalError(n.info, "genIf()")
|
|
if sonsLen(n) > 1: fixLabel(p, lend)
|
|
|
|
|
|
proc blockLeaveActions(p: BProc, howManyTrys, howManyExcepts: int) =
|
|
# Called by return and break stmts.
|
|
# Deals with issues faced when jumping out of try/except/finally stmts,
|
|
|
|
var stack: seq[PNode]
|
|
newSeq(stack, 0)
|
|
|
|
var alreadyPoppedCnt = p.inExceptBlock
|
|
for i in countup(1, howManyTrys):
|
|
if not p.module.compileToCpp:
|
|
# Pop safe points generated by try
|
|
if alreadyPoppedCnt > 0:
|
|
dec alreadyPoppedCnt
|
|
else:
|
|
linefmt(p, cpsStmts, "#popSafePoint();$n")
|
|
|
|
# Pop this try-stmt of the list of nested trys
|
|
# so we don't infinite recurse on it in the next step.
|
|
var tryStmt = p.nestedTryStmts.pop
|
|
stack.add(tryStmt)
|
|
|
|
# Find finally-stmt for this try-stmt
|
|
# and generate a copy of its sons
|
|
var finallyStmt = lastSon(tryStmt)
|
|
if finallyStmt.kind == nkFinally:
|
|
genStmts(p, finallyStmt.sons[0])
|
|
|
|
# push old elements again:
|
|
for i in countdown(howManyTrys-1, 0):
|
|
p.nestedTryStmts.add(stack[i])
|
|
|
|
if not p.module.compileToCpp:
|
|
# Pop exceptions that was handled by the
|
|
# except-blocks we are in
|
|
for i in countdown(howManyExcepts-1, 0):
|
|
linefmt(p, cpsStmts, "#popCurrentException();$n")
|
|
|
|
proc genReturnStmt(p: BProc, t: PNode) =
|
|
p.beforeRetNeeded = true
|
|
genLineDir(p, t)
|
|
if (t.sons[0].kind != nkEmpty): genStmts(p, t.sons[0])
|
|
blockLeaveActions(p,
|
|
howManyTrys = p.nestedTryStmts.len,
|
|
howManyExcepts = p.inExceptBlock)
|
|
if (p.finallySafePoints.len > 0):
|
|
# If we're in a finally block, and we came here by exception
|
|
# consume it before we return.
|
|
var safePoint = p.finallySafePoints[p.finallySafePoints.len-1]
|
|
linefmt(p, cpsStmts, "if ($1.status != 0) #popCurrentException();$n", safePoint)
|
|
lineF(p, cpsStmts, "goto BeforeRet;$n", [])
|
|
|
|
proc genGotoForCase(p: BProc; caseStmt: PNode) =
|
|
for i in 1 .. <caseStmt.len:
|
|
startBlock(p)
|
|
let it = caseStmt.sons[i]
|
|
for j in 0 .. it.len-2:
|
|
if it.sons[j].kind == nkRange:
|
|
localError(it.info, "range notation not available for computed goto")
|
|
return
|
|
let val = getOrdValue(it.sons[j])
|
|
lineF(p, cpsStmts, "NIMSTATE_$#:$n", [val.rope])
|
|
genStmts(p, it.lastSon)
|
|
endBlock(p)
|
|
|
|
proc genComputedGoto(p: BProc; n: PNode) =
|
|
# first pass: Generate array of computed labels:
|
|
var casePos = -1
|
|
var arraySize: int
|
|
for i in 0 .. <n.len:
|
|
let it = n.sons[i]
|
|
if it.kind == nkCaseStmt:
|
|
if lastSon(it).kind != nkOfBranch:
|
|
localError(it.info,
|
|
"case statement must be exhaustive for computed goto"); return
|
|
casePos = i
|
|
let aSize = lengthOrd(it.sons[0].typ)
|
|
if aSize > 10_000:
|
|
localError(it.info,
|
|
"case statement has too many cases for computed goto"); return
|
|
arraySize = aSize.int
|
|
if firstOrd(it.sons[0].typ) != 0:
|
|
localError(it.info,
|
|
"case statement has to start at 0 for computed goto"); return
|
|
if casePos < 0:
|
|
localError(n.info, "no case statement found for computed goto"); return
|
|
var id = p.labels+1
|
|
inc p.labels, arraySize+1
|
|
let tmp = "TMP$1" % [id.rope]
|
|
var gotoArray = "static void* $#[$#] = {" % [tmp, arraySize.rope]
|
|
for i in 1..arraySize-1:
|
|
gotoArray.addf("&&TMP$#, ", [(id+i).rope])
|
|
gotoArray.addf("&&TMP$#};$n", [(id+arraySize).rope])
|
|
line(p, cpsLocals, gotoArray)
|
|
|
|
let topBlock = p.blocks.len-1
|
|
let oldBody = p.blocks[topBlock].sections[cpsStmts]
|
|
p.blocks[topBlock].sections[cpsStmts] = nil
|
|
|
|
for j in casePos+1 .. <n.len: genStmts(p, n.sons[j])
|
|
let tailB = p.blocks[topBlock].sections[cpsStmts]
|
|
|
|
p.blocks[topBlock].sections[cpsStmts] = nil
|
|
for j in 0 .. casePos-1: genStmts(p, n.sons[j])
|
|
let tailA = p.blocks[topBlock].sections[cpsStmts]
|
|
|
|
p.blocks[topBlock].sections[cpsStmts] = oldBody & tailA
|
|
|
|
let caseStmt = n.sons[casePos]
|
|
var a: TLoc
|
|
initLocExpr(p, caseStmt.sons[0], a)
|
|
# first goto:
|
|
lineF(p, cpsStmts, "goto *$#[$#];$n", [tmp, a.rdLoc])
|
|
|
|
for i in 1 .. <caseStmt.len:
|
|
startBlock(p)
|
|
let it = caseStmt.sons[i]
|
|
for j in 0 .. it.len-2:
|
|
if it.sons[j].kind == nkRange:
|
|
localError(it.info, "range notation not available for computed goto")
|
|
return
|
|
let val = getOrdValue(it.sons[j])
|
|
lineF(p, cpsStmts, "TMP$#:$n", [intLiteral(val+id+1)])
|
|
genStmts(p, it.lastSon)
|
|
#for j in casePos+1 .. <n.len: genStmts(p, n.sons[j]) # tailB
|
|
#for j in 0 .. casePos-1: genStmts(p, n.sons[j]) # tailA
|
|
add(p.s(cpsStmts), tailB)
|
|
add(p.s(cpsStmts), tailA)
|
|
|
|
var a: TLoc
|
|
initLocExpr(p, caseStmt.sons[0], a)
|
|
lineF(p, cpsStmts, "goto *$#[$#];$n", [tmp, a.rdLoc])
|
|
endBlock(p)
|
|
|
|
proc genWhileStmt(p: BProc, t: PNode) =
|
|
# we don't generate labels here as for example GCC would produce
|
|
# significantly worse code
|
|
var
|
|
a: TLoc
|
|
labl: TLabel
|
|
assert(sonsLen(t) == 2)
|
|
inc(p.withinLoop)
|
|
genLineDir(p, t)
|
|
|
|
preserveBreakIdx:
|
|
p.breakIdx = startBlock(p, "while (1) {$n")
|
|
p.blocks[p.breakIdx].isLoop = true
|
|
initLocExpr(p, t.sons[0], a)
|
|
if (t.sons[0].kind != nkIntLit) or (t.sons[0].intVal == 0):
|
|
let label = assignLabel(p.blocks[p.breakIdx])
|
|
lineF(p, cpsStmts, "if (!$1) goto $2;$n", [rdLoc(a), label])
|
|
var loopBody = t.sons[1]
|
|
if loopBody.stmtsContainPragma(wComputedGoto) and
|
|
hasComputedGoto in CC[cCompiler].props:
|
|
# for closure support weird loop bodies are generated:
|
|
if loopBody.len == 2 and loopBody.sons[0].kind == nkEmpty:
|
|
loopBody = loopBody.sons[1]
|
|
genComputedGoto(p, loopBody)
|
|
else:
|
|
genStmts(p, loopBody)
|
|
|
|
if optProfiler in p.options:
|
|
# invoke at loop body exit:
|
|
linefmt(p, cpsStmts, "#nimProfile();$n")
|
|
endBlock(p)
|
|
|
|
dec(p.withinLoop)
|
|
|
|
proc genBlock(p: BProc, t: PNode, d: var TLoc) =
|
|
preserveBreakIdx:
|
|
p.breakIdx = startBlock(p)
|
|
if t.sons[0].kind != nkEmpty:
|
|
# named block?
|
|
assert(t.sons[0].kind == nkSym)
|
|
var sym = t.sons[0].sym
|
|
sym.loc.k = locOther
|
|
sym.position = p.breakIdx+1
|
|
expr(p, t.sons[1], d)
|
|
endBlock(p)
|
|
|
|
proc genParForStmt(p: BProc, t: PNode) =
|
|
assert(sonsLen(t) == 3)
|
|
inc(p.withinLoop)
|
|
genLineDir(p, t)
|
|
|
|
preserveBreakIdx:
|
|
let forLoopVar = t.sons[0].sym
|
|
var rangeA, rangeB: TLoc
|
|
assignLocalVar(p, forLoopVar)
|
|
#initLoc(forLoopVar.loc, locLocalVar, forLoopVar.typ, onStack)
|
|
#discard mangleName(forLoopVar)
|
|
let call = t.sons[1]
|
|
initLocExpr(p, call.sons[1], rangeA)
|
|
initLocExpr(p, call.sons[2], rangeB)
|
|
|
|
lineF(p, cpsStmts, "#pragma omp parallel for $4$n" &
|
|
"for ($1 = $2; $1 <= $3; ++$1)",
|
|
[forLoopVar.loc.rdLoc,
|
|
rangeA.rdLoc, rangeB.rdLoc,
|
|
call.sons[3].getStr.rope])
|
|
|
|
p.breakIdx = startBlock(p)
|
|
p.blocks[p.breakIdx].isLoop = true
|
|
genStmts(p, t.sons[2])
|
|
endBlock(p)
|
|
|
|
dec(p.withinLoop)
|
|
|
|
proc genBreakStmt(p: BProc, t: PNode) =
|
|
var idx = p.breakIdx
|
|
if t.sons[0].kind != nkEmpty:
|
|
# named break?
|
|
assert(t.sons[0].kind == nkSym)
|
|
var sym = t.sons[0].sym
|
|
assert(sym.loc.k == locOther)
|
|
idx = sym.position-1
|
|
else:
|
|
# an unnamed 'break' can only break a loop after 'transf' pass:
|
|
while idx >= 0 and not p.blocks[idx].isLoop: dec idx
|
|
if idx < 0 or not p.blocks[idx].isLoop:
|
|
internalError(t.info, "no loop to break")
|
|
let label = assignLabel(p.blocks[idx])
|
|
blockLeaveActions(p,
|
|
p.nestedTryStmts.len - p.blocks[idx].nestedTryStmts,
|
|
p.inExceptBlock - p.blocks[idx].nestedExceptStmts)
|
|
genLineDir(p, t)
|
|
lineF(p, cpsStmts, "goto $1;$n", [label])
|
|
|
|
proc getRaiseFrmt(p: BProc): string =
|
|
result = "#raiseException((#Exception*)$1, $2);$n"
|
|
|
|
proc genRaiseStmt(p: BProc, t: PNode) =
|
|
if p.inExceptBlock > 0:
|
|
# if the current try stmt have a finally block,
|
|
# we must execute it before reraising
|
|
var finallyBlock = p.nestedTryStmts[p.nestedTryStmts.len - 1].lastSon
|
|
if finallyBlock.kind == nkFinally:
|
|
genSimpleBlock(p, finallyBlock.sons[0])
|
|
if t.sons[0].kind != nkEmpty:
|
|
var a: TLoc
|
|
initLocExpr(p, t.sons[0], a)
|
|
var e = rdLoc(a)
|
|
var typ = skipTypes(t.sons[0].typ, abstractPtrs)
|
|
genLineDir(p, t)
|
|
lineCg(p, cpsStmts, getRaiseFrmt(p), [e, makeCString(typ.sym.name.s)])
|
|
else:
|
|
genLineDir(p, t)
|
|
# reraise the last exception:
|
|
if p.module.compileToCpp:
|
|
line(p, cpsStmts, ~"throw;$n")
|
|
else:
|
|
linefmt(p, cpsStmts, "#reraiseException();$n")
|
|
|
|
proc genCaseGenericBranch(p: BProc, b: PNode, e: TLoc,
|
|
rangeFormat, eqFormat: FormatStr, labl: TLabel) =
|
|
var
|
|
x, y: TLoc
|
|
var length = sonsLen(b)
|
|
for i in countup(0, length - 2):
|
|
if b.sons[i].kind == nkRange:
|
|
initLocExpr(p, b.sons[i].sons[0], x)
|
|
initLocExpr(p, b.sons[i].sons[1], y)
|
|
lineCg(p, cpsStmts, rangeFormat,
|
|
[rdCharLoc(e), rdCharLoc(x), rdCharLoc(y), labl])
|
|
else:
|
|
initLocExpr(p, b.sons[i], x)
|
|
lineCg(p, cpsStmts, eqFormat, [rdCharLoc(e), rdCharLoc(x), labl])
|
|
|
|
proc genCaseSecondPass(p: BProc, t: PNode, d: var TLoc,
|
|
labId, until: int): TLabel =
|
|
var lend = getLabel(p)
|
|
for i in 1..until:
|
|
lineF(p, cpsStmts, "LA$1: ;$n", [rope(labId + i)])
|
|
if t.sons[i].kind == nkOfBranch:
|
|
var length = sonsLen(t.sons[i])
|
|
exprBlock(p, t.sons[i].sons[length - 1], d)
|
|
lineF(p, cpsStmts, "goto $1;$n", [lend])
|
|
else:
|
|
exprBlock(p, t.sons[i].sons[0], d)
|
|
result = lend
|
|
|
|
proc genIfForCaseUntil(p: BProc, t: PNode, d: var TLoc,
|
|
rangeFormat, eqFormat: FormatStr,
|
|
until: int, a: TLoc): TLabel =
|
|
# generate a C-if statement for a Nim case statement
|
|
var labId = p.labels
|
|
for i in 1..until:
|
|
inc(p.labels)
|
|
if t.sons[i].kind == nkOfBranch: # else statement
|
|
genCaseGenericBranch(p, t.sons[i], a, rangeFormat, eqFormat,
|
|
"LA" & rope(p.labels))
|
|
else:
|
|
lineF(p, cpsStmts, "goto LA$1;$n", [rope(p.labels)])
|
|
if until < t.len-1:
|
|
inc(p.labels)
|
|
var gotoTarget = p.labels
|
|
lineF(p, cpsStmts, "goto LA$1;$n", [rope(gotoTarget)])
|
|
result = genCaseSecondPass(p, t, d, labId, until)
|
|
lineF(p, cpsStmts, "LA$1: ;$n", [rope(gotoTarget)])
|
|
else:
|
|
result = genCaseSecondPass(p, t, d, labId, until)
|
|
|
|
proc genCaseGeneric(p: BProc, t: PNode, d: var TLoc,
|
|
rangeFormat, eqFormat: FormatStr) =
|
|
var a: TLoc
|
|
initLocExpr(p, t.sons[0], a)
|
|
var lend = genIfForCaseUntil(p, t, d, rangeFormat, eqFormat, sonsLen(t)-1, a)
|
|
fixLabel(p, lend)
|
|
|
|
proc genCaseStringBranch(p: BProc, b: PNode, e: TLoc, labl: TLabel,
|
|
branches: var openArray[Rope]) =
|
|
var x: TLoc
|
|
var length = sonsLen(b)
|
|
for i in countup(0, length - 2):
|
|
assert(b.sons[i].kind != nkRange)
|
|
initLocExpr(p, b.sons[i], x)
|
|
assert(b.sons[i].kind in {nkStrLit..nkTripleStrLit})
|
|
var j = int(hashString(b.sons[i].strVal) and high(branches))
|
|
appcg(p.module, branches[j], "if (#eqStrings($1, $2)) goto $3;$n",
|
|
[rdLoc(e), rdLoc(x), labl])
|
|
|
|
proc genStringCase(p: BProc, t: PNode, d: var TLoc) =
|
|
# count how many constant strings there are in the case:
|
|
var strings = 0
|
|
for i in countup(1, sonsLen(t) - 1):
|
|
if t.sons[i].kind == nkOfBranch: inc(strings, sonsLen(t.sons[i]) - 1)
|
|
if strings > stringCaseThreshold:
|
|
var bitMask = math.nextPowerOfTwo(strings) - 1
|
|
var branches: seq[Rope]
|
|
newSeq(branches, bitMask + 1)
|
|
var a: TLoc
|
|
initLocExpr(p, t.sons[0], a) # fist pass: gnerate ifs+goto:
|
|
var labId = p.labels
|
|
for i in countup(1, sonsLen(t) - 1):
|
|
inc(p.labels)
|
|
if t.sons[i].kind == nkOfBranch:
|
|
genCaseStringBranch(p, t.sons[i], a, "LA" & rope(p.labels),
|
|
branches)
|
|
else:
|
|
# else statement: nothing to do yet
|
|
# but we reserved a label, which we use later
|
|
discard
|
|
linefmt(p, cpsStmts, "switch (#hashString($1) & $2) {$n",
|
|
rdLoc(a), rope(bitMask))
|
|
for j in countup(0, high(branches)):
|
|
if branches[j] != nil:
|
|
lineF(p, cpsStmts, "case $1: $n$2break;$n",
|
|
[intLiteral(j), branches[j]])
|
|
lineF(p, cpsStmts, "}$n", []) # else statement:
|
|
if t.sons[sonsLen(t)-1].kind != nkOfBranch:
|
|
lineF(p, cpsStmts, "goto LA$1;$n", [rope(p.labels)])
|
|
# third pass: generate statements
|
|
var lend = genCaseSecondPass(p, t, d, labId, sonsLen(t)-1)
|
|
fixLabel(p, lend)
|
|
else:
|
|
genCaseGeneric(p, t, d, "", "if (#eqStrings($1, $2)) goto $3;$n")
|
|
|
|
proc branchHasTooBigRange(b: PNode): bool =
|
|
for i in countup(0, sonsLen(b)-2):
|
|
# last son is block
|
|
if (b.sons[i].kind == nkRange) and
|
|
b.sons[i].sons[1].intVal - b.sons[i].sons[0].intVal > RangeExpandLimit:
|
|
return true
|
|
|
|
proc ifSwitchSplitPoint(p: BProc, n: PNode): int =
|
|
for i in 1..n.len-1:
|
|
var branch = n[i]
|
|
var stmtBlock = lastSon(branch)
|
|
if stmtBlock.stmtsContainPragma(wLinearScanEnd):
|
|
result = i
|
|
elif hasSwitchRange notin CC[cCompiler].props:
|
|
if branch.kind == nkOfBranch and branchHasTooBigRange(branch):
|
|
result = i
|
|
|
|
proc genCaseRange(p: BProc, branch: PNode) =
|
|
var length = branch.len
|
|
for j in 0 .. length-2:
|
|
if branch[j].kind == nkRange:
|
|
if hasSwitchRange in CC[cCompiler].props:
|
|
lineF(p, cpsStmts, "case $1 ... $2:$n", [
|
|
genLiteral(p, branch[j][0]),
|
|
genLiteral(p, branch[j][1])])
|
|
else:
|
|
var v = copyNode(branch[j][0])
|
|
while v.intVal <= branch[j][1].intVal:
|
|
lineF(p, cpsStmts, "case $1:$n", [genLiteral(p, v)])
|
|
inc(v.intVal)
|
|
else:
|
|
lineF(p, cpsStmts, "case $1:$n", [genLiteral(p, branch[j])])
|
|
|
|
proc genOrdinalCase(p: BProc, n: PNode, d: var TLoc) =
|
|
# analyse 'case' statement:
|
|
var splitPoint = ifSwitchSplitPoint(p, n)
|
|
|
|
# generate if part (might be empty):
|
|
var a: TLoc
|
|
initLocExpr(p, n.sons[0], a)
|
|
var lend = if splitPoint > 0: genIfForCaseUntil(p, n, d,
|
|
rangeFormat = "if ($1 >= $2 && $1 <= $3) goto $4;$n",
|
|
eqFormat = "if ($1 == $2) goto $3;$n",
|
|
splitPoint, a) else: nil
|
|
|
|
# generate switch part (might be empty):
|
|
if splitPoint+1 < n.len:
|
|
lineF(p, cpsStmts, "switch ($1) {$n", [rdCharLoc(a)])
|
|
var hasDefault = false
|
|
for i in splitPoint+1 .. < n.len:
|
|
var branch = n[i]
|
|
if branch.kind == nkOfBranch:
|
|
genCaseRange(p, branch)
|
|
else:
|
|
# else part of case statement:
|
|
lineF(p, cpsStmts, "default:$n", [])
|
|
hasDefault = true
|
|
exprBlock(p, branch.lastSon, d)
|
|
lineF(p, cpsStmts, "break;$n", [])
|
|
if (hasAssume in CC[cCompiler].props) and not hasDefault:
|
|
lineF(p, cpsStmts, "default: __assume(0);$n", [])
|
|
lineF(p, cpsStmts, "}$n", [])
|
|
if lend != nil: fixLabel(p, lend)
|
|
|
|
proc genCase(p: BProc, t: PNode, d: var TLoc) =
|
|
genLineDir(p, t)
|
|
if not isEmptyType(t.typ) and d.k == locNone:
|
|
getTemp(p, t.typ, d)
|
|
case skipTypes(t.sons[0].typ, abstractVarRange).kind
|
|
of tyString:
|
|
genStringCase(p, t, d)
|
|
of tyFloat..tyFloat128:
|
|
genCaseGeneric(p, t, d, "if ($1 >= $2 && $1 <= $3) goto $4;$n",
|
|
"if ($1 == $2) goto $3;$n")
|
|
else:
|
|
if t.sons[0].kind == nkSym and sfGoto in t.sons[0].sym.flags:
|
|
genGotoForCase(p, t)
|
|
else:
|
|
genOrdinalCase(p, t, d)
|
|
|
|
proc hasGeneralExceptSection(t: PNode): bool =
|
|
var length = sonsLen(t)
|
|
var i = 1
|
|
while (i < length) and (t.sons[i].kind == nkExceptBranch):
|
|
var blen = sonsLen(t.sons[i])
|
|
if blen == 1:
|
|
return true
|
|
inc(i)
|
|
result = false
|
|
|
|
proc genTryCpp(p: BProc, t: PNode, d: var TLoc) =
|
|
# code to generate:
|
|
#
|
|
# XXX: There should be a standard dispatch algorithm
|
|
# that's used both here and with multi-methods
|
|
#
|
|
# try
|
|
# {
|
|
# myDiv(4, 9);
|
|
# } catch (NimException& exp) {
|
|
# if (isObj(exp, EIO) {
|
|
# ...
|
|
# } else if (isObj(exp, ESystem) {
|
|
# ...
|
|
# finallyPart()
|
|
# raise;
|
|
# } else {
|
|
# // general handler
|
|
# }
|
|
# }
|
|
# finallyPart();
|
|
if not isEmptyType(t.typ) and d.k == locNone:
|
|
getTemp(p, t.typ, d)
|
|
genLineDir(p, t)
|
|
let exc = getTempName()
|
|
if getCompilerProc("Exception") != nil:
|
|
discard cgsym(p.module, "Exception")
|
|
else:
|
|
discard cgsym(p.module, "E_Base")
|
|
add(p.nestedTryStmts, t)
|
|
startBlock(p, "try {$n")
|
|
expr(p, t.sons[0], d)
|
|
let length = sonsLen(t)
|
|
endBlock(p, ropecg(p.module, "} catch (NimException& $1) {$n", [exc]))
|
|
if optStackTrace in p.options:
|
|
linefmt(p, cpsStmts, "#setFrame((TFrame*)&FR);$n")
|
|
inc p.inExceptBlock
|
|
var i = 1
|
|
var catchAllPresent = false
|
|
while (i < length) and (t.sons[i].kind == nkExceptBranch):
|
|
let blen = sonsLen(t.sons[i])
|
|
if i > 1: addf(p.s(cpsStmts), "else ", [])
|
|
if blen == 1:
|
|
# general except section:
|
|
catchAllPresent = true
|
|
startBlock(p)
|
|
expr(p, t.sons[i].sons[0], d)
|
|
linefmt(p, cpsStmts, "#popCurrentException();$n")
|
|
endBlock(p)
|
|
else:
|
|
var orExpr: Rope = nil
|
|
for j in countup(0, blen - 2):
|
|
assert(t.sons[i].sons[j].kind == nkType)
|
|
if orExpr != nil: add(orExpr, "||")
|
|
appcg(p.module, orExpr,
|
|
"#isObj($1.exp->m_type, $2)",
|
|
[exc, genTypeInfo(p.module, t.sons[i].sons[j].typ)])
|
|
lineF(p, cpsStmts, "if ($1) ", [orExpr])
|
|
startBlock(p)
|
|
expr(p, t.sons[i].sons[blen-1], d)
|
|
linefmt(p, cpsStmts, "#popCurrentException();$n")
|
|
endBlock(p)
|
|
inc(i)
|
|
|
|
# reraise the exception if there was no catch all
|
|
# and none of the handlers matched
|
|
if not catchAllPresent:
|
|
if i > 1: lineF(p, cpsStmts, "else ", [])
|
|
startBlock(p)
|
|
var finallyBlock = t.lastSon
|
|
if finallyBlock.kind == nkFinally:
|
|
#expr(p, finallyBlock.sons[0], d)
|
|
genStmts(p, finallyBlock.sons[0])
|
|
|
|
line(p, cpsStmts, ~"throw;$n")
|
|
endBlock(p)
|
|
|
|
lineF(p, cpsStmts, "}$n", []) # end of catch block
|
|
dec p.inExceptBlock
|
|
|
|
discard pop(p.nestedTryStmts)
|
|
if (i < length) and (t.sons[i].kind == nkFinally):
|
|
genSimpleBlock(p, t.sons[i].sons[0])
|
|
|
|
proc genTry(p: BProc, t: PNode, d: var TLoc) =
|
|
# code to generate:
|
|
#
|
|
# XXX: There should be a standard dispatch algorithm
|
|
# that's used both here and with multi-methods
|
|
#
|
|
# TSafePoint sp;
|
|
# pushSafePoint(&sp);
|
|
# sp.status = setjmp(sp.context);
|
|
# if (sp.status == 0) {
|
|
# myDiv(4, 9);
|
|
# popSafePoint();
|
|
# } else {
|
|
# popSafePoint();
|
|
# /* except DivisionByZero: */
|
|
# if (sp.status == DivisionByZero) {
|
|
# printf('Division by Zero\n');
|
|
# clearException();
|
|
# } else {
|
|
# clearException();
|
|
# }
|
|
# }
|
|
# {
|
|
# /* finally: */
|
|
# printf('fin!\n');
|
|
# }
|
|
# if (exception not cleared)
|
|
# propagateCurrentException();
|
|
#
|
|
if not isEmptyType(t.typ) and d.k == locNone:
|
|
getTemp(p, t.typ, d)
|
|
discard lists.includeStr(p.module.headerFiles, "<setjmp.h>")
|
|
genLineDir(p, t)
|
|
var safePoint = getTempName()
|
|
if getCompilerProc("Exception") != nil:
|
|
discard cgsym(p.module, "Exception")
|
|
else:
|
|
discard cgsym(p.module, "E_Base")
|
|
linefmt(p, cpsLocals, "#TSafePoint $1;$n", safePoint)
|
|
linefmt(p, cpsStmts, "#pushSafePoint(&$1);$n", safePoint)
|
|
if isDefined("nimStdSetjmp"):
|
|
linefmt(p, cpsStmts, "$1.status = setjmp($1.context);$n", safePoint)
|
|
elif isDefined("nimSigSetjmp"):
|
|
linefmt(p, cpsStmts, "$1.status = sigsetjmp($1.context, 0);$n", safePoint)
|
|
elif isDefined("nimRawSetjmp"):
|
|
linefmt(p, cpsStmts, "$1.status = _setjmp($1.context);$n", safePoint)
|
|
else:
|
|
linefmt(p, cpsStmts, "$1.status = setjmp($1.context);$n", safePoint)
|
|
startBlock(p, "if ($1.status == 0) {$n", [safePoint])
|
|
var length = sonsLen(t)
|
|
add(p.nestedTryStmts, t)
|
|
expr(p, t.sons[0], d)
|
|
linefmt(p, cpsStmts, "#popSafePoint();$n")
|
|
endBlock(p)
|
|
startBlock(p, "else {$n")
|
|
linefmt(p, cpsStmts, "#popSafePoint();$n")
|
|
if optStackTrace in p.options:
|
|
linefmt(p, cpsStmts, "#setFrame((TFrame*)&FR);$n")
|
|
inc p.inExceptBlock
|
|
var i = 1
|
|
while (i < length) and (t.sons[i].kind == nkExceptBranch):
|
|
var blen = sonsLen(t.sons[i])
|
|
if blen == 1:
|
|
# general except section:
|
|
if i > 1: lineF(p, cpsStmts, "else", [])
|
|
startBlock(p)
|
|
linefmt(p, cpsStmts, "$1.status = 0;$n", safePoint)
|
|
expr(p, t.sons[i].sons[0], d)
|
|
linefmt(p, cpsStmts, "#popCurrentException();$n")
|
|
endBlock(p)
|
|
else:
|
|
var orExpr: Rope = nil
|
|
for j in countup(0, blen - 2):
|
|
assert(t.sons[i].sons[j].kind == nkType)
|
|
if orExpr != nil: add(orExpr, "||")
|
|
appcg(p.module, orExpr,
|
|
"#isObj(#getCurrentException()->Sup.m_type, $1)",
|
|
[genTypeInfo(p.module, t.sons[i].sons[j].typ)])
|
|
if i > 1: line(p, cpsStmts, "else ")
|
|
startBlock(p, "if ($1) {$n", [orExpr])
|
|
linefmt(p, cpsStmts, "$1.status = 0;$n", safePoint)
|
|
expr(p, t.sons[i].sons[blen-1], d)
|
|
linefmt(p, cpsStmts, "#popCurrentException();$n")
|
|
endBlock(p)
|
|
inc(i)
|
|
dec p.inExceptBlock
|
|
discard pop(p.nestedTryStmts)
|
|
endBlock(p) # end of else block
|
|
if i < length and t.sons[i].kind == nkFinally:
|
|
p.finallySafePoints.add(safePoint)
|
|
genSimpleBlock(p, t.sons[i].sons[0])
|
|
discard pop(p.finallySafePoints)
|
|
linefmt(p, cpsStmts, "if ($1.status != 0) #reraiseException();$n", safePoint)
|
|
|
|
proc genAsmOrEmitStmt(p: BProc, t: PNode, isAsmStmt=false): Rope =
|
|
var res = ""
|
|
for i in countup(0, sonsLen(t) - 1):
|
|
case t.sons[i].kind
|
|
of nkStrLit..nkTripleStrLit:
|
|
res.add(t.sons[i].strVal)
|
|
of nkSym:
|
|
var sym = t.sons[i].sym
|
|
if sym.kind in {skProc, skIterator, skClosureIterator, skMethod}:
|
|
var a: TLoc
|
|
initLocExpr(p, t.sons[i], a)
|
|
res.add($rdLoc(a))
|
|
else:
|
|
var r = sym.loc.r
|
|
if r == nil:
|
|
# if no name has already been given,
|
|
# it doesn't matter much:
|
|
r = mangleName(sym)
|
|
sym.loc.r = r # but be consequent!
|
|
res.add($r)
|
|
else: internalError(t.sons[i].info, "genAsmOrEmitStmt()")
|
|
|
|
if isAsmStmt and hasGnuAsm in CC[cCompiler].props:
|
|
for x in splitLines(res):
|
|
var j = 0
|
|
while x[j] in {' ', '\t'}: inc(j)
|
|
if x[j] in {'"', ':'}:
|
|
# don't modify the line if already in quotes or
|
|
# some clobber register list:
|
|
add(result, x); add(result, tnl)
|
|
elif x[j] != '\0':
|
|
# ignore empty lines
|
|
add(result, "\"")
|
|
add(result, x)
|
|
add(result, "\\n\"\n")
|
|
else:
|
|
res.add(tnl)
|
|
result = res.rope
|
|
|
|
proc genAsmStmt(p: BProc, t: PNode) =
|
|
assert(t.kind == nkAsmStmt)
|
|
genLineDir(p, t)
|
|
var s = genAsmOrEmitStmt(p, t, isAsmStmt=true)
|
|
# see bug #2362, "top level asm statements" seem to be a mis-feature
|
|
# but even if we don't do this, the example in #2362 cannot possibly
|
|
# work:
|
|
if p.prc == nil:
|
|
# top level asm statement?
|
|
addf(p.module.s[cfsProcHeaders], CC[cCompiler].asmStmtFrmt, [s])
|
|
else:
|
|
lineF(p, cpsStmts, CC[cCompiler].asmStmtFrmt, [s])
|
|
|
|
proc determineSection(n: PNode): TCFileSection =
|
|
result = cfsProcHeaders
|
|
if n.len >= 1 and n.sons[0].kind in {nkStrLit..nkTripleStrLit}:
|
|
let sec = n.sons[0].strVal
|
|
if sec.startsWith("/*TYPESECTION*/"): result = cfsTypes
|
|
elif sec.startsWith("/*VARSECTION*/"): result = cfsVars
|
|
elif sec.startsWith("/*INCLUDESECTION*/"): result = cfsHeaders
|
|
|
|
proc genEmit(p: BProc, t: PNode) =
|
|
var s = genAsmOrEmitStmt(p, t.sons[1])
|
|
if p.prc == nil:
|
|
# top level emit pragma?
|
|
let section = determineSection(t[1])
|
|
genCLineDir(p.module.s[section], t.info)
|
|
add(p.module.s[section], s)
|
|
else:
|
|
genLineDir(p, t)
|
|
line(p, cpsStmts, s)
|
|
|
|
var
|
|
breakPointId: int = 0
|
|
gBreakpoints: Rope # later the breakpoints are inserted into the main proc
|
|
|
|
proc genBreakPoint(p: BProc, t: PNode) =
|
|
var name: string
|
|
if optEndb in p.options:
|
|
if t.kind == nkExprColonExpr:
|
|
assert(t.sons[1].kind in {nkStrLit..nkTripleStrLit})
|
|
name = normalize(t.sons[1].strVal)
|
|
else:
|
|
inc(breakPointId)
|
|
name = "bp" & $breakPointId
|
|
genLineDir(p, t) # BUGFIX
|
|
appcg(p.module, gBreakpoints,
|
|
"#dbgRegisterBreakpoint($1, (NCSTRING)$2, (NCSTRING)$3);$n", [
|
|
rope(toLinenumber(t.info)), makeCString(toFilename(t.info)),
|
|
makeCString(name)])
|
|
|
|
proc genWatchpoint(p: BProc, n: PNode) =
|
|
if optEndb notin p.options: return
|
|
var a: TLoc
|
|
initLocExpr(p, n.sons[1], a)
|
|
let typ = skipTypes(n.sons[1].typ, abstractVarRange)
|
|
lineCg(p, cpsStmts, "#dbgRegisterWatchpoint($1, (NCSTRING)$2, $3);$n",
|
|
[a.addrLoc, makeCString(renderTree(n.sons[1])),
|
|
genTypeInfo(p.module, typ)])
|
|
|
|
proc genPragma(p: BProc, n: PNode) =
|
|
for i in countup(0, sonsLen(n) - 1):
|
|
var it = n.sons[i]
|
|
case whichPragma(it)
|
|
of wEmit: genEmit(p, it)
|
|
of wBreakpoint: genBreakPoint(p, it)
|
|
of wWatchPoint: genWatchpoint(p, it)
|
|
of wInjectStmt:
|
|
var p = newProc(nil, p.module)
|
|
p.options = p.options - {optLineTrace, optStackTrace}
|
|
genStmts(p, it.sons[1])
|
|
p.module.injectStmt = p.s(cpsStmts)
|
|
else: discard
|
|
|
|
proc fieldDiscriminantCheckNeeded(p: BProc, asgn: PNode): bool =
|
|
if optFieldCheck in p.options:
|
|
var le = asgn.sons[0]
|
|
if le.kind == nkCheckedFieldExpr:
|
|
var field = le.sons[0].sons[1].sym
|
|
result = sfDiscriminant in field.flags
|
|
elif le.kind == nkDotExpr:
|
|
var field = le.sons[1].sym
|
|
result = sfDiscriminant in field.flags
|
|
|
|
proc genDiscriminantCheck(p: BProc, a, tmp: TLoc, objtype: PType,
|
|
field: PSym) =
|
|
var t = skipTypes(objtype, abstractVar)
|
|
assert t.kind == tyObject
|
|
discard genTypeInfo(p.module, t)
|
|
var L = lengthOrd(field.typ)
|
|
if not containsOrIncl(p.module.declaredThings, field.id):
|
|
appcg(p.module, cfsVars, "extern $1",
|
|
discriminatorTableDecl(p.module, t, field))
|
|
lineCg(p, cpsStmts,
|
|
"#FieldDiscriminantCheck((NI)(NU)($1), (NI)(NU)($2), $3, $4);$n",
|
|
[rdLoc(a), rdLoc(tmp), discriminatorTableName(p.module, t, field),
|
|
intLiteral(L+1)])
|
|
|
|
proc asgnFieldDiscriminant(p: BProc, e: PNode) =
|
|
var a, tmp: TLoc
|
|
var dotExpr = e.sons[0]
|
|
var d: PSym
|
|
if dotExpr.kind == nkCheckedFieldExpr: dotExpr = dotExpr.sons[0]
|
|
initLocExpr(p, e.sons[0], a)
|
|
getTemp(p, a.t, tmp)
|
|
expr(p, e.sons[1], tmp)
|
|
genDiscriminantCheck(p, a, tmp, dotExpr.sons[0].typ, dotExpr.sons[1].sym)
|
|
genAssignment(p, a, tmp, {})
|
|
|
|
proc genAsgn(p: BProc, e: PNode, fastAsgn: bool) =
|
|
genLineDir(p, e)
|
|
if e.sons[0].kind == nkSym and sfGoto in e.sons[0].sym.flags:
|
|
genGotoVar(p, e.sons[1])
|
|
elif not fieldDiscriminantCheckNeeded(p, e):
|
|
var a: TLoc
|
|
if e[0].kind in {nkDerefExpr, nkHiddenDeref}:
|
|
genDeref(p, e[0], a, enforceDeref=true)
|
|
else:
|
|
initLocExpr(p, e.sons[0], a)
|
|
if fastAsgn: incl(a.flags, lfNoDeepCopy)
|
|
assert(a.t != nil)
|
|
loadInto(p, e.sons[0], e.sons[1], a)
|
|
else:
|
|
asgnFieldDiscriminant(p, e)
|
|
|
|
proc genStmts(p: BProc, t: PNode) =
|
|
var a: TLoc
|
|
expr(p, t, a)
|
|
internalAssert a.k in {locNone, locTemp, locLocalVar}
|