jsgen.nim 101 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105
  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. # This is the JavaScript code generator.
  10. discard """
  11. The JS code generator contains only 2 tricks:
  12. Trick 1
  13. -------
  14. Some locations (for example 'var int') require "fat pointers" (`etyBaseIndex`)
  15. which are pairs (array, index). The derefence operation is then 'array[index]'.
  16. Check `mapType` for the details.
  17. Trick 2
  18. -------
  19. It is preferable to generate '||' and '&&' if possible since that is more
  20. idiomatic and hence should be friendlier for the JS JIT implementation. However
  21. code like `foo and (let bar = baz())` cannot be translated this way. Instead
  22. the expressions need to be transformed into statements. `isSimpleExpr`
  23. implements the required case distinction.
  24. """
  25. import
  26. ast, trees, magicsys, options,
  27. nversion, msgs, idents, types,
  28. ropes, ccgutils, wordrecg, renderer,
  29. cgmeth, lowerings, sighashes, modulegraphs, lineinfos, rodutils,
  30. transf, injectdestructors, sourcemap, astmsgs, backendpragmas
  31. import pipelineutils
  32. import json, sets, math, tables, intsets
  33. import strutils except addf
  34. when defined(nimPreviewSlimSystem):
  35. import std/[assertions, syncio]
  36. type
  37. TJSGen = object of PPassContext
  38. module: PSym
  39. graph: ModuleGraph
  40. config: ConfigRef
  41. sigConflicts: CountTable[SigHash]
  42. initProc: PProc
  43. BModule = ref TJSGen
  44. TJSTypeKind = enum # necessary JS "types"
  45. etyNone, # no type
  46. etyNull, # null type
  47. etyProc, # proc type
  48. etyBool, # bool type
  49. etySeq, # Nim seq or string type
  50. etyInt, # JavaScript's int
  51. etyFloat, # JavaScript's float
  52. etyString, # JavaScript's string
  53. etyObject, # JavaScript's reference to an object
  54. etyBaseIndex # base + index needed
  55. TResKind = enum
  56. resNone, # not set
  57. resExpr, # is some complex expression
  58. resVal, # is a temporary/value/l-value
  59. resCallee # expression is callee
  60. TCompRes = object
  61. kind: TResKind
  62. typ: TJSTypeKind
  63. res: Rope # result part; index if this is an
  64. # (address, index)-tuple
  65. address: Rope # address of an (address, index)-tuple
  66. tmpLoc: Rope # tmp var which stores the (address, index)
  67. # pair to prevent multiple evals.
  68. # the tmp is initialized upon evaling the
  69. # address.
  70. # might be nil.
  71. # (see `maybeMakeTemp`)
  72. TBlock = object
  73. id: int # the ID of the label; positive means that it
  74. # has been used (i.e. the label should be emitted)
  75. isLoop: bool # whether it's a 'block' or 'while'
  76. PGlobals = ref object of RootObj
  77. typeInfo, constants, code: Rope
  78. forwarded: seq[PSym]
  79. generatedSyms: IntSet
  80. typeInfoGenerated: IntSet
  81. unique: int # for temp identifier generation
  82. inSystem: bool
  83. PProc = ref TProc
  84. TProc = object
  85. procDef: PNode
  86. prc: PSym
  87. globals, locals, body: Rope
  88. options: TOptions
  89. optionsStack: seq[TOptions]
  90. module: BModule
  91. g: PGlobals
  92. generatedParamCopies: IntSet
  93. beforeRetNeeded: bool
  94. unique: int # for temp identifier generation
  95. blocks: seq[TBlock]
  96. extraIndent: int
  97. up: PProc # up the call chain; required for closure support
  98. declaredGlobals: IntSet
  99. previousFileName: string # For frameInfo inside templates.
  100. template config*(p: PProc): ConfigRef = p.module.config
  101. proc indentLine(p: PProc, r: Rope): Rope =
  102. var p = p
  103. var ind = 0
  104. while true:
  105. inc ind, p.blocks.len + p.extraIndent
  106. if p.up == nil or p.up.prc != p.prc.owner:
  107. break
  108. p = p.up
  109. result = repeat(' ', ind*2) & r
  110. template line(p: PProc, added: string) =
  111. p.body.add(indentLine(p, rope(added)))
  112. template lineF(p: PProc, frmt: FormatStr, args: varargs[Rope]) =
  113. p.body.add(indentLine(p, ropes.`%`(frmt, args)))
  114. template nested(p, body) =
  115. inc p.extraIndent
  116. body
  117. dec p.extraIndent
  118. proc newGlobals(): PGlobals =
  119. new(result)
  120. result.forwarded = @[]
  121. result.generatedSyms = initIntSet()
  122. result.typeInfoGenerated = initIntSet()
  123. proc initCompRes(r: var TCompRes) =
  124. r.address = ""
  125. r.res = ""
  126. r.tmpLoc = ""
  127. r.typ = etyNone
  128. r.kind = resNone
  129. proc rdLoc(a: TCompRes): Rope {.inline.} =
  130. if a.typ != etyBaseIndex:
  131. result = a.res
  132. else:
  133. result = "$1[$2]" % [a.address, a.res]
  134. proc newProc(globals: PGlobals, module: BModule, procDef: PNode,
  135. options: TOptions): PProc =
  136. result = PProc(
  137. blocks: @[],
  138. optionsStack: if module.initProc != nil: module.initProc.optionsStack
  139. else: @[],
  140. options: options,
  141. module: module,
  142. procDef: procDef,
  143. g: globals,
  144. extraIndent: int(procDef != nil))
  145. if procDef != nil: result.prc = procDef[namePos].sym
  146. proc initProcOptions(module: BModule): TOptions =
  147. result = module.config.options
  148. if PGlobals(module.graph.backend).inSystem:
  149. result.excl(optStackTrace)
  150. proc newInitProc(globals: PGlobals, module: BModule): PProc =
  151. result = newProc(globals, module, nil, initProcOptions(module))
  152. proc declareGlobal(p: PProc; id: int; r: Rope) =
  153. if p.prc != nil and not p.declaredGlobals.containsOrIncl(id):
  154. p.locals.addf("global $1;$n", [r])
  155. const
  156. MappedToObject = {tyObject, tyArray, tyTuple, tyOpenArray,
  157. tySet, tyVarargs}
  158. proc mapType(typ: PType): TJSTypeKind =
  159. let t = skipTypes(typ, abstractInst)
  160. case t.kind
  161. of tyVar, tyRef, tyPtr:
  162. if skipTypes(t.lastSon, abstractInst).kind in MappedToObject:
  163. result = etyObject
  164. else:
  165. result = etyBaseIndex
  166. of tyPointer:
  167. # treat a tyPointer like a typed pointer to an array of bytes
  168. result = etyBaseIndex
  169. of tyRange, tyDistinct, tyOrdinal, tyProxy, tyLent:
  170. # tyLent is no-op as JS has pass-by-reference semantics
  171. result = mapType(t[0])
  172. of tyInt..tyInt64, tyUInt..tyUInt64, tyEnum, tyChar: result = etyInt
  173. of tyBool: result = etyBool
  174. of tyFloat..tyFloat128: result = etyFloat
  175. of tySet: result = etyObject # map a set to a table
  176. of tyString, tySequence: result = etySeq
  177. of tyObject, tyArray, tyTuple, tyOpenArray, tyVarargs, tyUncheckedArray:
  178. result = etyObject
  179. of tyNil: result = etyNull
  180. of tyGenericParam, tyGenericBody, tyGenericInvocation,
  181. tyNone, tyFromExpr, tyForward, tyEmpty,
  182. tyUntyped, tyTyped, tyTypeDesc, tyBuiltInTypeClass, tyCompositeTypeClass,
  183. tyAnd, tyOr, tyNot, tyAnything, tyVoid:
  184. result = etyNone
  185. of tyGenericInst, tyInferred, tyAlias, tyUserTypeClass, tyUserTypeClassInst,
  186. tySink, tyOwned:
  187. result = mapType(typ.lastSon)
  188. of tyStatic:
  189. if t.n != nil: result = mapType(lastSon t)
  190. else: result = etyNone
  191. of tyProc: result = etyProc
  192. of tyCstring: result = etyString
  193. of tyConcept, tyIterable: doAssert false
  194. proc mapType(p: PProc; typ: PType): TJSTypeKind =
  195. result = mapType(typ)
  196. proc mangleName(m: BModule, s: PSym): Rope =
  197. proc validJsName(name: string): bool =
  198. result = true
  199. const reservedWords = ["abstract", "await", "boolean", "break", "byte",
  200. "case", "catch", "char", "class", "const", "continue", "debugger",
  201. "default", "delete", "do", "double", "else", "enum", "export", "extends",
  202. "false", "final", "finally", "float", "for", "function", "goto", "if",
  203. "implements", "import", "in", "instanceof", "int", "interface", "let",
  204. "long", "native", "new", "null", "package", "private", "protected",
  205. "public", "return", "short", "static", "super", "switch", "synchronized",
  206. "this", "throw", "throws", "transient", "true", "try", "typeof", "var",
  207. "void", "volatile", "while", "with", "yield"]
  208. case name
  209. of reservedWords:
  210. return false
  211. else:
  212. discard
  213. if name[0] in {'0'..'9'}: return false
  214. for chr in name:
  215. if chr notin {'A'..'Z','a'..'z','_','$','0'..'9'}:
  216. return false
  217. result = s.loc.r
  218. if result == "":
  219. if s.kind == skField and s.name.s.validJsName:
  220. result = rope(s.name.s)
  221. elif s.kind == skTemp:
  222. result = rope(mangle(s.name.s))
  223. else:
  224. var x = newStringOfCap(s.name.s.len)
  225. var i = 0
  226. while i < s.name.s.len:
  227. let c = s.name.s[i]
  228. case c
  229. of 'A'..'Z', 'a'..'z', '_', '0'..'9':
  230. x.add c
  231. else:
  232. x.add("HEX" & toHex(ord(c), 2))
  233. inc i
  234. result = rope(x)
  235. # From ES5 on reserved words can be used as object field names
  236. if s.kind != skField:
  237. if m.config.hcrOn:
  238. # When hot reloading is enabled, we must ensure that the names
  239. # of functions and types will be preserved across rebuilds:
  240. result.add(idOrSig(s, m.module.name.s, m.sigConflicts, m.config))
  241. else:
  242. result.add("_")
  243. result.add(rope(s.id))
  244. s.loc.r = result
  245. proc escapeJSString(s: string): string =
  246. result = newStringOfCap(s.len + s.len shr 2)
  247. result.add("\"")
  248. for c in items(s):
  249. case c
  250. of '\l': result.add("\\n")
  251. of '\r': result.add("\\r")
  252. of '\t': result.add("\\t")
  253. of '\b': result.add("\\b")
  254. of '\a': result.add("\\a")
  255. of '\e': result.add("\\e")
  256. of '\v': result.add("\\v")
  257. of '\\': result.add("\\\\")
  258. of '\"': result.add("\\\"")
  259. else: result.add(c)
  260. result.add("\"")
  261. proc makeJSString(s: string, escapeNonAscii = true): Rope =
  262. if escapeNonAscii:
  263. result = strutils.escape(s).rope
  264. else:
  265. result = escapeJSString(s).rope
  266. proc makeJsNimStrLit(s: string): Rope =
  267. var x = newStringOfCap(4*s.len+1)
  268. x.add "["
  269. var i = 0
  270. if i < s.len:
  271. x.addInt int64(s[i])
  272. inc i
  273. while i < s.len:
  274. x.add ","
  275. x.addInt int64(s[i])
  276. inc i
  277. x.add "]"
  278. result = rope(x)
  279. include jstypes
  280. proc gen(p: PProc, n: PNode, r: var TCompRes)
  281. proc genStmt(p: PProc, n: PNode)
  282. proc genProc(oldProc: PProc, prc: PSym): Rope
  283. proc genConstant(p: PProc, c: PSym)
  284. proc useMagic(p: PProc, name: string) =
  285. if name.len == 0: return
  286. var s = magicsys.getCompilerProc(p.module.graph, name)
  287. if s != nil:
  288. internalAssert p.config, s.kind in {skProc, skFunc, skMethod, skConverter}
  289. if not p.g.generatedSyms.containsOrIncl(s.id):
  290. let code = genProc(p, s)
  291. p.g.constants.add(code)
  292. else:
  293. if p.prc != nil:
  294. globalError(p.config, p.prc.info, "system module needs: " & name)
  295. else:
  296. rawMessage(p.config, errGenerated, "system module needs: " & name)
  297. proc isSimpleExpr(p: PProc; n: PNode): bool =
  298. # calls all the way down --> can stay expression based
  299. case n.kind
  300. of nkCallKinds, nkBracketExpr, nkDotExpr, nkPar, nkTupleConstr,
  301. nkObjConstr, nkBracket, nkCurly,
  302. nkDerefExpr, nkHiddenDeref, nkAddr, nkHiddenAddr,
  303. nkConv, nkHiddenStdConv, nkHiddenSubConv:
  304. for c in n:
  305. if not p.isSimpleExpr(c): return false
  306. result = true
  307. of nkStmtListExpr:
  308. for i in 0..<n.len-1:
  309. if n[i].kind notin {nkCommentStmt, nkEmpty}: return false
  310. result = isSimpleExpr(p, n.lastSon)
  311. else:
  312. if n.isAtom:
  313. result = true
  314. proc getTemp(p: PProc, defineInLocals: bool = true): Rope =
  315. inc(p.unique)
  316. result = "Temporary$1" % [rope(p.unique)]
  317. if defineInLocals:
  318. p.locals.add(p.indentLine("var $1;$n" % [result]))
  319. proc genAnd(p: PProc, a, b: PNode, r: var TCompRes) =
  320. assert r.kind == resNone
  321. var x, y: TCompRes
  322. if p.isSimpleExpr(a) and p.isSimpleExpr(b):
  323. gen(p, a, x)
  324. gen(p, b, y)
  325. r.kind = resExpr
  326. r.res = "($1 && $2)" % [x.rdLoc, y.rdLoc]
  327. else:
  328. r.res = p.getTemp
  329. r.kind = resVal
  330. # while a and b:
  331. # -->
  332. # while true:
  333. # aa
  334. # if not a: tmp = false
  335. # else:
  336. # bb
  337. # tmp = b
  338. # tmp
  339. gen(p, a, x)
  340. lineF(p, "if (!$1) $2 = false; else {", [x.rdLoc, r.rdLoc])
  341. p.nested:
  342. gen(p, b, y)
  343. lineF(p, "$2 = $1;", [y.rdLoc, r.rdLoc])
  344. line(p, "}")
  345. proc genOr(p: PProc, a, b: PNode, r: var TCompRes) =
  346. assert r.kind == resNone
  347. var x, y: TCompRes
  348. if p.isSimpleExpr(a) and p.isSimpleExpr(b):
  349. gen(p, a, x)
  350. gen(p, b, y)
  351. r.kind = resExpr
  352. r.res = "($1 || $2)" % [x.rdLoc, y.rdLoc]
  353. else:
  354. r.res = p.getTemp
  355. r.kind = resVal
  356. gen(p, a, x)
  357. lineF(p, "if ($1) $2 = true; else {", [x.rdLoc, r.rdLoc])
  358. p.nested:
  359. gen(p, b, y)
  360. lineF(p, "$2 = $1;", [y.rdLoc, r.rdLoc])
  361. line(p, "}")
  362. type
  363. TMagicFrmt = array[0..1, string]
  364. TMagicOps = array[mAddI..mStrToStr, TMagicFrmt]
  365. const # magic checked op; magic unchecked op;
  366. jsMagics: TMagicOps = [
  367. mAddI: ["addInt", ""],
  368. mSubI: ["subInt", ""],
  369. mMulI: ["mulInt", ""],
  370. mDivI: ["divInt", ""],
  371. mModI: ["modInt", ""],
  372. mSucc: ["addInt", ""],
  373. mPred: ["subInt", ""],
  374. mAddF64: ["", ""],
  375. mSubF64: ["", ""],
  376. mMulF64: ["", ""],
  377. mDivF64: ["", ""],
  378. mShrI: ["", ""],
  379. mShlI: ["", ""],
  380. mAshrI: ["", ""],
  381. mBitandI: ["", ""],
  382. mBitorI: ["", ""],
  383. mBitxorI: ["", ""],
  384. mMinI: ["nimMin", "nimMin"],
  385. mMaxI: ["nimMax", "nimMax"],
  386. mAddU: ["", ""],
  387. mSubU: ["", ""],
  388. mMulU: ["", ""],
  389. mDivU: ["", ""],
  390. mModU: ["", ""],
  391. mEqI: ["", ""],
  392. mLeI: ["", ""],
  393. mLtI: ["", ""],
  394. mEqF64: ["", ""],
  395. mLeF64: ["", ""],
  396. mLtF64: ["", ""],
  397. mLeU: ["", ""],
  398. mLtU: ["", ""],
  399. mEqEnum: ["", ""],
  400. mLeEnum: ["", ""],
  401. mLtEnum: ["", ""],
  402. mEqCh: ["", ""],
  403. mLeCh: ["", ""],
  404. mLtCh: ["", ""],
  405. mEqB: ["", ""],
  406. mLeB: ["", ""],
  407. mLtB: ["", ""],
  408. mEqRef: ["", ""],
  409. mLePtr: ["", ""],
  410. mLtPtr: ["", ""],
  411. mXor: ["", ""],
  412. mEqCString: ["", ""],
  413. mEqProc: ["", ""],
  414. mUnaryMinusI: ["negInt", ""],
  415. mUnaryMinusI64: ["negInt64", ""],
  416. mAbsI: ["absInt", ""],
  417. mNot: ["", ""],
  418. mUnaryPlusI: ["", ""],
  419. mBitnotI: ["", ""],
  420. mUnaryPlusF64: ["", ""],
  421. mUnaryMinusF64: ["", ""],
  422. mCharToStr: ["nimCharToStr", "nimCharToStr"],
  423. mBoolToStr: ["nimBoolToStr", "nimBoolToStr"],
  424. mIntToStr: ["cstrToNimstr", "cstrToNimstr"],
  425. mInt64ToStr: ["cstrToNimstr", "cstrToNimstr"],
  426. mFloatToStr: ["cstrToNimstr", "cstrToNimstr"],
  427. mCStrToStr: ["cstrToNimstr", "cstrToNimstr"],
  428. mStrToStr: ["", ""]]
  429. proc needsTemp(p: PProc; n: PNode): bool =
  430. # check if n contains a call to determine
  431. # if a temp should be made to prevent multiple evals
  432. if n.kind in nkCallKinds + {nkTupleConstr, nkObjConstr, nkBracket, nkCurly}:
  433. return true
  434. for c in n:
  435. if needsTemp(p, c):
  436. return true
  437. proc maybeMakeTemp(p: PProc, n: PNode; x: TCompRes): tuple[a, tmp: Rope] =
  438. var
  439. a = x.rdLoc
  440. b = a
  441. if needsTemp(p, n):
  442. # if we have tmp just use it
  443. if x.tmpLoc != "" and (mapType(n.typ) == etyBaseIndex or n.kind in {nkHiddenDeref, nkDerefExpr}):
  444. b = "$1[0][$1[1]]" % [x.tmpLoc]
  445. (a: a, tmp: b)
  446. else:
  447. let tmp = p.getTemp
  448. b = tmp
  449. a = "($1 = $2, $1)" % [tmp, a]
  450. (a: a, tmp: b)
  451. else:
  452. (a: a, tmp: b)
  453. proc maybeMakeTempAssignable(p: PProc, n: PNode; x: TCompRes): tuple[a, tmp: Rope] =
  454. var
  455. a = x.rdLoc
  456. b = a
  457. if needsTemp(p, n):
  458. # if we have tmp just use it
  459. if x.tmpLoc != "" and (mapType(n.typ) == etyBaseIndex or n.kind in {nkHiddenDeref, nkDerefExpr}):
  460. b = "$1[0][$1[1]]" % [x.tmpLoc]
  461. result = (a: a, tmp: b)
  462. elif x.tmpLoc != "" and n.kind == nkBracketExpr:
  463. # genArrayAddr
  464. var
  465. address, index: TCompRes
  466. first: Int128
  467. gen(p, n[0], address)
  468. gen(p, n[1], index)
  469. let (m1, tmp1) = maybeMakeTemp(p, n[0], address)
  470. let typ = skipTypes(n[0].typ, abstractPtrs)
  471. if typ.kind == tyArray:
  472. first = firstOrd(p.config, typ[0])
  473. if optBoundsCheck in p.options:
  474. useMagic(p, "chckIndx")
  475. if first == 0: # save a couple chars
  476. index.res = "chckIndx($1, 0, ($2).length - 1)" % [index.res, tmp1]
  477. else:
  478. index.res = "chckIndx($1, $2, ($3).length + ($2) - 1) - ($2)" % [
  479. index.res, rope(first), tmp1]
  480. elif first != 0:
  481. index.res = "($1) - ($2)" % [index.res, rope(first)]
  482. else:
  483. discard # index.res = index.res
  484. let (n1, tmp2) = maybeMakeTemp(p, n[1], index)
  485. result = (a: "$1[$2]" % [m1, n1], tmp: "$1[$2]" % [tmp1, tmp2])
  486. # could also put here: nkDotExpr -> genFieldAccess, nkCheckedFieldExpr -> genCheckedFieldOp
  487. # but the uses of maybeMakeTempAssignable don't need them
  488. else:
  489. result = (a: a, tmp: b)
  490. else:
  491. result = (a: a, tmp: b)
  492. template binaryExpr(p: PProc, n: PNode, r: var TCompRes, magic, frmt: string,
  493. reassign = false) =
  494. # $1 and $2 in the `frmt` string bind to lhs and rhs of the expr,
  495. # if $3 or $4 are present they will be substituted with temps for
  496. # lhs and rhs respectively
  497. var x, y: TCompRes
  498. useMagic(p, magic)
  499. gen(p, n[1], x)
  500. gen(p, n[2], y)
  501. var
  502. a, tmp = x.rdLoc
  503. b, tmp2 = y.rdLoc
  504. when reassign:
  505. (a, tmp) = maybeMakeTempAssignable(p, n[1], x)
  506. else:
  507. when "$3" in frmt: (a, tmp) = maybeMakeTemp(p, n[1], x)
  508. when "$4" in frmt: (b, tmp2) = maybeMakeTemp(p, n[2], y)
  509. r.res = frmt % [a, b, tmp, tmp2]
  510. r.kind = resExpr
  511. proc unsignedTrimmer(size: BiggestInt): string =
  512. case size
  513. of 1: "& 0xff"
  514. of 2: "& 0xffff"
  515. of 4: ">>> 0"
  516. else: ""
  517. proc signedTrimmer(size: BiggestInt): string =
  518. # sign extension is done by shifting to the left and then back to the right
  519. "<< $1 >> $1" % [$(32 - size * 8)]
  520. proc binaryUintExpr(p: PProc, n: PNode, r: var TCompRes, op: string,
  521. reassign: static[bool] = false) =
  522. var x, y: TCompRes
  523. gen(p, n[1], x)
  524. gen(p, n[2], y)
  525. let size = n[1].typ.skipTypes(abstractRange).size
  526. when reassign:
  527. let (a, tmp) = maybeMakeTempAssignable(p, n[1], x)
  528. if size == 8 and optJsBigInt64 in p.config.globalOptions:
  529. r.res = "$1 = BigInt.asUintN(64, ($4 $2 $3))" % [a, rope op, y.rdLoc, tmp]
  530. else:
  531. let trimmer = unsignedTrimmer(size)
  532. r.res = "$1 = (($5 $2 $3) $4)" % [a, rope op, y.rdLoc, trimmer, tmp]
  533. else:
  534. if size == 8 and optJsBigInt64 in p.config.globalOptions:
  535. r.res = "BigInt.asUintN(64, ($1 $2 $3))" % [x.rdLoc, rope op, y.rdLoc]
  536. else:
  537. let trimmer = unsignedTrimmer(size)
  538. r.res = "(($1 $2 $3) $4)" % [x.rdLoc, rope op, y.rdLoc, trimmer]
  539. r.kind = resExpr
  540. template ternaryExpr(p: PProc, n: PNode, r: var TCompRes, magic, frmt: string) =
  541. var x, y, z: TCompRes
  542. useMagic(p, magic)
  543. gen(p, n[1], x)
  544. gen(p, n[2], y)
  545. gen(p, n[3], z)
  546. r.res = frmt % [x.rdLoc, y.rdLoc, z.rdLoc]
  547. r.kind = resExpr
  548. template unaryExpr(p: PProc, n: PNode, r: var TCompRes, magic, frmt: string) =
  549. # $1 binds to n[1], if $2 is present it will be substituted to a tmp of $1
  550. useMagic(p, magic)
  551. gen(p, n[1], r)
  552. var a, tmp = r.rdLoc
  553. if "$2" in frmt: (a, tmp) = maybeMakeTemp(p, n[1], r)
  554. r.res = frmt % [a, tmp]
  555. r.kind = resExpr
  556. proc arithAux(p: PProc, n: PNode, r: var TCompRes, op: TMagic) =
  557. var
  558. x, y: TCompRes
  559. xLoc, yLoc: Rope
  560. let i = ord(optOverflowCheck notin p.options)
  561. useMagic(p, jsMagics[op][i])
  562. if n.len > 2:
  563. gen(p, n[1], x)
  564. gen(p, n[2], y)
  565. xLoc = x.rdLoc
  566. yLoc = y.rdLoc
  567. else:
  568. gen(p, n[1], r)
  569. xLoc = r.rdLoc
  570. template applyFormat(frmt) =
  571. r.res = frmt % [xLoc, yLoc]
  572. template applyFormat(frmtA, frmtB) =
  573. if i == 0: applyFormat(frmtA) else: applyFormat(frmtB)
  574. template bitwiseExpr(op: string) =
  575. let typ = n[1].typ.skipTypes(abstractVarRange)
  576. if typ.kind in {tyUInt, tyUInt32}:
  577. r.res = "(($1 $2 $3) >>> 0)" % [xLoc, op, yLoc]
  578. else:
  579. r.res = "($1 $2 $3)" % [xLoc, op, yLoc]
  580. case op
  581. of mAddI:
  582. if i == 0:
  583. if n[1].typ.size == 8 and optJsBigInt64 in p.config.globalOptions:
  584. useMagic(p, "addInt64")
  585. applyFormat("addInt64($1, $2)")
  586. else:
  587. applyFormat("addInt($1, $2)")
  588. else:
  589. applyFormat("($1 + $2)")
  590. of mSubI:
  591. if i == 0:
  592. if n[1].typ.size == 8 and optJsBigInt64 in p.config.globalOptions:
  593. useMagic(p, "subInt64")
  594. applyFormat("subInt64($1, $2)")
  595. else:
  596. applyFormat("subInt($1, $2)")
  597. else:
  598. applyFormat("($1 - $2)")
  599. of mMulI:
  600. if i == 0:
  601. if n[1].typ.size == 8 and optJsBigInt64 in p.config.globalOptions:
  602. useMagic(p, "mulInt64")
  603. applyFormat("mulInt64($1, $2)")
  604. else:
  605. applyFormat("mulInt($1, $2)")
  606. else:
  607. applyFormat("($1 * $2)")
  608. of mDivI:
  609. if n[1].typ.size == 8 and optJsBigInt64 in p.config.globalOptions:
  610. useMagic(p, "divInt64")
  611. applyFormat("divInt64($1, $2)", "$1 / $2")
  612. else:
  613. applyFormat("divInt($1, $2)", "Math.trunc($1 / $2)")
  614. of mModI:
  615. if n[1].typ.size == 8 and optJsBigInt64 in p.config.globalOptions:
  616. useMagic(p, "modInt64")
  617. applyFormat("modInt64($1, $2)", "$1 % $2")
  618. else:
  619. applyFormat("modInt($1, $2)", "Math.trunc($1 % $2)")
  620. of mSucc: applyFormat("addInt($1, $2)", "($1 + $2)")
  621. of mPred: applyFormat("subInt($1, $2)", "($1 - $2)")
  622. of mAddF64: applyFormat("($1 + $2)", "($1 + $2)")
  623. of mSubF64: applyFormat("($1 - $2)", "($1 - $2)")
  624. of mMulF64: applyFormat("($1 * $2)", "($1 * $2)")
  625. of mDivF64: applyFormat("($1 / $2)", "($1 / $2)")
  626. of mShrI:
  627. let typ = n[1].typ.skipTypes(abstractVarRange)
  628. if typ.kind == tyInt64 and optJsBigInt64 in p.config.globalOptions:
  629. applyFormat("BigInt.asIntN(64, BigInt.asUintN(64, $1) >> BigInt($2))")
  630. elif typ.kind == tyUInt64 and optJsBigInt64 in p.config.globalOptions:
  631. applyFormat("($1 >> BigInt($2))")
  632. else:
  633. if typ.kind in {tyInt..tyInt32}:
  634. let trimmerU = unsignedTrimmer(typ.size)
  635. let trimmerS = signedTrimmer(typ.size)
  636. r.res = "((($1 $2) >>> $3) $4)" % [xLoc, trimmerU, yLoc, trimmerS]
  637. else:
  638. applyFormat("($1 >>> $2)")
  639. of mShlI:
  640. let typ = n[1].typ.skipTypes(abstractVarRange)
  641. if typ.size == 8:
  642. if typ.kind == tyInt64 and optJsBigInt64 in p.config.globalOptions:
  643. applyFormat("BigInt.asIntN(64, $1 << BigInt($2))")
  644. elif typ.kind == tyUInt64 and optJsBigInt64 in p.config.globalOptions:
  645. applyFormat("BigInt.asUintN(64, $1 << BigInt($2))")
  646. else:
  647. applyFormat("($1 * Math.pow(2, $2))")
  648. else:
  649. if typ.kind in {tyUInt..tyUInt32}:
  650. let trimmer = unsignedTrimmer(typ.size)
  651. r.res = "(($1 << $2) $3)" % [xLoc, yLoc, trimmer]
  652. else:
  653. let trimmer = signedTrimmer(typ.size)
  654. r.res = "(($1 << $2) $3)" % [xLoc, yLoc, trimmer]
  655. of mAshrI:
  656. let typ = n[1].typ.skipTypes(abstractVarRange)
  657. if typ.size == 8:
  658. if optJsBigInt64 in p.config.globalOptions:
  659. applyFormat("($1 >> BigInt($2))")
  660. else:
  661. applyFormat("Math.floor($1 / Math.pow(2, $2))")
  662. else:
  663. if typ.kind in {tyUInt..tyUInt32}:
  664. applyFormat("($1 >>> $2)")
  665. else:
  666. applyFormat("($1 >> $2)")
  667. of mBitandI: bitwiseExpr("&")
  668. of mBitorI: bitwiseExpr("|")
  669. of mBitxorI: bitwiseExpr("^")
  670. of mMinI: applyFormat("nimMin($1, $2)", "nimMin($1, $2)")
  671. of mMaxI: applyFormat("nimMax($1, $2)", "nimMax($1, $2)")
  672. of mAddU: applyFormat("", "")
  673. of mSubU: applyFormat("", "")
  674. of mMulU: applyFormat("", "")
  675. of mDivU: applyFormat("", "")
  676. of mModU: applyFormat("($1 % $2)", "($1 % $2)")
  677. of mEqI: applyFormat("($1 == $2)", "($1 == $2)")
  678. of mLeI: applyFormat("($1 <= $2)", "($1 <= $2)")
  679. of mLtI: applyFormat("($1 < $2)", "($1 < $2)")
  680. of mEqF64: applyFormat("($1 == $2)", "($1 == $2)")
  681. of mLeF64: applyFormat("($1 <= $2)", "($1 <= $2)")
  682. of mLtF64: applyFormat("($1 < $2)", "($1 < $2)")
  683. of mLeU: applyFormat("($1 <= $2)", "($1 <= $2)")
  684. of mLtU: applyFormat("($1 < $2)", "($1 < $2)")
  685. of mEqEnum: applyFormat("($1 == $2)", "($1 == $2)")
  686. of mLeEnum: applyFormat("($1 <= $2)", "($1 <= $2)")
  687. of mLtEnum: applyFormat("($1 < $2)", "($1 < $2)")
  688. of mEqCh: applyFormat("($1 == $2)", "($1 == $2)")
  689. of mLeCh: applyFormat("($1 <= $2)", "($1 <= $2)")
  690. of mLtCh: applyFormat("($1 < $2)", "($1 < $2)")
  691. of mEqB: applyFormat("($1 == $2)", "($1 == $2)")
  692. of mLeB: applyFormat("($1 <= $2)", "($1 <= $2)")
  693. of mLtB: applyFormat("($1 < $2)", "($1 < $2)")
  694. of mEqRef: applyFormat("($1 == $2)", "($1 == $2)")
  695. of mLePtr: applyFormat("($1 <= $2)", "($1 <= $2)")
  696. of mLtPtr: applyFormat("($1 < $2)", "($1 < $2)")
  697. of mXor: applyFormat("($1 != $2)", "($1 != $2)")
  698. of mEqCString: applyFormat("($1 == $2)", "($1 == $2)")
  699. of mEqProc: applyFormat("($1 == $2)", "($1 == $2)")
  700. of mUnaryMinusI: applyFormat("negInt($1)", "-($1)")
  701. of mUnaryMinusI64: applyFormat("negInt64($1)", "-($1)")
  702. of mAbsI: applyFormat("absInt($1)", "Math.abs($1)")
  703. of mNot: applyFormat("!($1)", "!($1)")
  704. of mUnaryPlusI: applyFormat("+($1)", "+($1)")
  705. of mBitnotI:
  706. let typ = n[1].typ.skipTypes(abstractVarRange)
  707. if typ.kind in {tyUInt..tyUInt64}:
  708. if typ.size == 8 and optJsBigInt64 in p.config.globalOptions:
  709. applyFormat("BigInt.asUintN(64, ~($1))")
  710. else:
  711. let trimmer = unsignedTrimmer(typ.size)
  712. r.res = "(~($1) $2)" % [xLoc, trimmer]
  713. else:
  714. applyFormat("~($1)")
  715. of mUnaryPlusF64: applyFormat("+($1)", "+($1)")
  716. of mUnaryMinusF64: applyFormat("-($1)", "-($1)")
  717. of mCharToStr: applyFormat("nimCharToStr($1)", "nimCharToStr($1)")
  718. of mBoolToStr: applyFormat("nimBoolToStr($1)", "nimBoolToStr($1)")
  719. of mIntToStr: applyFormat("cstrToNimstr(($1) + \"\")", "cstrToNimstr(($1) + \"\")")
  720. of mInt64ToStr: applyFormat("cstrToNimstr(($1) + \"\")", "cstrToNimstr(($1) + \"\")")
  721. of mCStrToStr: applyFormat("cstrToNimstr($1)", "cstrToNimstr($1)")
  722. of mStrToStr, mUnown, mIsolate, mFinished: applyFormat("$1", "$1")
  723. else:
  724. assert false, $op
  725. proc arith(p: PProc, n: PNode, r: var TCompRes, op: TMagic) =
  726. case op
  727. of mAddU: binaryUintExpr(p, n, r, "+")
  728. of mSubU: binaryUintExpr(p, n, r, "-")
  729. of mMulU: binaryUintExpr(p, n, r, "*")
  730. of mDivU:
  731. binaryUintExpr(p, n, r, "/")
  732. if optJsBigInt64 notin p.config.globalOptions and
  733. n[1].typ.skipTypes(abstractRange).size == 8:
  734. # bigint / already truncates
  735. r.res = "Math.trunc($1)" % [r.res]
  736. of mDivI:
  737. arithAux(p, n, r, op)
  738. of mModI:
  739. arithAux(p, n, r, op)
  740. of mCharToStr, mBoolToStr, mIntToStr, mInt64ToStr, mCStrToStr, mStrToStr, mEnumToStr:
  741. arithAux(p, n, r, op)
  742. of mEqRef:
  743. if mapType(n[1].typ) != etyBaseIndex:
  744. arithAux(p, n, r, op)
  745. else:
  746. var x, y: TCompRes
  747. gen(p, n[1], x)
  748. gen(p, n[2], y)
  749. r.res = "($# == $# && $# == $#)" % [x.address, y.address, x.res, y.res]
  750. else:
  751. arithAux(p, n, r, op)
  752. r.kind = resExpr
  753. proc hasFrameInfo(p: PProc): bool =
  754. ({optLineTrace, optStackTrace} * p.options == {optLineTrace, optStackTrace}) and
  755. ((p.prc == nil) or not (sfPure in p.prc.flags))
  756. proc lineDir(config: ConfigRef, info: TLineInfo, line: int): Rope =
  757. "/* line $2:$3 \"$1\" */$n" % [
  758. rope(toFullPath(config, info)), rope(line), rope(info.toColumn)
  759. ]
  760. proc genLineDir(p: PProc, n: PNode) =
  761. let line = toLinenumber(n.info)
  762. if line < 0:
  763. return
  764. if optEmbedOrigSrc in p.config.globalOptions:
  765. lineF(p, "//$1$n", [sourceLine(p.config, n.info)])
  766. if optLineDir in p.options or optLineDir in p.config.options:
  767. lineF(p, "$1", [lineDir(p.config, n.info, line)])
  768. if hasFrameInfo(p):
  769. lineF(p, "F.line = $1;$n", [rope(line)])
  770. let currentFileName = toFilename(p.config, n.info)
  771. if p.previousFileName != currentFileName:
  772. lineF(p, "F.filename = $1;$n", [makeJSString(currentFileName)])
  773. p.previousFileName = currentFileName
  774. proc genWhileStmt(p: PProc, n: PNode) =
  775. var cond: TCompRes
  776. internalAssert p.config, isEmptyType(n.typ)
  777. genLineDir(p, n)
  778. inc(p.unique)
  779. setLen(p.blocks, p.blocks.len + 1)
  780. p.blocks[^1].id = -p.unique
  781. p.blocks[^1].isLoop = true
  782. let labl = p.unique.rope
  783. lineF(p, "Label$1: while (true) {$n", [labl])
  784. p.nested: gen(p, n[0], cond)
  785. lineF(p, "if (!$1) break Label$2;$n",
  786. [cond.res, labl])
  787. p.nested: genStmt(p, n[1])
  788. lineF(p, "}$n", [labl])
  789. setLen(p.blocks, p.blocks.len - 1)
  790. proc moveInto(p: PProc, src: var TCompRes, dest: TCompRes) =
  791. if src.kind != resNone:
  792. if dest.kind != resNone:
  793. lineF(p, "$1 = $2;$n", [dest.rdLoc, src.rdLoc])
  794. else:
  795. lineF(p, "$1;$n", [src.rdLoc])
  796. src.kind = resNone
  797. src.res = ""
  798. proc genTry(p: PProc, n: PNode, r: var TCompRes) =
  799. # code to generate:
  800. #
  801. # ++excHandler;
  802. # var tmpFramePtr = framePtr;
  803. # try {
  804. # stmts;
  805. # --excHandler;
  806. # } catch (EXCEPTION) {
  807. # var prevJSError = lastJSError; lastJSError = EXCEPTION;
  808. # framePtr = tmpFramePtr;
  809. # --excHandler;
  810. # if (e.typ && e.typ == NTI433 || e.typ == NTI2321) {
  811. # stmts;
  812. # } else if (e.typ && e.typ == NTI32342) {
  813. # stmts;
  814. # } else {
  815. # stmts;
  816. # }
  817. # lastJSError = prevJSError;
  818. # } finally {
  819. # framePtr = tmpFramePtr;
  820. # stmts;
  821. # }
  822. genLineDir(p, n)
  823. if not isEmptyType(n.typ):
  824. r.kind = resVal
  825. r.res = getTemp(p)
  826. inc(p.unique)
  827. var i = 1
  828. var catchBranchesExist = n.len > 1 and n[i].kind == nkExceptBranch
  829. if catchBranchesExist:
  830. p.body.add("++excHandler;\L")
  831. var tmpFramePtr = rope"F"
  832. lineF(p, "try {$n", [])
  833. var a: TCompRes = default(TCompRes)
  834. gen(p, n[0], a)
  835. moveInto(p, a, r)
  836. var generalCatchBranchExists = false
  837. if catchBranchesExist:
  838. p.body.addf("--excHandler;$n} catch (EXCEPTION) {$n var prevJSError = lastJSError;$n" &
  839. " lastJSError = EXCEPTION;$n --excHandler;$n", [])
  840. if hasFrameInfo(p):
  841. line(p, "framePtr = $1;$n" % [tmpFramePtr])
  842. while i < n.len and n[i].kind == nkExceptBranch:
  843. if n[i].len == 1:
  844. # general except section:
  845. generalCatchBranchExists = true
  846. if i > 1: lineF(p, "else {$n", [])
  847. gen(p, n[i][0], a)
  848. moveInto(p, a, r)
  849. if i > 1: lineF(p, "}$n", [])
  850. else:
  851. var orExpr: Rope = ""
  852. var excAlias: PNode = nil
  853. useMagic(p, "isObj")
  854. for j in 0..<n[i].len - 1:
  855. var throwObj: PNode
  856. let it = n[i][j]
  857. if it.isInfixAs():
  858. throwObj = it[1]
  859. excAlias = it[2]
  860. # If this is a ``except exc as sym`` branch there must be no following
  861. # nodes
  862. doAssert orExpr == ""
  863. elif it.kind == nkType:
  864. throwObj = it
  865. else:
  866. internalError(p.config, n.info, "genTryStmt")
  867. if orExpr != "": orExpr.add("||")
  868. # Generate the correct type checking code depending on whether this is a
  869. # NIM-native or a JS-native exception
  870. # if isJsObject(throwObj.typ):
  871. if isImportedException(throwObj.typ, p.config):
  872. orExpr.addf("lastJSError instanceof $1",
  873. [throwObj.typ.sym.loc.r])
  874. else:
  875. orExpr.addf("isObj(lastJSError.m_type, $1)",
  876. [genTypeInfo(p, throwObj.typ)])
  877. if i > 1: line(p, "else ")
  878. lineF(p, "if (lastJSError && ($1)) {$n", [orExpr])
  879. # If some branch requires a local alias introduce it here. This is needed
  880. # since JS cannot do ``catch x as y``.
  881. if excAlias != nil:
  882. excAlias.sym.loc.r = mangleName(p.module, excAlias.sym)
  883. lineF(p, "var $1 = lastJSError;$n", excAlias.sym.loc.r)
  884. gen(p, n[i][^1], a)
  885. moveInto(p, a, r)
  886. lineF(p, "}$n", [])
  887. inc(i)
  888. if catchBranchesExist:
  889. if not generalCatchBranchExists:
  890. useMagic(p, "reraiseException")
  891. line(p, "else {\L")
  892. line(p, "\treraiseException();\L")
  893. line(p, "}\L")
  894. lineF(p, "lastJSError = prevJSError;$n")
  895. line(p, "} finally {\L")
  896. if hasFrameInfo(p):
  897. line(p, "framePtr = $1;$n" % [tmpFramePtr])
  898. if i < n.len and n[i].kind == nkFinally:
  899. genStmt(p, n[i][0])
  900. line(p, "}\L")
  901. proc genRaiseStmt(p: PProc, n: PNode) =
  902. if n[0].kind != nkEmpty:
  903. var a: TCompRes
  904. gen(p, n[0], a)
  905. let typ = skipTypes(n[0].typ, abstractPtrs)
  906. genLineDir(p, n)
  907. useMagic(p, "raiseException")
  908. lineF(p, "raiseException($1, $2);$n",
  909. [a.rdLoc, makeJSString(typ.sym.name.s)])
  910. else:
  911. genLineDir(p, n)
  912. useMagic(p, "reraiseException")
  913. line(p, "reraiseException();\L")
  914. proc genCaseJS(p: PProc, n: PNode, r: var TCompRes) =
  915. var
  916. a, b, cond, stmt: TCompRes
  917. genLineDir(p, n)
  918. gen(p, n[0], cond)
  919. let typeKind = skipTypes(n[0].typ, abstractVar).kind
  920. var transferRange = false
  921. let anyString = typeKind in {tyString, tyCstring}
  922. case typeKind
  923. of tyString:
  924. useMagic(p, "toJSStr")
  925. lineF(p, "switch (toJSStr($1)) {$n", [cond.rdLoc])
  926. of tyFloat..tyFloat128, tyInt..tyInt64, tyUInt..tyUInt64:
  927. transferRange = true
  928. else:
  929. lineF(p, "switch ($1) {$n", [cond.rdLoc])
  930. if not isEmptyType(n.typ):
  931. r.kind = resVal
  932. r.res = getTemp(p)
  933. for i in 1..<n.len:
  934. let it = n[i]
  935. let itLen = it.len
  936. case it.kind
  937. of nkOfBranch:
  938. if transferRange:
  939. if i == 1:
  940. lineF(p, "if (", [])
  941. else:
  942. lineF(p, "else if (", [])
  943. for j in 0..<itLen - 1:
  944. let e = it[j]
  945. if e.kind == nkRange:
  946. if transferRange:
  947. gen(p, e[0], a)
  948. gen(p, e[1], b)
  949. if j != itLen - 2:
  950. lineF(p, "$1 >= $2 && $1 <= $3 || $n", [cond.rdLoc, a.rdLoc, b.rdLoc])
  951. else:
  952. lineF(p, "$1 >= $2 && $1 <= $3", [cond.rdLoc, a.rdLoc, b.rdLoc])
  953. else:
  954. var v = copyNode(e[0])
  955. while v.intVal <= e[1].intVal:
  956. gen(p, v, cond)
  957. lineF(p, "case $1:$n", [cond.rdLoc])
  958. inc(v.intVal)
  959. else:
  960. if anyString:
  961. case e.kind
  962. of nkStrLit..nkTripleStrLit: lineF(p, "case $1:$n",
  963. [makeJSString(e.strVal, false)])
  964. of nkNilLit: lineF(p, "case null:$n", [])
  965. else: internalError(p.config, e.info, "jsgen.genCaseStmt: 2")
  966. else:
  967. if transferRange:
  968. gen(p, e, a)
  969. if j != itLen - 2:
  970. lineF(p, "$1 == $2 || $n", [cond.rdLoc, a.rdLoc])
  971. else:
  972. lineF(p, "$1 == $2", [cond.rdLoc, a.rdLoc])
  973. else:
  974. gen(p, e, a)
  975. lineF(p, "case $1:$n", [a.rdLoc])
  976. if transferRange:
  977. lineF(p, "){", [])
  978. p.nested:
  979. gen(p, lastSon(it), stmt)
  980. moveInto(p, stmt, r)
  981. if transferRange:
  982. lineF(p, "}$n", [])
  983. else:
  984. lineF(p, "break;$n", [])
  985. of nkElse:
  986. if transferRange:
  987. lineF(p, "else{$n", [])
  988. else:
  989. lineF(p, "default: $n", [])
  990. p.nested:
  991. gen(p, it[0], stmt)
  992. moveInto(p, stmt, r)
  993. if transferRange:
  994. lineF(p, "}$n", [])
  995. else:
  996. lineF(p, "break;$n", [])
  997. else: internalError(p.config, it.info, "jsgen.genCaseStmt")
  998. if not transferRange:
  999. lineF(p, "}$n", [])
  1000. proc genBlock(p: PProc, n: PNode, r: var TCompRes) =
  1001. inc(p.unique)
  1002. let idx = p.blocks.len
  1003. if n[0].kind != nkEmpty:
  1004. # named block?
  1005. if (n[0].kind != nkSym): internalError(p.config, n.info, "genBlock")
  1006. var sym = n[0].sym
  1007. sym.loc.k = locOther
  1008. sym.position = idx+1
  1009. let labl = p.unique
  1010. lineF(p, "Label$1: {$n", [labl.rope])
  1011. setLen(p.blocks, idx + 1)
  1012. p.blocks[idx].id = - p.unique # negative because it isn't used yet
  1013. gen(p, n[1], r)
  1014. setLen(p.blocks, idx)
  1015. lineF(p, "};$n", [labl.rope])
  1016. proc genBreakStmt(p: PProc, n: PNode) =
  1017. var idx: int
  1018. genLineDir(p, n)
  1019. if n[0].kind != nkEmpty:
  1020. # named break?
  1021. assert(n[0].kind == nkSym)
  1022. let sym = n[0].sym
  1023. assert(sym.loc.k == locOther)
  1024. idx = sym.position-1
  1025. else:
  1026. # an unnamed 'break' can only break a loop after 'transf' pass:
  1027. idx = p.blocks.len - 1
  1028. while idx >= 0 and not p.blocks[idx].isLoop: dec idx
  1029. if idx < 0 or not p.blocks[idx].isLoop:
  1030. internalError(p.config, n.info, "no loop to break")
  1031. p.blocks[idx].id = abs(p.blocks[idx].id) # label is used
  1032. lineF(p, "break Label$1;$n", [rope(p.blocks[idx].id)])
  1033. proc genAsmOrEmitStmt(p: PProc, n: PNode) =
  1034. genLineDir(p, n)
  1035. p.body.add p.indentLine("")
  1036. for i in 0..<n.len:
  1037. let it = n[i]
  1038. case it.kind
  1039. of nkStrLit..nkTripleStrLit:
  1040. p.body.add(it.strVal)
  1041. of nkSym:
  1042. let v = it.sym
  1043. # for backwards compatibility we don't deref syms here :-(
  1044. if false:
  1045. discard
  1046. else:
  1047. var r: TCompRes
  1048. gen(p, it, r)
  1049. if it.typ.kind == tyPointer:
  1050. # A fat pointer is disguised as an array
  1051. r.res = r.address
  1052. r.address = ""
  1053. r.typ = etyNone
  1054. elif r.typ == etyBaseIndex:
  1055. # Deference first
  1056. r.res = "$1[$2]" % [r.address, r.res]
  1057. r.address = ""
  1058. r.typ = etyNone
  1059. p.body.add(r.rdLoc)
  1060. else:
  1061. var r: TCompRes
  1062. gen(p, it, r)
  1063. p.body.add(r.rdLoc)
  1064. p.body.add "\L"
  1065. proc genIf(p: PProc, n: PNode, r: var TCompRes) =
  1066. var cond, stmt: TCompRes
  1067. var toClose = 0
  1068. if not isEmptyType(n.typ):
  1069. r.kind = resVal
  1070. r.res = getTemp(p)
  1071. for i in 0..<n.len:
  1072. let it = n[i]
  1073. if it.len != 1:
  1074. if i > 0:
  1075. lineF(p, "else {$n", [])
  1076. inc(toClose)
  1077. p.nested: gen(p, it[0], cond)
  1078. lineF(p, "if ($1) {$n", [cond.rdLoc])
  1079. gen(p, it[1], stmt)
  1080. else:
  1081. # else part:
  1082. lineF(p, "else {$n", [])
  1083. p.nested: gen(p, it[0], stmt)
  1084. moveInto(p, stmt, r)
  1085. lineF(p, "}$n", [])
  1086. line(p, repeat('}', toClose) & "\L")
  1087. proc generateHeader(p: PProc, typ: PType): Rope =
  1088. result = ""
  1089. for i in 1..<typ.n.len:
  1090. assert(typ.n[i].kind == nkSym)
  1091. var param = typ.n[i].sym
  1092. if isCompileTimeOnly(param.typ): continue
  1093. if result != "": result.add(", ")
  1094. var name = mangleName(p.module, param)
  1095. result.add(name)
  1096. if mapType(param.typ) == etyBaseIndex:
  1097. result.add(", ")
  1098. result.add(name)
  1099. result.add("_Idx")
  1100. proc countJsParams(typ: PType): int =
  1101. for i in 1..<typ.n.len:
  1102. assert(typ.n[i].kind == nkSym)
  1103. var param = typ.n[i].sym
  1104. if isCompileTimeOnly(param.typ): continue
  1105. if mapType(param.typ) == etyBaseIndex:
  1106. inc result, 2
  1107. else:
  1108. inc result
  1109. const
  1110. nodeKindsNeedNoCopy = {nkCharLit..nkInt64Lit, nkStrLit..nkTripleStrLit,
  1111. nkFloatLit..nkFloat64Lit, nkPar, nkStringToCString,
  1112. nkObjConstr, nkTupleConstr, nkBracket,
  1113. nkCStringToString, nkCall, nkPrefix, nkPostfix, nkInfix,
  1114. nkCommand, nkHiddenCallConv, nkCallStrLit}
  1115. proc needsNoCopy(p: PProc; y: PNode): bool =
  1116. return y.kind in nodeKindsNeedNoCopy or
  1117. ((mapType(y.typ) != etyBaseIndex or (y.kind == nkSym and y.sym.kind == skParam)) and
  1118. (skipTypes(y.typ, abstractInst).kind in
  1119. {tyRef, tyPtr, tyLent, tyVar, tyCstring, tyProc, tyOwned, tyOpenArray} + IntegralTypes))
  1120. proc genAsgnAux(p: PProc, x, y: PNode, noCopyNeeded: bool) =
  1121. var a, b: TCompRes
  1122. var xtyp = mapType(p, x.typ)
  1123. # disable `[]=` for cstring
  1124. if x.kind == nkBracketExpr and x.len >= 2 and x[0].typ.skipTypes(abstractInst).kind == tyCstring:
  1125. localError(p.config, x.info, "cstring doesn't support `[]=` operator")
  1126. gen(p, x, a)
  1127. genLineDir(p, y)
  1128. gen(p, y, b)
  1129. # we don't care if it's an etyBaseIndex (global) of a string, it's
  1130. # still a string that needs to be copied properly:
  1131. if x.typ.skipTypes(abstractInst).kind in {tySequence, tyString}:
  1132. xtyp = etySeq
  1133. case xtyp
  1134. of etySeq:
  1135. if x.typ.kind in {tyVar, tyLent} or (needsNoCopy(p, y) and needsNoCopy(p, x)) or noCopyNeeded:
  1136. lineF(p, "$1 = $2;$n", [a.rdLoc, b.rdLoc])
  1137. else:
  1138. useMagic(p, "nimCopy")
  1139. lineF(p, "$1 = nimCopy(null, $2, $3);$n",
  1140. [a.rdLoc, b.res, genTypeInfo(p, y.typ)])
  1141. of etyObject:
  1142. if x.typ.kind in {tyVar, tyLent} or (needsNoCopy(p, y) and needsNoCopy(p, x)) or noCopyNeeded:
  1143. lineF(p, "$1 = $2;$n", [a.rdLoc, b.rdLoc])
  1144. else:
  1145. useMagic(p, "nimCopy")
  1146. # supports proc getF(): var T
  1147. if x.kind in {nkHiddenDeref, nkDerefExpr} and x[0].kind in nkCallKinds:
  1148. lineF(p, "nimCopy($1, $2, $3);$n",
  1149. [a.res, b.res, genTypeInfo(p, x.typ)])
  1150. else:
  1151. lineF(p, "$1 = nimCopy($1, $2, $3);$n",
  1152. [a.res, b.res, genTypeInfo(p, x.typ)])
  1153. of etyBaseIndex:
  1154. if a.typ != etyBaseIndex or b.typ != etyBaseIndex:
  1155. if y.kind == nkCall:
  1156. let tmp = p.getTemp(false)
  1157. lineF(p, "var $1 = $4; $2 = $1[0]; $3 = $1[1];$n", [tmp, a.address, a.res, b.rdLoc])
  1158. elif b.typ == etyBaseIndex:
  1159. lineF(p, "$# = [$#, $#];$n", [a.res, b.address, b.res])
  1160. elif b.typ == etyNone:
  1161. internalAssert p.config, b.address == ""
  1162. lineF(p, "$# = [$#, 0];$n", [a.address, b.res])
  1163. elif x.typ.kind == tyVar and y.typ.kind == tyPtr:
  1164. lineF(p, "$# = [$#, $#];$n", [a.res, b.address, b.res])
  1165. lineF(p, "$1 = $2;$n", [a.address, b.res])
  1166. lineF(p, "$1 = $2;$n", [a.rdLoc, b.rdLoc])
  1167. elif a.typ == etyBaseIndex:
  1168. # array indexing may not map to var type
  1169. if b.address != "":
  1170. lineF(p, "$1 = $2; $3 = $4;$n", [a.address, b.address, a.res, b.res])
  1171. else:
  1172. lineF(p, "$1 = $2;$n", [a.address, b.res])
  1173. else:
  1174. internalError(p.config, x.info, $("genAsgn", b.typ, a.typ))
  1175. elif b.address != "":
  1176. lineF(p, "$1 = $2; $3 = $4;$n", [a.address, b.address, a.res, b.res])
  1177. else:
  1178. lineF(p, "$1 = $2;$n", [a.address, b.res])
  1179. else:
  1180. lineF(p, "$1 = $2;$n", [a.rdLoc, b.rdLoc])
  1181. proc genAsgn(p: PProc, n: PNode) =
  1182. genAsgnAux(p, n[0], n[1], noCopyNeeded=false)
  1183. proc genFastAsgn(p: PProc, n: PNode) =
  1184. # 'shallowCopy' always produced 'noCopyNeeded = true' here but this is wrong
  1185. # for code like
  1186. # while j >= pos:
  1187. # dest[i].shallowCopy(dest[j])
  1188. # See bug #5933. So we try to be more compatible with the C backend semantics
  1189. # here for 'shallowCopy'. This is an educated guess and might require further
  1190. # changes later:
  1191. let noCopy = n[0].typ.skipTypes(abstractInst).kind in {tySequence, tyString}
  1192. genAsgnAux(p, n[0], n[1], noCopyNeeded=noCopy)
  1193. proc genSwap(p: PProc, n: PNode) =
  1194. var a, b: TCompRes
  1195. gen(p, n[1], a)
  1196. gen(p, n[2], b)
  1197. var tmp = p.getTemp(false)
  1198. if mapType(p, skipTypes(n[1].typ, abstractVar)) == etyBaseIndex:
  1199. let tmp2 = p.getTemp(false)
  1200. if a.typ != etyBaseIndex or b.typ != etyBaseIndex:
  1201. internalError(p.config, n.info, "genSwap")
  1202. lineF(p, "var $1 = $2; $2 = $3; $3 = $1;$n",
  1203. [tmp, a.address, b.address])
  1204. tmp = tmp2
  1205. lineF(p, "var $1 = $2; $2 = $3; $3 = $1;",
  1206. [tmp, a.res, b.res])
  1207. proc getFieldPosition(p: PProc; f: PNode): int =
  1208. case f.kind
  1209. of nkIntLit..nkUInt64Lit: result = int(f.intVal)
  1210. of nkSym: result = f.sym.position
  1211. else: internalError(p.config, f.info, "genFieldPosition")
  1212. proc genFieldAddr(p: PProc, n: PNode, r: var TCompRes) =
  1213. var a: TCompRes
  1214. r.typ = etyBaseIndex
  1215. let b = if n.kind == nkHiddenAddr: n[0] else: n
  1216. gen(p, b[0], a)
  1217. if skipTypes(b[0].typ, abstractVarRange).kind == tyTuple:
  1218. r.res = makeJSString("Field" & $getFieldPosition(p, b[1]))
  1219. else:
  1220. if b[1].kind != nkSym: internalError(p.config, b[1].info, "genFieldAddr")
  1221. var f = b[1].sym
  1222. if f.loc.r == "": f.loc.r = mangleName(p.module, f)
  1223. r.res = makeJSString($f.loc.r)
  1224. internalAssert p.config, a.typ != etyBaseIndex
  1225. r.address = a.res
  1226. r.kind = resExpr
  1227. proc genFieldAccess(p: PProc, n: PNode, r: var TCompRes) =
  1228. gen(p, n[0], r)
  1229. r.typ = mapType(n.typ)
  1230. let otyp = skipTypes(n[0].typ, abstractVarRange)
  1231. template mkTemp(i: int) =
  1232. if r.typ == etyBaseIndex:
  1233. if needsTemp(p, n[i]):
  1234. let tmp = p.getTemp
  1235. r.address = "($1 = $2, $1)[0]" % [tmp, r.res]
  1236. r.res = "$1[1]" % [tmp]
  1237. r.tmpLoc = tmp
  1238. else:
  1239. r.address = "$1[0]" % [r.res]
  1240. r.res = "$1[1]" % [r.res]
  1241. if otyp.kind == tyTuple:
  1242. r.res = ("$1.Field$2") %
  1243. [r.res, getFieldPosition(p, n[1]).rope]
  1244. mkTemp(0)
  1245. else:
  1246. if n[1].kind != nkSym: internalError(p.config, n[1].info, "genFieldAccess")
  1247. var f = n[1].sym
  1248. if f.loc.r == "": f.loc.r = mangleName(p.module, f)
  1249. r.res = "$1.$2" % [r.res, f.loc.r]
  1250. mkTemp(1)
  1251. r.kind = resExpr
  1252. proc genAddr(p: PProc, n: PNode, r: var TCompRes)
  1253. proc genCheckedFieldOp(p: PProc, n: PNode, addrTyp: PType, r: var TCompRes) =
  1254. internalAssert p.config, n.kind == nkCheckedFieldExpr
  1255. # nkDotExpr to access the requested field
  1256. let accessExpr = n[0]
  1257. # nkCall to check if the discriminant is valid
  1258. var checkExpr = n[1]
  1259. let negCheck = checkExpr[0].sym.magic == mNot
  1260. if negCheck:
  1261. checkExpr = checkExpr[^1]
  1262. # Field symbol
  1263. var field = accessExpr[1].sym
  1264. internalAssert p.config, field.kind == skField
  1265. if field.loc.r == "": field.loc.r = mangleName(p.module, field)
  1266. # Discriminant symbol
  1267. let disc = checkExpr[2].sym
  1268. internalAssert p.config, disc.kind == skField
  1269. if disc.loc.r == "": disc.loc.r = mangleName(p.module, disc)
  1270. var setx: TCompRes
  1271. gen(p, checkExpr[1], setx)
  1272. var obj: TCompRes
  1273. gen(p, accessExpr[0], obj)
  1274. # Avoid evaluating the LHS twice (one to read the discriminant and one to read
  1275. # the field)
  1276. let tmp = p.getTemp()
  1277. lineF(p, "var $1 = $2;$n", tmp, obj.res)
  1278. useMagic(p, "raiseFieldError2")
  1279. useMagic(p, "makeNimstrLit")
  1280. useMagic(p, "reprDiscriminant") # no need to offset by firstOrd unlike for cgen
  1281. let msg = genFieldDefect(p.config, field.name.s, disc)
  1282. lineF(p, "if ($1[$2.$3]$4undefined) { raiseFieldError2(makeNimstrLit($5), reprDiscriminant($2.$3, $6)); }$n",
  1283. setx.res, tmp, disc.loc.r, if negCheck: "!==" else: "===",
  1284. makeJSString(msg), genTypeInfo(p, disc.typ))
  1285. if addrTyp != nil and mapType(p, addrTyp) == etyBaseIndex:
  1286. r.typ = etyBaseIndex
  1287. r.res = makeJSString($field.loc.r)
  1288. r.address = tmp
  1289. else:
  1290. r.typ = etyNone
  1291. r.res = "$1.$2" % [tmp, field.loc.r]
  1292. r.kind = resExpr
  1293. proc genArrayAddr(p: PProc, n: PNode, r: var TCompRes) =
  1294. var
  1295. a, b: TCompRes
  1296. first: Int128
  1297. r.typ = etyBaseIndex
  1298. let m = if n.kind == nkHiddenAddr: n[0] else: n
  1299. gen(p, m[0], a)
  1300. gen(p, m[1], b)
  1301. #internalAssert p.config, a.typ != etyBaseIndex and b.typ != etyBaseIndex
  1302. let (x, tmp) = maybeMakeTemp(p, m[0], a)
  1303. r.address = x
  1304. var typ = skipTypes(m[0].typ, abstractPtrs)
  1305. if typ.kind == tyArray:
  1306. first = firstOrd(p.config, typ[0])
  1307. if optBoundsCheck in p.options:
  1308. useMagic(p, "chckIndx")
  1309. if first == 0: # save a couple chars
  1310. r.res = "chckIndx($1, 0, ($2).length - 1)" % [b.res, tmp]
  1311. else:
  1312. r.res = "chckIndx($1, $2, ($3).length + ($2) - 1) - ($2)" % [
  1313. b.res, rope(first), tmp]
  1314. elif first != 0:
  1315. r.res = "($1) - ($2)" % [b.res, rope(first)]
  1316. else:
  1317. r.res = b.res
  1318. r.kind = resExpr
  1319. proc genArrayAccess(p: PProc, n: PNode, r: var TCompRes) =
  1320. var ty = skipTypes(n[0].typ, abstractVarRange)
  1321. if ty.kind in {tyRef, tyPtr, tyLent, tyOwned}: ty = skipTypes(ty.lastSon, abstractVarRange)
  1322. case ty.kind
  1323. of tyArray, tyOpenArray, tySequence, tyString, tyCstring, tyVarargs:
  1324. genArrayAddr(p, n, r)
  1325. of tyTuple:
  1326. genFieldAddr(p, n, r)
  1327. else: internalError(p.config, n.info, "expr(nkBracketExpr, " & $ty.kind & ')')
  1328. r.typ = mapType(n.typ)
  1329. if r.res == "": internalError(p.config, n.info, "genArrayAccess")
  1330. if ty.kind == tyCstring:
  1331. r.res = "$1.charCodeAt($2)" % [r.address, r.res]
  1332. elif r.typ == etyBaseIndex:
  1333. if needsTemp(p, n[0]):
  1334. let tmp = p.getTemp
  1335. r.address = "($1 = $2, $1)[0]" % [tmp, r.rdLoc]
  1336. r.res = "$1[1]" % [tmp]
  1337. r.tmpLoc = tmp
  1338. else:
  1339. let x = r.rdLoc
  1340. r.address = "$1[0]" % [x]
  1341. r.res = "$1[1]" % [x]
  1342. else:
  1343. r.res = "$1[$2]" % [r.address, r.res]
  1344. r.kind = resExpr
  1345. template isIndirect(x: PSym): bool =
  1346. let v = x
  1347. ({sfAddrTaken, sfGlobal} * v.flags != {} and
  1348. #(mapType(v.typ) != etyObject) and
  1349. {sfImportc, sfExportc} * v.flags == {} and
  1350. v.kind notin {skProc, skFunc, skConverter, skMethod, skIterator,
  1351. skConst, skTemp, skLet})
  1352. proc genSymAddr(p: PProc, n: PNode, typ: PType, r: var TCompRes) =
  1353. let s = n.sym
  1354. if s.loc.r == "": internalError(p.config, n.info, "genAddr: 3")
  1355. case s.kind
  1356. of skParam:
  1357. r.res = s.loc.r
  1358. r.address = ""
  1359. r.typ = etyNone
  1360. of skVar, skLet, skResult:
  1361. r.kind = resExpr
  1362. let jsType = mapType(p):
  1363. if typ.isNil:
  1364. n.typ
  1365. else:
  1366. typ
  1367. if jsType == etyObject:
  1368. # make addr() a no-op:
  1369. r.typ = etyNone
  1370. if isIndirect(s):
  1371. r.res = s.loc.r & "[0]"
  1372. else:
  1373. r.res = s.loc.r
  1374. r.address = ""
  1375. elif {sfGlobal, sfAddrTaken} * s.flags != {} or jsType == etyBaseIndex:
  1376. # for ease of code generation, we do not distinguish between
  1377. # sfAddrTaken and sfGlobal.
  1378. r.typ = etyBaseIndex
  1379. r.address = s.loc.r
  1380. r.res = rope("0")
  1381. else:
  1382. # 'var openArray' for instance produces an 'addr' but this is harmless:
  1383. gen(p, n, r)
  1384. #internalError(p.config, n.info, "genAddr: 4 " & renderTree(n))
  1385. else: internalError(p.config, n.info, $("genAddr: 2", s.kind))
  1386. proc genAddr(p: PProc, n: PNode, r: var TCompRes) =
  1387. if n.kind == nkSym:
  1388. genSymAddr(p, n, nil, r)
  1389. else:
  1390. case n[0].kind
  1391. of nkSym:
  1392. genSymAddr(p, n[0], n.typ, r)
  1393. of nkCheckedFieldExpr:
  1394. genCheckedFieldOp(p, n[0], n.typ, r)
  1395. of nkDotExpr:
  1396. if mapType(p, n.typ) == etyBaseIndex:
  1397. genFieldAddr(p, n[0], r)
  1398. else:
  1399. genFieldAccess(p, n[0], r)
  1400. of nkBracketExpr:
  1401. var ty = skipTypes(n[0].typ, abstractVarRange)
  1402. if ty.kind in MappedToObject:
  1403. gen(p, n[0], r)
  1404. else:
  1405. let kindOfIndexedExpr = skipTypes(n[0][0].typ, abstractVarRange).kind
  1406. case kindOfIndexedExpr
  1407. of tyArray, tyOpenArray, tySequence, tyString, tyCstring, tyVarargs:
  1408. genArrayAddr(p, n[0], r)
  1409. of tyTuple:
  1410. genFieldAddr(p, n[0], r)
  1411. of tyGenericBody:
  1412. genAddr(p, n[^1], r)
  1413. else: internalError(p.config, n[0].info, "expr(nkBracketExpr, " & $kindOfIndexedExpr & ')')
  1414. of nkObjDownConv:
  1415. gen(p, n[0], r)
  1416. of nkHiddenDeref, nkDerefExpr:
  1417. gen(p, n[0], r)
  1418. of nkHiddenAddr:
  1419. gen(p, n[0], r)
  1420. of nkConv:
  1421. genAddr(p, n[0], r)
  1422. of nkStmtListExpr:
  1423. if n.len == 1: gen(p, n[0], r)
  1424. else: internalError(p.config, n[0].info, "genAddr for complex nkStmtListExpr")
  1425. of nkCallKinds:
  1426. if n[0].typ.kind == tyOpenArray:
  1427. # 'var openArray' for instance produces an 'addr' but this is harmless:
  1428. # namely toOpenArray(a, 1, 3)
  1429. gen(p, n[0], r)
  1430. else:
  1431. internalError(p.config, n[0].info, "genAddr: " & $n[0].kind)
  1432. else:
  1433. internalError(p.config, n[0].info, "genAddr: " & $n[0].kind)
  1434. proc attachProc(p: PProc; content: Rope; s: PSym) =
  1435. p.g.code.add(content)
  1436. proc attachProc(p: PProc; s: PSym) =
  1437. let newp = genProc(p, s)
  1438. attachProc(p, newp, s)
  1439. proc genProcForSymIfNeeded(p: PProc, s: PSym) =
  1440. if not p.g.generatedSyms.containsOrIncl(s.id):
  1441. let newp = genProc(p, s)
  1442. var owner = p
  1443. while owner != nil and owner.prc != s.owner:
  1444. owner = owner.up
  1445. if owner != nil: owner.locals.add(newp)
  1446. else: attachProc(p, newp, s)
  1447. proc genCopyForParamIfNeeded(p: PProc, n: PNode) =
  1448. let s = n.sym
  1449. if p.prc == s.owner or needsNoCopy(p, n):
  1450. return
  1451. var owner = p.up
  1452. while true:
  1453. if owner == nil:
  1454. internalError(p.config, n.info, "couldn't find the owner proc of the closed over param: " & s.name.s)
  1455. if owner.prc == s.owner:
  1456. if not owner.generatedParamCopies.containsOrIncl(s.id):
  1457. let copy = "$1 = nimCopy(null, $1, $2);$n" % [s.loc.r, genTypeInfo(p, s.typ)]
  1458. owner.locals.add(owner.indentLine(copy))
  1459. return
  1460. owner = owner.up
  1461. proc genVarInit(p: PProc, v: PSym, n: PNode)
  1462. proc genSym(p: PProc, n: PNode, r: var TCompRes) =
  1463. var s = n.sym
  1464. case s.kind
  1465. of skVar, skLet, skParam, skTemp, skResult, skForVar:
  1466. if s.loc.r == "":
  1467. internalError(p.config, n.info, "symbol has no generated name: " & s.name.s)
  1468. if sfCompileTime in s.flags:
  1469. genVarInit(p, s, if s.astdef != nil: s.astdef else: newNodeI(nkEmpty, s.info))
  1470. if s.kind == skParam:
  1471. genCopyForParamIfNeeded(p, n)
  1472. let k = mapType(p, s.typ)
  1473. if k == etyBaseIndex:
  1474. r.typ = etyBaseIndex
  1475. if {sfAddrTaken, sfGlobal} * s.flags != {}:
  1476. if isIndirect(s):
  1477. r.address = "$1[0][0]" % [s.loc.r]
  1478. r.res = "$1[0][1]" % [s.loc.r]
  1479. else:
  1480. r.address = "$1[0]" % [s.loc.r]
  1481. r.res = "$1[1]" % [s.loc.r]
  1482. else:
  1483. r.address = s.loc.r
  1484. r.res = s.loc.r & "_Idx"
  1485. elif isIndirect(s):
  1486. r.res = "$1[0]" % [s.loc.r]
  1487. else:
  1488. r.res = s.loc.r
  1489. of skConst:
  1490. genConstant(p, s)
  1491. if s.loc.r == "":
  1492. internalError(p.config, n.info, "symbol has no generated name: " & s.name.s)
  1493. r.res = s.loc.r
  1494. of skProc, skFunc, skConverter, skMethod:
  1495. if sfCompileTime in s.flags:
  1496. localError(p.config, n.info, "request to generate code for .compileTime proc: " &
  1497. s.name.s)
  1498. discard mangleName(p.module, s)
  1499. r.res = s.loc.r
  1500. if lfNoDecl in s.loc.flags or s.magic notin generatedMagics or
  1501. {sfImportc, sfInfixCall} * s.flags != {}:
  1502. discard
  1503. elif s.kind == skMethod and getBody(p.module.graph, s).kind == nkEmpty:
  1504. # we cannot produce code for the dispatcher yet:
  1505. discard
  1506. elif sfForward in s.flags:
  1507. p.g.forwarded.add(s)
  1508. else:
  1509. genProcForSymIfNeeded(p, s)
  1510. else:
  1511. if s.loc.r == "":
  1512. internalError(p.config, n.info, "symbol has no generated name: " & s.name.s)
  1513. if mapType(p, s.typ) == etyBaseIndex:
  1514. r.address = s.loc.r
  1515. r.res = s.loc.r & "_Idx"
  1516. else:
  1517. r.res = s.loc.r
  1518. r.kind = resVal
  1519. proc genDeref(p: PProc, n: PNode, r: var TCompRes) =
  1520. let it = n[0]
  1521. let t = mapType(p, it.typ)
  1522. if t == etyObject or it.typ.kind == tyLent:
  1523. gen(p, it, r)
  1524. else:
  1525. var a: TCompRes
  1526. gen(p, it, a)
  1527. r.kind = a.kind
  1528. r.typ = mapType(p, n.typ)
  1529. if r.typ == etyBaseIndex:
  1530. let tmp = p.getTemp
  1531. r.address = "($1 = $2, $1)[0]" % [tmp, a.rdLoc]
  1532. r.res = "$1[1]" % [tmp]
  1533. r.tmpLoc = tmp
  1534. elif a.typ == etyBaseIndex:
  1535. if a.tmpLoc != "":
  1536. r.tmpLoc = a.tmpLoc
  1537. r.res = a.rdLoc
  1538. else:
  1539. internalError(p.config, n.info, "genDeref")
  1540. proc genArgNoParam(p: PProc, n: PNode, r: var TCompRes) =
  1541. var a: TCompRes
  1542. gen(p, n, a)
  1543. if a.typ == etyBaseIndex:
  1544. r.res.add(a.address)
  1545. r.res.add(", ")
  1546. r.res.add(a.res)
  1547. else:
  1548. r.res.add(a.res)
  1549. proc genArg(p: PProc, n: PNode, param: PSym, r: var TCompRes; emitted: ptr int = nil) =
  1550. var a: TCompRes
  1551. gen(p, n, a)
  1552. if skipTypes(param.typ, abstractVar).kind in {tyOpenArray, tyVarargs} and
  1553. a.typ == etyBaseIndex:
  1554. r.res.add("$1[$2]" % [a.address, a.res])
  1555. elif a.typ == etyBaseIndex:
  1556. r.res.add(a.address)
  1557. r.res.add(", ")
  1558. r.res.add(a.res)
  1559. if emitted != nil: inc emitted[]
  1560. elif n.typ.kind in {tyVar, tyPtr, tyRef, tyLent, tyOwned} and
  1561. n.kind in nkCallKinds and mapType(param.typ) == etyBaseIndex:
  1562. # this fixes bug #5608:
  1563. let tmp = getTemp(p)
  1564. r.res.add("($1 = $2, $1[0]), $1[1]" % [tmp, a.rdLoc])
  1565. if emitted != nil: inc emitted[]
  1566. else:
  1567. r.res.add(a.res)
  1568. proc genArgs(p: PProc, n: PNode, r: var TCompRes; start=1) =
  1569. r.res.add("(")
  1570. var hasArgs = false
  1571. var typ = skipTypes(n[0].typ, abstractInst)
  1572. assert(typ.kind == tyProc)
  1573. assert(typ.len == typ.n.len)
  1574. var emitted = start-1
  1575. for i in start..<n.len:
  1576. let it = n[i]
  1577. var paramType: PNode = nil
  1578. if i < typ.len:
  1579. assert(typ.n[i].kind == nkSym)
  1580. paramType = typ.n[i]
  1581. if paramType.typ.isCompileTimeOnly: continue
  1582. if hasArgs: r.res.add(", ")
  1583. if paramType.isNil:
  1584. genArgNoParam(p, it, r)
  1585. else:
  1586. genArg(p, it, paramType.sym, r, addr emitted)
  1587. inc emitted
  1588. hasArgs = true
  1589. r.res.add(")")
  1590. when false:
  1591. # XXX look into this:
  1592. let jsp = countJsParams(typ)
  1593. if emitted != jsp and tfVarargs notin typ.flags:
  1594. localError(p.config, n.info, "wrong number of parameters emitted; expected: " & $jsp &
  1595. " but got: " & $emitted)
  1596. r.kind = resExpr
  1597. proc genOtherArg(p: PProc; n: PNode; i: int; typ: PType;
  1598. generated: var int; r: var TCompRes) =
  1599. if i >= n.len:
  1600. globalError(p.config, n.info, "wrong importcpp pattern; expected parameter at position " & $i &
  1601. " but got only: " & $(n.len-1))
  1602. let it = n[i]
  1603. var paramType: PNode = nil
  1604. if i < typ.len:
  1605. assert(typ.n[i].kind == nkSym)
  1606. paramType = typ.n[i]
  1607. if paramType.typ.isCompileTimeOnly: return
  1608. if paramType.isNil:
  1609. genArgNoParam(p, it, r)
  1610. else:
  1611. genArg(p, it, paramType.sym, r)
  1612. inc generated
  1613. proc genPatternCall(p: PProc; n: PNode; pat: string; typ: PType;
  1614. r: var TCompRes) =
  1615. var i = 0
  1616. var j = 1
  1617. r.kind = resExpr
  1618. while i < pat.len:
  1619. case pat[i]
  1620. of '@':
  1621. var generated = 0
  1622. for k in j..<n.len:
  1623. if generated > 0: r.res.add(", ")
  1624. genOtherArg(p, n, k, typ, generated, r)
  1625. inc i
  1626. of '#':
  1627. var generated = 0
  1628. genOtherArg(p, n, j, typ, generated, r)
  1629. inc j
  1630. inc i
  1631. of '\31':
  1632. # unit separator
  1633. r.res.add("#")
  1634. inc i
  1635. of '\29':
  1636. # group separator
  1637. r.res.add("@")
  1638. inc i
  1639. else:
  1640. let start = i
  1641. while i < pat.len:
  1642. if pat[i] notin {'@', '#', '\31', '\29'}: inc(i)
  1643. else: break
  1644. if i - 1 >= start:
  1645. r.res.add(substr(pat, start, i - 1))
  1646. proc genInfixCall(p: PProc, n: PNode, r: var TCompRes) =
  1647. # don't call '$' here for efficiency:
  1648. let f = n[0].sym
  1649. if f.loc.r == "": f.loc.r = mangleName(p.module, f)
  1650. if sfInfixCall in f.flags:
  1651. let pat = $n[0].sym.loc.r
  1652. internalAssert p.config, pat.len > 0
  1653. if pat.contains({'#', '(', '@'}):
  1654. var typ = skipTypes(n[0].typ, abstractInst)
  1655. assert(typ.kind == tyProc)
  1656. genPatternCall(p, n, pat, typ, r)
  1657. return
  1658. if n.len != 1:
  1659. gen(p, n[1], r)
  1660. if r.typ == etyBaseIndex:
  1661. if r.address == "":
  1662. globalError(p.config, n.info, "cannot invoke with infix syntax")
  1663. r.res = "$1[$2]" % [r.address, r.res]
  1664. r.address = ""
  1665. r.typ = etyNone
  1666. r.res.add(".")
  1667. var op: TCompRes
  1668. gen(p, n[0], op)
  1669. r.res.add(op.res)
  1670. genArgs(p, n, r, 2)
  1671. proc genCall(p: PProc, n: PNode, r: var TCompRes) =
  1672. gen(p, n[0], r)
  1673. genArgs(p, n, r)
  1674. if n.typ != nil:
  1675. let t = mapType(n.typ)
  1676. if t == etyBaseIndex:
  1677. let tmp = p.getTemp
  1678. r.address = "($1 = $2, $1)[0]" % [tmp, r.rdLoc]
  1679. r.res = "$1[1]" % [tmp]
  1680. r.tmpLoc = tmp
  1681. r.typ = t
  1682. proc genEcho(p: PProc, n: PNode, r: var TCompRes) =
  1683. let n = n[1].skipConv
  1684. internalAssert p.config, n.kind == nkBracket
  1685. useMagic(p, "toJSStr") # Used in rawEcho
  1686. useMagic(p, "rawEcho")
  1687. r.res.add("rawEcho(")
  1688. for i in 0..<n.len:
  1689. let it = n[i]
  1690. if it.typ.isCompileTimeOnly: continue
  1691. if i > 0: r.res.add(", ")
  1692. genArgNoParam(p, it, r)
  1693. r.res.add(")")
  1694. r.kind = resExpr
  1695. proc putToSeq(s: string, indirect: bool): Rope =
  1696. result = rope(s)
  1697. if indirect: result = "[$1]" % [result]
  1698. proc createVar(p: PProc, typ: PType, indirect: bool): Rope
  1699. proc createRecordVarAux(p: PProc, rec: PNode, excludedFieldIDs: IntSet, output: var Rope) =
  1700. case rec.kind
  1701. of nkRecList:
  1702. for i in 0..<rec.len:
  1703. createRecordVarAux(p, rec[i], excludedFieldIDs, output)
  1704. of nkRecCase:
  1705. createRecordVarAux(p, rec[0], excludedFieldIDs, output)
  1706. for i in 1..<rec.len:
  1707. createRecordVarAux(p, lastSon(rec[i]), excludedFieldIDs, output)
  1708. of nkSym:
  1709. # Do not produce code for void types
  1710. if isEmptyType(rec.sym.typ): return
  1711. if rec.sym.id notin excludedFieldIDs:
  1712. if output.len > 0: output.add(", ")
  1713. output.addf("$#: ", [mangleName(p.module, rec.sym)])
  1714. output.add(createVar(p, rec.sym.typ, false))
  1715. else: internalError(p.config, rec.info, "createRecordVarAux")
  1716. proc createObjInitList(p: PProc, typ: PType, excludedFieldIDs: IntSet, output: var Rope) =
  1717. var t = typ
  1718. if objHasTypeField(t):
  1719. if output.len > 0: output.add(", ")
  1720. output.addf("m_type: $1", [genTypeInfo(p, t)])
  1721. while t != nil:
  1722. t = t.skipTypes(skipPtrs)
  1723. createRecordVarAux(p, t.n, excludedFieldIDs, output)
  1724. t = t[0]
  1725. proc arrayTypeForElemType(conf: ConfigRef; typ: PType): string =
  1726. let typ = typ.skipTypes(abstractRange)
  1727. case typ.kind
  1728. of tyInt, tyInt32: "Int32Array"
  1729. of tyInt16: "Int16Array"
  1730. of tyInt8: "Int8Array"
  1731. of tyInt64:
  1732. if optJsBigInt64 in conf.globalOptions:
  1733. "BigInt64Array"
  1734. else:
  1735. ""
  1736. of tyUInt, tyUInt32: "Uint32Array"
  1737. of tyUInt16: "Uint16Array"
  1738. of tyUInt8, tyChar, tyBool: "Uint8Array"
  1739. of tyUInt64:
  1740. if optJsBigInt64 in conf.globalOptions:
  1741. "BigUint64Array"
  1742. else:
  1743. ""
  1744. of tyFloat32: "Float32Array"
  1745. of tyFloat64, tyFloat: "Float64Array"
  1746. of tyEnum:
  1747. case typ.size
  1748. of 1: "Uint8Array"
  1749. of 2: "Uint16Array"
  1750. of 4: "Uint32Array"
  1751. else: ""
  1752. else: ""
  1753. proc createVar(p: PProc, typ: PType, indirect: bool): Rope =
  1754. var t = skipTypes(typ, abstractInst)
  1755. case t.kind
  1756. of tyInt8..tyInt32, tyUInt8..tyUInt32, tyEnum, tyChar:
  1757. result = putToSeq("0", indirect)
  1758. of tyInt, tyUInt:
  1759. if $t.sym.loc.r == "bigint":
  1760. result = putToSeq("0n", indirect)
  1761. else:
  1762. result = putToSeq("0", indirect)
  1763. of tyInt64, tyUInt64:
  1764. if optJsBigInt64 in p.config.globalOptions:
  1765. result = putToSeq("0n", indirect)
  1766. else:
  1767. result = putToSeq("0", indirect)
  1768. of tyFloat..tyFloat128:
  1769. result = putToSeq("0.0", indirect)
  1770. of tyRange, tyGenericInst, tyAlias, tySink, tyOwned, tyLent:
  1771. result = createVar(p, lastSon(typ), indirect)
  1772. of tySet:
  1773. result = putToSeq("{}", indirect)
  1774. of tyBool:
  1775. result = putToSeq("false", indirect)
  1776. of tyNil:
  1777. result = putToSeq("null", indirect)
  1778. of tyArray:
  1779. let length = toInt(lengthOrd(p.config, t))
  1780. let e = elemType(t)
  1781. let jsTyp = arrayTypeForElemType(p.config, e)
  1782. if jsTyp.len > 0:
  1783. result = "new $1($2)" % [rope(jsTyp), rope(length)]
  1784. elif length > 32:
  1785. useMagic(p, "arrayConstr")
  1786. # XXX: arrayConstr depends on nimCopy. This line shouldn't be necessary.
  1787. useMagic(p, "nimCopy")
  1788. result = "arrayConstr($1, $2, $3)" % [rope(length),
  1789. createVar(p, e, false), genTypeInfo(p, e)]
  1790. else:
  1791. result = rope("[")
  1792. var i = 0
  1793. while i < length:
  1794. if i > 0: result.add(", ")
  1795. result.add(createVar(p, e, false))
  1796. inc(i)
  1797. result.add("]")
  1798. if indirect: result = "[$1]" % [result]
  1799. of tyTuple:
  1800. result = rope("{")
  1801. for i in 0..<t.len:
  1802. if i > 0: result.add(", ")
  1803. result.addf("Field$1: $2", [i.rope,
  1804. createVar(p, t[i], false)])
  1805. result.add("}")
  1806. if indirect: result = "[$1]" % [result]
  1807. of tyObject:
  1808. var initList: Rope
  1809. createObjInitList(p, t, initIntSet(), initList)
  1810. result = ("({$1})") % [initList]
  1811. if indirect: result = "[$1]" % [result]
  1812. of tyVar, tyPtr, tyRef, tyPointer:
  1813. if mapType(p, t) == etyBaseIndex:
  1814. result = putToSeq("[null, 0]", indirect)
  1815. else:
  1816. result = putToSeq("null", indirect)
  1817. of tySequence, tyString:
  1818. result = putToSeq("[]", indirect)
  1819. of tyCstring, tyProc, tyOpenArray:
  1820. result = putToSeq("null", indirect)
  1821. of tyStatic:
  1822. if t.n != nil:
  1823. result = createVar(p, lastSon t, indirect)
  1824. else:
  1825. internalError(p.config, "createVar: " & $t.kind)
  1826. result = ""
  1827. else:
  1828. internalError(p.config, "createVar: " & $t.kind)
  1829. result = ""
  1830. template returnType: untyped = ""
  1831. proc genVarInit(p: PProc, v: PSym, n: PNode) =
  1832. var
  1833. a: TCompRes
  1834. s: Rope
  1835. varCode: string
  1836. varName = mangleName(p.module, v)
  1837. useReloadingGuard = sfGlobal in v.flags and p.config.hcrOn
  1838. useGlobalPragmas = sfGlobal in v.flags and ({sfPure, sfThread} * v.flags != {})
  1839. if v.constraint.isNil:
  1840. if useReloadingGuard:
  1841. lineF(p, "var $1;$n", varName)
  1842. lineF(p, "if ($1 === undefined) {$n", varName)
  1843. varCode = $varName
  1844. inc p.extraIndent
  1845. elif useGlobalPragmas:
  1846. lineF(p, "if (globalThis.$1 === undefined) {$n", varName)
  1847. varCode = "globalThis." & $varName
  1848. inc p.extraIndent
  1849. else:
  1850. varCode = "var $2"
  1851. else:
  1852. # Is this really a thought through feature? this basically unused
  1853. # feature makes it impossible for almost all format strings in
  1854. # this function to be checked at compile time.
  1855. varCode = v.constraint.strVal
  1856. if n.kind == nkEmpty:
  1857. if not isIndirect(v) and
  1858. v.typ.kind in {tyVar, tyPtr, tyLent, tyRef, tyOwned} and mapType(p, v.typ) == etyBaseIndex:
  1859. lineF(p, "var $1 = null;$n", [varName])
  1860. lineF(p, "var $1_Idx = 0;$n", [varName])
  1861. else:
  1862. line(p, runtimeFormat(varCode & " = $3;$n", [returnType, varName, createVar(p, v.typ, isIndirect(v))]))
  1863. else:
  1864. gen(p, n, a)
  1865. case mapType(p, v.typ)
  1866. of etyObject, etySeq:
  1867. if needsNoCopy(p, n):
  1868. s = a.res
  1869. else:
  1870. useMagic(p, "nimCopy")
  1871. s = "nimCopy(null, $1, $2)" % [a.res, genTypeInfo(p, n.typ)]
  1872. of etyBaseIndex:
  1873. let targetBaseIndex = {sfAddrTaken, sfGlobal} * v.flags == {}
  1874. if a.typ == etyBaseIndex:
  1875. if targetBaseIndex:
  1876. line(p, runtimeFormat(varCode & " = $3, $2_Idx = $4;$n",
  1877. [returnType, v.loc.r, a.address, a.res]))
  1878. else:
  1879. if isIndirect(v):
  1880. line(p, runtimeFormat(varCode & " = [[$3, $4]];$n",
  1881. [returnType, v.loc.r, a.address, a.res]))
  1882. else:
  1883. line(p, runtimeFormat(varCode & " = [$3, $4];$n",
  1884. [returnType, v.loc.r, a.address, a.res]))
  1885. else:
  1886. if targetBaseIndex:
  1887. let tmp = p.getTemp
  1888. lineF(p, "var $1 = $2, $3 = $1[0], $3_Idx = $1[1];$n",
  1889. [tmp, a.res, v.loc.r])
  1890. else:
  1891. line(p, runtimeFormat(varCode & " = $3;$n", [returnType, v.loc.r, a.res]))
  1892. return
  1893. else:
  1894. s = a.res
  1895. if isIndirect(v):
  1896. line(p, runtimeFormat(varCode & " = [$3];$n", [returnType, v.loc.r, s]))
  1897. else:
  1898. line(p, runtimeFormat(varCode & " = $3;$n", [returnType, v.loc.r, s]))
  1899. if useReloadingGuard or useGlobalPragmas:
  1900. dec p.extraIndent
  1901. lineF(p, "}$n")
  1902. proc genVarStmt(p: PProc, n: PNode) =
  1903. for i in 0..<n.len:
  1904. var a = n[i]
  1905. if a.kind != nkCommentStmt:
  1906. if a.kind == nkVarTuple:
  1907. let unpacked = lowerTupleUnpacking(p.module.graph, a, p.module.idgen, p.prc)
  1908. genStmt(p, unpacked)
  1909. else:
  1910. assert(a.kind == nkIdentDefs)
  1911. assert(a[0].kind == nkSym)
  1912. var v = a[0].sym
  1913. if lfNoDecl notin v.loc.flags and sfImportc notin v.flags:
  1914. genLineDir(p, a)
  1915. if sfCompileTime notin v.flags:
  1916. genVarInit(p, v, a[2])
  1917. else:
  1918. # lazy emit, done when it's actually used.
  1919. if v.ast == nil: v.ast = a[2]
  1920. proc genConstant(p: PProc, c: PSym) =
  1921. if lfNoDecl notin c.loc.flags and not p.g.generatedSyms.containsOrIncl(c.id):
  1922. let oldBody = move p.body
  1923. #genLineDir(p, c.astdef)
  1924. genVarInit(p, c, c.astdef)
  1925. p.g.constants.add(p.body)
  1926. p.body = oldBody
  1927. proc genNew(p: PProc, n: PNode) =
  1928. var a: TCompRes
  1929. gen(p, n[1], a)
  1930. var t = skipTypes(n[1].typ, abstractVar)[0]
  1931. if mapType(t) == etyObject:
  1932. lineF(p, "$1 = $2;$n", [a.rdLoc, createVar(p, t, false)])
  1933. elif a.typ == etyBaseIndex:
  1934. lineF(p, "$1 = [$3]; $2 = 0;$n", [a.address, a.res, createVar(p, t, false)])
  1935. else:
  1936. lineF(p, "$1 = [[$2], 0];$n", [a.rdLoc, createVar(p, t, false)])
  1937. proc genNewSeq(p: PProc, n: PNode) =
  1938. var x, y: TCompRes
  1939. gen(p, n[1], x)
  1940. gen(p, n[2], y)
  1941. let t = skipTypes(n[1].typ, abstractVar)[0]
  1942. lineF(p, "$1 = new Array($2); for (var i = 0 ; i < $2 ; ++i) { $1[i] = $3; }", [
  1943. x.rdLoc, y.rdLoc, createVar(p, t, false)])
  1944. proc genOrd(p: PProc, n: PNode, r: var TCompRes) =
  1945. case skipTypes(n[1].typ, abstractVar + abstractRange).kind
  1946. of tyEnum, tyInt..tyInt32, tyUInt..tyUInt32, tyChar: gen(p, n[1], r)
  1947. of tyInt64, tyUInt64:
  1948. if optJsBigInt64 in p.config.globalOptions:
  1949. unaryExpr(p, n, r, "", "Number($1)")
  1950. else: gen(p, n[1], r)
  1951. of tyBool: unaryExpr(p, n, r, "", "($1 ? 1 : 0)")
  1952. else: internalError(p.config, n.info, "genOrd")
  1953. proc genConStrStr(p: PProc, n: PNode, r: var TCompRes) =
  1954. var a: TCompRes
  1955. gen(p, n[1], a)
  1956. r.kind = resExpr
  1957. if skipTypes(n[1].typ, abstractVarRange).kind == tyChar:
  1958. r.res.add("[$1].concat(" % [a.res])
  1959. else:
  1960. r.res.add("($1 || []).concat(" % [a.res])
  1961. for i in 2..<n.len - 1:
  1962. gen(p, n[i], a)
  1963. if skipTypes(n[i].typ, abstractVarRange).kind == tyChar:
  1964. r.res.add("[$1]," % [a.res])
  1965. else:
  1966. r.res.add("$1 || []," % [a.res])
  1967. gen(p, n[^1], a)
  1968. if skipTypes(n[^1].typ, abstractVarRange).kind == tyChar:
  1969. r.res.add("[$1])" % [a.res])
  1970. else:
  1971. r.res.add("$1 || [])" % [a.res])
  1972. proc genReprAux(p: PProc, n: PNode, r: var TCompRes, magic: string, typ: Rope = "") =
  1973. useMagic(p, magic)
  1974. r.res.add(magic & "(")
  1975. var a: TCompRes
  1976. gen(p, n[1], a)
  1977. if magic == "reprAny":
  1978. # the pointer argument in reprAny is expandend to
  1979. # (pointedto, pointer), so we need to fill it
  1980. if a.address.len == 0:
  1981. r.res.add(a.res)
  1982. r.res.add(", null")
  1983. else:
  1984. r.res.add("$1, $2" % [a.address, a.res])
  1985. else:
  1986. r.res.add(a.res)
  1987. if typ != "":
  1988. r.res.add(", ")
  1989. r.res.add(typ)
  1990. r.res.add(")")
  1991. proc genRepr(p: PProc, n: PNode, r: var TCompRes) =
  1992. let t = skipTypes(n[1].typ, abstractVarRange)
  1993. case t.kind
  1994. of tyInt..tyInt64, tyUInt..tyUInt64:
  1995. genReprAux(p, n, r, "reprInt")
  1996. of tyChar:
  1997. genReprAux(p, n, r, "reprChar")
  1998. of tyBool:
  1999. genReprAux(p, n, r, "reprBool")
  2000. of tyFloat..tyFloat128:
  2001. genReprAux(p, n, r, "reprFloat")
  2002. of tyString:
  2003. genReprAux(p, n, r, "reprStr")
  2004. of tyEnum, tyOrdinal:
  2005. genReprAux(p, n, r, "reprEnum", genTypeInfo(p, t))
  2006. of tySet:
  2007. genReprAux(p, n, r, "reprSet", genTypeInfo(p, t))
  2008. of tyEmpty, tyVoid:
  2009. localError(p.config, n.info, "'repr' doesn't support 'void' type")
  2010. of tyPointer:
  2011. genReprAux(p, n, r, "reprPointer")
  2012. of tyOpenArray, tyVarargs:
  2013. genReprAux(p, n, r, "reprJSONStringify")
  2014. else:
  2015. genReprAux(p, n, r, "reprAny", genTypeInfo(p, t))
  2016. r.kind = resExpr
  2017. proc genOf(p: PProc, n: PNode, r: var TCompRes) =
  2018. var x: TCompRes
  2019. let t = skipTypes(n[2].typ,
  2020. abstractVarRange+{tyRef, tyPtr, tyLent, tyTypeDesc, tyOwned})
  2021. gen(p, n[1], x)
  2022. if tfFinal in t.flags:
  2023. r.res = "($1.m_type == $2)" % [x.res, genTypeInfo(p, t)]
  2024. else:
  2025. useMagic(p, "isObj")
  2026. r.res = "isObj($1.m_type, $2)" % [x.res, genTypeInfo(p, t)]
  2027. r.kind = resExpr
  2028. proc genDefault(p: PProc, n: PNode; r: var TCompRes) =
  2029. r.res = createVar(p, n.typ, indirect = false)
  2030. r.kind = resExpr
  2031. proc genReset(p: PProc, n: PNode) =
  2032. var x: TCompRes
  2033. useMagic(p, "genericReset")
  2034. gen(p, n[1], x)
  2035. if x.typ == etyBaseIndex:
  2036. lineF(p, "$1 = null, $2 = 0;$n", [x.address, x.res])
  2037. else:
  2038. let (a, tmp) = maybeMakeTempAssignable(p, n[1], x)
  2039. lineF(p, "$1 = genericReset($3, $2);$n", [a,
  2040. genTypeInfo(p, n[1].typ), tmp])
  2041. proc genMove(p: PProc; n: PNode; r: var TCompRes) =
  2042. var a: TCompRes
  2043. r.kind = resVal
  2044. r.res = p.getTemp()
  2045. gen(p, n[1], a)
  2046. lineF(p, "$1 = $2;$n", [r.rdLoc, a.rdLoc])
  2047. genReset(p, n)
  2048. #lineF(p, "$1 = $2;$n", [dest.rdLoc, src.rdLoc])
  2049. proc genDup(p: PProc; n: PNode; r: var TCompRes) =
  2050. var a: TCompRes
  2051. r.kind = resVal
  2052. r.res = p.getTemp()
  2053. gen(p, n[1], a)
  2054. lineF(p, "$1 = $2;$n", [r.rdLoc, a.rdLoc])
  2055. proc genJSArrayConstr(p: PProc, n: PNode, r: var TCompRes) =
  2056. var a: TCompRes
  2057. r.res = rope("[")
  2058. r.kind = resExpr
  2059. for i in 0 ..< n.len:
  2060. if i > 0: r.res.add(", ")
  2061. gen(p, n[i], a)
  2062. if a.typ == etyBaseIndex:
  2063. r.res.addf("[$1, $2]", [a.address, a.res])
  2064. else:
  2065. if not needsNoCopy(p, n[i]):
  2066. let typ = n[i].typ.skipTypes(abstractInst)
  2067. useMagic(p, "nimCopy")
  2068. a.res = "nimCopy(null, $1, $2)" % [a.rdLoc, genTypeInfo(p, typ)]
  2069. r.res.add(a.res)
  2070. r.res.add("]")
  2071. proc genMagic(p: PProc, n: PNode, r: var TCompRes) =
  2072. var
  2073. a: TCompRes
  2074. line, filen: Rope
  2075. var op = n[0].sym.magic
  2076. case op
  2077. of mOr: genOr(p, n[1], n[2], r)
  2078. of mAnd: genAnd(p, n[1], n[2], r)
  2079. of mAddI..mStrToStr: arith(p, n, r, op)
  2080. of mRepr: genRepr(p, n, r)
  2081. of mSwap: genSwap(p, n)
  2082. of mAppendStrCh:
  2083. binaryExpr(p, n, r, "addChar",
  2084. "addChar($1, $2);")
  2085. of mAppendStrStr:
  2086. var lhs, rhs: TCompRes
  2087. gen(p, n[1], lhs)
  2088. gen(p, n[2], rhs)
  2089. if skipTypes(n[1].typ, abstractVarRange).kind == tyCstring:
  2090. let (b, tmp) = maybeMakeTemp(p, n[2], rhs)
  2091. r.res = "if (null != $1) { if (null == $2) $2 = $3; else $2 += $3; }" %
  2092. [b, lhs.rdLoc, tmp]
  2093. else:
  2094. let (a, tmp) = maybeMakeTemp(p, n[1], lhs)
  2095. r.res = "$1.push.apply($3, $2);" % [a, rhs.rdLoc, tmp]
  2096. r.kind = resExpr
  2097. of mAppendSeqElem:
  2098. var x, y: TCompRes
  2099. gen(p, n[1], x)
  2100. gen(p, n[2], y)
  2101. if mapType(n[2].typ) == etyBaseIndex:
  2102. let c = "[$1, $2]" % [y.address, y.res]
  2103. r.res = "$1.push($2);" % [x.rdLoc, c]
  2104. elif needsNoCopy(p, n[2]):
  2105. r.res = "$1.push($2);" % [x.rdLoc, y.rdLoc]
  2106. else:
  2107. useMagic(p, "nimCopy")
  2108. let c = getTemp(p, defineInLocals=false)
  2109. lineF(p, "var $1 = nimCopy(null, $2, $3);$n",
  2110. [c, y.rdLoc, genTypeInfo(p, n[2].typ)])
  2111. r.res = "$1.push($2);" % [x.rdLoc, c]
  2112. r.kind = resExpr
  2113. of mConStrStr:
  2114. genConStrStr(p, n, r)
  2115. of mEqStr:
  2116. binaryExpr(p, n, r, "eqStrings", "eqStrings($1, $2)")
  2117. of mLeStr:
  2118. binaryExpr(p, n, r, "cmpStrings", "(cmpStrings($1, $2) <= 0)")
  2119. of mLtStr:
  2120. binaryExpr(p, n, r, "cmpStrings", "(cmpStrings($1, $2) < 0)")
  2121. of mIsNil:
  2122. # we want to accept undefined, so we ==
  2123. if mapType(n[1].typ) != etyBaseIndex:
  2124. unaryExpr(p, n, r, "", "($1 == null)")
  2125. else:
  2126. var x: TCompRes
  2127. gen(p, n[1], x)
  2128. r.res = "($# == null && $# === 0)" % [x.address, x.res]
  2129. of mEnumToStr: genRepr(p, n, r)
  2130. of mNew, mNewFinalize: genNew(p, n)
  2131. of mChr: gen(p, n[1], r)
  2132. of mArrToSeq:
  2133. # only array literals doesn't need copy
  2134. if n[1].kind == nkBracket:
  2135. genJSArrayConstr(p, n[1], r)
  2136. else:
  2137. var x: TCompRes
  2138. gen(p, n[1], x)
  2139. useMagic(p, "nimCopy")
  2140. r.res = "nimCopy(null, $1, $2)" % [x.rdLoc, genTypeInfo(p, n.typ)]
  2141. of mOpenArrayToSeq:
  2142. genCall(p, n, r)
  2143. of mDestroy, mTrace: discard "ignore calls to the default destructor"
  2144. of mOrd: genOrd(p, n, r)
  2145. of mLengthStr, mLengthSeq, mLengthOpenArray, mLengthArray:
  2146. var x: TCompRes
  2147. gen(p, n[1], x)
  2148. if skipTypes(n[1].typ, abstractInst).kind == tyCstring:
  2149. let (a, tmp) = maybeMakeTemp(p, n[1], x)
  2150. r.res = "(($1) == null ? 0 : ($2).length)" % [a, tmp]
  2151. else:
  2152. r.res = "($1).length" % [x.rdLoc]
  2153. r.kind = resExpr
  2154. of mHigh:
  2155. var x: TCompRes
  2156. gen(p, n[1], x)
  2157. if skipTypes(n[1].typ, abstractInst).kind == tyCstring:
  2158. let (a, tmp) = maybeMakeTemp(p, n[1], x)
  2159. r.res = "(($1) == null ? -1 : ($2).length - 1)" % [a, tmp]
  2160. else:
  2161. r.res = "($1).length - 1" % [x.rdLoc]
  2162. r.kind = resExpr
  2163. of mInc:
  2164. let typ = n[1].typ.skipTypes(abstractVarRange)
  2165. case typ.kind
  2166. of tyUInt..tyUInt32:
  2167. binaryUintExpr(p, n, r, "+", true)
  2168. of tyUInt64:
  2169. if optJsBigInt64 in p.config.globalOptions:
  2170. binaryExpr(p, n, r, "", "$1 = BigInt.asUintN(64, $3 + BigInt($2))", true)
  2171. else: binaryUintExpr(p, n, r, "+", true)
  2172. elif typ.kind == tyInt64 and optJsBigInt64 in p.config.globalOptions:
  2173. if optOverflowCheck notin p.options:
  2174. binaryExpr(p, n, r, "", "$1 = BigInt.asIntN(64, $3 + BigInt($2))", true)
  2175. else: binaryExpr(p, n, r, "addInt64", "$1 = addInt64($3, BigInt($2))", true)
  2176. else:
  2177. if optOverflowCheck notin p.options: binaryExpr(p, n, r, "", "$1 += $2")
  2178. else: binaryExpr(p, n, r, "addInt", "$1 = addInt($3, $2)", true)
  2179. of ast.mDec:
  2180. let typ = n[1].typ.skipTypes(abstractVarRange)
  2181. case typ.kind
  2182. of tyUInt..tyUInt32:
  2183. binaryUintExpr(p, n, r, "-", true)
  2184. of tyUInt64:
  2185. if optJsBigInt64 in p.config.globalOptions:
  2186. binaryExpr(p, n, r, "", "$1 = BigInt.asUintN(64, $3 - BigInt($2))", true)
  2187. else: binaryUintExpr(p, n, r, "+", true)
  2188. elif typ.kind == tyInt64 and optJsBigInt64 in p.config.globalOptions:
  2189. if optOverflowCheck notin p.options:
  2190. binaryExpr(p, n, r, "", "$1 = BigInt.asIntN(64, $3 - BigInt($2))", true)
  2191. else: binaryExpr(p, n, r, "subInt64", "$1 = subInt64($3, BigInt($2))", true)
  2192. else:
  2193. if optOverflowCheck notin p.options: binaryExpr(p, n, r, "", "$1 -= $2")
  2194. else: binaryExpr(p, n, r, "subInt", "$1 = subInt($3, $2)", true)
  2195. of mSetLengthStr:
  2196. binaryExpr(p, n, r, "mnewString",
  2197. """if ($1.length < $2) { for (var i = $3.length; i < $4; ++i) $3.push(0); }
  2198. else {$3.length = $4; }""")
  2199. of mSetLengthSeq:
  2200. var x, y: TCompRes
  2201. gen(p, n[1], x)
  2202. gen(p, n[2], y)
  2203. let t = skipTypes(n[1].typ, abstractVar)[0]
  2204. let (a, tmp) = maybeMakeTemp(p, n[1], x)
  2205. let (b, tmp2) = maybeMakeTemp(p, n[2], y)
  2206. r.res = """if ($1.length < $2) { for (var i = $4.length ; i < $5 ; ++i) $4.push($3); }
  2207. else { $4.length = $5; }""" % [a, b, createVar(p, t, false), tmp, tmp2]
  2208. r.kind = resExpr
  2209. of mCard: unaryExpr(p, n, r, "SetCard", "SetCard($1)")
  2210. of mLtSet: binaryExpr(p, n, r, "SetLt", "SetLt($1, $2)")
  2211. of mLeSet: binaryExpr(p, n, r, "SetLe", "SetLe($1, $2)")
  2212. of mEqSet: binaryExpr(p, n, r, "SetEq", "SetEq($1, $2)")
  2213. of mMulSet: binaryExpr(p, n, r, "SetMul", "SetMul($1, $2)")
  2214. of mPlusSet: binaryExpr(p, n, r, "SetPlus", "SetPlus($1, $2)")
  2215. of mMinusSet: binaryExpr(p, n, r, "SetMinus", "SetMinus($1, $2)")
  2216. of mIncl: binaryExpr(p, n, r, "", "$1[$2] = true")
  2217. of mExcl: binaryExpr(p, n, r, "", "delete $1[$2]")
  2218. of mInSet:
  2219. binaryExpr(p, n, r, "", "($1[$2] != undefined)")
  2220. of mNewSeq: genNewSeq(p, n)
  2221. of mNewSeqOfCap: unaryExpr(p, n, r, "", "[]")
  2222. of mOf: genOf(p, n, r)
  2223. of mDefault, mZeroDefault: genDefault(p, n, r)
  2224. of mReset, mWasMoved: genReset(p, n)
  2225. of mEcho: genEcho(p, n, r)
  2226. of mNLen..mNError, mSlurp, mStaticExec:
  2227. localError(p.config, n.info, errXMustBeCompileTime % n[0].sym.name.s)
  2228. of mNewString: unaryExpr(p, n, r, "mnewString", "mnewString($1)")
  2229. of mNewStringOfCap:
  2230. unaryExpr(p, n, r, "mnewString", "mnewString(0)")
  2231. of mDotDot:
  2232. genProcForSymIfNeeded(p, n[0].sym)
  2233. genCall(p, n, r)
  2234. of mParseBiggestFloat:
  2235. useMagic(p, "nimParseBiggestFloat")
  2236. genCall(p, n, r)
  2237. of mSlice:
  2238. # arr.slice([begin[, end]]): 'end' is exclusive
  2239. var x, y, z: TCompRes
  2240. gen(p, n[1], x)
  2241. gen(p, n[2], y)
  2242. gen(p, n[3], z)
  2243. r.res = "($1.slice($2, $3 + 1))" % [x.rdLoc, y.rdLoc, z.rdLoc]
  2244. r.kind = resExpr
  2245. of mMove:
  2246. genMove(p, n, r)
  2247. of mDup:
  2248. genDup(p, n, r)
  2249. of mEnsureMove:
  2250. gen(p, n[1], r)
  2251. else:
  2252. genCall(p, n, r)
  2253. #else internalError(p.config, e.info, 'genMagic: ' + magicToStr[op]);
  2254. proc genSetConstr(p: PProc, n: PNode, r: var TCompRes) =
  2255. var
  2256. a, b: TCompRes
  2257. useMagic(p, "setConstr")
  2258. r.res = rope("setConstr(")
  2259. r.kind = resExpr
  2260. for i in 0..<n.len:
  2261. if i > 0: r.res.add(", ")
  2262. var it = n[i]
  2263. if it.kind == nkRange:
  2264. gen(p, it[0], a)
  2265. gen(p, it[1], b)
  2266. if it[0].typ.kind == tyBool:
  2267. r.res.addf("$1, $2", [a.res, b.res])
  2268. else:
  2269. r.res.addf("[$1, $2]", [a.res, b.res])
  2270. else:
  2271. gen(p, it, a)
  2272. r.res.add(a.res)
  2273. r.res.add(")")
  2274. # emit better code for constant sets:
  2275. if isDeepConstExpr(n):
  2276. inc(p.g.unique)
  2277. let tmp = rope("ConstSet") & rope(p.g.unique)
  2278. p.g.constants.addf("var $1 = $2;$n", [tmp, r.res])
  2279. r.res = tmp
  2280. proc genArrayConstr(p: PProc, n: PNode, r: var TCompRes) =
  2281. ## Constructs array or sequence.
  2282. ## Nim array of uint8..uint32, int8..int32 maps to JS typed arrays.
  2283. ## Nim sequence maps to JS array.
  2284. var t = skipTypes(n.typ, abstractInst)
  2285. let e = elemType(t)
  2286. let jsTyp = arrayTypeForElemType(p.config, e)
  2287. if skipTypes(n.typ, abstractVarRange).kind != tySequence and jsTyp.len > 0:
  2288. # generate typed array
  2289. # for example Nim generates `new Uint8Array([1, 2, 3])` for `[byte(1), 2, 3]`
  2290. # TODO use `set` or loop to initialize typed array which improves performances in some situations
  2291. var a: TCompRes
  2292. r.res = "new $1([" % [rope(jsTyp)]
  2293. r.kind = resExpr
  2294. for i in 0 ..< n.len:
  2295. if i > 0: r.res.add(", ")
  2296. gen(p, n[i], a)
  2297. r.res.add(a.res)
  2298. r.res.add("])")
  2299. else:
  2300. genJSArrayConstr(p, n, r)
  2301. proc genTupleConstr(p: PProc, n: PNode, r: var TCompRes) =
  2302. var a: TCompRes
  2303. r.res = rope("{")
  2304. r.kind = resExpr
  2305. for i in 0..<n.len:
  2306. if i > 0: r.res.add(", ")
  2307. var it = n[i]
  2308. if it.kind == nkExprColonExpr: it = it[1]
  2309. gen(p, it, a)
  2310. let typ = it.typ.skipTypes(abstractInst)
  2311. if a.typ == etyBaseIndex:
  2312. r.res.addf("Field$#: [$#, $#]", [i.rope, a.address, a.res])
  2313. else:
  2314. if not needsNoCopy(p, it):
  2315. useMagic(p, "nimCopy")
  2316. a.res = "nimCopy(null, $1, $2)" % [a.rdLoc, genTypeInfo(p, typ)]
  2317. r.res.addf("Field$#: $#", [i.rope, a.res])
  2318. r.res.add("}")
  2319. proc genObjConstr(p: PProc, n: PNode, r: var TCompRes) =
  2320. var a: TCompRes
  2321. r.kind = resExpr
  2322. var initList : Rope
  2323. var fieldIDs = initIntSet()
  2324. let nTyp = n.typ.skipTypes(abstractInst)
  2325. for i in 1..<n.len:
  2326. if i > 1: initList.add(", ")
  2327. var it = n[i]
  2328. internalAssert p.config, it.kind == nkExprColonExpr
  2329. let val = it[1]
  2330. gen(p, val, a)
  2331. var f = it[0].sym
  2332. if f.loc.r == "": f.loc.r = mangleName(p.module, f)
  2333. fieldIDs.incl(lookupFieldAgain(n.typ.skipTypes({tyDistinct}), f).id)
  2334. let typ = val.typ.skipTypes(abstractInst)
  2335. if a.typ == etyBaseIndex:
  2336. initList.addf("$#: [$#, $#]", [f.loc.r, a.address, a.res])
  2337. else:
  2338. if not needsNoCopy(p, val):
  2339. useMagic(p, "nimCopy")
  2340. a.res = "nimCopy(null, $1, $2)" % [a.rdLoc, genTypeInfo(p, typ)]
  2341. initList.addf("$#: $#", [f.loc.r, a.res])
  2342. let t = skipTypes(n.typ, abstractInst + skipPtrs)
  2343. createObjInitList(p, t, fieldIDs, initList)
  2344. r.res = ("{$1}") % [initList]
  2345. proc genConv(p: PProc, n: PNode, r: var TCompRes) =
  2346. var dest = skipTypes(n.typ, abstractVarRange)
  2347. var src = skipTypes(n[1].typ, abstractVarRange)
  2348. gen(p, n[1], r)
  2349. if dest.kind == src.kind:
  2350. # no-op conversion
  2351. return
  2352. let toInt = (dest.kind in tyInt..tyInt32)
  2353. let fromInt = (src.kind in tyInt..tyInt32)
  2354. let toUint = (dest.kind in tyUInt..tyUInt32)
  2355. let fromUint = (src.kind in tyUInt..tyUInt32)
  2356. if toUint and (fromInt or fromUint):
  2357. let trimmer = unsignedTrimmer(dest.size)
  2358. r.res = "($1 $2)" % [r.res, trimmer]
  2359. elif dest.kind == tyBool:
  2360. r.res = "(!!($1))" % [r.res]
  2361. r.kind = resExpr
  2362. elif toInt:
  2363. if src.kind in {tyInt64, tyUInt64} and optJsBigInt64 in p.config.globalOptions:
  2364. r.res = "Number($1)" % [r.res]
  2365. else:
  2366. r.res = "(($1) | 0)" % [r.res]
  2367. elif dest.kind == tyInt64 and optJsBigInt64 in p.config.globalOptions:
  2368. if fromInt or fromUint or src.kind in {tyBool, tyChar, tyEnum}:
  2369. r.res = "BigInt($1)" % [r.res]
  2370. elif src.kind in {tyFloat..tyFloat64}:
  2371. r.res = "BigInt(Math.trunc($1))" % [r.res]
  2372. elif src.kind == tyUInt64:
  2373. r.res = "BigInt.asIntN(64, $1)" % [r.res]
  2374. elif dest.kind == tyUInt64 and optJsBigInt64 in p.config.globalOptions:
  2375. if fromUint or src.kind in {tyBool, tyChar, tyEnum}:
  2376. r.res = "BigInt($1)" % [r.res]
  2377. elif fromInt: # could be negative
  2378. r.res = "BigInt.asUintN(64, BigInt($1))" % [r.res]
  2379. elif src.kind in {tyFloat..tyFloat64}:
  2380. r.res = "BigInt.asUintN(64, BigInt(Math.trunc($1)))" % [r.res]
  2381. elif src.kind == tyInt64:
  2382. r.res = "BigInt.asUintN(64, $1)" % [r.res]
  2383. elif toUint or dest.kind in tyFloat..tyFloat64:
  2384. if src.kind in {tyInt64, tyUInt64} and optJsBigInt64 in p.config.globalOptions:
  2385. r.res = "Number($1)" % [r.res]
  2386. else:
  2387. # TODO: What types must we handle here?
  2388. discard
  2389. proc upConv(p: PProc, n: PNode, r: var TCompRes) =
  2390. gen(p, n[0], r) # XXX
  2391. proc genRangeChck(p: PProc, n: PNode, r: var TCompRes, magic: string) =
  2392. var a, b: TCompRes
  2393. gen(p, n[0], r)
  2394. let src = skipTypes(n[0].typ, abstractVarRange)
  2395. let dest = skipTypes(n.typ, abstractVarRange)
  2396. if src.kind in {tyInt64, tyUInt64} and dest.kind notin {tyInt64, tyUInt64} and optJsBigInt64 in p.config.globalOptions:
  2397. r.res = "Number($1)" % [r.res]
  2398. if optRangeCheck notin p.options or (dest.kind in {tyUInt..tyUInt64} and
  2399. checkUnsignedConversions notin p.config.legacyFeatures):
  2400. discard "XXX maybe emit masking instructions here"
  2401. else:
  2402. gen(p, n[1], a)
  2403. gen(p, n[2], b)
  2404. useMagic(p, "chckRange")
  2405. r.res = "chckRange($1, $2, $3)" % [r.res, a.res, b.res]
  2406. r.kind = resExpr
  2407. proc convStrToCStr(p: PProc, n: PNode, r: var TCompRes) =
  2408. # we do an optimization here as this is likely to slow down
  2409. # much of the code otherwise:
  2410. if n[0].kind == nkCStringToString:
  2411. gen(p, n[0][0], r)
  2412. else:
  2413. gen(p, n[0], r)
  2414. if r.res == "": internalError(p.config, n.info, "convStrToCStr")
  2415. useMagic(p, "toJSStr")
  2416. r.res = "toJSStr($1)" % [r.res]
  2417. r.kind = resExpr
  2418. proc convCStrToStr(p: PProc, n: PNode, r: var TCompRes) =
  2419. # we do an optimization here as this is likely to slow down
  2420. # much of the code otherwise:
  2421. if n[0].kind == nkStringToCString:
  2422. gen(p, n[0][0], r)
  2423. else:
  2424. gen(p, n[0], r)
  2425. if r.res == "": internalError(p.config, n.info, "convCStrToStr")
  2426. useMagic(p, "cstrToNimstr")
  2427. r.res = "cstrToNimstr($1)" % [r.res]
  2428. r.kind = resExpr
  2429. proc genReturnStmt(p: PProc, n: PNode) =
  2430. if p.procDef == nil: internalError(p.config, n.info, "genReturnStmt")
  2431. p.beforeRetNeeded = true
  2432. if n[0].kind != nkEmpty:
  2433. genStmt(p, n[0])
  2434. else:
  2435. genLineDir(p, n)
  2436. lineF(p, "break BeforeRet;$n", [])
  2437. proc frameCreate(p: PProc; procname, filename: Rope): Rope =
  2438. const frameFmt =
  2439. "var F = {procname: $1, prev: framePtr, filename: $2, line: 0};$n"
  2440. result = p.indentLine(frameFmt % [procname, filename])
  2441. result.add p.indentLine(ropes.`%`("framePtr = F;$n", []))
  2442. proc frameDestroy(p: PProc): Rope =
  2443. result = p.indentLine rope(("framePtr = F.prev;") & "\L")
  2444. proc genProcBody(p: PProc, prc: PSym): Rope =
  2445. if hasFrameInfo(p):
  2446. result = frameCreate(p,
  2447. makeJSString(prc.owner.name.s & '.' & prc.name.s),
  2448. makeJSString(toFilenameOption(p.config, prc.info.fileIndex, foStacktrace)))
  2449. else:
  2450. result = ""
  2451. if p.beforeRetNeeded:
  2452. result.add p.indentLine("BeforeRet: {\n")
  2453. result.add p.body
  2454. result.add p.indentLine("};\n")
  2455. else:
  2456. result.add(p.body)
  2457. if prc.typ.callConv == ccSysCall:
  2458. result = ("try {$n$1} catch (e) {$n" &
  2459. " alert(\"Unhandled exception:\\n\" + e.message + \"\\n\"$n}") % [result]
  2460. if hasFrameInfo(p):
  2461. result.add(frameDestroy(p))
  2462. proc optionalLine(p: Rope): Rope =
  2463. if p == "":
  2464. return ""
  2465. else:
  2466. return p & "\L"
  2467. proc genProc(oldProc: PProc, prc: PSym): Rope =
  2468. ## Generate a JS procedure ('function').
  2469. var
  2470. resultSym: PSym
  2471. a: TCompRes
  2472. #if gVerbosity >= 3:
  2473. # echo "BEGIN generating code for: " & prc.name.s
  2474. var p = newProc(oldProc.g, oldProc.module, prc.ast, prc.options)
  2475. p.up = oldProc
  2476. var returnStmt: Rope = ""
  2477. var resultAsgn: Rope = ""
  2478. var name = mangleName(p.module, prc)
  2479. let header = generateHeader(p, prc.typ)
  2480. if prc.typ[0] != nil and sfPure notin prc.flags:
  2481. resultSym = prc.ast[resultPos].sym
  2482. let mname = mangleName(p.module, resultSym)
  2483. # otherwise uses "fat pointers"
  2484. let useRawPointer = not isIndirect(resultSym) and
  2485. resultSym.typ.kind in {tyVar, tyPtr, tyLent, tyRef, tyOwned} and
  2486. mapType(p, resultSym.typ) == etyBaseIndex
  2487. if useRawPointer:
  2488. resultAsgn = p.indentLine(("var $# = null;$n") % [mname])
  2489. resultAsgn.add p.indentLine("var $#_Idx = 0;$n" % [mname])
  2490. else:
  2491. let resVar = createVar(p, resultSym.typ, isIndirect(resultSym))
  2492. resultAsgn = p.indentLine(("var $# = $#;$n") % [mname, resVar])
  2493. gen(p, prc.ast[resultPos], a)
  2494. if mapType(p, resultSym.typ) == etyBaseIndex:
  2495. returnStmt = "return [$#, $#];$n" % [a.address, a.res]
  2496. else:
  2497. returnStmt = "return $#;$n" % [a.res]
  2498. var transformedBody = transformBody(p.module.graph, p.module.idgen, prc, dontUseCache)
  2499. if sfInjectDestructors in prc.flags:
  2500. transformedBody = injectDestructorCalls(p.module.graph, p.module.idgen, prc, transformedBody)
  2501. p.nested: genStmt(p, transformedBody)
  2502. if optLineDir in p.config.options:
  2503. result = lineDir(p.config, prc.info, toLinenumber(prc.info))
  2504. var def: Rope
  2505. if not prc.constraint.isNil:
  2506. def = runtimeFormat(prc.constraint.strVal & " {$n$#$#$#$#$#",
  2507. [ returnType,
  2508. name,
  2509. header,
  2510. optionalLine(p.globals),
  2511. optionalLine(p.locals),
  2512. optionalLine(resultAsgn),
  2513. optionalLine(genProcBody(p, prc)),
  2514. optionalLine(p.indentLine(returnStmt))])
  2515. else:
  2516. # if optLineDir in p.config.options:
  2517. # result.add("\L")
  2518. if p.config.hcrOn:
  2519. # Here, we introduce thunks that create the equivalent of a jump table
  2520. # for all global functions, because references to them may be stored
  2521. # in JavaScript variables. The added indirection ensures that such
  2522. # references will end up calling the reloaded code.
  2523. var thunkName = name
  2524. name = name & "IMLP"
  2525. result.add("\Lfunction $#() { return $#.apply(this, arguments); }$n" %
  2526. [thunkName, name])
  2527. def = "\Lfunction $#($#) {$n$#$#$#$#$#" %
  2528. [ name,
  2529. header,
  2530. optionalLine(p.globals),
  2531. optionalLine(p.locals),
  2532. optionalLine(resultAsgn),
  2533. optionalLine(genProcBody(p, prc)),
  2534. optionalLine(p.indentLine(returnStmt))]
  2535. dec p.extraIndent
  2536. result.add p.indentLine(def)
  2537. result.add p.indentLine("}\n")
  2538. #if gVerbosity >= 3:
  2539. # echo "END generated code for: " & prc.name.s
  2540. proc genStmt(p: PProc, n: PNode) =
  2541. var r: TCompRes
  2542. gen(p, n, r)
  2543. if r.res != "": lineF(p, "$#;$n", [r.res])
  2544. proc genPragma(p: PProc, n: PNode) =
  2545. for i in 0..<n.len:
  2546. let it = n[i]
  2547. case whichPragma(it)
  2548. of wEmit: genAsmOrEmitStmt(p, it[1])
  2549. of wPush:
  2550. processPushBackendOption(p.optionsStack, p.options, n, i+1)
  2551. of wPop:
  2552. processPopBackendOption(p.optionsStack, p.options)
  2553. else: discard
  2554. proc genCast(p: PProc, n: PNode, r: var TCompRes) =
  2555. var dest = skipTypes(n.typ, abstractVarRange)
  2556. var src = skipTypes(n[1].typ, abstractVarRange)
  2557. gen(p, n[1], r)
  2558. if dest.kind == src.kind:
  2559. # no-op conversion
  2560. return
  2561. let toInt = (dest.kind in tyInt..tyInt32)
  2562. let toUint = (dest.kind in tyUInt..tyUInt32)
  2563. let fromInt = (src.kind in tyInt..tyInt32)
  2564. let fromUint = (src.kind in tyUInt..tyUInt32)
  2565. if toUint:
  2566. if fromInt or fromUint:
  2567. r.res = "Number(BigInt.asUintN($1, BigInt($2)))" % [$(dest.size * 8), r.res]
  2568. elif src.kind in {tyInt64, tyUInt64} and optJsBigInt64 in p.config.globalOptions:
  2569. r.res = "Number(BigInt.asUintN($1, $2))" % [$(dest.size * 8), r.res]
  2570. elif toInt:
  2571. if fromInt or fromUint:
  2572. r.res = "Number(BigInt.asIntN($1, BigInt($2)))" % [$(dest.size * 8), r.res]
  2573. elif src.kind in {tyInt64, tyUInt64} and optJsBigInt64 in p.config.globalOptions:
  2574. r.res = "Number(BigInt.asIntN($1, $2))" % [$(dest.size * 8), r.res]
  2575. elif dest.kind == tyInt64 and optJsBigInt64 in p.config.globalOptions:
  2576. if fromInt or fromUint or src.kind in {tyBool, tyChar, tyEnum}:
  2577. r.res = "BigInt($1)" % [r.res]
  2578. elif src.kind in {tyFloat..tyFloat64}:
  2579. r.res = "BigInt(Math.trunc($1))" % [r.res]
  2580. elif src.kind == tyUInt64:
  2581. r.res = "BigInt.asIntN(64, $1)" % [r.res]
  2582. elif dest.kind == tyUInt64 and optJsBigInt64 in p.config.globalOptions:
  2583. if fromUint or src.kind in {tyBool, tyChar, tyEnum}:
  2584. r.res = "BigInt($1)" % [r.res]
  2585. elif fromInt: # could be negative
  2586. r.res = "BigInt.asUintN(64, BigInt($1))" % [r.res]
  2587. elif src.kind in {tyFloat..tyFloat64}:
  2588. r.res = "BigInt.asUintN(64, BigInt(Math.trunc($1)))" % [r.res]
  2589. elif src.kind == tyInt64:
  2590. r.res = "BigInt.asUintN(64, $1)" % [r.res]
  2591. elif dest.kind in tyFloat..tyFloat64:
  2592. if src.kind in {tyInt64, tyUInt64} and optJsBigInt64 in p.config.globalOptions:
  2593. r.res = "Number($1)" % [r.res]
  2594. elif (src.kind == tyPtr and mapType(p, src) == etyObject) and dest.kind == tyPointer:
  2595. r.address = r.res
  2596. r.res = "null"
  2597. r.typ = etyBaseIndex
  2598. elif (dest.kind == tyPtr and mapType(p, dest) == etyObject) and src.kind == tyPointer:
  2599. r.res = r.address
  2600. r.typ = etyObject
  2601. proc gen(p: PProc, n: PNode, r: var TCompRes) =
  2602. r.typ = etyNone
  2603. if r.kind != resCallee: r.kind = resNone
  2604. #r.address = ""
  2605. r.res = ""
  2606. case n.kind
  2607. of nkSym:
  2608. genSym(p, n, r)
  2609. of nkCharLit..nkUInt64Lit:
  2610. case n.typ.skipTypes(abstractVarRange).kind
  2611. of tyBool:
  2612. r.res = if n.intVal == 0: rope"false" else: rope"true"
  2613. of tyUInt64:
  2614. r.res = rope($cast[BiggestUInt](n.intVal))
  2615. if optJsBigInt64 in p.config.globalOptions:
  2616. r.res.add('n')
  2617. of tyInt64:
  2618. let wrap = n.intVal < 0 # wrap negative integers with parens
  2619. if wrap: r.res.add '('
  2620. r.res.addInt n.intVal
  2621. if optJsBigInt64 in p.config.globalOptions:
  2622. r.res.add('n')
  2623. if wrap: r.res.add ')'
  2624. else:
  2625. let wrap = n.intVal < 0 # wrap negative integers with parens
  2626. if wrap: r.res.add '('
  2627. r.res.addInt n.intVal
  2628. if wrap: r.res.add ')'
  2629. r.kind = resExpr
  2630. of nkNilLit:
  2631. if isEmptyType(n.typ):
  2632. discard
  2633. elif mapType(p, n.typ) == etyBaseIndex:
  2634. r.typ = etyBaseIndex
  2635. r.address = rope"null"
  2636. r.res = rope"0"
  2637. r.kind = resExpr
  2638. else:
  2639. r.res = rope"null"
  2640. r.kind = resExpr
  2641. of nkStrLit..nkTripleStrLit:
  2642. if skipTypes(n.typ, abstractVarRange).kind == tyString:
  2643. if n.strVal.len <= 64:
  2644. r.res = makeJsNimStrLit(n.strVal)
  2645. else:
  2646. useMagic(p, "makeNimstrLit")
  2647. r.res = "makeNimstrLit($1)" % [makeJSString(n.strVal)]
  2648. else:
  2649. r.res = makeJSString(n.strVal, false)
  2650. r.kind = resExpr
  2651. of nkFloatLit..nkFloat64Lit:
  2652. let f = n.floatVal
  2653. case classify(f)
  2654. of fcNan:
  2655. if signbit(f):
  2656. r.res = rope"-NaN"
  2657. else:
  2658. r.res = rope"NaN"
  2659. of fcNegZero:
  2660. r.res = rope"-0.0"
  2661. of fcZero:
  2662. r.res = rope"0.0"
  2663. of fcInf:
  2664. r.res = rope"Infinity"
  2665. of fcNegInf:
  2666. r.res = rope"-Infinity"
  2667. else: r.res = rope(f.toStrMaxPrecision)
  2668. r.kind = resExpr
  2669. of nkCallKinds:
  2670. if isEmptyType(n.typ):
  2671. genLineDir(p, n)
  2672. if (n[0].kind == nkSym) and (n[0].sym.magic != mNone):
  2673. genMagic(p, n, r)
  2674. elif n[0].kind == nkSym and sfInfixCall in n[0].sym.flags and
  2675. n.len >= 1:
  2676. genInfixCall(p, n, r)
  2677. else:
  2678. genCall(p, n, r)
  2679. of nkClosure: gen(p, n[0], r)
  2680. of nkCurly: genSetConstr(p, n, r)
  2681. of nkBracket: genArrayConstr(p, n, r)
  2682. of nkPar, nkTupleConstr: genTupleConstr(p, n, r)
  2683. of nkObjConstr: genObjConstr(p, n, r)
  2684. of nkHiddenStdConv, nkHiddenSubConv, nkConv: genConv(p, n, r)
  2685. of nkAddr, nkHiddenAddr:
  2686. if n.typ.kind in {tyLent}:
  2687. gen(p, n[0], r)
  2688. else:
  2689. genAddr(p, n, r)
  2690. of nkDerefExpr, nkHiddenDeref:
  2691. if n.typ.kind in {tyLent}:
  2692. gen(p, n[0], r)
  2693. else:
  2694. genDeref(p, n, r)
  2695. of nkBracketExpr: genArrayAccess(p, n, r)
  2696. of nkDotExpr: genFieldAccess(p, n, r)
  2697. of nkCheckedFieldExpr: genCheckedFieldOp(p, n, nil, r)
  2698. of nkObjDownConv: gen(p, n[0], r)
  2699. of nkObjUpConv: upConv(p, n, r)
  2700. of nkCast: genCast(p, n, r)
  2701. of nkChckRangeF: genRangeChck(p, n, r, "chckRangeF")
  2702. of nkChckRange64: genRangeChck(p, n, r, "chckRange64")
  2703. of nkChckRange: genRangeChck(p, n, r, "chckRange")
  2704. of nkStringToCString: convStrToCStr(p, n, r)
  2705. of nkCStringToString: convCStrToStr(p, n, r)
  2706. of nkEmpty: discard
  2707. of nkLambdaKinds:
  2708. let s = n[namePos].sym
  2709. discard mangleName(p.module, s)
  2710. r.res = s.loc.r
  2711. if lfNoDecl in s.loc.flags or s.magic notin generatedMagics: discard
  2712. elif not p.g.generatedSyms.containsOrIncl(s.id):
  2713. p.locals.add(genProc(p, s))
  2714. of nkType: r.res = genTypeInfo(p, n.typ)
  2715. of nkStmtList, nkStmtListExpr:
  2716. # this shows the distinction is nice for backends and should be kept
  2717. # in the frontend
  2718. let isExpr = not isEmptyType(n.typ)
  2719. for i in 0..<n.len - isExpr.ord:
  2720. genStmt(p, n[i])
  2721. if isExpr:
  2722. gen(p, lastSon(n), r)
  2723. of nkBlockStmt, nkBlockExpr: genBlock(p, n, r)
  2724. of nkIfStmt, nkIfExpr: genIf(p, n, r)
  2725. of nkWhen:
  2726. # This is "when nimvm" node
  2727. gen(p, n[1][0], r)
  2728. of nkWhileStmt: genWhileStmt(p, n)
  2729. of nkVarSection, nkLetSection: genVarStmt(p, n)
  2730. of nkConstSection: discard
  2731. of nkForStmt, nkParForStmt:
  2732. internalError(p.config, n.info, "for statement not eliminated")
  2733. of nkCaseStmt: genCaseJS(p, n, r)
  2734. of nkReturnStmt: genReturnStmt(p, n)
  2735. of nkBreakStmt: genBreakStmt(p, n)
  2736. of nkAsgn: genAsgn(p, n)
  2737. of nkFastAsgn, nkSinkAsgn: genFastAsgn(p, n)
  2738. of nkDiscardStmt:
  2739. if n[0].kind != nkEmpty:
  2740. genLineDir(p, n)
  2741. gen(p, n[0], r)
  2742. r.res = "var _ = " & r.res
  2743. of nkAsmStmt: genAsmOrEmitStmt(p, n)
  2744. of nkTryStmt, nkHiddenTryStmt: genTry(p, n, r)
  2745. of nkRaiseStmt: genRaiseStmt(p, n)
  2746. of nkTypeSection, nkCommentStmt, nkIncludeStmt,
  2747. nkImportStmt, nkImportExceptStmt, nkExportStmt, nkExportExceptStmt,
  2748. nkFromStmt, nkTemplateDef, nkMacroDef, nkStaticStmt,
  2749. nkMixinStmt, nkBindStmt: discard
  2750. of nkIteratorDef:
  2751. if n[0].sym.typ.callConv == TCallingConvention.ccClosure:
  2752. globalError(p.config, n.info, "Closure iterators are not supported by JS backend!")
  2753. of nkPragma: genPragma(p, n)
  2754. of nkProcDef, nkFuncDef, nkMethodDef, nkConverterDef:
  2755. var s = n[namePos].sym
  2756. if {sfExportc, sfCompilerProc} * s.flags == {sfExportc}:
  2757. genSym(p, n[namePos], r)
  2758. r.res = ""
  2759. of nkGotoState, nkState:
  2760. globalError(p.config, n.info, "First class iterators not implemented")
  2761. of nkPragmaBlock: gen(p, n.lastSon, r)
  2762. of nkComesFrom:
  2763. discard "XXX to implement for better stack traces"
  2764. else: internalError(p.config, n.info, "gen: unknown node type: " & $n.kind)
  2765. proc newModule(g: ModuleGraph; module: PSym): BModule =
  2766. ## Create a new JS backend module node.
  2767. new(result)
  2768. result.module = module
  2769. result.sigConflicts = initCountTable[SigHash]()
  2770. if g.backend == nil:
  2771. g.backend = newGlobals()
  2772. result.graph = g
  2773. result.config = g.config
  2774. if sfSystemModule in module.flags:
  2775. PGlobals(g.backend).inSystem = true
  2776. proc genHeader(): Rope =
  2777. ## Generate the JS header.
  2778. result = rope("""/* Generated by the Nim Compiler v$1 */
  2779. var framePtr = null;
  2780. var excHandler = 0;
  2781. var lastJSError = null;
  2782. """.unindent.format(VersionAsString))
  2783. proc addHcrInitGuards(p: PProc, n: PNode,
  2784. moduleLoadedVar: Rope, inInitGuard: var bool) =
  2785. if n.kind == nkStmtList:
  2786. for child in n:
  2787. addHcrInitGuards(p, child, moduleLoadedVar, inInitGuard)
  2788. else:
  2789. let stmtShouldExecute = n.kind in {
  2790. nkProcDef, nkFuncDef, nkMethodDef,nkConverterDef,
  2791. nkVarSection, nkLetSection} or nfExecuteOnReload in n.flags
  2792. if inInitGuard:
  2793. if stmtShouldExecute:
  2794. dec p.extraIndent
  2795. line(p, "}\L")
  2796. inInitGuard = false
  2797. else:
  2798. if not stmtShouldExecute:
  2799. lineF(p, "if ($1 == undefined) {$n", [moduleLoadedVar])
  2800. inc p.extraIndent
  2801. inInitGuard = true
  2802. genStmt(p, n)
  2803. proc genModule(p: PProc, n: PNode) =
  2804. ## Generate the JS module code.
  2805. ## Called for each top level node in a Nim module.
  2806. if optStackTrace in p.options:
  2807. p.body.add(frameCreate(p,
  2808. makeJSString("module " & p.module.module.name.s),
  2809. makeJSString(toFilenameOption(p.config, p.module.module.info.fileIndex, foStacktrace))))
  2810. var transformedN = transformStmt(p.module.graph, p.module.idgen, p.module.module, n)
  2811. if sfInjectDestructors in p.module.module.flags:
  2812. transformedN = injectDestructorCalls(p.module.graph, p.module.idgen, p.module.module, transformedN)
  2813. if p.config.hcrOn and n.kind == nkStmtList:
  2814. let moduleSym = p.module.module
  2815. var moduleLoadedVar = rope(moduleSym.name.s) & "_loaded" &
  2816. idOrSig(moduleSym, moduleSym.name.s, p.module.sigConflicts, p.config)
  2817. lineF(p, "var $1;$n", [moduleLoadedVar])
  2818. var inGuardedBlock = false
  2819. addHcrInitGuards(p, transformedN, moduleLoadedVar, inGuardedBlock)
  2820. if inGuardedBlock:
  2821. dec p.extraIndent
  2822. line(p, "}\L")
  2823. lineF(p, "$1 = true;$n", [moduleLoadedVar])
  2824. else:
  2825. genStmt(p, transformedN)
  2826. if optStackTrace in p.options:
  2827. p.body.add(frameDestroy(p))
  2828. proc processJSCodeGen*(b: PPassContext, n: PNode): PNode =
  2829. ## Generate JS code for a node.
  2830. result = n
  2831. let m = BModule(b)
  2832. if pipelineutils.skipCodegen(m.config, n): return n
  2833. if m.module == nil: internalError(m.config, n.info, "myProcess")
  2834. let globals = PGlobals(m.graph.backend)
  2835. var p = newInitProc(globals, m)
  2836. m.initProc = p
  2837. p.unique = globals.unique
  2838. genModule(p, n)
  2839. p.g.code.add(p.locals)
  2840. p.g.code.add(p.body)
  2841. proc wholeCode(graph: ModuleGraph; m: BModule): Rope =
  2842. ## Combine source code from all nodes.
  2843. let globals = PGlobals(graph.backend)
  2844. for prc in globals.forwarded:
  2845. if not globals.generatedSyms.containsOrIncl(prc.id):
  2846. var p = newInitProc(globals, m)
  2847. attachProc(p, prc)
  2848. var disp = generateMethodDispatchers(graph, m.idgen)
  2849. for i in 0..<disp.len:
  2850. let prc = disp[i].sym
  2851. if not globals.generatedSyms.containsOrIncl(prc.id):
  2852. var p = newInitProc(globals, m)
  2853. attachProc(p, prc)
  2854. result = globals.typeInfo & globals.constants & globals.code
  2855. proc getClassName(t: PType): Rope =
  2856. var s = t.sym
  2857. if s.isNil or sfAnon in s.flags:
  2858. s = skipTypes(t, abstractPtrs).sym
  2859. if s.isNil or sfAnon in s.flags:
  2860. doAssert(false, "cannot retrieve class name")
  2861. if s.loc.r != "": result = s.loc.r
  2862. else: result = rope(s.name.s)
  2863. proc finalJSCodeGen*(graph: ModuleGraph; b: PPassContext, n: PNode): PNode =
  2864. ## Finalize JS code generation of a Nim module.
  2865. ## Param `n` may contain nodes returned from the last module close call.
  2866. var m = BModule(b)
  2867. if sfMainModule in m.module.flags:
  2868. # Add global destructors to the module.
  2869. # This must come before the last call to `myProcess`.
  2870. for i in countdown(high(graph.globalDestructors), 0):
  2871. n.add graph.globalDestructors[i]
  2872. # Process any nodes left over from the last call to `myClose`.
  2873. result = processJSCodeGen(b, n)
  2874. # Some codegen is different (such as no stacktraces; see `initProcOptions`)
  2875. # when `std/system` is being processed.
  2876. if sfSystemModule in m.module.flags:
  2877. PGlobals(graph.backend).inSystem = false
  2878. # Check if codegen should continue before any files are generated.
  2879. # It may bail early is if too many errors have been raised.
  2880. if pipelineutils.skipCodegen(m.config, n): return n
  2881. # Nim modules are compiled into a single JS file.
  2882. # If this is the main module, then this is the final call to `myClose`.
  2883. if sfMainModule in m.module.flags:
  2884. var code = genHeader() & wholeCode(graph, m)
  2885. let outFile = m.config.prepareToWriteOutput()
  2886. # Generate an optional source map.
  2887. if optSourcemap in m.config.globalOptions:
  2888. var map: SourceMap
  2889. map = genSourceMap($code, outFile.string)
  2890. code &= "\n//# sourceMappingURL=$#.map" % [outFile.string]
  2891. writeFile(outFile.string & ".map", $(%map))
  2892. # Check if the generated JS code matches the output file, or else
  2893. # write it to the file.
  2894. if not equalsFile(code, outFile):
  2895. if not writeRope(code, outFile):
  2896. rawMessage(m.config, errCannotOpenFile, outFile.string)
  2897. proc setupJSgen*(graph: ModuleGraph; s: PSym; idgen: IdGenerator): PPassContext =
  2898. result = newModule(graph, s)
  2899. result.idgen = idgen