astspec.txt 26 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405
  1. The AST in Nim
  2. ==============
  3. This section describes how the AST is modelled with Nim's type system.
  4. The AST consists of nodes (``NimNode``) with a variable number of
  5. children. Each node has a field named ``kind`` which describes what the node
  6. contains:
  7. .. code-block:: nim
  8. type
  9. NimNodeKind = enum ## kind of a node; only explanatory
  10. nnkNone, ## invalid node kind
  11. nnkEmpty, ## empty node
  12. nnkIdent, ## node contains an identifier
  13. nnkIntLit, ## node contains an int literal (example: 10)
  14. nnkStrLit, ## node contains a string literal (example: "abc")
  15. nnkNilLit, ## node contains a nil literal (example: nil)
  16. nnkCaseStmt, ## node represents a case statement
  17. ... ## many more
  18. NimNode = ref NimNodeObj
  19. NimNodeObj = object
  20. case kind: NimNodeKind ## the node's kind
  21. of nnkNone, nnkEmpty, nnkNilLit:
  22. discard ## node contains no additional fields
  23. of nnkCharLit..nnkUInt64Lit:
  24. intVal: BiggestInt ## the int literal
  25. of nnkFloatLit..nnkFloat64Lit:
  26. floatVal: BiggestFloat ## the float literal
  27. of nnkStrLit..nnkTripleStrLit, nnkCommentStmt, nnkIdent, nnkSym:
  28. strVal: string ## the string literal
  29. else:
  30. sons: seq[NimNode] ## the node's sons (or children)
  31. For the ``NimNode`` type, the ``[]`` operator has been overloaded:
  32. ``n[i]`` is ``n``'s ``i``-th child.
  33. To specify the AST for the different Nim constructs, the notation
  34. ``nodekind(son1, son2, ...)`` or ``nodekind(value)`` or
  35. ``nodekind(field=value)`` is used.
  36. Some child may be missing. A missing child is a node of kind ``nnkEmpty``;
  37. a child can never be nil.
  38. Leaf nodes/Atoms
  39. ================
  40. A leaf of the AST often corresponds to a terminal symbol in the concrete
  41. syntax. Note that the default ``float`` in Nim maps to ``float64`` such that
  42. the default AST for a float is ``nnkFloat64Lit`` as below.
  43. ----------------- ---------------------------------------------
  44. Nim expression Corresponding AST
  45. ----------------- ---------------------------------------------
  46. ``42`` ``nnkIntLit(intVal = 42)``
  47. ``42'i8`` ``nnkInt8Lit(intVal = 42)``
  48. ``42'i16`` ``nnkInt16Lit(intVal = 42)``
  49. ``42'i32`` ``nnkInt32Lit(intVal = 42)``
  50. ``42'i64`` ``nnkInt64Lit(intVal = 42)``
  51. ``42'u8`` ``nnkUInt8Lit(intVal = 42)``
  52. ``42'u16`` ``nnkUInt16Lit(intVal = 42)``
  53. ``42'u32`` ``nnkUInt32Lit(intVal = 42)``
  54. ``42'u64`` ``nnkUInt64Lit(intVal = 42)``
  55. ``42.0`` ``nnkFloat64Lit(floatVal = 42.0)``
  56. ``42.0'f32`` ``nnkFloat32Lit(floatVal = 42.0)``
  57. ``42.0'f64`` ``nnkFloat64Lit(floatVal = 42.0)``
  58. ``"abc"`` ``nnkStrLit(strVal = "abc")``
  59. ``r"abc"`` ``nnkRStrLit(strVal = "abc")``
  60. ``"""abc"""`` ``nnkTripleStrLit(strVal = "abc")``
  61. ``' '`` ``nnkCharLit(intVal = 32)``
  62. ``nil`` ``nnkNilLit()``
  63. ``myIdentifier`` ``nnkIdent(strVal = "myIdentifier")``
  64. ``myIdentifier`` after lookup pass: ``nnkSym(strVal = "myIdentifier", ...)``
  65. ----------------- ---------------------------------------------
  66. Identifiers are ``nnkIdent`` nodes. After the name lookup pass these nodes
  67. get transferred into ``nnkSym`` nodes.
  68. Calls/expressions
  69. =================
  70. Command call
  71. ------------
  72. Concrete syntax:
  73. .. code-block:: nim
  74. echo "abc", "xyz"
  75. AST:
  76. .. code-block:: nim
  77. nnkCommand(
  78. nnkIdent("echo"),
  79. nnkStrLit("abc"),
  80. nnkStrLit("xyz")
  81. )
  82. Call with ``()``
  83. ----------------
  84. Concrete syntax:
  85. .. code-block:: nim
  86. echo("abc", "xyz")
  87. AST:
  88. .. code-block:: nim
  89. nnkCall(
  90. nnkIdent("echo"),
  91. nnkStrLit("abc"),
  92. nnkStrLit("xyz")
  93. )
  94. Infix operator call
  95. -------------------
  96. Concrete syntax:
  97. .. code-block:: nim
  98. "abc" & "xyz"
  99. AST:
  100. .. code-block:: nim
  101. nnkInfix(
  102. nnkIdent("&"),
  103. nnkStrLit("abc"),
  104. nnkStrLit("xyz")
  105. )
  106. Note that with multiple infix operators, the command is parsed by operator
  107. precedence.
  108. Concrete syntax:
  109. .. code-block:: nim
  110. 5 + 3 * 4
  111. AST:
  112. .. code-block:: nim
  113. nnkInfix(
  114. nnkIdent("+"),
  115. nnkIntLit(5),
  116. nnkInfix(
  117. nnkIdent("*"),
  118. nnkIntLit(3),
  119. nnkIntLit(4)
  120. )
  121. )
  122. As a side note, if you choose to use infix operators in a prefix form, the AST
  123. behaves as a
  124. [parenthetical function call](#callsslashexpressions-call-with) with
  125. ``nnkAccQuoted``, as follows:
  126. Concrete syntax:
  127. .. code-block:: nim
  128. `+`(3, 4)
  129. AST:
  130. .. code-block:: nim
  131. nnkCall(
  132. nnkAccQuoted(
  133. nnkIdent("+")
  134. ),
  135. nnkIntLit(3),
  136. nnkIntLit(4)
  137. )
  138. Prefix operator call
  139. --------------------
  140. Concrete syntax:
  141. .. code-block:: nim
  142. ? "xyz"
  143. AST:
  144. .. code-block:: nim
  145. nnkPrefix(
  146. nnkIdent("?"),
  147. nnkStrLit("abc")
  148. )
  149. Postfix operator call
  150. ---------------------
  151. **Note:** There are no postfix operators in Nim. However, the
  152. ``nnkPostfix`` node is used for the *asterisk export marker* ``*``:
  153. Concrete syntax:
  154. .. code-block:: nim
  155. identifier*
  156. AST:
  157. .. code-block:: nim
  158. nnkPostfix(
  159. nnkIdent("*"),
  160. nnkIdent("identifier")
  161. )
  162. Call with named arguments
  163. -------------------------
  164. Concrete syntax:
  165. .. code-block:: nim
  166. writeLine(file=stdout, "hallo")
  167. AST:
  168. .. code-block:: nim
  169. nnkCall(
  170. nnkIdent("writeLine"),
  171. nnkExprEqExpr(
  172. nnkIdent("file"),
  173. nnkIdent("stdout")
  174. ),
  175. nnkStrLit("hallo")
  176. )
  177. Call with raw string literal
  178. ----------------------------
  179. This is used, for example, in the ``bindSym`` examples
  180. [here](manual.html#macros-bindsym) and with
  181. ``re"some regexp"`` in the regular expression module.
  182. Concrete syntax:
  183. .. code-block:: nim
  184. echo"abc"
  185. AST:
  186. .. code-block:: nim
  187. nnkCallStrLit(
  188. nnkIdent("echo"),
  189. nnkRStrLit("hello")
  190. )
  191. Dereference operator ``[]``
  192. ---------------------------
  193. Concrete syntax:
  194. .. code-block:: nim
  195. x[]
  196. AST:
  197. .. code-block:: nim
  198. nnkDerefExpr(nnkIdent("x"))
  199. Addr operator
  200. -------------
  201. Concrete syntax:
  202. .. code-block:: nim
  203. addr(x)
  204. AST:
  205. .. code-block:: nim
  206. nnkAddr(nnkIdent("x"))
  207. Cast operator
  208. -------------
  209. Concrete syntax:
  210. .. code-block:: nim
  211. cast[T](x)
  212. AST:
  213. .. code-block:: nim
  214. nnkCast(nnkIdent("T"), nnkIdent("x"))
  215. Object access operator ``.``
  216. ----------------------------
  217. Concrete syntax:
  218. .. code-block:: nim
  219. x.y
  220. AST:
  221. .. code-block:: nim
  222. nnkDotExpr(nnkIdent("x"), nnkIdent("y"))
  223. If you use Nim's flexible calling syntax (as in ``x.len()``), the result is the
  224. same as above but wrapped in an ``nnkCall``.
  225. Array access operator ``[]``
  226. ----------------------------
  227. Concrete syntax:
  228. .. code-block:: nim
  229. x[y]
  230. AST:
  231. .. code-block:: nim
  232. nnkBracketExpr(nnkIdent("x"), nnkIdent("y"))
  233. Parentheses
  234. -----------
  235. Parentheses for affecting operator precedence or tuple construction
  236. are built with the ``nnkPar`` node.
  237. Concrete syntax:
  238. .. code-block:: nim
  239. (1, 2, (3))
  240. AST:
  241. .. code-block:: nim
  242. nnkPar(nnkIntLit(1), nnkIntLit(2), nnkPar(nnkIntLit(3)))
  243. Curly braces
  244. ------------
  245. Curly braces are used as the set constructor.
  246. Concrete syntax:
  247. .. code-block:: nim
  248. {1, 2, 3}
  249. AST:
  250. .. code-block:: nim
  251. nnkCurly(nnkIntLit(1), nnkIntLit(2), nnkIntLit(3))
  252. When used as a table constructor, the syntax is different.
  253. Concrete syntax:
  254. .. code-block:: nim
  255. {a: 3, b: 5}
  256. AST:
  257. .. code-block:: nim
  258. nnkTableConstr(
  259. nnkExprColonExpr(nnkIdent("a"), nnkIntLit(3)),
  260. nnkExprColonExpr(nnkIdent("b"), nnkIntLit(5))
  261. )
  262. Brackets
  263. --------
  264. Brackets are used as the array constructor.
  265. Concrete syntax:
  266. .. code-block:: nim
  267. [1, 2, 3]
  268. AST:
  269. .. code-block:: nim
  270. nnkBracket(nnkIntLit(1), nnkIntLit(2), nnkIntLit(3))
  271. Ranges
  272. ------
  273. Ranges occur in set constructors, case statement branches, or array slices.
  274. Internally, the node kind ``nnkRange`` is used, but when constructing the
  275. AST, construction with ``..`` as an infix operator should be used instead.
  276. Concrete syntax:
  277. .. code-block:: nim
  278. 1..3
  279. AST:
  280. .. code-block:: nim
  281. nnkInfix(
  282. nnkIdent(".."),
  283. nnkIntLit(1),
  284. nnkIntLit(3)
  285. )
  286. Example code:
  287. .. code-block:: nim
  288. macro genRepeatEcho(): stmt =
  289. result = newNimNode(nnkStmtList)
  290. var forStmt = newNimNode(nnkForStmt) # generate a for statement
  291. forStmt.add(ident("i")) # use the variable `i` for iteration
  292. var rangeDef = newNimNode(nnkInfix).add(
  293. ident("..")).add(
  294. newIntLitNode(3),newIntLitNode(5)) # iterate over the range 3..5
  295. forStmt.add(rangeDef)
  296. forStmt.add(newCall(ident("echo"), newIntLitNode(3))) # meat of the loop
  297. result.add(forStmt)
  298. genRepeatEcho() # gives:
  299. # 3
  300. # 3
  301. # 3
  302. If expression
  303. -------------
  304. The representation of the ``if`` expression is subtle, but easy to traverse.
  305. Concrete syntax:
  306. .. code-block:: nim
  307. if cond1: expr1 elif cond2: expr2 else: expr3
  308. AST:
  309. .. code-block:: nim
  310. nnkIfExpr(
  311. nnkElifExpr(cond1, expr1),
  312. nnkElifExpr(cond2, expr2),
  313. nnkElseExpr(expr3)
  314. )
  315. Documentation Comments
  316. ----------------------
  317. Double-hash (``##``) comments in the code actually have their own format,
  318. using ``strVal`` to get and set the comment text. Single-hash (``#``)
  319. comments are ignored.
  320. Concrete syntax:
  321. .. code-block:: nim
  322. ## This is a comment
  323. ## This is part of the first comment
  324. stmt1
  325. ## Yet another
  326. AST:
  327. .. code-block:: nim
  328. nnkCommentStmt() # only appears once for the first two lines!
  329. stmt1
  330. nnkCommentStmt() # another nnkCommentStmt because there is another comment
  331. # (separate from the first)
  332. Pragmas
  333. -------
  334. One of Nim's cool features is pragmas, which allow fine-tuning of various
  335. aspects of the language. They come in all types, such as adorning procs and
  336. objects, but the standalone ``emit`` pragma shows the basics with the AST.
  337. Concrete syntax:
  338. .. code-block:: nim
  339. {.emit: "#include <stdio.h>".}
  340. AST:
  341. .. code-block:: nim
  342. nnkPragma(
  343. nnkExprColonExpr(
  344. nnkIdent("emit"),
  345. nnkStrLit("#include <stdio.h>") # the "argument"
  346. )
  347. )
  348. As many ``nnkIdent`` appear as there are pragmas between ``{..}``. Note that
  349. the declaration of new pragmas is essentially the same:
  350. Concrete syntax:
  351. .. code-block:: nim
  352. {.pragma: cdeclRename, cdecl.}
  353. AST:
  354. .. code-block:: nim
  355. nnkPragma(
  356. nnkExprColonExpr(
  357. nnkIdent("pragma"), # this is always first when declaring a new pragma
  358. nnkIdent("cdeclRename") # the name of the pragma
  359. ),
  360. nnkIdent("cdecl")
  361. )
  362. Statements
  363. ==========
  364. If statement
  365. ------------
  366. The representation of the if statement is subtle, but easy to traverse. If
  367. there is no ``else`` branch, no ``nnkElse`` child exists.
  368. Concrete syntax:
  369. .. code-block:: nim
  370. if cond1:
  371. stmt1
  372. elif cond2:
  373. stmt2
  374. elif cond3:
  375. stmt3
  376. else:
  377. stmt4
  378. AST:
  379. .. code-block:: nim
  380. nnkIfStmt(
  381. nnkElifBranch(cond1, stmt1),
  382. nnkElifBranch(cond2, stmt2),
  383. nnkElifBranch(cond3, stmt3),
  384. nnkElse(stmt4)
  385. )
  386. When statement
  387. --------------
  388. Like the ``if`` statement, but the root has the kind ``nnkWhenStmt``.
  389. Assignment
  390. ----------
  391. Concrete syntax:
  392. .. code-block:: nim
  393. x = 42
  394. AST:
  395. .. code-block:: nim
  396. nnkAsgn(nnkIdent("x"), nnkIntLit(42))
  397. This is not the syntax for assignment when combined with ``var``, ``let``,
  398. or ``const``.
  399. Statement list
  400. --------------
  401. Concrete syntax:
  402. .. code-block:: nim
  403. stmt1
  404. stmt2
  405. stmt3
  406. AST:
  407. .. code-block:: nim
  408. nnkStmtList(stmt1, stmt2, stmt3)
  409. Case statement
  410. --------------
  411. Concrete syntax:
  412. .. code-block:: nim
  413. case expr1
  414. of expr2, expr3..expr4:
  415. stmt1
  416. of expr5:
  417. stmt2
  418. elif cond1:
  419. stmt3
  420. else:
  421. stmt4
  422. AST:
  423. .. code-block:: nim
  424. nnkCaseStmt(
  425. expr1,
  426. nnkOfBranch(expr2, nnkRange(expr3, expr4), stmt1),
  427. nnkOfBranch(expr5, stmt2),
  428. nnkElifBranch(cond1, stmt3),
  429. nnkElse(stmt4)
  430. )
  431. The ``nnkElifBranch`` and ``nnkElse`` parts may be missing.
  432. While statement
  433. ---------------
  434. Concrete syntax:
  435. .. code-block:: nim
  436. while expr1:
  437. stmt1
  438. AST:
  439. .. code-block:: nim
  440. nnkWhileStmt(expr1, stmt1)
  441. For statement
  442. -------------
  443. Concrete syntax:
  444. .. code-block:: nim
  445. for ident1, ident2 in expr1:
  446. stmt1
  447. AST:
  448. .. code-block:: nim
  449. nnkForStmt(ident1, ident2, expr1, stmt1)
  450. Try statement
  451. -------------
  452. Concrete syntax:
  453. .. code-block:: nim
  454. try:
  455. stmt1
  456. except e1, e2:
  457. stmt2
  458. except e3:
  459. stmt3
  460. except:
  461. stmt4
  462. finally:
  463. stmt5
  464. AST:
  465. .. code-block:: nim
  466. nnkTryStmt(
  467. stmt1,
  468. nnkExceptBranch(e1, e2, stmt2),
  469. nnkExceptBranch(e3, stmt3),
  470. nnkExceptBranch(stmt4),
  471. nnkFinally(stmt5)
  472. )
  473. Return statement
  474. ----------------
  475. Concrete syntax:
  476. .. code-block:: nim
  477. return expr1
  478. AST:
  479. .. code-block:: nim
  480. nnkReturnStmt(expr1)
  481. Yield statement
  482. ---------------
  483. Like ``return``, but with ``nnkYieldStmt`` kind.
  484. .. code-block:: nim
  485. nnkYieldStmt(expr1)
  486. Discard statement
  487. -----------------
  488. Like ``return``, but with ``nnkDiscardStmt`` kind.
  489. .. code-block:: nim
  490. nnkDiscardStmt(expr1)
  491. Continue statement
  492. ------------------
  493. Concrete syntax:
  494. .. code-block:: nim
  495. continue
  496. AST:
  497. .. code-block:: nim
  498. nnkContinueStmt()
  499. Break statement
  500. ---------------
  501. Concrete syntax:
  502. .. code-block:: nim
  503. break otherLocation
  504. AST:
  505. .. code-block:: nim
  506. nnkBreakStmt(nnkIdent("otherLocation"))
  507. If ``break`` is used without a jump-to location, ``nnkEmpty`` replaces ``nnkIdent``.
  508. Block statement
  509. ---------------
  510. Concrete syntax:
  511. .. code-block:: nim
  512. block name:
  513. AST:
  514. .. code-block:: nim
  515. nnkBlockStmt(nnkIdent("name"), nnkStmtList(...))
  516. A ``block`` doesn't need an name, in which case ``nnkEmpty`` is used.
  517. Asm statement
  518. -------------
  519. Concrete syntax:
  520. .. code-block:: nim
  521. asm """
  522. some asm
  523. """
  524. AST:
  525. .. code-block:: nim
  526. nnkAsmStmt(
  527. nnkEmpty(), # for pragmas
  528. nnkTripleStrLit("some asm"),
  529. )
  530. Import section
  531. --------------
  532. Nim's ``import`` statement actually takes different variations depending
  533. on what keywords are present. Let's start with the simplest form.
  534. Concrete syntax:
  535. .. code-block:: nim
  536. import math
  537. AST:
  538. .. code-block:: nim
  539. nnkImportStmt(nnkIdent("math"))
  540. With ``except``, we get ``nnkImportExceptStmt``.
  541. Concrete syntax:
  542. .. code-block:: nim
  543. import math except pow
  544. AST:
  545. .. code-block:: nim
  546. nnkImportExceptStmt(nnkIdent("math"),nnkIdent("pow"))
  547. Note that ``import math as m`` does not use a different node; rather,
  548. we use ``nnkImportStmt`` with ``as`` as an infix operator.
  549. Concrete syntax:
  550. .. code-block:: nim
  551. import strutils as su
  552. AST:
  553. .. code-block:: nim
  554. nnkImportStmt(
  555. nnkInfix(
  556. nnkIdent("as"),
  557. nnkIdent("strutils"),
  558. nnkIdent("su")
  559. )
  560. )
  561. From statement
  562. --------------
  563. If we use ``from ... import``, the result is different, too.
  564. Concrete syntax:
  565. .. code-block:: nim
  566. from math import pow
  567. AST:
  568. .. code-block:: nim
  569. nnkFromStmt(nnkIdent("math"), nnkIdent("pow"))
  570. Using ``from math as m import pow`` works identically to the ``as`` modifier
  571. with the ``import`` statement, but wrapped in ``nnkFromStmt``.
  572. Export statement
  573. ----------------
  574. When you are making an imported module accessible by modules that import yours,
  575. the ``export`` syntax is pretty straightforward.
  576. Concrete syntax:
  577. .. code-block:: nim
  578. export unsigned
  579. AST:
  580. .. code-block:: nim
  581. nnkExportStmt(nnkIdent("unsigned"))
  582. Similar to the ``import`` statement, the AST is different for
  583. ``export ... except``.
  584. Concrete syntax:
  585. .. code-block:: nim
  586. export math except pow # we're going to implement our own exponentiation
  587. AST:
  588. .. code-block:: nim
  589. nnkExportExceptStmt(nnkIdent("math"),nnkIdent("pow"))
  590. Include statement
  591. -----------------
  592. Like a plain ``import`` statement but with ``nnkIncludeStmt``.
  593. Concrete syntax:
  594. .. code-block:: nim
  595. include blocks
  596. AST:
  597. .. code-block:: nim
  598. nnkIncludeStmt(nnkIdent("blocks"))
  599. Var section
  600. -----------
  601. Concrete syntax:
  602. .. code-block:: nim
  603. var a = 3
  604. AST:
  605. .. code-block:: nim
  606. nnkVarSection(
  607. nnkIdentDefs(
  608. nnkIdent("a"),
  609. nnkEmpty(), # or nnkIdent(...) if the variable declares the type
  610. nnkIntLit(3),
  611. )
  612. )
  613. Note that either the second or third (or both) parameters above must exist,
  614. as the compiler needs to know the type somehow (which it can infer from
  615. the given assignment).
  616. This is not the same AST for all uses of ``var``. See
  617. [Procedure declaration](macros.html#statements-procedure-declaration)
  618. for details.
  619. Let section
  620. -----------
  621. This is equivalent to ``var``, but with ``nnkLetSection`` rather than
  622. ``nnkVarSection``.
  623. Concrete syntax:
  624. .. code-block:: nim
  625. let a = 3
  626. AST:
  627. .. code-block:: nim
  628. nnkLetSection(
  629. nnkIdentDefs(
  630. nnkIdent("a"),
  631. nnkEmpty(), # or nnkIdent(...) for the type
  632. nnkIntLit(3),
  633. )
  634. )
  635. Const section
  636. -------------
  637. Concrete syntax:
  638. .. code-block:: nim
  639. const a = 3
  640. AST:
  641. .. code-block:: nim
  642. nnkConstSection(
  643. nnkConstDef( # not nnkConstDefs!
  644. nnkIdent("a"),
  645. nnkEmpty(), # or nnkIdent(...) if the variable declares the type
  646. nnkIntLit(3), # required in a const declaration!
  647. )
  648. )
  649. Type section
  650. ------------
  651. Starting with the simplest case, a ``type`` section appears much like ``var``
  652. and ``const``.
  653. Concrete syntax:
  654. .. code-block:: nim
  655. type A = int
  656. AST:
  657. .. code-block:: nim
  658. nnkTypeSection(
  659. nnkTypeDef(
  660. nnkIdent("A"),
  661. nnkEmpty(),
  662. nnkIdent("int")
  663. )
  664. )
  665. Declaring ``distinct`` types is similar, with the last ``nnkIdent`` wrapped
  666. in ``nnkDistinctTy``.
  667. Concrete syntax:
  668. .. code-block:: nim
  669. type MyInt = distinct int
  670. AST:
  671. .. code-block:: nim
  672. # ...
  673. nnkTypeDef(
  674. nnkIdent("MyInt"),
  675. nnkEmpty(),
  676. nnkDistinctTy(
  677. nnkIdent("int")
  678. )
  679. )
  680. If a type section uses generic parameters, they are treated here:
  681. Concrete syntax:
  682. .. code-block:: nim
  683. type A[T] = expr1
  684. AST:
  685. .. code-block:: nim
  686. nnkTypeSection(
  687. nnkTypeDef(
  688. nnkIdent("A"),
  689. nnkGenericParams(
  690. nnkIdentDefs(
  691. nnkIdent("T"),
  692. nnkEmpty(), # if the type is declared with options, like
  693. # ``[T: SomeInteger]``, they are given here
  694. nnkEmpty(),
  695. )
  696. )
  697. expr1,
  698. )
  699. )
  700. Note that not all ``nnkTypeDef`` utilize ``nnkIdent`` as their
  701. parameter. One of the most common uses of type declarations
  702. is to work with objects.
  703. Concrete syntax:
  704. .. code-block:: nim
  705. type IO = object of RootObj
  706. AST:
  707. .. code-block:: nim
  708. # ...
  709. nnkTypeDef(
  710. nnkIdent("IO"),
  711. nnkEmpty(),
  712. nnkObjectTy(
  713. nnkEmpty(), # no pragmas here
  714. nnkOfInherit(
  715. nnkIdent("RootObj") # inherits from RootObj
  716. )
  717. nnkEmpty()
  718. )
  719. )
  720. Nim's object syntax is rich. Let's take a look at an involved example in
  721. its entirety to see some of the complexities.
  722. Concrete syntax:
  723. .. code-block:: nim
  724. type Obj[T] = object {.inheritable.}
  725. name: string
  726. case isFat: bool
  727. of true:
  728. m: array[100_000, T]
  729. of false:
  730. m: array[10, T]
  731. AST:
  732. .. code-block:: nim
  733. # ...
  734. nnkObjectTy(
  735. nnkPragma(
  736. nnkIdent("inheritable")
  737. ),
  738. nnkEmpty(),
  739. nnkRecList( # list of object parameters
  740. nnkIdentDefs(
  741. nnkIdent("name"),
  742. nnkIdent("string"),
  743. nnkEmpty()
  744. ),
  745. nnkRecCase( # case statement within object (not nnkCaseStmt)
  746. nnkIdentDefs(
  747. nnkIdent("isFat"),
  748. nnkIdent("bool"),
  749. nnkEmpty()
  750. ),
  751. nnkOfBranch(
  752. nnkIdent("true"),
  753. nnkRecList( # again, a list of object parameters
  754. nnkIdentDefs(
  755. nnkIdent("m"),
  756. nnkBracketExpr(
  757. nnkIdent("array"),
  758. nnkIntLit(100000),
  759. nnkIdent("T")
  760. ),
  761. nnkEmpty()
  762. )
  763. ),
  764. nnkOfBranch(
  765. nnkIdent("false"),
  766. nnkRecList(
  767. nnkIdentDefs(
  768. nnkIdent("m"),
  769. nnkBracketExpr(
  770. nnkIdent("array"),
  771. nnkIntLit(10),
  772. nnkIdent("T")
  773. ),
  774. nnkEmpty()
  775. )
  776. )
  777. )
  778. )
  779. )
  780. )
  781. Using an ``enum`` is similar to using an ``object``.
  782. Concrete syntax:
  783. .. code-block:: nim
  784. type X = enum
  785. First
  786. AST:
  787. .. code-block:: nim
  788. # ...
  789. nnkEnumTy(
  790. nnkEmpty(),
  791. nnkIdent("First") # you need at least one nnkIdent or the compiler complains
  792. )
  793. The usage of ``concept`` (experimental) is similar to objects.
  794. Concrete syntax:
  795. .. code-block:: nim
  796. type Con = concept x,y,z
  797. (x & y & z) is string
  798. AST:
  799. .. code-block:: nim
  800. # ...
  801. nnkTypeClassTy( # note this isn't nnkConceptTy!
  802. nnkArglist(
  803. # ... idents for x, y, z
  804. )
  805. # ...
  806. )
  807. Static types, like ``static[int]``, use ``nnkIdent`` wrapped in
  808. ``nnkStaticTy``.
  809. Concrete syntax:
  810. .. code-block:: nim
  811. type A[T: static[int]] = object
  812. AST:
  813. .. code-block:: nim
  814. # ... within nnkGenericParams
  815. nnkIdentDefs(
  816. nnkIdent("T"),
  817. nnkStaticTy(
  818. nnkIdent("int")
  819. ),
  820. nnkEmpty()
  821. )
  822. # ...
  823. In general, declaring types mirrors this syntax (i.e., ``nnkStaticTy`` for
  824. ``static``, etc.). Examples follow (exceptions marked by ``*``):
  825. ------------- ---------------------------------------------
  826. Nim type Corresponding AST
  827. ------------- ---------------------------------------------
  828. ``static`` ``nnkStaticTy``
  829. ``tuple`` ``nnkTupleTy``
  830. ``var`` ``nnkVarTy``
  831. ``ptr`` ``nnkPtrTy``
  832. ``ref`` ``nnkRefTy``
  833. ``distinct`` ``nnkDistinctTy``
  834. ``enum`` ``nnkEnumTy``
  835. ``concept`` ``nnkTypeClassTy``\*
  836. ``array`` ``nnkBracketExpr(nnkIdent("array"),...``\*
  837. ``proc`` ``nnkProcTy``
  838. ``iterator`` ``nnkIteratorTy``
  839. ``object`` ``nnkObjectTy``
  840. ------------- ---------------------------------------------
  841. Take special care when declaring types as ``proc``. The behavior is similar
  842. to ``Procedure declaration``, below, but does not treat ``nnkGenericParams``.
  843. Generic parameters are treated in the type, not the ``proc`` itself.
  844. Concrete syntax:
  845. .. code-block:: nim
  846. type MyProc[T] = proc(x: T)
  847. AST:
  848. .. code-block:: nim
  849. # ...
  850. nnkTypeDef(
  851. nnkIdent("MyProc"),
  852. nnkGenericParams( # here, not with the proc
  853. # ...
  854. )
  855. nnkProcTy( # behaves like a procedure declaration from here on
  856. nnkFormalParams(
  857. # ...
  858. )
  859. )
  860. )
  861. The same syntax applies to ``iterator`` (with ``nnkIteratorTy``), but
  862. *does not* apply to ``converter`` or ``template``.
  863. Mixin statement
  864. ---------------
  865. Concrete syntax:
  866. .. code-block:: nim
  867. mixin x
  868. AST:
  869. .. code-block:: nim
  870. nnkMixinStmt(nnkIdent("x"))
  871. Bind statement
  872. --------------
  873. Concrete syntax:
  874. .. code-block:: nim
  875. bind x
  876. AST:
  877. .. code-block:: nim
  878. nnkBindStmt(nnkIdent("x"))
  879. Procedure declaration
  880. ---------------------
  881. Let's take a look at a procedure with a lot of interesting aspects to get
  882. a feel for how procedure calls are broken down.
  883. Concrete syntax:
  884. .. code-block:: nim
  885. proc hello*[T: SomeInteger](x: int = 3, y: float32): int {.inline.} = discard
  886. AST:
  887. .. code-block:: nim
  888. nnkProcDef(
  889. nnkPostfix(nnkIdent("*"), nnkIdent("hello")), # the exported proc name
  890. nnkEmpty(), # patterns for term rewriting in templates and macros (not procs)
  891. nnkGenericParams( # generic type parameters, like with type declaration
  892. nnkIdentDefs(
  893. nnkIdent("T"),
  894. nnkIdent("SomeInteger"),
  895. nnkEmpty()
  896. )
  897. ),
  898. nnkFormalParams(
  899. nnkIdent("int"), # the first FormalParam is the return type. nnkEmpty() if there is none
  900. nnkIdentDefs(
  901. nnkIdent("x"),
  902. nnkIdent("int"), # type type (required for procs, not for templates)
  903. nnkIntLit(3) # a default value
  904. ),
  905. nnkIdentDefs(
  906. nnkIdent("y"),
  907. nnkIdent("float32"),
  908. nnkEmpty()
  909. )
  910. ),
  911. nnkPragma(nnkIdent("inline")),
  912. nnkEmpty(), # reserved slot for future use
  913. nnkStmtList(nnkDiscardStmt(nnkEmpty())) # the meat of the proc
  914. )
  915. There is another consideration. Nim has flexible type identification for
  916. its procs. Even though ``proc(a: int, b: int)`` and ``proc(a, b: int)``
  917. are equivalent in the code, the AST is a little different for the latter.
  918. Concrete syntax:
  919. .. code-block:: nim
  920. proc(a, b: int)
  921. AST:
  922. .. code-block:: nim
  923. # ...AST as above...
  924. nnkFormalParams(
  925. nnkEmpty(), # no return here
  926. nnkIdentDefs(
  927. nnkIdent("a"), # the first parameter
  928. nnkIdent("b"), # directly to the second parameter
  929. nnkIdent("int"), # their shared type identifier
  930. nnkEmpty(), # default value would go here
  931. )
  932. ),
  933. # ...
  934. When a procedure uses the special ``var`` type return variable, the result
  935. is different from that of a var section.
  936. Concrete syntax:
  937. .. code-block:: nim
  938. proc hello(): var int
  939. AST:
  940. .. code-block:: nim
  941. # ...
  942. nnkFormalParams(
  943. nnkVarTy(
  944. nnkIdent("int")
  945. )
  946. )
  947. Iterator declaration
  948. --------------------
  949. The syntax for iterators is similar to procs, but with ``nnkIteratorDef``
  950. replacing ``nnkProcDef``.
  951. Concrete syntax:
  952. .. code-block:: nim
  953. iterator nonsense[T](x: seq[T]): float {.closure.} = ...
  954. AST:
  955. .. code-block:: nim
  956. nnkIteratorDef(
  957. nnkIdent("nonsense"),
  958. nnkEmpty(),
  959. ...
  960. )
  961. Converter declaration
  962. ---------------------
  963. A converter is similar to a proc.
  964. Concrete syntax:
  965. .. code-block:: nim
  966. converter toBool(x: float): bool
  967. AST:
  968. .. code-block:: nim
  969. nnkConverterDef(
  970. nnkIdent("toBool"),
  971. # ...
  972. )
  973. Template declaration
  974. --------------------
  975. Templates (as well as macros, as we'll see) have a slightly expanded AST when
  976. compared to procs and iterators. The reason for this is [term-rewriting
  977. macros](manual.html#term-rewriting-macros). Notice
  978. the ``nnkEmpty()`` as the second argument to ``nnkProcDef`` and
  979. ``nnkIteratorDef`` above? That's where the term-rewriting macros go.
  980. Concrete syntax:
  981. .. code-block:: nim
  982. template optOpt{expr1}(a: int): int
  983. AST:
  984. .. code-block:: nim
  985. nnkTemplateDef(
  986. nnkIdent("optOpt"),
  987. nnkStmtList( # instead of nnkEmpty()
  988. expr1
  989. ),
  990. # follows like a proc or iterator
  991. )
  992. If the template does not have types for its parameters, the type identifiers
  993. inside ``nnkFormalParams`` just becomes ``nnkEmpty``.
  994. Macro declaration
  995. -----------------
  996. Macros behave like templates, but ``nnkTemplateDef`` is replaced with
  997. ``nnkMacroDef``.
  998. Special node kinds
  999. ==================
  1000. There are several node kinds that are used for semantic checking or code
  1001. generation. These are accessible from this module, but should not be used.
  1002. Other node kinds are especially designed to make AST manipulations easier.
  1003. These are explained here.
  1004. To be written.