Skip to content

Commit

Permalink
getCustomPragma is split up in more usable chunks (nim-lang#11526)
Browse files Browse the repository at this point in the history
* getCustomPragma is split up in more usable chunks
* changelog entry
* fix for style checks
* shitty typedesc special casing
* Add since annotation and remove typedesc comments
* Fix typo
* Revert since annotation because it breaks bootstrapping
* Export getCustomPragmaNode conditionally
* Reduce code duplication
* Update since
* Update lib/core/macros.nim
* Apply suggestions from code review

Co-authored-by: Clyybber <darkmine956@gmail.com>
Co-authored-by: Andreas Rumpf <rumpf_a@web.de>
  • Loading branch information
3 people authored Apr 14, 2021
1 parent 44657b7 commit 56c3775
Show file tree
Hide file tree
Showing 3 changed files with 193 additions and 96 deletions.
1 change: 1 addition & 0 deletions changelog.md
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@
implementations. Old behavior can be obtained with
`-d:nimLegacyParseQueryStrict`. `cgi.decodeData` which uses the same
underlying code is also updated the same way.
- Custom pragma values have now an API for use in macros.

- In `std/os`, `getHomeDir`, `expandTilde`, `getTempDir`, `getConfigDir` now do not include trailing `DirSep`,
unless `-d:nimLegacyHomeDir` is specified (for a transition period).
Expand Down
258 changes: 165 additions & 93 deletions lib/core/macros.nim
Original file line number Diff line number Diff line change
Expand Up @@ -1474,80 +1474,153 @@ macro expandMacros*(body: typed): untyped =
echo body.toStrLit
result = body

proc customPragmaNode(n: NimNode): NimNode =
expectKind(n, {nnkSym, nnkDotExpr, nnkBracketExpr, nnkTypeOfExpr, nnkCheckedFieldExpr})
let
typ = n.getTypeInst()

if typ.kind == nnkBracketExpr and typ.len > 1 and typ[1].kind == nnkProcTy:
return typ[1][1]
elif typ.typeKind == ntyTypeDesc:
let impl = typ[1].getImpl()
if impl[0].kind == nnkPragmaExpr:
return impl[0][1]
proc findPragmaExprForFieldSym(arg, fieldSym: NimNode): NimNode =
case arg.kind
of nnkRecList, nnkRecCase:
for it in arg.children:
result = findPragmaExprForFieldSym(it, fieldSym)
if result != nil:
return
of nnkOfBranch:
return findPragmaExprForFieldSym(arg[1], fieldSym)
of nnkElse:
return findPragmaExprForFieldSym(arg[0], fieldSym)
of nnkIdentDefs:
for i in 0 ..< arg.len-2:
let child = arg[i]
result = findPragmaExprForFieldSym(child, fieldSym)
if result != nil:
return
of nnkAccQuoted, nnkIdent, nnkSym, nnkPostfix:
return nil
of nnkPragmaExpr:
var ident = arg[0]
if ident.kind == nnkPostfix: ident = ident[1]
if ident.kind == nnkAccQuoted: ident = ident[0]
if eqIdent(ident, fieldSym):
return arg[1]
else:
error("illegal arg: ", arg)

proc getPragmaByName(pragmaExpr: NimNode, name: string): NimNode =
if pragmaExpr.kind == nnkPragma:
for it in pragmaExpr:
if it.kind in nnkPragmaCallKinds:
if eqIdent(it[0], name):
return it
elif it.kind == nnkSym:
if eqIdent(it, name):
return it

proc getCustomPragmaNodeFromProcSym(sym: NimNode, name: string): NimNode =
sym.expectKind nnkSym
if sym.symKind != nskProc: error("expected proc sym", sym)

let impl = sym.getImpl
expectKind(impl, nnkProcDef)
result = getPragmaByName(impl[4], name)

proc getCustomPragmaNodeFromObjFieldSym(sym: NimNode, name: string): NimNode =
sym.expectKind nnkSym
if sym.symKind != nskField: error("expected field sym", sym)

# note this is not ``getTypeImpl``, because the result of
# ``getTypeImpl`` is cleaned up of any pragma expressions.
let impl = sym.owner.getImpl
impl.expectKind nnkTypeDef

let objectTy = if impl[2].kind == nnkRefTy: impl[2][0]
else: impl[2]

# only works on object types
objectTy.expectKind nnkObjectTy

let recList = objectTy[2]
recList.expectKind nnkRecList
result = getPragmaByName(findPragmaExprForFieldSym(recList, sym), name)

proc getCustomPragmaNodeFromTypeSym(sym: NimNode, name: string): NimNode =
sym.expectKind nnkSym
if sym.symKind != nskType: error("expected type sym", sym)
let impl = sym.getImpl
if impl.len > 0:
impl.expectKind nnkTypeDef
let pragmaExpr = impl[0]
if pragmaExpr.kind == nnkPragmaExpr:
result = getPragmaByName(pragmaExpr[1], name)

proc getCustomPragmaNodeFromVarLetSym(sym: NimNode, name: string): NimNode =
sym.expectKind nnkSym
if sym.symKind notin {nskVar, nskLet}: error("expected var/let sym", sym)
let impl = sym.getImpl
impl.expectKind nnkIdentDefs
impl.expectLen 3
let pragmaExpr = impl[0]
if pragmaExpr.kind == nnkPragmaExpr:
result = getPragmaByName(pragmaExpr[1], name)

proc getCustomPragmaNode(sym: NimNode, name: string): NimNode =
sym.expectKind nnkSym
case sym.symKind
of nskField:
result = getCustomPragmaNodeFromObjFieldSym(sym, name)
of nskProc:
result = getCustomPragmaNodeFromProcSym(sym, name)
of nskType:
result = getCustomPragmaNodeFromTypeSym(sym, name)
of nskParam:
# When a typedesc parameter is passed to the macro, it will be of nskParam.
let typeInst = getTypeInst(sym)
if typeInst.kind == nnkBracketExpr and eqIdent(typeInst[0], "typeDesc"):
result = getCustomPragmaNodeFromTypeSym(typeInst[1], name)
else:
return impl[0] # handle types which don't have macro at all

if n.kind == nnkSym: # either an variable or a proc
let impl = n.getImpl()
if impl.kind in RoutineNodes:
return impl.pragma
elif impl.kind == nnkIdentDefs and impl[0].kind == nnkPragmaExpr:
return impl[0][1]
error("illegal sym kind for argument: " & $sym.symKind, sym)
of nskVar, nskLet:
# I think it is a bad idea to fall back to the typeSym. The API
# explicity requests a var/let symbol, not a type symbol.
result =
getCustomPragmaNodeFromVarLetSym(sym, name) or
getCustomPragmaNodeFromTypeSym(sym.getTypeInst, name)
else:
error("illegal sym kind for argument: " & $sym.symKind, sym)

since (1, 5):
export getCustomPragmaNode

proc hasCustomPragma*(n: NimNode, name: string): bool =
n.expectKind nnkSym
let pragmaNode = getCustomPragmaNode(n, name)
result = pragmaNode != nil

proc getCustomPragmaNodeSmart(n: NimNode, name: string): NimNode =
case n.kind
of nnkDotExpr:
result = getCustomPragmaNode(n[1], name)
of nnkCheckedFieldExpr:
expectKind n[0], nnkDotExpr
result = getCustomPragmaNode(n[0][1], name)
of nnkSym:
result = getCustomPragmaNode(n, name)
of nnkTypeOfExpr:
var typeSym = n.getTypeInst
while typeSym.kind == nnkBracketExpr and typeSym[0].eqIdent "typeDesc":
typeSym = typeSym[1]
case typeSym.kind:
of nnkSym:
result = getCustomPragmaNode(typeSym, name)
of nnkProcTy:
# It is a bad idea to support this. The annotation can't be part
# of a symbol.
let pragmaExpr = typeSym[1]
result = getPragmaByName(pragmaExpr, name)
else:
let timpl = typ.getImpl()
if timpl.len>0 and timpl[0].len>1:
return timpl[0][1]
else:
return timpl

if n.kind in {nnkDotExpr, nnkCheckedFieldExpr}:
let name = $(if n.kind == nnkCheckedFieldExpr: n[0][1] else: n[1])
let typInst = getTypeInst(if n.kind == nnkCheckedFieldExpr or n[0].kind == nnkHiddenDeref: n[0][0] else: n[0])
var typDef = getImpl(if typInst.kind == nnkVarTy: typInst[0] else: typInst)
while typDef != nil:
typDef.expectKind(nnkTypeDef)
let typ = typDef[2]
typ.expectKind({nnkRefTy, nnkPtrTy, nnkObjectTy})
let isRef = typ.kind in {nnkRefTy, nnkPtrTy}
if isRef and typ[0].kind in {nnkSym, nnkBracketExpr}: # defines ref type for another object(e.g. X = ref X)
typDef = getImpl(typ[0])
else: # object definition, maybe an object directly defined as a ref type
let
obj = (if isRef: typ[0] else: typ)
var identDefsStack = newSeq[NimNode](obj[2].len)
for i in 0..<identDefsStack.len: identDefsStack[i] = obj[2][i]
while identDefsStack.len > 0:
var identDefs = identDefsStack.pop()
if identDefs.kind == nnkRecCase:
identDefsStack.add(identDefs[0])
for i in 1..<identDefs.len:
let varNode = identDefs[i]
# if it is and empty branch, skip
if varNode[0].kind == nnkNilLit: continue
if varNode[1].kind == nnkIdentDefs:
identDefsStack.add(varNode[1])
else: # nnkRecList
for j in 0 ..< varNode[1].len:
identDefsStack.add(varNode[1][j])

else:
for i in 0 .. identDefs.len - 3:
let varNode = identDefs[i]
if varNode.kind == nnkPragmaExpr:
var varName = varNode[0]
if varName.kind == nnkPostfix:
# This is a public field. We are skipping the postfix *
varName = varName[1]
if eqIdent($varName, name):
return varNode[1]

if obj[1].kind == nnkOfInherit: # explore the parent object
typDef = getImpl(obj[1][0])
else:
typDef = nil
typeSym.expectKind nnkSym
of nnkBracketExpr:
result = nil #false
else:
n.expectKind({nnkDotExpr, nnkCheckedFieldExpr, nnkSym, nnkTypeOfExpr})

macro hasCustomPragma*(n: typed, cp: typed{nkSym}): untyped =
macro hasCustomPragma*(n: typed, cp: typed{nkSym}): bool =
## Expands to `true` if expression `n` which is expected to be `nnkDotExpr`
## (if checking a field), a proc or a type has custom pragma `cp`.
##
Expand All @@ -1564,12 +1637,7 @@ macro hasCustomPragma*(n: typed, cp: typed{nkSym}): untyped =
## var o: MyObj
## assert(o.myField.hasCustomPragma(myAttr))
## assert(myProc.hasCustomPragma(myAttr))
let pragmaNode = customPragmaNode(n)
for p in pragmaNode:
if (p.kind == nnkSym and p == cp) or
(p.kind in nnkPragmaCallKinds and p.len > 0 and p[0].kind == nnkSym and p[0] == cp):
return newLit(true)
return newLit(false)
result = newLit(getCustomPragmaNodeSmart(n, $cp) != nil)

macro getCustomPragmaVal*(n: typed, cp: typed{nkSym}): untyped =
## Expands to value of custom pragma `cp` of expression `n` which is expected
Expand All @@ -1586,22 +1654,26 @@ macro getCustomPragmaVal*(n: typed, cp: typed{nkSym}): untyped =
## assert(o.myField.getCustomPragmaVal(serializationKey) == "mf")
## assert(o.getCustomPragmaVal(serializationKey) == "mo")
## assert(MyObj.getCustomPragmaVal(serializationKey) == "mo")
result = nil
let pragmaNode = customPragmaNode(n)
for p in pragmaNode:
if p.kind in nnkPragmaCallKinds and p.len > 0 and p[0].kind == nnkSym and p[0] == cp:
if p.len == 2:
result = p[1]
else:
let def = p[0].getImpl[3]
result = newTree(nnkPar)
for i in 1 ..< def.len:
let key = def[i][0]
let val = p[i]
result.add newTree(nnkExprColonExpr, key, val)
break
if result.kind == nnkEmpty:
error(n.repr & " doesn't have a pragma named " & cp.repr()) # returning an empty node results in most cases in a cryptic error,
n.expectKind({nnkDotExpr, nnkCheckedFieldExpr, nnkSym, nnkTypeOfExpr})
let pragmaNode = getCustomPragmaNodeSmart(n, $cp)

case pragmaNode.kind
of nnkPragmaCallKinds:
assert pragmaNode[0] == cp
if pragmaNode.len == 2:
result = pragmaNode[1]
else:
# create a named tuple expression for pragmas with multiple arguments
let def = pragmaNode[0].getImpl[3]
result = newTree(nnkPar)
for i in 1 ..< def.len:
let key = def[i][0]
let val = pragmaNode[i]
result.add nnkExprColonExpr.newTree(key, val)
of nnkSym:
error("The named pragma " & cp.repr & " in " & n.repr & " has no arguments and therefore no value.")
else:
error(n.repr & " doesn't have a pragma named " & cp.repr, n)

macro unpackVarargs*(callee: untyped; args: varargs[untyped]): untyped =
result = newCall(callee)
Expand Down
30 changes: 27 additions & 3 deletions tests/pragmas/tcustom_pragma.nim
Original file line number Diff line number Diff line change
Expand Up @@ -156,13 +156,20 @@ block:
proc generic_proc[T]() =
doAssert Annotated.hasCustomPragma(simpleAttr)


#--------------------------------------------------------------------------
# Pragma on proc type

let a: proc(x: int) {.defaultValue(5).} = nil
type
MyAnnotatedProcType {.defaultValue(4).} = proc(x: int)

let a {.defaultValue(4).}: proc(x: int) = nil
var b: MyAnnotatedProcType = nil
var c: proc(x: int): void {.defaultValue(5).} = nil
static:
doAssert hasCustomPragma(a.type, defaultValue)
doAssert hasCustomPragma(a, defaultValue)
doAssert hasCustomPragma(MyAnnotatedProcType, defaultValue)
doAssert hasCustomPragma(b, defaultValue)
doAssert hasCustomPragma(typeof(c), defaultValue)

# bug #8371
template thingy {.pragma.}
Expand Down Expand Up @@ -378,3 +385,20 @@ block:
b {.world.}: int

discard Hello(a: 1.0, b: 12)

# issue #11511
block:
template myAttr {.pragma.}

type TObj = object
a {.myAttr.}: int

macro hasMyAttr(t: typedesc): untyped =
let objTy = t.getType[1].getType
let recList = objTy[2]
let sym = recList[0]
assert sym.kind == nnkSym and sym.eqIdent("a")
let hasAttr = sym.hasCustomPragma("myAttr")
newLit(hasAttr)

doAssert hasMyAttr(TObj)

0 comments on commit 56c3775

Please sign in to comment.