ast.nim 86 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301
  1. #
  2. #
  3. # The Nim Compiler
  4. # (c) Copyright 2015 Andreas Rumpf
  5. #
  6. # See the file "copying.txt", included in this
  7. # distribution, for details about the copyright.
  8. #
  9. # abstract syntax tree + symbol table
  10. import
  11. lineinfos, hashes, options, ropes, idents, int128, tables, wordrecg
  12. from strutils import toLowerAscii
  13. when defined(nimPreviewSlimSystem):
  14. import std/assertions
  15. export int128
  16. type
  17. TCallingConvention* = enum
  18. ccNimCall = "nimcall" # nimcall, also the default
  19. ccStdCall = "stdcall" # procedure is stdcall
  20. ccCDecl = "cdecl" # cdecl
  21. ccSafeCall = "safecall" # safecall
  22. ccSysCall = "syscall" # system call
  23. ccInline = "inline" # proc should be inlined
  24. ccNoInline = "noinline" # proc should not be inlined
  25. ccFastCall = "fastcall" # fastcall (pass parameters in registers)
  26. ccThisCall = "thiscall" # thiscall (parameters are pushed right-to-left)
  27. ccClosure = "closure" # proc has a closure
  28. ccNoConvention = "noconv" # needed for generating proper C procs sometimes
  29. type
  30. TNodeKind* = enum # order is extremely important, because ranges are used
  31. # to check whether a node belongs to a certain class
  32. nkNone, # unknown node kind: indicates an error
  33. # Expressions:
  34. # Atoms:
  35. nkEmpty, # the node is empty
  36. nkIdent, # node is an identifier
  37. nkSym, # node is a symbol
  38. nkType, # node is used for its typ field
  39. nkCharLit, # a character literal ''
  40. nkIntLit, # an integer literal
  41. nkInt8Lit,
  42. nkInt16Lit,
  43. nkInt32Lit,
  44. nkInt64Lit,
  45. nkUIntLit, # an unsigned integer literal
  46. nkUInt8Lit,
  47. nkUInt16Lit,
  48. nkUInt32Lit,
  49. nkUInt64Lit,
  50. nkFloatLit, # a floating point literal
  51. nkFloat32Lit,
  52. nkFloat64Lit,
  53. nkFloat128Lit,
  54. nkStrLit, # a string literal ""
  55. nkRStrLit, # a raw string literal r""
  56. nkTripleStrLit, # a triple string literal """
  57. nkNilLit, # the nil literal
  58. # end of atoms
  59. nkComesFrom, # "comes from" template/macro information for
  60. # better stack trace generation
  61. nkDotCall, # used to temporarily flag a nkCall node;
  62. # this is used
  63. # for transforming ``s.len`` to ``len(s)``
  64. nkCommand, # a call like ``p 2, 4`` without parenthesis
  65. nkCall, # a call like p(x, y) or an operation like +(a, b)
  66. nkCallStrLit, # a call with a string literal
  67. # x"abc" has two sons: nkIdent, nkRStrLit
  68. # x"""abc""" has two sons: nkIdent, nkTripleStrLit
  69. nkInfix, # a call like (a + b)
  70. nkPrefix, # a call like !a
  71. nkPostfix, # something like a! (also used for visibility)
  72. nkHiddenCallConv, # an implicit type conversion via a type converter
  73. nkExprEqExpr, # a named parameter with equals: ''expr = expr''
  74. nkExprColonExpr, # a named parameter with colon: ''expr: expr''
  75. nkIdentDefs, # a definition like `a, b: typeDesc = expr`
  76. # either typeDesc or expr may be nil; used in
  77. # formal parameters, var statements, etc.
  78. nkVarTuple, # a ``var (a, b) = expr`` construct
  79. nkPar, # syntactic (); may be a tuple constructor
  80. nkObjConstr, # object constructor: T(a: 1, b: 2)
  81. nkCurly, # syntactic {}
  82. nkCurlyExpr, # an expression like a{i}
  83. nkBracket, # syntactic []
  84. nkBracketExpr, # an expression like a[i..j, k]
  85. nkPragmaExpr, # an expression like a{.pragmas.}
  86. nkRange, # an expression like i..j
  87. nkDotExpr, # a.b
  88. nkCheckedFieldExpr, # a.b, but b is a field that needs to be checked
  89. nkDerefExpr, # a^
  90. nkIfExpr, # if as an expression
  91. nkElifExpr,
  92. nkElseExpr,
  93. nkLambda, # lambda expression
  94. nkDo, # lambda block appering as trailing proc param
  95. nkAccQuoted, # `a` as a node
  96. nkTableConstr, # a table constructor {expr: expr}
  97. nkBind, # ``bind expr`` node
  98. nkClosedSymChoice, # symbol choice node; a list of nkSyms (closed)
  99. nkOpenSymChoice, # symbol choice node; a list of nkSyms (open)
  100. nkHiddenStdConv, # an implicit standard type conversion
  101. nkHiddenSubConv, # an implicit type conversion from a subtype
  102. # to a supertype
  103. nkConv, # a type conversion
  104. nkCast, # a type cast
  105. nkStaticExpr, # a static expr
  106. nkAddr, # a addr expression
  107. nkHiddenAddr, # implicit address operator
  108. nkHiddenDeref, # implicit ^ operator
  109. nkObjDownConv, # down conversion between object types
  110. nkObjUpConv, # up conversion between object types
  111. nkChckRangeF, # range check for floats
  112. nkChckRange64, # range check for 64 bit ints
  113. nkChckRange, # range check for ints
  114. nkStringToCString, # string to cstring
  115. nkCStringToString, # cstring to string
  116. # end of expressions
  117. nkAsgn, # a = b
  118. nkFastAsgn, # internal node for a fast ``a = b``
  119. # (no string copy)
  120. nkGenericParams, # generic parameters
  121. nkFormalParams, # formal parameters
  122. nkOfInherit, # inherited from symbol
  123. nkImportAs, # a 'as' b in an import statement
  124. nkProcDef, # a proc
  125. nkMethodDef, # a method
  126. nkConverterDef, # a converter
  127. nkMacroDef, # a macro
  128. nkTemplateDef, # a template
  129. nkIteratorDef, # an iterator
  130. nkOfBranch, # used inside case statements
  131. # for (cond, action)-pairs
  132. nkElifBranch, # used in if statements
  133. nkExceptBranch, # an except section
  134. nkElse, # an else part
  135. nkAsmStmt, # an assembler block
  136. nkPragma, # a pragma statement
  137. nkPragmaBlock, # a pragma with a block
  138. nkIfStmt, # an if statement
  139. nkWhenStmt, # a when expression or statement
  140. nkForStmt, # a for statement
  141. nkParForStmt, # a parallel for statement
  142. nkWhileStmt, # a while statement
  143. nkCaseStmt, # a case statement
  144. nkTypeSection, # a type section (consists of type definitions)
  145. nkVarSection, # a var section
  146. nkLetSection, # a let section
  147. nkConstSection, # a const section
  148. nkConstDef, # a const definition
  149. nkTypeDef, # a type definition
  150. nkYieldStmt, # the yield statement as a tree
  151. nkDefer, # the 'defer' statement
  152. nkTryStmt, # a try statement
  153. nkFinally, # a finally section
  154. nkRaiseStmt, # a raise statement
  155. nkReturnStmt, # a return statement
  156. nkBreakStmt, # a break statement
  157. nkContinueStmt, # a continue statement
  158. nkBlockStmt, # a block statement
  159. nkStaticStmt, # a static statement
  160. nkDiscardStmt, # a discard statement
  161. nkStmtList, # a list of statements
  162. nkImportStmt, # an import statement
  163. nkImportExceptStmt, # an import x except a statement
  164. nkExportStmt, # an export statement
  165. nkExportExceptStmt, # an 'export except' statement
  166. nkFromStmt, # a from * import statement
  167. nkIncludeStmt, # an include statement
  168. nkBindStmt, # a bind statement
  169. nkMixinStmt, # a mixin statement
  170. nkUsingStmt, # an using statement
  171. nkCommentStmt, # a comment statement
  172. nkStmtListExpr, # a statement list followed by an expr; this is used
  173. # to allow powerful multi-line templates
  174. nkBlockExpr, # a statement block ending in an expr; this is used
  175. # to allow powerful multi-line templates that open a
  176. # temporary scope
  177. nkStmtListType, # a statement list ending in a type; for macros
  178. nkBlockType, # a statement block ending in a type; for macros
  179. # types as syntactic trees:
  180. nkWith, # distinct with `foo`
  181. nkWithout, # distinct without `foo`
  182. nkTypeOfExpr, # type(1+2)
  183. nkObjectTy, # object body
  184. nkTupleTy, # tuple body
  185. nkTupleClassTy, # tuple type class
  186. nkTypeClassTy, # user-defined type class
  187. nkStaticTy, # ``static[T]``
  188. nkRecList, # list of object parts
  189. nkRecCase, # case section of object
  190. nkRecWhen, # when section of object
  191. nkRefTy, # ``ref T``
  192. nkPtrTy, # ``ptr T``
  193. nkVarTy, # ``var T``
  194. nkConstTy, # ``const T``
  195. nkOutTy, # ``out T``
  196. nkDistinctTy, # distinct type
  197. nkProcTy, # proc type
  198. nkIteratorTy, # iterator type
  199. nkSinkAsgn, # '=sink(x, y)'
  200. nkEnumTy, # enum body
  201. nkEnumFieldDef, # `ident = expr` in an enumeration
  202. nkArgList, # argument list
  203. nkPattern, # a special pattern; used for matching
  204. nkHiddenTryStmt, # a hidden try statement
  205. nkClosure, # (prc, env)-pair (internally used for code gen)
  206. nkGotoState, # used for the state machine (for iterators)
  207. nkState, # give a label to a code section (for iterators)
  208. nkBreakState, # special break statement for easier code generation
  209. nkFuncDef, # a func
  210. nkTupleConstr # a tuple constructor
  211. nkError # erroneous AST node
  212. nkModuleRef # for .rod file support: A (moduleId, itemId) pair
  213. nkReplayAction # for .rod file support: A replay action
  214. nkNilRodNode # for .rod file support: a 'nil' PNode
  215. nkOpenSym # container for captured sym that can be overriden by local symbols
  216. TNodeKinds* = set[TNodeKind]
  217. type
  218. TSymFlag* = enum # 52 flags!
  219. sfUsed, # read access of sym (for warnings) or simply used
  220. sfExported, # symbol is exported from module
  221. sfFromGeneric, # symbol is instantiation of a generic; this is needed
  222. # for symbol file generation; such symbols should always
  223. # be written into the ROD file
  224. sfGlobal, # symbol is at global scope
  225. sfForward, # symbol is forward declared
  226. sfWasForwarded, # symbol had a forward declaration
  227. # (implies it's too dangerous to patch its type signature)
  228. sfImportc, # symbol is external; imported
  229. sfExportc, # symbol is exported (under a specified name)
  230. sfMangleCpp, # mangle as cpp (combines with `sfExportc`)
  231. sfVolatile, # variable is volatile
  232. sfRegister, # variable should be placed in a register
  233. sfPure, # object is "pure" that means it has no type-information
  234. # enum is "pure", its values need qualified access
  235. # variable is "pure"; it's an explicit "global"
  236. sfNoSideEffect, # proc has no side effects
  237. sfSideEffect, # proc may have side effects; cannot prove it has none
  238. sfMainModule, # module is the main module
  239. sfSystemModule, # module is the system module
  240. sfNoReturn, # proc never returns (an exit proc)
  241. sfAddrTaken, # the variable's address is taken (ex- or implicitly);
  242. # *OR*: a proc is indirectly called (used as first class)
  243. sfCompilerProc, # proc is a compiler proc, that is a C proc that is
  244. # needed for the code generator
  245. sfEscapes # param escapes
  246. # currently unimplemented
  247. sfDiscriminant, # field is a discriminant in a record/object
  248. sfRequiresInit, # field must be initialized during construction
  249. sfDeprecated, # symbol is deprecated
  250. sfExplain, # provide more diagnostics when this symbol is used
  251. sfError, # usage of symbol should trigger a compile-time error
  252. sfShadowed, # a symbol that was shadowed in some inner scope
  253. sfThread, # proc will run as a thread
  254. # variable is a thread variable
  255. sfCppNonPod, # tells compiler to treat such types as non-pod's, so that
  256. # `thread_local` is used instead of `__thread` for
  257. # {.threadvar.} + `--threads`. Only makes sense for importcpp types.
  258. # This has a performance impact so isn't set by default.
  259. sfCompileTime, # proc can be evaluated at compile time
  260. sfConstructor, # proc is a C++ constructor
  261. sfDispatcher, # copied method symbol is the dispatcher
  262. # deprecated and unused, except for the con
  263. sfBorrow, # proc is borrowed
  264. sfInfixCall, # symbol needs infix call syntax in target language;
  265. # for interfacing with C++, JS
  266. sfNamedParamCall, # symbol needs named parameter call syntax in target
  267. # language; for interfacing with Objective C
  268. sfDiscardable, # returned value may be discarded implicitly
  269. sfOverridden, # proc is overridden
  270. sfCallsite # A flag for template symbols to tell the
  271. # compiler it should use line information from
  272. # the calling side of the macro, not from the
  273. # implementation.
  274. sfGenSym # symbol is 'gensym'ed; do not add to symbol table
  275. sfNonReloadable # symbol will be left as-is when hot code reloading is on -
  276. # meaning that it won't be renamed and/or changed in any way
  277. sfGeneratedOp # proc is a generated '='; do not inject destructors in it
  278. # variable is generated closure environment; requires early
  279. # destruction for --newruntime.
  280. sfTemplateParam # symbol is a template parameter
  281. sfCursor # variable/field is a cursor, see RFC 177 for details
  282. sfInjectDestructors # whether the proc needs the 'injectdestructors' transformation
  283. sfNeverRaises # proc can never raise an exception, not even OverflowDefect
  284. # or out-of-memory
  285. sfSystemRaisesDefect # proc in the system can raise defects
  286. sfUsedInFinallyOrExcept # symbol is used inside an 'except' or 'finally'
  287. sfSingleUsedTemp # For temporaries that we know will only be used once
  288. sfNoalias # 'noalias' annotation, means C's 'restrict'
  289. # for templates and macros, means cannot be called
  290. # as a lone symbol (cannot use alias syntax)
  291. sfEffectsDelayed # an 'effectsDelayed' parameter
  292. sfGeneratedType # A anonymous generic type that is generated by the compiler for
  293. # objects that do not have generic parameters in case one of the
  294. # object fields has one.
  295. #
  296. # This is disallowed but can cause the typechecking to go into
  297. # an infinite loop, this flag is used as a sentinel to stop it.
  298. sfVirtual # proc is a C++ virtual function
  299. sfByCopy # param is marked as pass bycopy
  300. sfCodegenDecl # type, proc, global or proc param is marked as codegenDecl
  301. sfWasGenSym # symbol was 'gensym'ed
  302. TSymFlags* = set[TSymFlag]
  303. const
  304. sfNoInit* = sfMainModule # don't generate code to init the variable
  305. sfAllUntyped* = sfVolatile # macro or template is immediately expanded \
  306. # in a generic context
  307. sfDirty* = sfPure
  308. # template is not hygienic (old styled template)
  309. # module, compiled from a dirty-buffer
  310. sfAnon* = sfDiscardable
  311. # symbol name that was generated by the compiler
  312. # the compiler will avoid printing such names
  313. # in user messages.
  314. sfNoForward* = sfRegister
  315. # forward declarations are not required (per module)
  316. sfReorder* = sfForward
  317. # reordering pass is enabled
  318. sfCompileToCpp* = sfInfixCall # compile the module as C++ code
  319. sfCompileToObjc* = sfNamedParamCall # compile the module as Objective-C code
  320. sfExperimental* = sfOverridden # module uses the .experimental switch
  321. sfGoto* = sfOverridden # var is used for 'goto' code generation
  322. sfWrittenTo* = sfBorrow # param is assigned to
  323. # currently unimplemented
  324. sfBase* = sfDiscriminant
  325. sfCustomPragma* = sfRegister # symbol is custom pragma template
  326. sfTemplateRedefinition* = sfExportc # symbol is a redefinition of an earlier template
  327. const
  328. # getting ready for the future expr/stmt merge
  329. nkWhen* = nkWhenStmt
  330. nkWhenExpr* = nkWhenStmt
  331. nkEffectList* = nkArgList
  332. # hacks ahead: an nkEffectList is a node with 4 children:
  333. exceptionEffects* = 0 # exceptions at position 0
  334. requiresEffects* = 1 # 'requires' annotation
  335. ensuresEffects* = 2 # 'ensures' annotation
  336. tagEffects* = 3 # user defined tag ('gc', 'time' etc.)
  337. pragmasEffects* = 4 # not an effect, but a slot for pragmas in proc type
  338. forbiddenEffects* = 5 # list of illegal effects
  339. effectListLen* = 6 # list of effects list
  340. nkLastBlockStmts* = {nkRaiseStmt, nkReturnStmt, nkBreakStmt, nkContinueStmt}
  341. # these must be last statements in a block
  342. type
  343. TTypeKind* = enum # order is important!
  344. # Don't forget to change hti.nim if you make a change here
  345. # XXX put this into an include file to avoid this issue!
  346. # several types are no longer used (guess which), but a
  347. # spot in the sequence is kept for backwards compatibility
  348. # (apparently something with bootstrapping)
  349. # if you need to add a type, they can apparently be reused
  350. tyNone, tyBool, tyChar,
  351. tyEmpty, tyAlias, tyNil, tyUntyped, tyTyped, tyTypeDesc,
  352. tyGenericInvocation, # ``T[a, b]`` for types to invoke
  353. tyGenericBody, # ``T[a, b, body]`` last parameter is the body
  354. tyGenericInst, # ``T[a, b, realInstance]`` instantiated generic type
  355. # realInstance will be a concrete type like tyObject
  356. # unless this is an instance of a generic alias type.
  357. # then realInstance will be the tyGenericInst of the
  358. # completely (recursively) resolved alias.
  359. tyGenericParam, # ``a`` in the above patterns
  360. tyDistinct,
  361. tyEnum,
  362. tyOrdinal, # integer types (including enums and boolean)
  363. tyArray,
  364. tyObject,
  365. tyTuple,
  366. tySet,
  367. tyRange,
  368. tyPtr, tyRef,
  369. tyVar,
  370. tySequence,
  371. tyProc,
  372. tyPointer, tyOpenArray,
  373. tyString, tyCstring, tyForward,
  374. tyInt, tyInt8, tyInt16, tyInt32, tyInt64, # signed integers
  375. tyFloat, tyFloat32, tyFloat64, tyFloat128,
  376. tyUInt, tyUInt8, tyUInt16, tyUInt32, tyUInt64,
  377. tyOwned, tySink, tyLent,
  378. tyVarargs,
  379. tyUncheckedArray
  380. # An array with boundaries [0,+∞]
  381. tyProxy # used as errornous type (for idetools)
  382. tyBuiltInTypeClass
  383. # Type such as the catch-all object, tuple, seq, etc
  384. tyUserTypeClass
  385. # the body of a user-defined type class
  386. tyUserTypeClassInst
  387. # Instance of a parametric user-defined type class.
  388. # Structured similarly to tyGenericInst.
  389. # tyGenericInst represents concrete types, while
  390. # this is still a "generic param" that will bind types
  391. # and resolves them during sigmatch and instantiation.
  392. tyCompositeTypeClass
  393. # Type such as seq[Number]
  394. # The notes for tyUserTypeClassInst apply here as well
  395. # sons[0]: the original expression used by the user.
  396. # sons[1]: fully expanded and instantiated meta type
  397. # (potentially following aliases)
  398. tyInferred
  399. # In the initial state `base` stores a type class constraining
  400. # the types that can be inferred. After a candidate type is
  401. # selected, it's stored in `lastSon`. Between `base` and `lastSon`
  402. # there may be 0, 2 or more types that were also considered as
  403. # possible candidates in the inference process (i.e. lastSon will
  404. # be updated to store a type best conforming to all candidates)
  405. tyAnd, tyOr, tyNot
  406. # boolean type classes such as `string|int`,`not seq`,
  407. # `Sortable and Enumable`, etc
  408. tyAnything
  409. # a type class matching any type
  410. tyStatic
  411. # a value known at compile type (the underlying type is .base)
  412. tyFromExpr
  413. # This is a type representing an expression that depends
  414. # on generic parameters (the expression is stored in t.n)
  415. # It will be converted to a real type only during generic
  416. # instantiation and prior to this it has the potential to
  417. # be any type.
  418. tyConcept
  419. # new style concept.
  420. tyVoid
  421. # now different from tyEmpty, hurray!
  422. tyIterable
  423. static:
  424. # remind us when TTypeKind stops to fit in a single 64-bit word
  425. # assert TTypeKind.high.ord <= 63
  426. discard
  427. const
  428. tyPureObject* = tyTuple
  429. GcTypeKinds* = {tyRef, tySequence, tyString}
  430. tyError* = tyProxy # as an errornous node should match everything
  431. tyUnknown* = tyFromExpr
  432. tyUnknownTypes* = {tyError, tyFromExpr}
  433. tyTypeClasses* = {tyBuiltInTypeClass, tyCompositeTypeClass,
  434. tyUserTypeClass, tyUserTypeClassInst,
  435. tyAnd, tyOr, tyNot, tyAnything}
  436. tyMetaTypes* = {tyGenericParam, tyTypeDesc, tyUntyped} + tyTypeClasses
  437. tyUserTypeClasses* = {tyUserTypeClass, tyUserTypeClassInst}
  438. # consider renaming as `tyAbstractVarRange`
  439. abstractVarRange* = {tyGenericInst, tyRange, tyVar, tyDistinct, tyOrdinal,
  440. tyTypeDesc, tyAlias, tyInferred, tySink, tyOwned}
  441. abstractInst* = {tyGenericInst, tyDistinct, tyOrdinal, tyTypeDesc, tyAlias,
  442. tyInferred, tySink, tyOwned} # xxx what about tyStatic?
  443. type
  444. TTypeKinds* = set[TTypeKind]
  445. TNodeFlag* = enum
  446. nfNone,
  447. nfBase2, # nfBase10 is default, so not needed
  448. nfBase8,
  449. nfBase16,
  450. nfAllConst, # used to mark complex expressions constant; easy to get rid of
  451. # but unfortunately it has measurable impact for compilation
  452. # efficiency
  453. nfTransf, # node has been transformed
  454. nfNoRewrite # node should not be transformed anymore
  455. nfSem # node has been checked for semantics
  456. nfLL # node has gone through lambda lifting
  457. nfDotField # the call can use a dot operator
  458. nfDotSetter # the call can use a setter dot operarator
  459. nfExplicitCall # x.y() was used instead of x.y
  460. nfExprCall # this is an attempt to call a regular expression
  461. nfIsRef # this node is a 'ref' node; used for the VM
  462. nfIsPtr # this node is a 'ptr' node; used for the VM
  463. nfPreventCg # this node should be ignored by the codegen
  464. nfBlockArg # this a stmtlist appearing in a call (e.g. a do block)
  465. nfFromTemplate # a top-level node returned from a template
  466. nfDefaultParam # an automatically inserter default parameter
  467. nfDefaultRefsParam # a default param value references another parameter
  468. # the flag is applied to proc default values and to calls
  469. nfExecuteOnReload # A top-level statement that will be executed during reloads
  470. nfLastRead # this node is a last read
  471. nfFirstWrite # this node is a first write
  472. nfHasComment # node has a comment
  473. nfSkipFieldChecking # node skips field visable checking
  474. nfDisabledOpenSym # temporary: node should be nkOpenSym but cannot
  475. # because openSym experimental switch is disabled
  476. # gives warning instead
  477. TNodeFlags* = set[TNodeFlag]
  478. TTypeFlag* = enum # keep below 32 for efficiency reasons (now: 47)
  479. tfVarargs, # procedure has C styled varargs
  480. # tyArray type represeting a varargs list
  481. tfNoSideEffect, # procedure type does not allow side effects
  482. tfFinal, # is the object final?
  483. tfInheritable, # is the object inheritable?
  484. tfHasOwned, # type contains an 'owned' type and must be moved
  485. tfEnumHasHoles, # enum cannot be mapped into a range
  486. tfShallow, # type can be shallow copied on assignment
  487. tfThread, # proc type is marked as ``thread``; alias for ``gcsafe``
  488. tfFromGeneric, # type is an instantiation of a generic; this is needed
  489. # because for instantiations of objects, structural
  490. # type equality has to be used
  491. tfUnresolved, # marks unresolved typedesc/static params: e.g.
  492. # proc foo(T: typedesc, list: seq[T]): var T
  493. # proc foo(L: static[int]): array[L, int]
  494. # can be attached to ranges to indicate that the range
  495. # can be attached to generic procs with free standing
  496. # type parameters: e.g. proc foo[T]()
  497. # depends on unresolved static params.
  498. tfResolved # marks a user type class, after it has been bound to a
  499. # concrete type (lastSon becomes the concrete type)
  500. tfRetType, # marks return types in proc (used to detect type classes
  501. # used as return types for return type inference)
  502. tfCapturesEnv, # whether proc really captures some environment
  503. tfByCopy, # pass object/tuple by copy (C backend)
  504. tfByRef, # pass object/tuple by reference (C backend)
  505. tfIterator, # type is really an iterator, not a tyProc
  506. tfPartial, # type is declared as 'partial'
  507. tfNotNil, # type cannot be 'nil'
  508. tfRequiresInit, # type constains a "not nil" constraint somewhere or
  509. # a `requiresInit` field, so the default zero init
  510. # is not appropriate
  511. tfNeedsFullInit, # object type marked with {.requiresInit.}
  512. # all fields must be initialized
  513. tfVarIsPtr, # 'var' type is translated like 'ptr' even in C++ mode
  514. tfHasMeta, # type contains "wildcard" sub-types such as generic params
  515. # or other type classes
  516. tfHasGCedMem, # type contains GC'ed memory
  517. tfPacked
  518. tfHasStatic
  519. tfGenericTypeParam
  520. tfImplicitTypeParam
  521. tfInferrableStatic
  522. tfConceptMatchedTypeSym
  523. tfExplicit # for typedescs, marks types explicitly prefixed with the
  524. # `type` operator (e.g. type int)
  525. tfWildcard # consider a proc like foo[T, I](x: Type[T, I])
  526. # T and I here can bind to both typedesc and static types
  527. # before this is determined, we'll consider them to be a
  528. # wildcard type.
  529. tfHasAsgn # type has overloaded assignment operator
  530. tfBorrowDot # distinct type borrows '.'
  531. tfTriggersCompileTime # uses the NimNode type which make the proc
  532. # implicitly '.compiletime'
  533. tfRefsAnonObj # used for 'ref object' and 'ptr object'
  534. tfCovariant # covariant generic param mimicking a ptr type
  535. tfWeakCovariant # covariant generic param mimicking a seq/array type
  536. tfContravariant # contravariant generic param
  537. tfCheckedForDestructor # type was checked for having a destructor.
  538. # If it has one, t.destructor is not nil.
  539. tfAcyclic # object type was annotated as .acyclic
  540. tfIncompleteStruct # treat this type as if it had sizeof(pointer)
  541. tfCompleteStruct
  542. # (for importc types); type is fully specified, allowing to compute
  543. # sizeof, alignof, offsetof at CT
  544. tfExplicitCallConv
  545. tfIsConstructor
  546. tfEffectSystemWorkaround
  547. tfIsOutParam
  548. tfSendable
  549. tfImplicitStatic
  550. TTypeFlags* = set[TTypeFlag]
  551. TSymKind* = enum # the different symbols (start with the prefix sk);
  552. # order is important for the documentation generator!
  553. skUnknown, # unknown symbol: used for parsing assembler blocks
  554. # and first phase symbol lookup in generics
  555. skConditional, # symbol for the preprocessor (may become obsolete)
  556. skDynLib, # symbol represents a dynamic library; this is used
  557. # internally; it does not exist in Nim code
  558. skParam, # a parameter
  559. skGenericParam, # a generic parameter; eq in ``proc x[eq=`==`]()``
  560. skTemp, # a temporary variable (introduced by compiler)
  561. skModule, # module identifier
  562. skType, # a type
  563. skVar, # a variable
  564. skLet, # a 'let' symbol
  565. skConst, # a constant
  566. skResult, # special 'result' variable
  567. skProc, # a proc
  568. skFunc, # a func
  569. skMethod, # a method
  570. skIterator, # an iterator
  571. skConverter, # a type converter
  572. skMacro, # a macro
  573. skTemplate, # a template; currently also misused for user-defined
  574. # pragmas
  575. skField, # a field in a record or object
  576. skEnumField, # an identifier in an enum
  577. skForVar, # a for loop variable
  578. skLabel, # a label (for block statement)
  579. skStub, # symbol is a stub and not yet loaded from the ROD
  580. # file (it is loaded on demand, which may
  581. # mean: never)
  582. skPackage, # symbol is a package (used for canonicalization)
  583. TSymKinds* = set[TSymKind]
  584. const
  585. routineKinds* = {skProc, skFunc, skMethod, skIterator,
  586. skConverter, skMacro, skTemplate}
  587. ExportableSymKinds* = {skVar, skLet, skConst, skType, skEnumField, skStub} + routineKinds
  588. tfUnion* = tfNoSideEffect
  589. tfGcSafe* = tfThread
  590. tfObjHasKids* = tfEnumHasHoles
  591. tfReturnsNew* = tfInheritable
  592. skError* = skUnknown
  593. var
  594. eqTypeFlags* = {tfIterator, tfNotNil, tfVarIsPtr, tfGcSafe, tfNoSideEffect, tfIsOutParam}
  595. ## type flags that are essential for type equality.
  596. ## This is now a variable because for emulation of version:1.0 we
  597. ## might exclude {tfGcSafe, tfNoSideEffect}.
  598. type
  599. TMagic* = enum # symbols that require compiler magic:
  600. mNone,
  601. mDefined, mDeclared, mDeclaredInScope, mCompiles, mArrGet, mArrPut, mAsgn,
  602. mLow, mHigh, mSizeOf, mAlignOf, mOffsetOf, mTypeTrait,
  603. mIs, mOf, mAddr, mType, mTypeOf,
  604. mPlugin, mEcho, mShallowCopy, mSlurp, mStaticExec, mStatic,
  605. mParseExprToAst, mParseStmtToAst, mExpandToAst, mQuoteAst,
  606. mInc, mDec, mOrd,
  607. mNew, mNewFinalize, mNewSeq, mNewSeqOfCap,
  608. mLengthOpenArray, mLengthStr, mLengthArray, mLengthSeq,
  609. mIncl, mExcl, mCard, mChr,
  610. mGCref, mGCunref,
  611. mAddI, mSubI, mMulI, mDivI, mModI,
  612. mSucc, mPred,
  613. mAddF64, mSubF64, mMulF64, mDivF64,
  614. mShrI, mShlI, mAshrI, mBitandI, mBitorI, mBitxorI,
  615. mMinI, mMaxI,
  616. mAddU, mSubU, mMulU, mDivU, mModU,
  617. mEqI, mLeI, mLtI,
  618. mEqF64, mLeF64, mLtF64,
  619. mLeU, mLtU,
  620. mEqEnum, mLeEnum, mLtEnum,
  621. mEqCh, mLeCh, mLtCh,
  622. mEqB, mLeB, mLtB,
  623. mEqRef, mLePtr, mLtPtr,
  624. mXor, mEqCString, mEqProc,
  625. mUnaryMinusI, mUnaryMinusI64, mAbsI, mNot,
  626. mUnaryPlusI, mBitnotI,
  627. mUnaryPlusF64, mUnaryMinusF64,
  628. mCharToStr, mBoolToStr,
  629. mIntToStr, mInt64ToStr, mFloatToStr, # for compiling nimStdlibVersion < 1.5.1 (not bootstrapping)
  630. mCStrToStr,
  631. mStrToStr, mEnumToStr,
  632. mAnd, mOr,
  633. mImplies, mIff, mExists, mForall, mOld,
  634. mEqStr, mLeStr, mLtStr,
  635. mEqSet, mLeSet, mLtSet, mMulSet, mPlusSet, mMinusSet,
  636. mConStrStr, mSlice,
  637. mDotDot, # this one is only necessary to give nice compile time warnings
  638. mFields, mFieldPairs, mOmpParFor,
  639. mAppendStrCh, mAppendStrStr, mAppendSeqElem,
  640. mInSet, mRepr, mExit,
  641. mSetLengthStr, mSetLengthSeq,
  642. mIsPartOf, mAstToStr, mParallel,
  643. mSwap, mIsNil, mArrToSeq, mOpenArrayToSeq,
  644. mNewString, mNewStringOfCap, mParseBiggestFloat,
  645. mMove, mEnsureMove, mWasMoved, mDup, mDestroy, mTrace,
  646. mDefault, mUnown, mFinished, mIsolate, mAccessEnv, mAccessTypeField, mReset,
  647. mArray, mOpenArray, mRange, mSet, mSeq, mVarargs,
  648. mRef, mPtr, mVar, mDistinct, mVoid, mTuple,
  649. mOrdinal, mIterableType,
  650. mInt, mInt8, mInt16, mInt32, mInt64,
  651. mUInt, mUInt8, mUInt16, mUInt32, mUInt64,
  652. mFloat, mFloat32, mFloat64, mFloat128,
  653. mBool, mChar, mString, mCstring,
  654. mPointer, mNil, mExpr, mStmt, mTypeDesc,
  655. mVoidType, mPNimrodNode, mSpawn, mDeepCopy,
  656. mIsMainModule, mCompileDate, mCompileTime, mProcCall,
  657. mCpuEndian, mHostOS, mHostCPU, mBuildOS, mBuildCPU, mAppType,
  658. mCompileOption, mCompileOptionArg,
  659. mNLen, mNChild, mNSetChild, mNAdd, mNAddMultiple, mNDel,
  660. mNKind, mNSymKind,
  661. mNccValue, mNccInc, mNcsAdd, mNcsIncl, mNcsLen, mNcsAt,
  662. mNctPut, mNctLen, mNctGet, mNctHasNext, mNctNext,
  663. mNIntVal, mNFloatVal, mNSymbol, mNIdent, mNGetType, mNStrVal, mNSetIntVal,
  664. mNSetFloatVal, mNSetSymbol, mNSetIdent, mNSetStrVal, mNLineInfo,
  665. mNNewNimNode, mNCopyNimNode, mNCopyNimTree, mStrToIdent, mNSigHash, mNSizeOf,
  666. mNBindSym, mNCallSite,
  667. mEqIdent, mEqNimrodNode, mSameNodeType, mGetImpl, mNGenSym,
  668. mNHint, mNWarning, mNError,
  669. mInstantiationInfo, mGetTypeInfo, mGetTypeInfoV2,
  670. mNimvm, mIntDefine, mStrDefine, mBoolDefine, mGenericDefine, mRunnableExamples,
  671. mException, mBuiltinType, mSymOwner, mUncheckedArray, mGetImplTransf,
  672. mSymIsInstantiationOf, mNodeId, mPrivateAccess, mZeroDefault
  673. const
  674. # things that we can evaluate safely at compile time, even if not asked for it:
  675. ctfeWhitelist* = {mNone, mSucc,
  676. mPred, mInc, mDec, mOrd, mLengthOpenArray,
  677. mLengthStr, mLengthArray, mLengthSeq,
  678. mArrGet, mArrPut, mAsgn, mDestroy,
  679. mIncl, mExcl, mCard, mChr,
  680. mAddI, mSubI, mMulI, mDivI, mModI,
  681. mAddF64, mSubF64, mMulF64, mDivF64,
  682. mShrI, mShlI, mBitandI, mBitorI, mBitxorI,
  683. mMinI, mMaxI,
  684. mAddU, mSubU, mMulU, mDivU, mModU,
  685. mEqI, mLeI, mLtI,
  686. mEqF64, mLeF64, mLtF64,
  687. mLeU, mLtU,
  688. mEqEnum, mLeEnum, mLtEnum,
  689. mEqCh, mLeCh, mLtCh,
  690. mEqB, mLeB, mLtB,
  691. mEqRef, mEqProc, mLePtr, mLtPtr, mEqCString, mXor,
  692. mUnaryMinusI, mUnaryMinusI64, mAbsI, mNot, mUnaryPlusI, mBitnotI,
  693. mUnaryPlusF64, mUnaryMinusF64,
  694. mCharToStr, mBoolToStr,
  695. mIntToStr, mInt64ToStr, mFloatToStr,
  696. mCStrToStr,
  697. mStrToStr, mEnumToStr,
  698. mAnd, mOr,
  699. mEqStr, mLeStr, mLtStr,
  700. mEqSet, mLeSet, mLtSet, mMulSet, mPlusSet, mMinusSet,
  701. mConStrStr, mAppendStrCh, mAppendStrStr, mAppendSeqElem,
  702. mInSet, mRepr, mOpenArrayToSeq}
  703. generatedMagics* = {mNone, mIsolate, mFinished, mOpenArrayToSeq}
  704. ## magics that are generated as normal procs in the backend
  705. type
  706. ItemId* = object
  707. module*: int32
  708. item*: int32
  709. proc `$`*(x: ItemId): string =
  710. "(module: " & $x.module & ", item: " & $x.item & ")"
  711. proc `==`*(a, b: ItemId): bool {.inline.} =
  712. a.item == b.item and a.module == b.module
  713. proc hash*(x: ItemId): Hash =
  714. var h: Hash = hash(x.module)
  715. h = h !& hash(x.item)
  716. result = !$h
  717. type
  718. TIdObj* {.acyclic.} = object of RootObj
  719. itemId*: ItemId
  720. PIdObj* = ref TIdObj
  721. PNode* = ref TNode
  722. TNodeSeq* = seq[PNode]
  723. PType* = ref TType
  724. PSym* = ref TSym
  725. TNode*{.final, acyclic.} = object # on a 32bit machine, this takes 32 bytes
  726. when defined(useNodeIds):
  727. id*: int
  728. typ*: PType
  729. info*: TLineInfo
  730. flags*: TNodeFlags
  731. case kind*: TNodeKind
  732. of nkCharLit..nkUInt64Lit:
  733. intVal*: BiggestInt
  734. of nkFloatLit..nkFloat128Lit:
  735. floatVal*: BiggestFloat
  736. of nkStrLit..nkTripleStrLit:
  737. strVal*: string
  738. of nkSym:
  739. sym*: PSym
  740. of nkIdent:
  741. ident*: PIdent
  742. else:
  743. sons*: TNodeSeq
  744. when defined(nimsuggest):
  745. endInfo*: TLineInfo
  746. TStrTable* = object # a table[PIdent] of PSym
  747. counter*: int
  748. data*: seq[PSym]
  749. # -------------- backend information -------------------------------
  750. TLocKind* = enum
  751. locNone, # no location
  752. locTemp, # temporary location
  753. locLocalVar, # location is a local variable
  754. locGlobalVar, # location is a global variable
  755. locParam, # location is a parameter
  756. locField, # location is a record field
  757. locExpr, # "location" is really an expression
  758. locProc, # location is a proc (an address of a procedure)
  759. locData, # location is a constant
  760. locCall, # location is a call expression
  761. locOther # location is something other
  762. TLocFlag* = enum
  763. lfIndirect, # backend introduced a pointer
  764. lfNoDeepCopy, # no need for a deep copy
  765. lfNoDecl, # do not declare it in C
  766. lfDynamicLib, # link symbol to dynamic library
  767. lfExportLib, # export symbol for dynamic library generation
  768. lfHeader, # include header file for symbol
  769. lfImportCompilerProc, # ``importc`` of a compilerproc
  770. lfSingleUse # no location yet and will only be used once
  771. lfEnforceDeref # a copyMem is required to dereference if this a
  772. # ptr array due to C array limitations.
  773. # See #1181, #6422, #11171
  774. lfPrepareForMutation # string location is about to be mutated (V2)
  775. TStorageLoc* = enum
  776. OnUnknown, # location is unknown (stack, heap or static)
  777. OnStatic, # in a static section
  778. OnStack, # location is on hardware stack
  779. OnHeap # location is on heap or global
  780. # (reference counting needed)
  781. TLocFlags* = set[TLocFlag]
  782. TLoc* = object
  783. k*: TLocKind # kind of location
  784. storage*: TStorageLoc
  785. flags*: TLocFlags # location's flags
  786. lode*: PNode # Node where the location came from; can be faked
  787. r*: Rope # rope value of location (code generators)
  788. # ---------------- end of backend information ------------------------------
  789. TLibKind* = enum
  790. libHeader, libDynamic
  791. TLib* = object # also misused for headers!
  792. # keep in sync with PackedLib
  793. kind*: TLibKind
  794. generated*: bool # needed for the backends:
  795. isOverridden*: bool
  796. name*: Rope
  797. path*: PNode # can be a string literal!
  798. CompilesId* = int ## id that is used for the caching logic within
  799. ## ``system.compiles``. See the seminst module.
  800. TInstantiation* = object
  801. sym*: PSym
  802. concreteTypes*: seq[PType]
  803. compilesId*: CompilesId
  804. PInstantiation* = ref TInstantiation
  805. TScope* {.acyclic.} = object
  806. depthLevel*: int
  807. symbols*: TStrTable
  808. parent*: PScope
  809. allowPrivateAccess*: seq[PSym] # # enable access to private fields
  810. PScope* = ref TScope
  811. PLib* = ref TLib
  812. TSym* {.acyclic.} = object of TIdObj # Keep in sync with PackedSym
  813. # proc and type instantiations are cached in the generic symbol
  814. case kind*: TSymKind
  815. of routineKinds:
  816. #procInstCache*: seq[PInstantiation]
  817. gcUnsafetyReason*: PSym # for better error messages regarding gcsafe
  818. transformedBody*: PNode # cached body after transf pass
  819. of skLet, skVar, skField, skForVar:
  820. guard*: PSym
  821. bitsize*: int
  822. alignment*: int # for alignment
  823. else: nil
  824. magic*: TMagic
  825. typ*: PType
  826. name*: PIdent
  827. info*: TLineInfo
  828. when defined(nimsuggest):
  829. endInfo*: TLineInfo
  830. hasUserSpecifiedType*: bool # used for determining whether to display inlay type hints
  831. owner*: PSym
  832. flags*: TSymFlags
  833. ast*: PNode # syntax tree of proc, iterator, etc.:
  834. # the whole proc including header; this is used
  835. # for easy generation of proper error messages
  836. # for variant record fields the discriminant
  837. # expression
  838. # for modules, it's a placeholder for compiler
  839. # generated code that will be appended to the
  840. # module after the sem pass (see appendToModule)
  841. options*: TOptions
  842. position*: int # used for many different things:
  843. # for enum fields its position;
  844. # for fields its offset
  845. # for parameters its position (starting with 0)
  846. # for a conditional:
  847. # 1 iff the symbol is defined, else 0
  848. # (or not in symbol table)
  849. # for modules, an unique index corresponding
  850. # to the module's fileIdx
  851. # for variables a slot index for the evaluator
  852. offset*: int32 # offset of record field
  853. disamb*: int32 # disambiguation number; the basic idea is that
  854. # `<procname>__<module>_<disamb>`
  855. loc*: TLoc
  856. annex*: PLib # additional fields (seldom used, so we use a
  857. # reference to another object to save space)
  858. when hasFFI:
  859. cname*: string # resolved C declaration name in importc decl, e.g.:
  860. # proc fun() {.importc: "$1aux".} => cname = funaux
  861. constraint*: PNode # additional constraints like 'lit|result'; also
  862. # misused for the codegenDecl and virtual pragmas in the hope
  863. # it won't cause problems
  864. # for skModule the string literal to output for
  865. # deprecated modules.
  866. when defined(nimsuggest):
  867. allUsages*: seq[TLineInfo]
  868. TTypeSeq* = seq[PType]
  869. TTypeAttachedOp* = enum ## as usual, order is important here
  870. attachedWasMoved,
  871. attachedDestructor,
  872. attachedAsgn,
  873. attachedDup,
  874. attachedSink,
  875. attachedTrace,
  876. attachedDeepCopy
  877. TType* {.acyclic.} = object of TIdObj # \
  878. # types are identical iff they have the
  879. # same id; there may be multiple copies of a type
  880. # in memory!
  881. # Keep in sync with PackedType
  882. kind*: TTypeKind # kind of type
  883. callConv*: TCallingConvention # for procs
  884. flags*: TTypeFlags # flags of the type
  885. sons*: TTypeSeq # base types, etc.
  886. n*: PNode # node for types:
  887. # for range types a nkRange node
  888. # for record types a nkRecord node
  889. # for enum types a list of symbols
  890. # if kind == tyInt: it is an 'int literal(x)' type
  891. # for procs and tyGenericBody, it's the
  892. # formal param list
  893. # for concepts, the concept body
  894. # else: unused
  895. owner*: PSym # the 'owner' of the type
  896. sym*: PSym # types have the sym associated with them
  897. # it is used for converting types to strings
  898. size*: BiggestInt # the size of the type in bytes
  899. # -1 means that the size is unkwown
  900. align*: int16 # the type's alignment requirements
  901. paddingAtEnd*: int16 #
  902. loc*: TLoc
  903. typeInst*: PType # for generic instantiations the tyGenericInst that led to this
  904. # type.
  905. uniqueId*: ItemId # due to a design mistake, we need to keep the real ID here as it
  906. # is required by the --incremental:on mode.
  907. TPair* = object
  908. key*, val*: RootRef
  909. TPairSeq* = seq[TPair]
  910. TIdPair* = object
  911. key*: PIdObj
  912. val*: RootRef
  913. TIdPairSeq* = seq[TIdPair]
  914. TIdTable* = object # the same as table[PIdent] of PObject
  915. counter*: int
  916. data*: TIdPairSeq
  917. TIdNodePair* = object
  918. key*: PIdObj
  919. val*: PNode
  920. TIdNodePairSeq* = seq[TIdNodePair]
  921. TIdNodeTable* = object # the same as table[PIdObj] of PNode
  922. counter*: int
  923. data*: TIdNodePairSeq
  924. TNodePair* = object
  925. h*: Hash # because it is expensive to compute!
  926. key*: PNode
  927. val*: int
  928. TNodePairSeq* = seq[TNodePair]
  929. TNodeTable* = object # the same as table[PNode] of int;
  930. # nodes are compared by structure!
  931. counter*: int
  932. data*: TNodePairSeq
  933. TObjectSeq* = seq[RootRef]
  934. TObjectSet* = object
  935. counter*: int
  936. data*: TObjectSeq
  937. TImplication* = enum
  938. impUnknown, impNo, impYes
  939. template nodeId(n: PNode): int = cast[int](n)
  940. type Gconfig = object
  941. # we put comments in a side channel to avoid increasing `sizeof(TNode)`, which
  942. # reduces memory usage given that `PNode` is the most allocated type by far.
  943. comments: Table[int, string] # nodeId => comment
  944. useIc*: bool
  945. var gconfig {.threadvar.}: Gconfig
  946. proc setUseIc*(useIc: bool) = gconfig.useIc = useIc
  947. proc comment*(n: PNode): string =
  948. if nfHasComment in n.flags and not gconfig.useIc:
  949. # IC doesn't track comments, see `packed_ast`, so this could fail
  950. result = gconfig.comments[n.nodeId]
  951. proc `comment=`*(n: PNode, a: string) =
  952. let id = n.nodeId
  953. if a.len > 0:
  954. # if needed, we could periodically cleanup gconfig.comments when its size increases,
  955. # to ensure only live nodes (and with nfHasComment) have an entry in gconfig.comments;
  956. # for compiling compiler, the waste is very small:
  957. # num calls to newNodeImpl: 14984160 (num of PNode allocations)
  958. # size of gconfig.comments: 33585
  959. # num of nodes with comments that were deleted and hence wasted: 3081
  960. n.flags.incl nfHasComment
  961. gconfig.comments[id] = a
  962. elif nfHasComment in n.flags:
  963. n.flags.excl nfHasComment
  964. gconfig.comments.del(id)
  965. # BUGFIX: a module is overloadable so that a proc can have the
  966. # same name as an imported module. This is necessary because of
  967. # the poor naming choices in the standard library.
  968. const
  969. OverloadableSyms* = {skProc, skFunc, skMethod, skIterator,
  970. skConverter, skModule, skTemplate, skMacro, skEnumField}
  971. GenericTypes*: TTypeKinds = {tyGenericInvocation, tyGenericBody,
  972. tyGenericParam}
  973. StructuralEquivTypes*: TTypeKinds = {tyNil, tyTuple, tyArray,
  974. tySet, tyRange, tyPtr, tyRef, tyVar, tyLent, tySequence, tyProc, tyOpenArray,
  975. tyVarargs}
  976. ConcreteTypes*: TTypeKinds = { # types of the expr that may occur in::
  977. # var x = expr
  978. tyBool, tyChar, tyEnum, tyArray, tyObject,
  979. tySet, tyTuple, tyRange, tyPtr, tyRef, tyVar, tyLent, tySequence, tyProc,
  980. tyPointer,
  981. tyOpenArray, tyString, tyCstring, tyInt..tyInt64, tyFloat..tyFloat128,
  982. tyUInt..tyUInt64}
  983. IntegralTypes* = {tyBool, tyChar, tyEnum, tyInt..tyInt64,
  984. tyFloat..tyFloat128, tyUInt..tyUInt64} # weird name because it contains tyFloat
  985. ConstantDataTypes*: TTypeKinds = {tyArray, tySet,
  986. tyTuple, tySequence}
  987. NilableTypes*: TTypeKinds = {tyPointer, tyCstring, tyRef, tyPtr,
  988. tyProc, tyError} # TODO
  989. PtrLikeKinds*: TTypeKinds = {tyPointer, tyPtr} # for VM
  990. PersistentNodeFlags*: TNodeFlags = {nfBase2, nfBase8, nfBase16,
  991. nfDotSetter, nfDotField,
  992. nfIsRef, nfIsPtr, nfPreventCg, nfLL,
  993. nfFromTemplate, nfDefaultRefsParam,
  994. nfExecuteOnReload, nfLastRead,
  995. nfFirstWrite, nfSkipFieldChecking,
  996. nfDisabledOpenSym}
  997. namePos* = 0
  998. patternPos* = 1 # empty except for term rewriting macros
  999. genericParamsPos* = 2
  1000. paramsPos* = 3
  1001. pragmasPos* = 4
  1002. miscPos* = 5 # used for undocumented and hacky stuff
  1003. bodyPos* = 6 # position of body; use rodread.getBody() instead!
  1004. resultPos* = 7
  1005. dispatcherPos* = 8
  1006. nfAllFieldsSet* = nfBase2
  1007. nkCallKinds* = {nkCall, nkInfix, nkPrefix, nkPostfix,
  1008. nkCommand, nkCallStrLit, nkHiddenCallConv}
  1009. nkIdentKinds* = {nkIdent, nkSym, nkAccQuoted, nkOpenSymChoice,
  1010. nkClosedSymChoice, nkOpenSym}
  1011. nkPragmaCallKinds* = {nkExprColonExpr, nkCall, nkCallStrLit}
  1012. nkLiterals* = {nkCharLit..nkTripleStrLit}
  1013. nkFloatLiterals* = {nkFloatLit..nkFloat128Lit}
  1014. nkLambdaKinds* = {nkLambda, nkDo}
  1015. declarativeDefs* = {nkProcDef, nkFuncDef, nkMethodDef, nkIteratorDef, nkConverterDef}
  1016. routineDefs* = declarativeDefs + {nkMacroDef, nkTemplateDef}
  1017. procDefs* = nkLambdaKinds + declarativeDefs
  1018. callableDefs* = nkLambdaKinds + routineDefs
  1019. nkSymChoices* = {nkClosedSymChoice, nkOpenSymChoice}
  1020. nkStrKinds* = {nkStrLit..nkTripleStrLit}
  1021. skLocalVars* = {skVar, skLet, skForVar, skParam, skResult}
  1022. skProcKinds* = {skProc, skFunc, skTemplate, skMacro, skIterator,
  1023. skMethod, skConverter}
  1024. defaultSize = -1
  1025. defaultAlignment = -1
  1026. defaultOffset* = -1
  1027. proc getPIdent*(a: PNode): PIdent {.inline.} =
  1028. ## Returns underlying `PIdent` for `{nkSym, nkIdent}`, or `nil`.
  1029. case a.kind
  1030. of nkSym: a.sym.name
  1031. of nkIdent: a.ident
  1032. of nkOpenSymChoice, nkClosedSymChoice: a.sons[0].sym.name
  1033. of nkOpenSym: getPIdent(a.sons[0])
  1034. else: nil
  1035. const
  1036. moduleShift = when defined(cpu32): 20 else: 24
  1037. template id*(a: PIdObj): int =
  1038. let x = a
  1039. (x.itemId.module.int shl moduleShift) + x.itemId.item.int
  1040. type
  1041. IdGenerator* = ref object # unfortunately, we really need the 'shared mutable' aspect here.
  1042. module*: int32
  1043. symId*: int32
  1044. typeId*: int32
  1045. sealed*: bool
  1046. disambTable*: CountTable[PIdent]
  1047. const
  1048. PackageModuleId* = -3'i32
  1049. proc idGeneratorFromModule*(m: PSym): IdGenerator =
  1050. assert m.kind == skModule
  1051. result = IdGenerator(module: m.itemId.module, symId: m.itemId.item, typeId: 0, disambTable: initCountTable[PIdent]())
  1052. proc idGeneratorForPackage*(nextIdWillBe: int32): IdGenerator =
  1053. result = IdGenerator(module: PackageModuleId, symId: nextIdWillBe - 1'i32, typeId: 0, disambTable: initCountTable[PIdent]())
  1054. proc nextSymId*(x: IdGenerator): ItemId {.inline.} =
  1055. assert(not x.sealed)
  1056. inc x.symId
  1057. result = ItemId(module: x.module, item: x.symId)
  1058. proc nextTypeId*(x: IdGenerator): ItemId {.inline.} =
  1059. assert(not x.sealed)
  1060. inc x.typeId
  1061. result = ItemId(module: x.module, item: x.typeId)
  1062. when false:
  1063. proc nextId*(x: IdGenerator): ItemId {.inline.} =
  1064. inc x.item
  1065. result = x[]
  1066. when false:
  1067. proc storeBack*(dest: var IdGenerator; src: IdGenerator) {.inline.} =
  1068. assert dest.ItemId.module == src.ItemId.module
  1069. if dest.ItemId.item > src.ItemId.item:
  1070. echo dest.ItemId.item, " ", src.ItemId.item, " ", src.ItemId.module
  1071. assert dest.ItemId.item <= src.ItemId.item
  1072. dest = src
  1073. var ggDebug* {.deprecated.}: bool ## convenience switch for trying out things
  1074. proc isCallExpr*(n: PNode): bool =
  1075. result = n.kind in nkCallKinds
  1076. proc discardSons*(father: PNode)
  1077. type Indexable = PNode | PType
  1078. proc len*(n: Indexable): int {.inline.} =
  1079. result = n.sons.len
  1080. proc safeLen*(n: PNode): int {.inline.} =
  1081. ## works even for leaves.
  1082. if n.kind in {nkNone..nkNilLit}: result = 0
  1083. else: result = n.len
  1084. proc safeArrLen*(n: PNode): int {.inline.} =
  1085. ## works for array-like objects (strings passed as openArray in VM).
  1086. if n.kind in {nkStrLit..nkTripleStrLit}: result = n.strVal.len
  1087. elif n.kind in {nkNone..nkFloat128Lit}: result = 0
  1088. else: result = n.len
  1089. proc add*(father, son: Indexable) =
  1090. assert son != nil
  1091. father.sons.add(son)
  1092. proc addAllowNil*(father, son: Indexable) {.inline.} =
  1093. father.sons.add(son)
  1094. template `[]`*(n: Indexable, i: int): Indexable = n.sons[i]
  1095. template `[]=`*(n: Indexable, i: int; x: Indexable) = n.sons[i] = x
  1096. template `[]`*(n: Indexable, i: BackwardsIndex): Indexable = n[n.len - i.int]
  1097. template `[]=`*(n: Indexable, i: BackwardsIndex; x: Indexable) = n[n.len - i.int] = x
  1098. proc getDeclPragma*(n: PNode): PNode =
  1099. ## return the `nkPragma` node for declaration `n`, or `nil` if no pragma was found.
  1100. ## Currently only supports routineDefs + {nkTypeDef}.
  1101. case n.kind
  1102. of routineDefs:
  1103. if n[pragmasPos].kind != nkEmpty: result = n[pragmasPos]
  1104. of nkTypeDef:
  1105. #[
  1106. type F3*{.deprecated: "x3".} = int
  1107. TypeSection
  1108. TypeDef
  1109. PragmaExpr
  1110. Postfix
  1111. Ident "*"
  1112. Ident "F3"
  1113. Pragma
  1114. ExprColonExpr
  1115. Ident "deprecated"
  1116. StrLit "x3"
  1117. Empty
  1118. Ident "int"
  1119. ]#
  1120. if n[0].kind == nkPragmaExpr:
  1121. result = n[0][1]
  1122. else:
  1123. # support as needed for `nkIdentDefs` etc.
  1124. result = nil
  1125. if result != nil:
  1126. assert result.kind == nkPragma, $(result.kind, n.kind)
  1127. proc extractPragma*(s: PSym): PNode =
  1128. ## gets the pragma node of routine/type/var/let/const symbol `s`
  1129. if s.kind in routineKinds: # bug #24167
  1130. if s.ast[pragmasPos] != nil and s.ast[pragmasPos].kind != nkEmpty:
  1131. result = s.ast[pragmasPos]
  1132. else:
  1133. result = nil
  1134. elif s.kind in {skType, skVar, skLet, skConst}:
  1135. if s.ast != nil and s.ast.len > 0:
  1136. if s.ast[0].kind == nkPragmaExpr and s.ast[0].len > 1:
  1137. # s.ast = nkTypedef / nkPragmaExpr / [nkSym, nkPragma]
  1138. result = s.ast[0][1]
  1139. assert result == nil or result.kind == nkPragma
  1140. proc skipPragmaExpr*(n: PNode): PNode =
  1141. ## if pragma expr, give the node the pragmas are applied to,
  1142. ## otherwise give node itself
  1143. if n.kind == nkPragmaExpr:
  1144. result = n[0]
  1145. else:
  1146. result = n
  1147. proc setInfoRecursive*(n: PNode, info: TLineInfo) =
  1148. ## set line info recursively
  1149. if n != nil:
  1150. for i in 0..<n.safeLen: setInfoRecursive(n[i], info)
  1151. n.info = info
  1152. when defined(useNodeIds):
  1153. const nodeIdToDebug* = -1 # 2322968
  1154. var gNodeId: int
  1155. template newNodeImpl(info2) =
  1156. result = PNode(kind: kind, info: info2)
  1157. when false:
  1158. # this would add overhead, so we skip it; it results in a small amount of leaked entries
  1159. # for old PNode that gets re-allocated at the same address as a PNode that
  1160. # has `nfHasComment` set (and an entry in that table). Only `nfHasComment`
  1161. # should be used to test whether a PNode has a comment; gconfig.comments
  1162. # can contain extra entries for deleted PNode's with comments.
  1163. gconfig.comments.del(cast[int](result))
  1164. template setIdMaybe() =
  1165. when defined(useNodeIds):
  1166. result.id = gNodeId
  1167. if result.id == nodeIdToDebug:
  1168. echo "KIND ", result.kind
  1169. writeStackTrace()
  1170. inc gNodeId
  1171. proc newNode*(kind: TNodeKind): PNode =
  1172. ## new node with unknown line info, no type, and no children
  1173. newNodeImpl(unknownLineInfo)
  1174. setIdMaybe()
  1175. proc newNodeI*(kind: TNodeKind, info: TLineInfo): PNode =
  1176. ## new node with line info, no type, and no children
  1177. newNodeImpl(info)
  1178. setIdMaybe()
  1179. proc newNodeI*(kind: TNodeKind, info: TLineInfo, children: int): PNode =
  1180. ## new node with line info, type, and children
  1181. newNodeImpl(info)
  1182. if children > 0:
  1183. newSeq(result.sons, children)
  1184. setIdMaybe()
  1185. proc newNodeIT*(kind: TNodeKind, info: TLineInfo, typ: PType): PNode =
  1186. ## new node with line info, type, and no children
  1187. result = newNode(kind)
  1188. result.info = info
  1189. result.typ = typ
  1190. proc newTree*(kind: TNodeKind; children: varargs[PNode]): PNode =
  1191. result = newNode(kind)
  1192. if children.len > 0:
  1193. result.info = children[0].info
  1194. result.sons = @children
  1195. proc newTreeI*(kind: TNodeKind; info: TLineInfo; children: varargs[PNode]): PNode =
  1196. result = newNodeI(kind, info)
  1197. if children.len > 0:
  1198. result.info = children[0].info
  1199. result.sons = @children
  1200. proc newTreeIT*(kind: TNodeKind; info: TLineInfo; typ: PType; children: varargs[PNode]): PNode =
  1201. result = newNodeIT(kind, info, typ)
  1202. if children.len > 0:
  1203. result.info = children[0].info
  1204. result.sons = @children
  1205. template previouslyInferred*(t: PType): PType =
  1206. if t.sons.len > 1: t.lastSon else: nil
  1207. when false:
  1208. import tables, strutils
  1209. var x: CountTable[string]
  1210. addQuitProc proc () {.noconv.} =
  1211. for k, v in pairs(x):
  1212. echo k
  1213. echo v
  1214. proc newSym*(symKind: TSymKind, name: PIdent, idgen: IdGenerator; owner: PSym,
  1215. info: TLineInfo; options: TOptions = {}): PSym =
  1216. # generates a symbol and initializes the hash field too
  1217. assert not name.isNil
  1218. let id = nextSymId idgen
  1219. result = PSym(name: name, kind: symKind, flags: {}, info: info, itemId: id,
  1220. options: options, owner: owner, offset: defaultOffset,
  1221. disamb: getOrDefault(idgen.disambTable, name).int32)
  1222. idgen.disambTable.inc name
  1223. when false:
  1224. if id.module == 48 and id.item == 39:
  1225. writeStackTrace()
  1226. echo "kind ", symKind, " ", name.s
  1227. if owner != nil: echo owner.name.s
  1228. proc astdef*(s: PSym): PNode =
  1229. # get only the definition (initializer) portion of the ast
  1230. if s.ast != nil and s.ast.kind in {nkIdentDefs, nkConstDef}:
  1231. s.ast[2]
  1232. else:
  1233. s.ast
  1234. proc isMetaType*(t: PType): bool =
  1235. return t.kind in tyMetaTypes or
  1236. (t.kind == tyStatic and t.n == nil) or
  1237. tfHasMeta in t.flags
  1238. proc isUnresolvedStatic*(t: PType): bool =
  1239. return t.kind == tyStatic and t.n == nil
  1240. proc linkTo*(t: PType, s: PSym): PType {.discardable.} =
  1241. t.sym = s
  1242. s.typ = t
  1243. result = t
  1244. proc linkTo*(s: PSym, t: PType): PSym {.discardable.} =
  1245. t.sym = s
  1246. s.typ = t
  1247. result = s
  1248. template fileIdx*(c: PSym): FileIndex =
  1249. # XXX: this should be used only on module symbols
  1250. c.position.FileIndex
  1251. template filename*(c: PSym): string =
  1252. # XXX: this should be used only on module symbols
  1253. c.position.FileIndex.toFilename
  1254. proc appendToModule*(m: PSym, n: PNode) =
  1255. ## The compiler will use this internally to add nodes that will be
  1256. ## appended to the module after the sem pass
  1257. if m.ast == nil:
  1258. m.ast = newNode(nkStmtList)
  1259. m.ast.sons = @[n]
  1260. else:
  1261. assert m.ast.kind == nkStmtList
  1262. m.ast.sons.add(n)
  1263. const # for all kind of hash tables:
  1264. GrowthFactor* = 2 # must be power of 2, > 0
  1265. StartSize* = 8 # must be power of 2, > 0
  1266. proc copyStrTable*(dest: var TStrTable, src: TStrTable) =
  1267. dest.counter = src.counter
  1268. setLen(dest.data, src.data.len)
  1269. for i in 0..high(src.data): dest.data[i] = src.data[i]
  1270. proc copyIdTable*(dest: var TIdTable, src: TIdTable) =
  1271. dest.counter = src.counter
  1272. newSeq(dest.data, src.data.len)
  1273. for i in 0..high(src.data): dest.data[i] = src.data[i]
  1274. proc copyObjectSet*(dest: var TObjectSet, src: TObjectSet) =
  1275. dest.counter = src.counter
  1276. setLen(dest.data, src.data.len)
  1277. for i in 0..high(src.data): dest.data[i] = src.data[i]
  1278. proc discardSons*(father: PNode) =
  1279. father.sons = @[]
  1280. proc withInfo*(n: PNode, info: TLineInfo): PNode =
  1281. n.info = info
  1282. return n
  1283. proc newIdentNode*(ident: PIdent, info: TLineInfo): PNode =
  1284. result = newNode(nkIdent)
  1285. result.ident = ident
  1286. result.info = info
  1287. proc newSymNode*(sym: PSym): PNode =
  1288. result = newNode(nkSym)
  1289. result.sym = sym
  1290. result.typ = sym.typ
  1291. result.info = sym.info
  1292. proc newSymNode*(sym: PSym, info: TLineInfo): PNode =
  1293. result = newNode(nkSym)
  1294. result.sym = sym
  1295. result.typ = sym.typ
  1296. result.info = info
  1297. proc newOpenSym*(n: PNode): PNode {.inline.} =
  1298. result = newTreeI(nkOpenSym, n.info, n)
  1299. proc newIntNode*(kind: TNodeKind, intVal: BiggestInt): PNode =
  1300. result = newNode(kind)
  1301. result.intVal = intVal
  1302. proc newIntNode*(kind: TNodeKind, intVal: Int128): PNode =
  1303. result = newNode(kind)
  1304. result.intVal = castToInt64(intVal)
  1305. proc lastSon*(n: Indexable): Indexable {.inline.} = n.sons[^1]
  1306. template setLastSon*(n: PNode, s: PNode) = n.sons[^1] = s
  1307. template firstSon*(n: PNode): PNode = n.sons[0]
  1308. template secondSon*(n: PNode): PNode = n.sons[1]
  1309. template hasSon*(n: PNode): bool = n.len > 0
  1310. template has2Sons*(n: PNode): bool = n.len > 1
  1311. proc replaceFirstSon*(n, newson: PNode) {.inline.} =
  1312. n.sons[0] = newson
  1313. proc replaceSon*(n: PNode; i: int; newson: PNode) {.inline.} =
  1314. n.sons[i] = newson
  1315. proc last*(n: PType): PType {.inline.} = n.sons[^1]
  1316. proc elementType*(n: PType): PType {.inline.} = n.sons[^1]
  1317. proc skipModifier*(n: PType): PType {.inline.} = n.sons[^1]
  1318. proc indexType*(n: PType): PType {.inline.} = n.sons[0]
  1319. proc baseClass*(n: PType): PType {.inline.} = n.sons[0]
  1320. proc base*(t: PType): PType {.inline.} =
  1321. result = t.sons[0]
  1322. proc returnType*(n: PType): PType {.inline.} = n.sons[0]
  1323. proc setReturnType*(n, r: PType) {.inline.} = n.sons[0] = r
  1324. proc setIndexType*(n, idx: PType) {.inline.} = n.sons[0] = idx
  1325. proc firstParamType*(n: PType): PType {.inline.} = n.sons[1]
  1326. proc firstGenericParam*(n: PType): PType {.inline.} = n.sons[1]
  1327. proc typeBodyImpl*(n: PType): PType {.inline.} = n.sons[^1]
  1328. proc genericHead*(n: PType): PType {.inline.} = n.sons[0]
  1329. proc skipTypes*(t: PType, kinds: TTypeKinds): PType =
  1330. ## Used throughout the compiler code to test whether a type tree contains or
  1331. ## doesn't contain a specific type/types - it is often the case that only the
  1332. ## last child nodes of a type tree need to be searched. This is a really hot
  1333. ## path within the compiler!
  1334. result = t
  1335. while result.kind in kinds: result = last(result)
  1336. proc newIntTypeNode*(intVal: BiggestInt, typ: PType): PNode =
  1337. let kind = skipTypes(typ, abstractVarRange).kind
  1338. case kind
  1339. of tyInt: result = newNode(nkIntLit)
  1340. of tyInt8: result = newNode(nkInt8Lit)
  1341. of tyInt16: result = newNode(nkInt16Lit)
  1342. of tyInt32: result = newNode(nkInt32Lit)
  1343. of tyInt64: result = newNode(nkInt64Lit)
  1344. of tyChar: result = newNode(nkCharLit)
  1345. of tyUInt: result = newNode(nkUIntLit)
  1346. of tyUInt8: result = newNode(nkUInt8Lit)
  1347. of tyUInt16: result = newNode(nkUInt16Lit)
  1348. of tyUInt32: result = newNode(nkUInt32Lit)
  1349. of tyUInt64: result = newNode(nkUInt64Lit)
  1350. of tyBool, tyEnum:
  1351. # XXX: does this really need to be the kind nkIntLit?
  1352. result = newNode(nkIntLit)
  1353. of tyStatic: # that's a pre-existing bug, will fix in another PR
  1354. result = newNode(nkIntLit)
  1355. else: doAssert false, $kind
  1356. result.intVal = intVal
  1357. result.typ = typ
  1358. proc newIntTypeNode*(intVal: Int128, typ: PType): PNode =
  1359. # XXX: introduce range check
  1360. newIntTypeNode(castToInt64(intVal), typ)
  1361. proc newFloatNode*(kind: TNodeKind, floatVal: BiggestFloat): PNode =
  1362. result = newNode(kind)
  1363. result.floatVal = floatVal
  1364. proc newStrNode*(kind: TNodeKind, strVal: string): PNode =
  1365. result = newNode(kind)
  1366. result.strVal = strVal
  1367. proc newStrNode*(strVal: string; info: TLineInfo): PNode =
  1368. result = newNodeI(nkStrLit, info)
  1369. result.strVal = strVal
  1370. proc newProcNode*(kind: TNodeKind, info: TLineInfo, body: PNode,
  1371. params,
  1372. name, pattern, genericParams,
  1373. pragmas, exceptions: PNode): PNode =
  1374. result = newNodeI(kind, info)
  1375. result.sons = @[name, pattern, genericParams, params,
  1376. pragmas, exceptions, body]
  1377. const
  1378. AttachedOpToStr*: array[TTypeAttachedOp, string] = [
  1379. "=wasMoved", "=destroy", "=copy", "=dup", "=sink", "=trace", "=deepcopy"]
  1380. proc `$`*(s: PSym): string =
  1381. if s != nil:
  1382. result = s.name.s & "@" & $s.id
  1383. else:
  1384. result = "<nil>"
  1385. when false:
  1386. iterator items*(t: PType): PType =
  1387. for i in 0..<t.sons.len: yield t.sons[i]
  1388. iterator pairs*(n: PType): tuple[i: int, n: PType] =
  1389. for i in 0..<n.sons.len: yield (i, n.sons[i])
  1390. when true:
  1391. proc len*(n: PType): int {.inline.} =
  1392. result = n.sons.len
  1393. proc sameTupleLengths*(a, b: PType): bool {.inline.} =
  1394. result = a.sons.len == b.sons.len
  1395. iterator tupleTypePairs*(a, b: PType): (int, PType, PType) =
  1396. for i in 0 ..< a.sons.len:
  1397. yield (i, a.sons[i], b.sons[i])
  1398. iterator underspecifiedPairs*(a, b: PType; start = 0; without = 0): (PType, PType) =
  1399. # XXX Figure out with what typekinds this is called.
  1400. for i in start ..< min(a.sons.len, b.sons.len) + without:
  1401. yield (a.sons[i], b.sons[i])
  1402. proc signatureLen*(t: PType): int {.inline.} =
  1403. result = t.sons.len
  1404. proc paramsLen*(t: PType): int {.inline.} =
  1405. result = t.sons.len - 1
  1406. proc genericParamsLen*(t: PType): int {.inline.} =
  1407. assert t.kind == tyGenericInst
  1408. result = t.sons.len - 2 # without 'head' and 'body'
  1409. proc genericInvocationParamsLen*(t: PType): int {.inline.} =
  1410. assert t.kind == tyGenericInvocation
  1411. result = t.sons.len - 1 # without 'head'
  1412. proc kidsLen*(t: PType): int {.inline.} =
  1413. result = t.sons.len
  1414. proc genericParamHasConstraints*(t: PType): bool {.inline.} = t.sons.len > 0
  1415. proc hasElementType*(t: PType): bool {.inline.} = t.sons.len > 0
  1416. proc isEmptyTupleType*(t: PType): bool {.inline.} = t.sons.len == 0
  1417. proc isSingletonTupleType*(t: PType): bool {.inline.} = t.sons.len == 1
  1418. proc genericConstraint*(t: PType): PType {.inline.} = t.sons[0]
  1419. iterator genericInstParams*(t: PType): (bool, PType) =
  1420. for i in 1..<t.sons.len-1:
  1421. yield (i!=1, t.sons[i])
  1422. iterator genericInstParamPairs*(a, b: PType): (int, PType, PType) =
  1423. for i in 1..<min(a.sons.len, b.sons.len)-1:
  1424. yield (i-1, a.sons[i], b.sons[i])
  1425. iterator genericInvocationParams*(t: PType): (bool, PType) =
  1426. for i in 1..<t.sons.len:
  1427. yield (i!=1, t.sons[i])
  1428. iterator genericInvocationAndBodyElements*(a, b: PType): (PType, PType) =
  1429. for i in 1..<a.sons.len:
  1430. yield (a.sons[i], b.sons[i-1])
  1431. iterator genericInvocationParamPairs*(a, b: PType): (bool, PType, PType) =
  1432. for i in 1..<a.sons.len:
  1433. if i >= b.sons.len:
  1434. yield (false, nil, nil)
  1435. else:
  1436. yield (true, a.sons[i], b.sons[i])
  1437. iterator genericBodyParams*(t: PType): (int, PType) =
  1438. for i in 0..<t.sons.len-1:
  1439. yield (i, t.sons[i])
  1440. iterator userTypeClassInstParams*(t: PType): (bool, PType) =
  1441. for i in 1..<t.sons.len-1:
  1442. yield (i!=1, t.sons[i])
  1443. iterator ikids*(t: PType): (int, PType) =
  1444. for i in 0..<t.sons.len: yield (i, t.sons[i])
  1445. const
  1446. FirstParamAt* = 1
  1447. FirstGenericParamAt* = 1
  1448. iterator paramTypes*(t: PType): (int, PType) =
  1449. for i in FirstParamAt..<t.sons.len: yield (i, t.sons[i])
  1450. iterator paramTypePairs*(a, b: PType): (PType, PType) =
  1451. for i in FirstParamAt..<a.sons.len: yield (a.sons[i], b.sons[i])
  1452. template paramTypeToNodeIndex*(x: int): int = x
  1453. iterator kids*(t: PType): PType =
  1454. for i in 0..<t.sons.len: yield t.sons[i]
  1455. iterator signature*(t: PType): PType =
  1456. # yields return type + parameter types
  1457. for i in 0..<t.sons.len: yield t.sons[i]
  1458. proc newType*(kind: TTypeKind, id: ItemId; owner: PSym): PType =
  1459. result = PType(kind: kind, owner: owner, size: defaultSize,
  1460. align: defaultAlignment, itemId: id,
  1461. uniqueId: id)
  1462. when false:
  1463. if result.itemId.module == 55 and result.itemId.item == 2:
  1464. echo "KNID ", kind
  1465. writeStackTrace()
  1466. proc mergeLoc(a: var TLoc, b: TLoc) =
  1467. if a.k == low(typeof(a.k)): a.k = b.k
  1468. if a.storage == low(typeof(a.storage)): a.storage = b.storage
  1469. a.flags.incl b.flags
  1470. if a.lode == nil: a.lode = b.lode
  1471. if a.r == "": a.r = b.r
  1472. proc newSons*(father: Indexable, length: int) =
  1473. setLen(father.sons, length)
  1474. proc assignType*(dest, src: PType) =
  1475. dest.kind = src.kind
  1476. dest.flags = src.flags
  1477. dest.callConv = src.callConv
  1478. dest.n = src.n
  1479. dest.size = src.size
  1480. dest.align = src.align
  1481. # this fixes 'type TLock = TSysLock':
  1482. if src.sym != nil:
  1483. if dest.sym != nil:
  1484. dest.sym.flags.incl src.sym.flags-{sfUsed, sfExported}
  1485. if dest.sym.annex == nil: dest.sym.annex = src.sym.annex
  1486. mergeLoc(dest.sym.loc, src.sym.loc)
  1487. else:
  1488. dest.sym = src.sym
  1489. newSons(dest, src.len)
  1490. for i in 0..<src.len: dest[i] = src[i]
  1491. proc copyType*(t: PType, id: ItemId, owner: PSym): PType =
  1492. result = newType(t.kind, id, owner)
  1493. assignType(result, t)
  1494. result.sym = t.sym # backend-info should not be copied
  1495. proc exactReplica*(t: PType): PType =
  1496. result = copyType(t, t.itemId, t.owner)
  1497. proc copySym*(s: PSym; idgen: IdGenerator): PSym =
  1498. result = newSym(s.kind, s.name, idgen, s.owner, s.info, s.options)
  1499. #result.ast = nil # BUGFIX; was: s.ast which made problems
  1500. result.typ = s.typ
  1501. result.flags = s.flags
  1502. result.magic = s.magic
  1503. result.options = s.options
  1504. result.position = s.position
  1505. result.loc = s.loc
  1506. result.annex = s.annex # BUGFIX
  1507. result.constraint = s.constraint
  1508. if result.kind in {skVar, skLet, skField}:
  1509. result.guard = s.guard
  1510. result.bitsize = s.bitsize
  1511. result.alignment = s.alignment
  1512. proc createModuleAlias*(s: PSym, idgen: IdGenerator, newIdent: PIdent, info: TLineInfo;
  1513. options: TOptions): PSym =
  1514. result = newSym(s.kind, newIdent, idgen, s.owner, info, options)
  1515. # keep ID!
  1516. result.ast = s.ast
  1517. #result.id = s.id # XXX figure out what to do with the ID.
  1518. result.flags = s.flags
  1519. result.options = s.options
  1520. result.position = s.position
  1521. result.loc = s.loc
  1522. result.annex = s.annex
  1523. proc initStrTable*(x: var TStrTable) =
  1524. x.counter = 0
  1525. newSeq(x.data, StartSize)
  1526. proc newStrTable*: TStrTable =
  1527. initStrTable(result)
  1528. proc initIdTable*(x: var TIdTable) =
  1529. x.counter = 0
  1530. newSeq(x.data, StartSize)
  1531. proc newIdTable*: TIdTable =
  1532. initIdTable(result)
  1533. proc resetIdTable*(x: var TIdTable) =
  1534. x.counter = 0
  1535. # clear and set to old initial size:
  1536. setLen(x.data, 0)
  1537. setLen(x.data, StartSize)
  1538. proc initObjectSet*(x: var TObjectSet) =
  1539. x.counter = 0
  1540. newSeq(x.data, StartSize)
  1541. proc initIdNodeTable*(x: var TIdNodeTable) =
  1542. x.counter = 0
  1543. newSeq(x.data, StartSize)
  1544. proc initNodeTable*(x: var TNodeTable) =
  1545. x.counter = 0
  1546. newSeq(x.data, StartSize)
  1547. proc skipTypes*(t: PType, kinds: TTypeKinds; maxIters: int): PType =
  1548. result = t
  1549. var i = maxIters
  1550. while result.kind in kinds:
  1551. result = last(result)
  1552. dec i
  1553. if i == 0: return nil
  1554. proc skipTypesOrNil*(t: PType, kinds: TTypeKinds): PType =
  1555. ## same as skipTypes but handles 'nil'
  1556. result = t
  1557. while result != nil and result.kind in kinds:
  1558. if result.len == 0: return nil
  1559. result = last(result)
  1560. proc isGCedMem*(t: PType): bool {.inline.} =
  1561. result = t.kind in {tyString, tyRef, tySequence} or
  1562. t.kind == tyProc and t.callConv == ccClosure
  1563. proc propagateToOwner*(owner, elem: PType; propagateHasAsgn = true) =
  1564. owner.flags.incl elem.flags * {tfHasMeta, tfTriggersCompileTime}
  1565. if tfNotNil in elem.flags:
  1566. if owner.kind in {tyGenericInst, tyGenericBody, tyGenericInvocation}:
  1567. owner.flags.incl tfNotNil
  1568. if elem.isMetaType:
  1569. owner.flags.incl tfHasMeta
  1570. let mask = elem.flags * {tfHasAsgn, tfHasOwned}
  1571. if mask != {} and propagateHasAsgn:
  1572. let o2 = owner.skipTypes({tyGenericInst, tyAlias, tySink})
  1573. if o2.kind in {tyTuple, tyObject, tyArray,
  1574. tySequence, tySet, tyDistinct}:
  1575. o2.flags.incl mask
  1576. owner.flags.incl mask
  1577. if owner.kind notin {tyProc, tyGenericInst, tyGenericBody,
  1578. tyGenericInvocation, tyPtr}:
  1579. let elemB = elem.skipTypes({tyGenericInst, tyAlias, tySink})
  1580. if elemB.isGCedMem or tfHasGCedMem in elemB.flags:
  1581. # for simplicity, we propagate this flag even to generics. We then
  1582. # ensure this doesn't bite us in sempass2.
  1583. owner.flags.incl tfHasGCedMem
  1584. proc rawAddSon*(father, son: PType; propagateHasAsgn = true) =
  1585. father.sons.add(son)
  1586. if not son.isNil: propagateToOwner(father, son, propagateHasAsgn)
  1587. proc rawAddSonNoPropagationOfTypeFlags*(father, son: PType) =
  1588. father.sons.add(son)
  1589. proc addSonNilAllowed*(father, son: PNode) =
  1590. father.sons.add(son)
  1591. proc delSon*(father: PNode, idx: int) =
  1592. if father.len == 0: return
  1593. for i in idx..<father.len - 1: father[i] = father[i + 1]
  1594. father.sons.setLen(father.len - 1)
  1595. proc copyNode*(src: PNode): PNode =
  1596. # does not copy its sons!
  1597. if src == nil:
  1598. return nil
  1599. result = newNode(src.kind)
  1600. result.info = src.info
  1601. result.typ = src.typ
  1602. result.flags = src.flags * PersistentNodeFlags
  1603. result.comment = src.comment
  1604. when defined(useNodeIds):
  1605. if result.id == nodeIdToDebug:
  1606. echo "COMES FROM ", src.id
  1607. case src.kind
  1608. of nkCharLit..nkUInt64Lit: result.intVal = src.intVal
  1609. of nkFloatLiterals: result.floatVal = src.floatVal
  1610. of nkSym: result.sym = src.sym
  1611. of nkIdent: result.ident = src.ident
  1612. of nkStrLit..nkTripleStrLit: result.strVal = src.strVal
  1613. else: discard
  1614. when defined(nimsuggest):
  1615. result.endInfo = src.endInfo
  1616. template transitionNodeKindCommon(k: TNodeKind) =
  1617. let obj {.inject.} = n[]
  1618. n[] = TNode(kind: k, typ: obj.typ, info: obj.info, flags: obj.flags)
  1619. # n.comment = obj.comment # shouldn't be needed, the address doesnt' change
  1620. when defined(useNodeIds):
  1621. n.id = obj.id
  1622. proc transitionSonsKind*(n: PNode, kind: range[nkComesFrom..nkTupleConstr]) =
  1623. transitionNodeKindCommon(kind)
  1624. n.sons = obj.sons
  1625. proc transitionIntKind*(n: PNode, kind: range[nkCharLit..nkUInt64Lit]) =
  1626. transitionNodeKindCommon(kind)
  1627. n.intVal = obj.intVal
  1628. proc transitionIntToFloatKind*(n: PNode, kind: range[nkFloatLit..nkFloat128Lit]) =
  1629. transitionNodeKindCommon(kind)
  1630. n.floatVal = BiggestFloat(obj.intVal)
  1631. proc transitionNoneToSym*(n: PNode) =
  1632. transitionNodeKindCommon(nkSym)
  1633. template transitionSymKindCommon*(k: TSymKind) =
  1634. let obj {.inject.} = s[]
  1635. s[] = TSym(kind: k, itemId: obj.itemId, magic: obj.magic, typ: obj.typ, name: obj.name,
  1636. info: obj.info, owner: obj.owner, flags: obj.flags, ast: obj.ast,
  1637. options: obj.options, position: obj.position, offset: obj.offset,
  1638. loc: obj.loc, annex: obj.annex, constraint: obj.constraint)
  1639. when hasFFI:
  1640. s.cname = obj.cname
  1641. when defined(nimsuggest):
  1642. s.allUsages = obj.allUsages
  1643. proc transitionGenericParamToType*(s: PSym) =
  1644. transitionSymKindCommon(skType)
  1645. proc transitionRoutineSymKind*(s: PSym, kind: range[skProc..skTemplate]) =
  1646. transitionSymKindCommon(kind)
  1647. s.gcUnsafetyReason = obj.gcUnsafetyReason
  1648. s.transformedBody = obj.transformedBody
  1649. proc transitionToLet*(s: PSym) =
  1650. transitionSymKindCommon(skLet)
  1651. s.guard = obj.guard
  1652. s.bitsize = obj.bitsize
  1653. s.alignment = obj.alignment
  1654. template copyNodeImpl(dst, src, processSonsStmt) =
  1655. if src == nil: return
  1656. dst = newNode(src.kind)
  1657. dst.info = src.info
  1658. when defined(nimsuggest):
  1659. result.endInfo = src.endInfo
  1660. dst.typ = src.typ
  1661. dst.flags = src.flags * PersistentNodeFlags
  1662. dst.comment = src.comment
  1663. when defined(useNodeIds):
  1664. if dst.id == nodeIdToDebug:
  1665. echo "COMES FROM ", src.id
  1666. case src.kind
  1667. of nkCharLit..nkUInt64Lit: dst.intVal = src.intVal
  1668. of nkFloatLiterals: dst.floatVal = src.floatVal
  1669. of nkSym: dst.sym = src.sym
  1670. of nkIdent: dst.ident = src.ident
  1671. of nkStrLit..nkTripleStrLit: dst.strVal = src.strVal
  1672. else: processSonsStmt
  1673. proc shallowCopy*(src: PNode): PNode =
  1674. # does not copy its sons, but provides space for them:
  1675. copyNodeImpl(result, src):
  1676. newSeq(result.sons, src.len)
  1677. proc copyTree*(src: PNode): PNode =
  1678. # copy a whole syntax tree; performs deep copying
  1679. copyNodeImpl(result, src):
  1680. newSeq(result.sons, src.len)
  1681. for i in 0..<src.len:
  1682. result[i] = copyTree(src[i])
  1683. proc copyTreeWithoutNode*(src, skippedNode: PNode): PNode =
  1684. copyNodeImpl(result, src):
  1685. result.sons = newSeqOfCap[PNode](src.len)
  1686. for n in src.sons:
  1687. if n != skippedNode:
  1688. result.sons.add copyTreeWithoutNode(n, skippedNode)
  1689. proc hasSonWith*(n: PNode, kind: TNodeKind): bool =
  1690. for i in 0..<n.len:
  1691. if n[i].kind == kind:
  1692. return true
  1693. result = false
  1694. proc hasNilSon*(n: PNode): bool =
  1695. for i in 0..<n.safeLen:
  1696. if n[i] == nil:
  1697. return true
  1698. elif hasNilSon(n[i]):
  1699. return true
  1700. result = false
  1701. proc containsNode*(n: PNode, kinds: TNodeKinds): bool =
  1702. if n == nil: return
  1703. case n.kind
  1704. of nkEmpty..nkNilLit: result = n.kind in kinds
  1705. else:
  1706. for i in 0..<n.len:
  1707. if n.kind in kinds or containsNode(n[i], kinds): return true
  1708. proc hasSubnodeWith*(n: PNode, kind: TNodeKind): bool =
  1709. case n.kind
  1710. of nkEmpty..nkNilLit, nkFormalParams: result = n.kind == kind
  1711. else:
  1712. for i in 0..<n.len:
  1713. if (n[i].kind == kind) or hasSubnodeWith(n[i], kind):
  1714. return true
  1715. result = false
  1716. proc getInt*(a: PNode): Int128 =
  1717. case a.kind
  1718. of nkCharLit, nkUIntLit..nkUInt64Lit:
  1719. result = toInt128(cast[uint64](a.intVal))
  1720. of nkInt8Lit..nkInt64Lit:
  1721. result = toInt128(a.intVal)
  1722. of nkIntLit:
  1723. # XXX: enable this assert
  1724. # assert a.typ.kind notin {tyChar, tyUint..tyUInt64}
  1725. result = toInt128(a.intVal)
  1726. else:
  1727. raiseRecoverableError("cannot extract number from invalid AST node")
  1728. proc getInt64*(a: PNode): int64 {.deprecated: "use getInt".} =
  1729. case a.kind
  1730. of nkCharLit, nkUIntLit..nkUInt64Lit, nkIntLit..nkInt64Lit:
  1731. result = a.intVal
  1732. else:
  1733. raiseRecoverableError("cannot extract number from invalid AST node")
  1734. proc getFloat*(a: PNode): BiggestFloat =
  1735. case a.kind
  1736. of nkFloatLiterals: result = a.floatVal
  1737. of nkCharLit, nkUIntLit..nkUInt64Lit, nkIntLit..nkInt64Lit:
  1738. result = BiggestFloat a.intVal
  1739. else:
  1740. raiseRecoverableError("cannot extract number from invalid AST node")
  1741. #doAssert false, "getFloat"
  1742. #internalError(a.info, "getFloat")
  1743. #result = 0.0
  1744. proc getStr*(a: PNode): string =
  1745. case a.kind
  1746. of nkStrLit..nkTripleStrLit: result = a.strVal
  1747. of nkNilLit:
  1748. # let's hope this fixes more problems than it creates:
  1749. result = ""
  1750. else:
  1751. raiseRecoverableError("cannot extract string from invalid AST node")
  1752. #doAssert false, "getStr"
  1753. #internalError(a.info, "getStr")
  1754. #result = ""
  1755. proc getStrOrChar*(a: PNode): string =
  1756. case a.kind
  1757. of nkStrLit..nkTripleStrLit: result = a.strVal
  1758. of nkCharLit..nkUInt64Lit: result = $chr(int(a.intVal))
  1759. else:
  1760. raiseRecoverableError("cannot extract string from invalid AST node")
  1761. #doAssert false, "getStrOrChar"
  1762. #internalError(a.info, "getStrOrChar")
  1763. #result = ""
  1764. proc isGenericParams*(n: PNode): bool {.inline.} =
  1765. ## used to judge whether a node is generic params.
  1766. n != nil and n.kind == nkGenericParams
  1767. proc isGenericRoutine*(n: PNode): bool {.inline.} =
  1768. n != nil and n.kind in callableDefs and n[genericParamsPos].isGenericParams
  1769. proc isGenericRoutineStrict*(s: PSym): bool {.inline.} =
  1770. ## determines if this symbol represents a generic routine
  1771. ## the unusual name is so it doesn't collide and eventually replaces
  1772. ## `isGenericRoutine`
  1773. s.kind in skProcKinds and s.ast.isGenericRoutine
  1774. proc isGenericRoutine*(s: PSym): bool {.inline.} =
  1775. ## determines if this symbol represents a generic routine or an instance of
  1776. ## one. This should be renamed accordingly and `isGenericRoutineStrict`
  1777. ## should take this name instead.
  1778. ##
  1779. ## Warning/XXX: Unfortunately, it considers a proc kind symbol flagged with
  1780. ## sfFromGeneric as a generic routine. Instead this should likely not be the
  1781. ## case and the concepts should be teased apart:
  1782. ## - generic definition
  1783. ## - generic instance
  1784. ## - either generic definition or instance
  1785. s.kind in skProcKinds and (sfFromGeneric in s.flags or
  1786. s.ast.isGenericRoutine)
  1787. proc skipGenericOwner*(s: PSym): PSym =
  1788. ## Generic instantiations are owned by their originating generic
  1789. ## symbol. This proc skips such owners and goes straight to the owner
  1790. ## of the generic itself (the module or the enclosing proc).
  1791. result = if s.kind == skModule:
  1792. s
  1793. elif s.kind in skProcKinds and sfFromGeneric in s.flags and s.owner.kind != skModule:
  1794. s.owner.owner
  1795. else:
  1796. s.owner
  1797. proc originatingModule*(s: PSym): PSym =
  1798. result = s
  1799. while result.kind != skModule: result = result.owner
  1800. proc isRoutine*(s: PSym): bool {.inline.} =
  1801. result = s.kind in skProcKinds
  1802. proc isCompileTimeProc*(s: PSym): bool {.inline.} =
  1803. result = s.kind == skMacro or
  1804. s.kind in {skProc, skFunc} and sfCompileTime in s.flags
  1805. proc hasPattern*(s: PSym): bool {.inline.} =
  1806. result = isRoutine(s) and s.ast[patternPos].kind != nkEmpty
  1807. iterator items*(n: PNode): PNode =
  1808. for i in 0..<n.safeLen: yield n[i]
  1809. iterator pairs*(n: PNode): tuple[i: int, n: PNode] =
  1810. for i in 0..<n.safeLen: yield (i, n[i])
  1811. proc isAtom*(n: PNode): bool {.inline.} =
  1812. result = n.kind >= nkNone and n.kind <= nkNilLit
  1813. proc isEmptyType*(t: PType): bool {.inline.} =
  1814. ## 'void' and 'typed' types are often equivalent to 'nil' these days:
  1815. result = t == nil or t.kind in {tyVoid, tyTyped}
  1816. proc makeStmtList*(n: PNode): PNode =
  1817. if n.kind == nkStmtList:
  1818. result = n
  1819. else:
  1820. result = newNodeI(nkStmtList, n.info)
  1821. result.add n
  1822. proc skipStmtList*(n: PNode): PNode =
  1823. if n.kind in {nkStmtList, nkStmtListExpr}:
  1824. for i in 0..<n.len-1:
  1825. if n[i].kind notin {nkEmpty, nkCommentStmt}: return n
  1826. result = n.lastSon
  1827. else:
  1828. result = n
  1829. proc toVar*(typ: PType; kind: TTypeKind; idgen: IdGenerator): PType =
  1830. ## If ``typ`` is not a tyVar then it is converted into a `var <typ>` and
  1831. ## returned. Otherwise ``typ`` is simply returned as-is.
  1832. result = typ
  1833. if typ.kind != kind:
  1834. result = newType(kind, nextTypeId(idgen), typ.owner)
  1835. rawAddSon(result, typ)
  1836. proc toRef*(typ: PType; idgen: IdGenerator): PType =
  1837. ## If ``typ`` is a tyObject then it is converted into a `ref <typ>` and
  1838. ## returned. Otherwise ``typ`` is simply returned as-is.
  1839. result = typ
  1840. if typ.skipTypes({tyAlias, tyGenericInst}).kind == tyObject:
  1841. result = newType(tyRef, nextTypeId(idgen), typ.owner)
  1842. rawAddSon(result, typ)
  1843. proc toObject*(typ: PType): PType =
  1844. ## If ``typ`` is a tyRef then its immediate son is returned (which in many
  1845. ## cases should be a ``tyObject``).
  1846. ## Otherwise ``typ`` is simply returned as-is.
  1847. let t = typ.skipTypes({tyAlias, tyGenericInst})
  1848. if t.kind == tyRef: t.elementType
  1849. else: typ
  1850. proc toObjectFromRefPtrGeneric*(typ: PType): PType =
  1851. #[
  1852. See also `toObject`.
  1853. Finds the underlying `object`, even in cases like these:
  1854. type
  1855. B[T] = object f0: int
  1856. A1[T] = ref B[T]
  1857. A2[T] = ref object f1: int
  1858. A3 = ref object f2: int
  1859. A4 = object f3: int
  1860. ]#
  1861. result = typ
  1862. while true:
  1863. case result.kind
  1864. of tyGenericBody: result = result.last
  1865. of tyRef, tyPtr, tyGenericInst, tyGenericInvocation, tyAlias: result = result[0]
  1866. # automatic dereferencing is deep, refs #18298.
  1867. else: break
  1868. # result does not have to be object type
  1869. proc isImportedException*(t: PType; conf: ConfigRef): bool =
  1870. assert t != nil
  1871. if conf.exc != excCpp:
  1872. return false
  1873. let base = t.skipTypes({tyAlias, tyPtr, tyDistinct, tyGenericInst})
  1874. if base.sym != nil and {sfCompileToCpp, sfImportc} * base.sym.flags != {}:
  1875. result = true
  1876. proc isInfixAs*(n: PNode): bool =
  1877. return n.kind == nkInfix and n[0].kind == nkIdent and n[0].ident.id == ord(wAs)
  1878. proc skipColon*(n: PNode): PNode =
  1879. result = n
  1880. if n.kind == nkExprColonExpr:
  1881. result = n[1]
  1882. proc findUnresolvedStatic*(n: PNode): PNode =
  1883. if n.kind == nkSym and n.typ != nil and n.typ.kind == tyStatic and n.typ.n == nil:
  1884. return n
  1885. if n.typ != nil and n.typ.kind == tyTypeDesc:
  1886. let t = skipTypes(n.typ, {tyTypeDesc})
  1887. if t.kind == tyGenericParam and t.len == 0:
  1888. return n
  1889. for son in n:
  1890. let n = son.findUnresolvedStatic
  1891. if n != nil: return n
  1892. return nil
  1893. when false:
  1894. proc containsNil*(n: PNode): bool =
  1895. # only for debugging
  1896. if n.isNil: return true
  1897. for i in 0..<n.safeLen:
  1898. if n[i].containsNil: return true
  1899. template hasDestructor*(t: PType): bool = {tfHasAsgn, tfHasOwned} * t.flags != {}
  1900. template incompleteType*(t: PType): bool =
  1901. t.sym != nil and {sfForward, sfNoForward} * t.sym.flags == {sfForward}
  1902. template typeCompleted*(s: PSym) =
  1903. incl s.flags, sfNoForward
  1904. template detailedInfo*(sym: PSym): string =
  1905. sym.name.s
  1906. proc isInlineIterator*(typ: PType): bool {.inline.} =
  1907. typ.kind == tyProc and tfIterator in typ.flags and typ.callConv != ccClosure
  1908. proc isIterator*(typ: PType): bool {.inline.} =
  1909. typ.kind == tyProc and tfIterator in typ.flags
  1910. proc isClosureIterator*(typ: PType): bool {.inline.} =
  1911. typ.kind == tyProc and tfIterator in typ.flags and typ.callConv == ccClosure
  1912. proc isClosure*(typ: PType): bool {.inline.} =
  1913. typ.kind == tyProc and typ.callConv == ccClosure
  1914. proc isNimcall*(s: PSym): bool {.inline.} =
  1915. s.typ.callConv == ccNimCall
  1916. proc isExplicitCallConv*(s: PSym): bool {.inline.} =
  1917. tfExplicitCallConv in s.typ.flags
  1918. proc isSinkParam*(s: PSym): bool {.inline.} =
  1919. s.kind == skParam and (s.typ.kind == tySink or tfHasOwned in s.typ.flags)
  1920. proc isSinkType*(t: PType): bool {.inline.} =
  1921. t.kind == tySink or tfHasOwned in t.flags
  1922. proc newProcType*(info: TLineInfo; id: ItemId; owner: PSym): PType =
  1923. result = newType(tyProc, id, owner)
  1924. result.n = newNodeI(nkFormalParams, info)
  1925. rawAddSon(result, nil) # return type
  1926. # result.n[0] used to be `nkType`, but now it's `nkEffectList` because
  1927. # the effects are now stored in there too ... this is a bit hacky, but as
  1928. # usual we desperately try to save memory:
  1929. result.n.add newNodeI(nkEffectList, info)
  1930. proc addParam*(procType: PType; param: PSym) =
  1931. param.position = procType.len-1
  1932. procType.n.add newSymNode(param)
  1933. rawAddSon(procType, param.typ)
  1934. const magicsThatCanRaise = {
  1935. mNone, mSlurp, mStaticExec, mParseExprToAst, mParseStmtToAst, mEcho}
  1936. proc canRaiseConservative*(fn: PNode): bool =
  1937. if fn.kind == nkSym and fn.sym.magic notin magicsThatCanRaise:
  1938. result = false
  1939. else:
  1940. result = true
  1941. proc canRaise*(fn: PNode): bool =
  1942. if fn.kind == nkSym and (fn.sym.magic notin magicsThatCanRaise or
  1943. {sfImportc, sfInfixCall} * fn.sym.flags == {sfImportc} or
  1944. sfGeneratedOp in fn.sym.flags):
  1945. result = false
  1946. elif fn.kind == nkSym and fn.sym.magic == mEcho:
  1947. result = true
  1948. else:
  1949. # TODO check for n having sons? or just return false for now if not
  1950. if fn.typ != nil and fn.typ.n != nil and fn.typ.n[0].kind == nkSym:
  1951. result = false
  1952. else:
  1953. result = fn.typ != nil and fn.typ.n != nil and ((fn.typ.n[0].len < effectListLen) or
  1954. (fn.typ.n[0][exceptionEffects] != nil and
  1955. fn.typ.n[0][exceptionEffects].safeLen > 0))
  1956. proc toHumanStrImpl[T](kind: T, num: static int): string =
  1957. result = $kind
  1958. result = result[num..^1]
  1959. result[0] = result[0].toLowerAscii
  1960. proc toHumanStr*(kind: TSymKind): string =
  1961. ## strips leading `sk`
  1962. result = toHumanStrImpl(kind, 2)
  1963. proc toHumanStr*(kind: TTypeKind): string =
  1964. ## strips leading `tk`
  1965. result = toHumanStrImpl(kind, 2)
  1966. proc skipAddr*(n: PNode): PNode {.inline.} =
  1967. (if n.kind == nkHiddenAddr: n[0] else: n)
  1968. proc isNewStyleConcept*(n: PNode): bool {.inline.} =
  1969. assert n.kind == nkTypeClassTy
  1970. result = n[0].kind == nkEmpty
  1971. proc isOutParam*(t: PType): bool {.inline.} = tfIsOutParam in t.flags
  1972. const
  1973. nodesToIgnoreSet* = {nkNone..pred(nkSym), succ(nkSym)..nkNilLit,
  1974. nkTypeSection, nkProcDef, nkConverterDef,
  1975. nkMethodDef, nkIteratorDef, nkMacroDef, nkTemplateDef, nkLambda, nkDo,
  1976. nkFuncDef, nkConstSection, nkConstDef, nkIncludeStmt, nkImportStmt,
  1977. nkExportStmt, nkPragma, nkCommentStmt, nkBreakState,
  1978. nkTypeOfExpr, nkMixinStmt, nkBindStmt}