pas2nim compiles; testing will be done later

This commit is contained in:
rumpf_a@web.de
2010-01-24 10:53:51 +01:00
parent 70465b5f06
commit 6bbed25d14
5 changed files with 311 additions and 395 deletions

View File

@@ -1,7 +1,7 @@
#
#
# The Nimrod Compiler
# (c) Copyright 2009 Andreas Rumpf
# (c) Copyright 2010 Andreas Rumpf
#
# See the file "copying.txt", included in this
# distribution, for details about the copyright.
@@ -17,7 +17,7 @@ var
cmdLineInfo: TLineInfo
proc ProcessCmdLine(pass: TCmdLinePass, command, filename: var string) =
var p = parseopt.init()
var p = parseopt.initOptParser()
while true:
parseopt.next(p)
case p.kind
@@ -40,7 +40,7 @@ proc ProcessCmdLine(pass: TCmdLinePass, command, filename: var string) =
filename = unixToNativePath(p.key) # BUGFIX for portable build scripts
break
if pass == passCmd2:
arguments = getRestOfCommandLine(p)
arguments = cmdLineRest(p)
if not (optRun in gGlobalOptions) and (arguments != ""):
rawMessage(errArgsNeedRunOption)

View File

@@ -1,115 +0,0 @@
#
#
# Nimrod's Runtime Library
# (c) Copyright 2008 Andreas Rumpf
#
# See the file "copying.txt", included in this
# distribution, for details about the copyright.
#
# A command line parser; the Nimrod version of this file
# will become part of the standard library.
import
os, strutils
type
TCmdLineKind* = enum
cmdEnd, # end of command line reached
cmdArgument, # argument detected
cmdLongoption, # a long option ``--option`` detected
cmdShortOption # a short option ``-c`` detected
TOptParser* = object of TObject
cmd*: string
pos*: int
inShortState*: bool
kind*: TCmdLineKind
key*, val*: string
proc init*(cmdline: string = ""): TOptParser
proc next*(p: var TOptParser)
proc getRestOfCommandLine*(p: TOptParser): string
# implementation
proc init(cmdline: string = ""): TOptParser =
result.pos = 0
result.inShortState = false
if cmdline != "":
result.cmd = cmdline
else:
result.cmd = ""
for i in countup(1, ParamCount()):
result.cmd = result.cmd & quoteIfContainsWhite(paramStr(i)) & ' '
result.kind = cmdEnd
result.key = ""
result.val = ""
proc parseWord(s: string, i: int, w: var string,
delim: TCharSet = {'\x09', ' ', '\0'}): int =
result = i
if s[result] == '\"':
inc(result)
while not (s[result] in {'\0', '\"'}):
add(w, s[result])
inc(result)
if s[result] == '\"': inc(result)
else:
while not (s[result] in delim):
add(w, s[result])
inc(result)
proc handleShortOption(p: var TOptParser) =
var i: int
i = p.pos
p.kind = cmdShortOption
add(p.key, p.cmd[i])
inc(i)
p.inShortState = true
while p.cmd[i] in {'\x09', ' '}:
inc(i)
p.inShortState = false
if p.cmd[i] in {':', '='}:
inc(i)
p.inShortState = false
while p.cmd[i] in {'\x09', ' '}: inc(i)
i = parseWord(p.cmd, i, p.val)
if p.cmd[i] == '\0': p.inShortState = false
p.pos = i
proc next(p: var TOptParser) =
var i: int
i = p.pos
while p.cmd[i] in {'\x09', ' '}: inc(i)
p.pos = i
setlen(p.key, 0)
setlen(p.val, 0)
if p.inShortState:
handleShortOption(p)
return
case p.cmd[i]
of '\0':
p.kind = cmdEnd
of '-':
inc(i)
if p.cmd[i] == '-':
p.kind = cmdLongOption
inc(i)
i = parseWord(p.cmd, i, p.key, {'\0', ' ', '\x09', ':', '='})
while p.cmd[i] in {'\x09', ' '}: inc(i)
if p.cmd[i] in {':', '='}:
inc(i)
while p.cmd[i] in {'\x09', ' '}: inc(i)
p.pos = parseWord(p.cmd, i, p.val)
else:
p.pos = i
else:
p.pos = i
handleShortOption(p)
else:
p.kind = cmdArgument
p.pos = parseWord(p.cmd, i, p.key)
proc getRestOfCommandLine(p: TOptParser): string =
result = strip(copy(p.cmd, p.pos + 0, len(p.cmd) - 1)) # always -1, because Pascal version uses a trailing zero here

View File

@@ -7,61 +7,58 @@
# distribution, for details about the copyright.
#
#
import
llstream, strutils, os, ast, rnimsyn, options, msgs,
strutils, os, parseopt, llstream, ast, rnimsyn, options, msgs,
paslex, pasparse
proc exSymbols(n: PNode) =
case n.kind
of nkEmpty..nkNilLit: nil
of nkProcDef..nkIteratorDef: exSymbol(n.sons[namePos])
of nkWhenStmt, nkStmtList:
for i in countup(0, sonsLen(n) - 1): exSymbols(n.sons[i])
of nkVarSection, nkConstSection:
for i in countup(0, sonsLen(n) - 1): exSymbol(n.sons[i].sons[0])
of nkTypeSection:
for i in countup(0, sonsLen(n) - 1):
exSymbol(n.sons[i].sons[0])
if (n.sons[i].sons[2] != nil) and
(n.sons[i].sons[2].kind == nkObjectTy):
fixRecordDef(n.sons[i].sons[2])
else: nil
proc CommandExportSymbols(filename: string) =
# now unused!
var module = parseFile(addFileExt(filename, NimExt))
if module != nil:
exSymbols(module)
renderModule(module, getOutFile(filename, "pretty." & NimExt))
proc CommandLexPas(filename: string) =
var f = addFileExt(filename, "pas")
var stream = LLStreamOpen(f, fmRead)
if stream != nil:
var
L: TPasLex
tok: TPasTok
OpenLexer(L, f, stream)
getPasTok(L, tok)
while tok.xkind != pxEof:
printPasTok(tok)
getPasTok(L, tok)
closeLexer(L)
else: rawMessage(errCannotOpenFile, f)
proc CommandPas(filename: string) =
var f = addFileExt(filename, "pas")
var stream = LLStreamOpen(f, fmRead)
if stream != nil:
var p: TPasParser
OpenPasParser(p, f, stream)
var module = parseUnit(p)
closePasParser(p)
renderModule(module, getOutFile(filename, NimExt))
else:
rawMessage(errCannotOpenFile, f)
const
Version = "0.8"
Usage = """
pas2nim - Pascal to Nimrod source converter
(c) 2010 Andreas Rumpf
Usage: pas2nim [options] inputfile [options]
Options:
-o, --out:FILE set output filename
--ref convert ^typ to ref typ (default: ptr typ)
--boot use special translation rules for the Nimrod compiler
-v, --version write pas2nim's version
-h, --help show this help
"""
proc main(infile, outfile: string, flags: set[TParserFlag]) =
var stream = LLStreamOpen(infile, fmRead)
if stream == nil: rawMessage(errCannotOpenFile, infile)
var p: TParser
OpenParser(p, infile, stream, flags)
var module = parseUnit(p)
closeParser(p)
renderModule(module, outfile)
var
infile = ""
outfile = ""
flags: set[TParserFlag] = {}
for kind, key, val in getopt():
case kind
of cmdArgument: infile = key
of cmdLongOption, cmdShortOption:
case key
of "help", "h":
stdout.write(Usage)
quit(0)
of "version", "v":
stdout.write(Version & "\n")
quit(0)
of "o", "out": outfile = key
of "ref": incl(flags, pfRefs)
of "boot": flags = flags + {pfRefs, pfMoreReplacements, pfImportBlackList}
else: stdout.write("[Error] unknown option: " & key)
of cmdEnd: assert(false)
if infile.len == 0:
# no filename has been given, so we show the help:
stdout.write(Usage)
else:
if outfile.len == 0:
outfile = changeFileExt(infile, "nim")
infile = addFileExt(infile, "pas")
main(infile, outfile, flags)

View File

@@ -11,7 +11,7 @@
# the scanner module.
import
nhashes, options, msgs, strutils, platform, idents, lexbase
nhashes, options, msgs, strutils, platform, idents, lexbase, llstream
const
MaxLineLength* = 80 # lines longer than this lead to a warning
@@ -24,7 +24,7 @@ const
# keywords are sorted!
type
TPasTokKind* = enum
TTokKind* = enum
pxInvalid, pxEof,
pxAnd, pxArray, pxAs, pxAsm, pxBegin, pxCase, pxClass, pxConst,
pxConstructor, pxDestructor, pxDiv, pxDo, pxDownto, pxElse, pxEnd, pxExcept,
@@ -46,7 +46,7 @@ type
pxAsgn, pxEquals, pxDot, pxDotDot, pxHat, pxPlus, pxMinus, pxStar, pxSlash,
pxLe, pxLt, pxGe, pxGt, pxNeq, pxAt, pxStarDirLe, pxStarDirRi, pxCurlyDirLe,
pxCurlyDirRi
TPasTokKinds* = set[TPasTokKind]
TTokKinds* = set[TTokKind]
const
Keywords = ["and", "array", "as", "asm", "begin", "case", "class", "const",
@@ -62,9 +62,10 @@ const
firstKeyword = pxAnd
lastKeyword = pxXor
type
TPasTok* = object
xkind*: TPasTokKind # the type of the token
type
TNumericalBase* = enum base10, base2, base8, base16
TToken* = object
xkind*: TTokKind # the type of the token
ident*: PIdent # the parsed identifier
iNumber*: BiggestInt # the parsed integer literal
fNumber*: BiggestFloat # the parsed floating point literal
@@ -72,79 +73,104 @@ type
# or float literals
literal*: string # the parsed (string) literal
TPasLex* = object
TLexer* = object of TBaseLexer
filename*: string
proc getPasTok*(L: var TPasLex, tok: var TPasTok)
proc PrintPasTok*(tok: TPasTok)
proc pasTokToStr*(tok: TPasTok): string
proc getTok*(L: var TLexer, tok: var TToken)
proc PrintTok*(tok: TToken)
proc `$`*(tok: TToken): string
# implementation
var dummyIdent: PIdent
var
dummyIdent: PIdent
gLinesCompiled: int
proc fillToken(L: var TToken) =
L.TokType = tkInvalid
L.xkind = pxInvalid
L.iNumber = 0
L.Indent = 0
L.literal = ""
L.fNumber = 0.0
L.base = base10
L.ident = dummyIdent # this prevents many bugs!
proc openLexer(lex: var TLexer, filename: string, inputstream: PLLStream) =
proc openLexer*(lex: var TLexer, filename: string, inputstream: PLLStream) =
openBaseLexer(lex, inputstream)
lex.indentStack = @[0]
lex.filename = filename
lex.indentAhead = - 1
proc closeLexer(lex: var TLexer) =
proc closeLexer*(lex: var TLexer) =
inc(gLinesCompiled, lex.LineNumber)
closeBaseLexer(lex)
proc getColumn(L: TLexer): int =
result = getColNumber(L, L.bufPos)
proc getLineInfo(L: TLexer): TLineInfo =
proc getLineInfo*(L: TLexer): TLineInfo =
result = newLineInfo(L.filename, L.linenumber, getColNumber(L, L.bufpos))
proc lexMessage(L: TLexer, msg: TMsgKind, arg = "") =
proc lexMessage*(L: TLexer, msg: TMsgKind, arg = "") =
msgs.liMessage(getLineInfo(L), msg, arg)
proc lexMessagePos(L: var TLexer, msg: TMsgKind, pos: int, arg = "") =
var info = newLineInfo(L.filename, L.linenumber, pos - L.lineStart)
msgs.liMessage(info, msg, arg)
proc binaryStrSearch(x: openarray[string], y: string): int =
var a = 0
var b = len(x)
while a < b:
var mid = (a + b) div 2
if x[mid] < y: a = mid + 1
else: b = mid
if a < len(x) and x[a] == y: result = a
else: result = -1
proc TokKindToStr*(k: TTokKind): string =
case k
of pxEof: result = "[EOF]"
of firstKeyword..lastKeyword:
result = keywords[ord(k)-ord(firstKeyword)]
of pxInvalid, pxComment, pxStrLit: result = "string literal"
of pxCommand: result = "{@"
of pxAmp: result = "{&"
of pxPer: result = "{%"
of pxSymbol: result = "identifier"
of pxIntLit, pxInt64Lit: result = "integer literal"
of pxFloatLit: result = "floating point literal"
of pxParLe: result = "("
of pxParRi: result = ")"
of pxBracketLe: result = "["
of pxBracketRi: result = "]"
of pxComma: result = ","
of pxSemiColon: result = ";"
of pxColon: result = ":"
of pxAsgn: result = ":="
of pxEquals: result = "="
of pxDot: result = "."
of pxDotDot: result = ".."
of pxHat: result = "^"
of pxPlus: result = "+"
of pxMinus: result = "-"
of pxStar: result = "*"
of pxSlash: result = "/"
of pxLe: result = "<="
of pxLt: result = "<"
of pxGe: result = ">="
of pxGt: result = ">"
of pxNeq: result = "<>"
of pxAt: result = "@"
of pxStarDirLe: result = "(*$"
of pxStarDirRi: result = "*)"
of pxCurlyDirLe: result = "{$"
of pxCurlyDirRi: result = "}"
proc pastokToStr(tok: TPasTok): string =
proc `$`(tok: TToken): string =
case tok.xkind
of pxIntLit, pxInt64Lit: result = $(tok.iNumber)
of pxFloatLit: result = $(tok.fNumber)
of pxInvalid, pxComment..pxStrLit: result = tok.literal
else:
if (tok.ident.s != ""): result = tok.ident.s
else: result = pasTokKindToStr[tok.xkind]
of pxInvalid, pxComment, pxStrLit: result = tok.literal
of pxSymbol: result = tok.ident.s
of pxIntLit, pxInt64Lit: result = $tok.iNumber
of pxFloatLit: result = $tok.fNumber
else: result = TokKindToStr(tok.xkind)
proc PrintPasTok(tok: TPasTok) =
write(stdout, pasTokKindToStr[tok.xkind])
write(stdout, ' ')
writeln(stdout, pastokToStr(tok))
proc PrintTok(tok: TToken) =
writeln(stdout, $tok)
proc setKeyword(L: var TPasLex, tok: var TPasTok) =
proc setKeyword(L: var TLexer, tok: var TToken) =
var x = binaryStrSearch(keywords, toLower(tok.ident.s))
if x < 0: tok.xkind = pxSymbol
else: tok.xKind = TPasTokKind(x + ord(firstKeyword))
else: tok.xKind = TTokKind(x + ord(firstKeyword))
proc matchUnderscoreChars(L: var TPasLex, tok: var TPasTok, chars: TCharSet) =
proc matchUnderscoreChars(L: var TLexer, tok: var TToken, chars: TCharSet) =
# matches ([chars]_)*
var pos = L.bufpos # use registers for pos, buf
var buf = L.buf
@@ -164,7 +190,7 @@ proc isFloatLiteral(s: string): bool =
if s[i] in {'.', 'e', 'E'}:
return true
proc getNumber2(L: var TPasLex, tok: var TPasTok) =
proc getNumber2(L: var TLexer, tok: var TToken) =
var pos = L.bufpos + 1 # skip %
if not (L.buf[pos] in {'0'..'1'}):
# BUGFIX for %date%
@@ -192,7 +218,7 @@ proc getNumber2(L: var TPasLex, tok: var TPasTok) =
else: tok.xkind = pxIntLit
L.bufpos = pos
proc getNumber16(L: var TPasLex, tok: var TPasTok) =
proc getNumber16(L: var TLexer, tok: var TToken) =
var pos = L.bufpos + 1 # skip $
tok.base = base16
var xi: biggestInt = 0
@@ -223,7 +249,7 @@ proc getNumber16(L: var TPasLex, tok: var TPasTok) =
tok.xkind = pxIntLit
L.bufpos = pos
proc getNumber10(L: var TPasLex, tok: var TPasTok) =
proc getNumber10(L: var TLexer, tok: var TToken) =
tok.base = base10
matchUnderscoreChars(L, tok, {'0'..'9'})
if (L.buf[L.bufpos] == '.') and (L.buf[L.bufpos + 1] in {'0'..'9'}):
@@ -251,7 +277,7 @@ proc HandleCRLF(L: var TLexer, pos: int): int =
of LF: result = lexbase.HandleLF(L, pos)
else: result = pos
proc getString(L: var TPasLex, tok: var TPasTok) =
proc getString(L: var TLexer, tok: var TToken) =
var xi: int
var pos = L.bufPos
var buf = L.buf
@@ -300,7 +326,7 @@ proc getString(L: var TPasLex, tok: var TPasTok) =
tok.xkind = pxStrLit
L.bufpos = pos
proc getSymbol(L: var TPasLex, tok: var TPasTok) =
proc getSymbol(L: var TLexer, tok: var TToken) =
var h: THash = 0
var pos = L.bufpos
var buf = L.buf
@@ -326,7 +352,7 @@ proc getSymbol(L: var TPasLex, tok: var TPasTok) =
L.bufpos = pos
setKeyword(L, tok)
proc scanLineComment(L: var TPasLex, tok: var TPasTok) =
proc scanLineComment(L: var TLexer, tok: var TToken) =
var pos = L.bufpos
var buf = L.buf
# a comment ends if the next line does not start with the // on the same
@@ -351,7 +377,7 @@ proc scanLineComment(L: var TPasLex, tok: var TPasTok) =
break
L.bufpos = pos
proc scanCurlyComment(L: var TPasLex, tok: var TPasTok) =
proc scanCurlyComment(L: var TLexer, tok: var TToken) =
var pos = L.bufpos
var buf = L.buf
tok.literal = "#"
@@ -371,7 +397,7 @@ proc scanCurlyComment(L: var TPasLex, tok: var TPasTok) =
inc(pos)
L.bufpos = pos
proc scanStarComment(L: var TPasLex, tok: var TPasTok) =
proc scanStarComment(L: var TLexer, tok: var TToken) =
var pos = L.bufpos
var buf = L.buf
tok.literal = "#"
@@ -396,7 +422,7 @@ proc scanStarComment(L: var TPasLex, tok: var TPasTok) =
inc(pos)
L.bufpos = pos
proc skip(L: var TPasLex, tok: var TPasTok) =
proc skip(L: var TLexer, tok: var TToken) =
var pos = L.bufpos
var buf = L.buf
while true:
@@ -410,7 +436,7 @@ proc skip(L: var TPasLex, tok: var TPasTok) =
break # EndOfFile also leaves the loop
L.bufpos = pos
proc getPasTok(L: var TPasLex, tok: var TPasTok) =
proc getTok(L: var TLexer, tok: var TToken) =
tok.xkind = pxInvalid
fillToken(tok)
skip(L, tok)

View File

@@ -15,18 +15,23 @@ import
os, llstream, paslex, idents, strutils, ast, astalgo, msgs, options
type
TPasSection* = enum
TSection = enum
seImplementation, seInterface
TPasContext* = enum
TContext = enum
conExpr, conStmt, conTypeDesc
TPasParser*{.final.} = object
section*: TPasSection
inParamList*: bool
context*: TPasContext # needed for the @emit command
lastVarSection*: PNode
lex*: TPasLex
tok*: TPasTok
repl*: TIdTable # replacements
TParserFlag* = enum
pfRefs, ## use "ref" instead of "ptr" for Pascal's ^typ
pfMoreReplacements, ## use more than the default replacements
pfImportBlackList ## use import blacklist
TParser*{.final.} = object
section: TSection
inParamList: bool
context: TContext # needed for the @emit command
lastVarSection: PNode
lex: TLexer
tok: TToken
repl: TIdTable # replacements
flags: set[TParserFlag]
TReplaceTuple* = array[0..1, string]
@@ -49,49 +54,42 @@ const
["ltu", "`<%`"], ["leu", "`<=%`"], ["shlu", "`shl`"], ["shru", "`shr`"],
["assigned", "not isNil"], ["eintoverflow", "EOverflow"], ["format", "`%`"],
["snil", "nil"], ["tostringf", "$"], ["ttextfile", "tfile"],
["tbinaryfile", "tfile"], ["strstart", "0"], ["nl", "\"\\n\""], ["tostring",
"$"]] #,
# ('NL', '"\n"'),
# ('tabulator', '''\t'''),
# ('esc', '''\e'''),
# ('cr', '''\r'''),
# ('lf', '''\l'''),
# ('ff', '''\f'''),
# ('bel', '''\a'''),
# ('backspace', '''\b'''),
# ('vt', '''\v''')
["tbinaryfile", "tfile"], ["strstart", "0"], ["nl", "\"\\n\""],
["tostring", "$"]]
proc ParseUnit*(p: var TPasParser): PNode
proc openPasParser*(p: var TPasParser, filename: string, inputStream: PLLStream)
proc closePasParser*(p: var TPasParser)
proc ParseUnit*(p: var TParser): PNode
proc openParser*(p: var TParser, filename: string, inputStream: PLLStream,
flags: set[TParserFlag] = {})
proc closeParser*(p: var TParser)
proc exSymbol*(n: var PNode)
proc fixRecordDef*(n: var PNode)
# XXX: move these two to an auxiliary module
# implementation
proc OpenPasParser(p: var TPasParser, filename: string,
inputStream: PLLStream) =
proc OpenParser(p: var TParser, filename: string,
inputStream: PLLStream, flags: set[TParserFlag] = {}) =
OpenLexer(p.lex, filename, inputStream)
initIdTable(p.repl)
for i in countup(low(stdReplacements), high(stdReplacements)):
IdTablePut(p.repl, getIdent(stdReplacements[i][0]),
getIdent(stdReplacements[i][1]))
if gCmd == cmdBoot:
if pfMoreReplacements in flags:
for i in countup(low(nimReplacements), high(nimReplacements)):
IdTablePut(p.repl, getIdent(nimReplacements[i][0]),
getIdent(nimReplacements[i][1]))
p.flags = flags
proc ClosePasParser(p: var TPasParser) = CloseLexer(p.lex)
proc getTok(p: var TPasParser) = getPasTok(p.lex, p.tok)
proc CloseParser(p: var TParser) = CloseLexer(p.lex)
proc getTok(p: var TParser) = getTok(p.lex, p.tok)
proc parMessage(p: TPasParser, msg: TMsgKind, arg = "") =
proc parMessage(p: TParser, msg: TMsgKind, arg = "") =
lexMessage(p.lex, msg, arg)
proc parLineInfo(p: TPasParser): TLineInfo =
proc parLineInfo(p: TParser): TLineInfo =
result = getLineInfo(p.lex)
proc skipCom(p: var TPasParser, n: PNode) =
proc skipCom(p: var TParser, n: PNode) =
while p.tok.xkind == pxComment:
if (n != nil):
if n.comment == nil: n.comment = p.tok.literal
@@ -100,48 +98,48 @@ proc skipCom(p: var TPasParser, n: PNode) =
parMessage(p, warnCommentXIgnored, p.tok.literal)
getTok(p)
proc ExpectIdent(p: TPasParser) =
proc ExpectIdent(p: TParser) =
if p.tok.xkind != pxSymbol:
lexMessage(p.lex, errIdentifierExpected, pasTokToStr(p.tok))
lexMessage(p.lex, errIdentifierExpected, $(p.tok))
proc Eat(p: var TPasParser, xkind: TPasTokKind) =
proc Eat(p: var TParser, xkind: TTokKind) =
if p.tok.xkind == xkind: getTok(p)
else: lexMessage(p.lex, errTokenExpected, PasTokKindToStr[xkind])
else: lexMessage(p.lex, errTokenExpected, TokKindToStr(xkind))
proc Opt(p: var TPasParser, xkind: TPasTokKind) =
proc Opt(p: var TParser, xkind: TTokKind) =
if p.tok.xkind == xkind: getTok(p)
proc newNodeP(kind: TNodeKind, p: TPasParser): PNode =
proc newNodeP(kind: TNodeKind, p: TParser): PNode =
result = newNodeI(kind, getLineInfo(p.lex))
proc newIntNodeP(kind: TNodeKind, intVal: BiggestInt, p: TPasParser): PNode =
proc newIntNodeP(kind: TNodeKind, intVal: BiggestInt, p: TParser): PNode =
result = newNodeP(kind, p)
result.intVal = intVal
proc newFloatNodeP(kind: TNodeKind, floatVal: BiggestFloat,
p: TPasParser): PNode =
p: TParser): PNode =
result = newNodeP(kind, p)
result.floatVal = floatVal
proc newStrNodeP(kind: TNodeKind, strVal: string, p: TPasParser): PNode =
proc newStrNodeP(kind: TNodeKind, strVal: string, p: TParser): PNode =
result = newNodeP(kind, p)
result.strVal = strVal
proc newIdentNodeP(ident: PIdent, p: TPasParser): PNode =
proc newIdentNodeP(ident: PIdent, p: TParser): PNode =
result = newNodeP(nkIdent, p)
result.ident = ident
proc createIdentNodeP(ident: PIdent, p: TPasParser): PNode =
proc createIdentNodeP(ident: PIdent, p: TParser): PNode =
result = newNodeP(nkIdent, p)
var x = PIdent(IdTableGet(p.repl, ident))
if x != nil: result.ident = x
else: result.ident = ident
proc parseExpr(p: var TPasParser): PNode
proc parseStmt(p: var TPasParser): PNode
proc parseTypeDesc(p: var TPasParser, definition: PNode = nil): PNode
proc parseExpr(p: var TParser): PNode
proc parseStmt(p: var TParser): PNode
proc parseTypeDesc(p: var TParser, definition: PNode = nil): PNode
proc parseEmit(p: var TPasParser, definition: PNode): PNode =
proc parseEmit(p: var TParser, definition: PNode): PNode =
getTok(p) # skip 'emit'
result = nil
if p.tok.xkind != pxCurlyDirRi:
@@ -160,7 +158,7 @@ proc parseEmit(p: var TPasParser, definition: PNode): PNode =
result = parseTypeDesc(p, definition)
eat(p, pxCurlyDirRi)
proc parseCommand(p: var TPasParser, definition: PNode = nil): PNode =
proc parseCommand(p: var TParser, definition: PNode = nil): PNode =
result = nil
getTok(p)
if p.tok.ident.id == getIdent("discard").id:
@@ -183,7 +181,7 @@ proc parseCommand(p: var TPasParser, definition: PNode = nil): PNode =
addSon(result, a.sons[0])
addSon(result, a.sons[1])
else:
parMessage(p, errInvalidDirectiveX, pasTokToStr(p.tok))
parMessage(p, errInvalidDirectiveX, $(p.tok))
result = a
elif p.tok.ident.id == getIdent("emit").id:
result = parseEmit(p, definition)
@@ -218,24 +216,21 @@ proc parseCommand(p: var TPasParser, definition: PNode = nil): PNode =
getTok(p)
eat(p, pxCurlyDirRi)
else:
parMessage(p, errInvalidDirectiveX, pasTokToStr(p.tok))
parMessage(p, errInvalidDirectiveX, $(p.tok))
while true:
getTok(p)
if (p.tok.xkind == pxCurlyDirRi) or (p.tok.xkind == pxEof): break
eat(p, pxCurlyDirRi)
result = nil
proc getPrecedence(kind: TPasTokKind): int =
proc getPrecedence(kind: TTokKind): int =
case kind
of pxDiv, pxMod, pxStar, pxSlash, pxShl, pxShr, pxAnd:
result = 5
of pxPlus, pxMinus, pxOr, pxXor:
result = 4
of pxIn, pxEquals, pxLe, pxLt, pxGe, pxGt, pxNeq, pxIs:
result = 3
of pxDiv, pxMod, pxStar, pxSlash, pxShl, pxShr, pxAnd: result = 5
of pxPlus, pxMinus, pxOr, pxXor: result = 4
of pxIn, pxEquals, pxLe, pxLt, pxGe, pxGt, pxNeq, pxIs: result = 3
else: result = -1
proc rangeExpr(p: var TPasParser): PNode =
proc rangeExpr(p: var TParser): PNode =
var a = parseExpr(p)
if p.tok.xkind == pxDotDot:
result = newNodeP(nkRange, p)
@@ -246,7 +241,7 @@ proc rangeExpr(p: var TPasParser): PNode =
else:
result = a
proc bracketExprList(p: var TPasParser, first: PNode): PNode =
proc bracketExprList(p: var TParser, first: PNode): PNode =
result = newNodeP(nkBracketExpr, p)
addSon(result, first)
getTok(p)
@@ -256,7 +251,7 @@ proc bracketExprList(p: var TPasParser, first: PNode): PNode =
getTok(p)
break
if p.tok.xkind == pxEof:
parMessage(p, errTokenExpected, PasTokKindToStr[pxBracketRi])
parMessage(p, errTokenExpected, TokKindToStr(pxBracketRi))
break
var a = rangeExpr(p)
skipCom(p, a)
@@ -265,8 +260,8 @@ proc bracketExprList(p: var TPasParser, first: PNode): PNode =
skipCom(p, a)
addSon(result, a)
proc exprColonEqExpr(p: var TPasParser, kind: TNodeKind,
tok: TPasTokKind): PNode =
proc exprColonEqExpr(p: var TParser, kind: TNodeKind,
tok: TTokKind): PNode =
var a = parseExpr(p)
if p.tok.xkind == tok:
result = newNodeP(kind, p)
@@ -277,8 +272,8 @@ proc exprColonEqExpr(p: var TPasParser, kind: TNodeKind,
else:
result = a
proc exprListAux(p: var TPasParser, elemKind: TNodeKind,
endTok, sepTok: TPasTokKind, result: PNode) =
proc exprListAux(p: var TParser, elemKind: TNodeKind,
endTok, sepTok: TTokKind, result: PNode) =
getTok(p)
skipCom(p, result)
while true:
@@ -286,7 +281,7 @@ proc exprListAux(p: var TPasParser, elemKind: TNodeKind,
getTok(p)
break
if p.tok.xkind == pxEof:
parMessage(p, errTokenExpected, PasTokKindToStr[endtok])
parMessage(p, errTokenExpected, TokKindToStr(endtok))
break
var a = exprColonEqExpr(p, elemKind, sepTok)
skipCom(p, a)
@@ -295,11 +290,11 @@ proc exprListAux(p: var TPasParser, elemKind: TNodeKind,
skipCom(p, a)
addSon(result, a)
proc qualifiedIdent(p: var TPasParser): PNode =
proc qualifiedIdent(p: var TParser): PNode =
if p.tok.xkind == pxSymbol:
result = createIdentNodeP(p.tok.ident, p)
else:
parMessage(p, errIdentifierExpected, pasTokToStr(p.tok))
parMessage(p, errIdentifierExpected, $p.tok)
return nil
getTok(p)
skipCom(p, result)
@@ -313,9 +308,9 @@ proc qualifiedIdent(p: var TPasParser): PNode =
addSon(result, createIdentNodeP(p.tok.ident, p))
getTok(p)
else:
parMessage(p, errIdentifierExpected, pasTokToStr(p.tok))
parMessage(p, errIdentifierExpected, $p.tok)
proc qualifiedIdentListAux(p: var TPasParser, endTok: TPasTokKind,
proc qualifiedIdentListAux(p: var TParser, endTok: TTokKind,
result: PNode) =
getTok(p)
skipCom(p, result)
@@ -324,7 +319,7 @@ proc qualifiedIdentListAux(p: var TPasParser, endTok: TPasTokKind,
getTok(p)
break
if p.tok.xkind == pxEof:
parMessage(p, errTokenExpected, PasTokKindToStr[endtok])
parMessage(p, errTokenExpected, TokKindToStr(endtok))
break
var a = qualifiedIdent(p)
skipCom(p, a)
@@ -333,8 +328,8 @@ proc qualifiedIdentListAux(p: var TPasParser, endTok: TPasTokKind,
skipCom(p, a)
addSon(result, a)
proc exprColonEqExprList(p: var TPasParser, kind, elemKind: TNodeKind,
endTok, sepTok: TPasTokKind): PNode =
proc exprColonEqExprList(p: var TParser, kind, elemKind: TNodeKind,
endTok, sepTok: TTokKind): PNode =
result = newNodeP(kind, p)
exprListAux(p, elemKind, endTok, sepTok, result)
@@ -345,7 +340,7 @@ proc setBaseFlags(n: PNode, base: TNumericalBase) =
of base8: incl(n.flags, nfBase8)
of base16: incl(n.flags, nfBase16)
proc identOrLiteral(p: var TPasParser): PNode =
proc identOrLiteral(p: var TParser): PNode =
case p.tok.xkind
of pxSymbol:
result = createIdentNodeP(p.tok.ident, p)
@@ -393,17 +388,17 @@ proc identOrLiteral(p: var TPasParser): PNode =
of pxCommand:
result = parseCommand(p)
else:
parMessage(p, errExprExpected, pasTokToStr(p.tok))
parMessage(p, errExprExpected, $(p.tok))
getTok(p) # we must consume a token here to prevend endless loops!
result = nil
if result != nil: skipCom(p, result)
proc primary(p: var TPasParser): PNode =
proc primary(p: var TParser): PNode =
# prefix operator?
if (p.tok.xkind == pxNot) or (p.tok.xkind == pxMinus) or
(p.tok.xkind == pxPlus):
result = newNodeP(nkPrefix, p)
var a = newIdentNodeP(getIdent(pasTokToStr(p.tok)), p)
var a = newIdentNodeP(getIdent($(p.tok)), p)
addSon(result, a)
getTok(p)
skipCom(p, a)
@@ -411,7 +406,7 @@ proc primary(p: var TPasParser): PNode =
return
elif p.tok.xkind == pxAt:
result = newNodeP(nkAddr, p)
var a = newIdentNodeP(getIdent(pasTokToStr(p.tok)), p)
var a = newIdentNodeP(getIdent($(p.tok)), p)
getTok(p)
if p.tok.xkind == pxBracketLe:
result = newNodeP(nkPrefix, p)
@@ -438,7 +433,7 @@ proc primary(p: var TPasParser): PNode =
addSon(result, createIdentNodeP(p.tok.ident, p))
getTok(p)
else:
parMessage(p, errIdentifierExpected, pasTokToStr(p.tok))
parMessage(p, errIdentifierExpected, $(p.tok))
of pxHat:
var a = result
result = newNodeP(nkDerefExpr, p)
@@ -448,16 +443,16 @@ proc primary(p: var TPasParser): PNode =
result = bracketExprList(p, result)
else: break
proc lowestExprAux(p: var TPasParser, v: var PNode, limit: int): TPasTokKind =
proc lowestExprAux(p: var TParser, v: var PNode, limit: int): TTokKind =
var
nextop: TPasTokKind
nextop: TTokKind
v2, node, opNode: PNode
v = primary(p) # expand while operators have priorities higher than 'limit'
var op = p.tok.xkind
var opPred = getPrecedence(op)
while (opPred > limit):
node = newNodeP(nkInfix, p)
opNode = newIdentNodeP(getIdent(pasTokToStr(p.tok)), p) # skip operator:
opNode = newIdentNodeP(getIdent($(p.tok)), p) # skip operator:
getTok(p)
case op
of pxPlus:
@@ -498,10 +493,8 @@ proc fixExpr(n: PNode): PNode =
if n == nil: return
case n.kind
of nkInfix:
if n.sons[1].kind == nkBracket:
n.sons[1].kind = nkCurly
if n.sons[2].kind == nkBracket:
n.sons[2].kind = nkCurly
if n.sons[1].kind == nkBracket: n.sons[1].kind = nkCurly
if n.sons[2].kind == nkBracket: n.sons[2].kind = nkCurly
if (n.sons[0].kind == nkIdent):
if (n.sons[0].ident.id == getIdent("+").id):
if (n.sons[1].kind == nkCharLit) and (n.sons[2].kind == nkStrLit) and
@@ -517,7 +510,7 @@ proc fixExpr(n: PNode): PNode =
if not (n.kind in {nkEmpty..nkNilLit}):
for i in countup(0, sonsLen(n) - 1): result.sons[i] = fixExpr(n.sons[i])
proc parseExpr(p: var TPasParser): PNode =
proc parseExpr(p: var TParser): PNode =
var oldcontext = p.context
p.context = conExpr
if p.tok.xkind == pxCommand:
@@ -527,7 +520,7 @@ proc parseExpr(p: var TPasParser): PNode =
result = fixExpr(result)
p.context = oldcontext
proc parseExprStmt(p: var TPasParser): PNode =
proc parseExprStmt(p: var TParser): PNode =
var info = parLineInfo(p)
var a = parseExpr(p)
if p.tok.xkind == pxAsgn:
@@ -545,7 +538,7 @@ proc inImportBlackList(ident: PIdent): bool =
if ident.id == getIdent(ImportBlackList[i]).id:
return true
proc parseUsesStmt(p: var TPasParser): PNode =
proc parseUsesStmt(p: var TParser): PNode =
var a: PNode
result = newNodeP(nkImportStmt, p)
getTok(p) # skip `import`
@@ -555,11 +548,11 @@ proc parseUsesStmt(p: var TPasParser): PNode =
of pxEof: break
of pxSymbol: a = newIdentNodeP(p.tok.ident, p)
else:
parMessage(p, errIdentifierExpected, pasTokToStr(p.tok))
parMessage(p, errIdentifierExpected, $(p.tok))
break
getTok(p) # skip identifier, string
skipCom(p, a)
if (gCmd != cmdBoot) or not inImportBlackList(a.ident):
if pfImportBlackList notin p.flags or not inImportBlackList(a.ident):
addSon(result, createIdentNodeP(a.ident, p))
if p.tok.xkind == pxComma:
getTok(p)
@@ -568,14 +561,14 @@ proc parseUsesStmt(p: var TPasParser): PNode =
break
if sonsLen(result) == 0: result = nil
proc parseIncludeDir(p: var TPasParser): PNode =
proc parseIncludeDir(p: var TParser): PNode =
result = newNodeP(nkIncludeStmt, p)
getTok(p) # skip `include`
var filename = ""
while true:
case p.tok.xkind
of pxSymbol, pxDot, pxDotDot, pxSlash:
filename = filename & pasTokToStr(p.tok)
filename = filename & $(p.tok)
getTok(p)
of pxStrLit:
filename = p.tok.literal
@@ -584,26 +577,26 @@ proc parseIncludeDir(p: var TPasParser): PNode =
of pxCurlyDirRi:
break
else:
parMessage(p, errIdentifierExpected, pasTokToStr(p.tok))
parMessage(p, errIdentifierExpected, $(p.tok))
break
addSon(result, newStrNodeP(nkStrLit, changeFileExt(filename, "nim"), p))
if filename == "config.inc": result = nil
proc definedExprAux(p: var TPasParser): PNode =
proc definedExprAux(p: var TParser): PNode =
result = newNodeP(nkCall, p)
addSon(result, newIdentNodeP(getIdent("defined"), p))
ExpectIdent(p)
addSon(result, createIdentNodeP(p.tok.ident, p))
getTok(p)
proc isHandledDirective(p: TPasParser): bool =
proc isHandledDirective(p: TParser): bool =
result = false
if p.tok.xkind in {pxCurlyDirLe, pxStarDirLe}:
case whichKeyword(p.tok.ident)
of wElse, wEndif: result = false
case toLower(p.tok.ident.s)
of "else", "endif": result = false
else: result = true
proc parseStmtList(p: var TPasParser): PNode =
proc parseStmtList(p: var TParser): PNode =
result = newNodeP(nkStmtList, p)
while true:
case p.tok.xkind
@@ -616,27 +609,27 @@ proc parseStmtList(p: var TPasParser): PNode =
addSon(result, parseStmt(p))
if sonsLen(result) == 1: result = result.sons[0]
proc parseIfDirAux(p: var TPasParser, result: PNode) =
proc parseIfDirAux(p: var TParser, result: PNode) =
addSon(result.sons[0], parseStmtList(p))
if p.tok.xkind in {pxCurlyDirLe, pxStarDirLe}:
var endMarker = succ(p.tok.xkind)
if whichKeyword(p.tok.ident) == wElse:
if toLower(p.tok.ident.s) == "else":
var s = newNodeP(nkElse, p)
while (p.tok.xkind != pxEof) and (p.tok.xkind != endMarker): getTok(p)
while p.tok.xkind != pxEof and p.tok.xkind != endMarker: getTok(p)
eat(p, endMarker)
addSon(s, parseStmtList(p))
addSon(result, s)
if p.tok.xkind in {pxCurlyDirLe, pxStarDirLe}:
endMarker = succ(p.tok.xkind)
if whichKeyword(p.tok.ident) == wEndif:
while (p.tok.xkind != pxEof) and (p.tok.xkind != endMarker): getTok(p)
if toLower(p.tok.ident.s) == "endif":
while p.tok.xkind != pxEof and p.tok.xkind != endMarker: getTok(p)
eat(p, endMarker)
else:
parMessage(p, errXExpected, "{$endif}")
else:
parMessage(p, errXExpected, "{$endif}")
proc parseIfdefDir(p: var TPasParser, endMarker: TPasTokKind): PNode =
proc parseIfdefDir(p: var TParser, endMarker: TTokKind): PNode =
result = newNodeP(nkWhenStmt, p)
addSon(result, newNodeP(nkElifBranch, p))
getTok(p)
@@ -644,7 +637,7 @@ proc parseIfdefDir(p: var TPasParser, endMarker: TPasTokKind): PNode =
eat(p, endMarker)
parseIfDirAux(p, result)
proc parseIfndefDir(p: var TPasParser, endMarker: TPasTokKind): PNode =
proc parseIfndefDir(p: var TParser, endMarker: TTokKind): PNode =
result = newNodeP(nkWhenStmt, p)
addSon(result, newNodeP(nkElifBranch, p))
getTok(p)
@@ -655,7 +648,7 @@ proc parseIfndefDir(p: var TPasParser, endMarker: TPasTokKind): PNode =
addSon(result.sons[0], e)
parseIfDirAux(p, result)
proc parseIfDir(p: var TPasParser, endMarker: TPasTokKind): PNode =
proc parseIfDir(p: var TParser, endMarker: TTokKind): PNode =
result = newNodeP(nkWhenStmt, p)
addSon(result, newNodeP(nkElifBranch, p))
getTok(p)
@@ -663,33 +656,33 @@ proc parseIfDir(p: var TPasParser, endMarker: TPasTokKind): PNode =
eat(p, endMarker)
parseIfDirAux(p, result)
proc parseDirective(p: var TPasParser): PNode =
proc parseDirective(p: var TParser): PNode =
result = nil
if not (p.tok.xkind in {pxCurlyDirLe, pxStarDirLe}): return
var endMarker = succ(p.tok.xkind)
if p.tok.ident != nil:
case whichKeyword(p.tok.ident)
of wInclude:
case toLower(p.tok.ident.s)
of "include":
result = parseIncludeDir(p)
eat(p, endMarker)
of wIf: result = parseIfDir(p, endMarker)
of wIfdef: result = parseIfdefDir(p, endMarker)
of wIfndef: result = parseIfndefDir(p, endMarker)
of "if": result = parseIfDir(p, endMarker)
of "ifdef": result = parseIfdefDir(p, endMarker)
of "ifndef": result = parseIfndefDir(p, endMarker)
else:
# skip unknown compiler directive
while (p.tok.xkind != pxEof) and (p.tok.xkind != endMarker): getTok(p)
while p.tok.xkind != pxEof and p.tok.xkind != endMarker: getTok(p)
eat(p, endMarker)
else:
eat(p, endMarker)
proc parseRaise(p: var TPasParser): PNode =
proc parseRaise(p: var TParser): PNode =
result = newNodeP(nkRaiseStmt, p)
getTok(p)
skipCom(p, result)
if p.tok.xkind != pxSemicolon: addSon(result, parseExpr(p))
else: addSon(result, nil)
proc parseIf(p: var TPasParser): PNode =
proc parseIf(p: var TParser): PNode =
result = newNodeP(nkIfStmt, p)
while true:
getTok(p) # skip ``if``
@@ -713,7 +706,7 @@ proc parseIf(p: var TPasParser): PNode =
else:
break
proc parseWhile(p: var TPasParser): PNode =
proc parseWhile(p: var TParser): PNode =
result = newNodeP(nkWhileStmt, p)
getTok(p)
skipCom(p, result)
@@ -722,7 +715,7 @@ proc parseWhile(p: var TPasParser): PNode =
skipCom(p, result)
addSon(result, parseStmt(p))
proc parseRepeat(p: var TPasParser): PNode =
proc parseRepeat(p: var TParser): PNode =
result = newNodeP(nkWhileStmt, p)
getTok(p)
skipCom(p, result)
@@ -746,7 +739,7 @@ proc parseRepeat(p: var TPasParser): PNode =
addSon(s, a)
addSon(result, s)
proc parseCase(p: var TPasParser): PNode =
proc parseCase(p: var TParser): PNode =
var b: PNode
result = newNodeP(nkCaseStmt, p)
getTok(p)
@@ -770,7 +763,7 @@ proc parseCase(p: var TPasParser): PNode =
if b.kind == nkElse: break
eat(p, pxEnd)
proc parseTry(p: var TPasParser): PNode =
proc parseTry(p: var TParser): PNode =
result = newNodeP(nkTryStmt, p)
getTok(p)
skipCom(p, result)
@@ -807,8 +800,7 @@ proc parseTry(p: var TPasParser): PNode =
addSon(result, e)
eat(p, pxEnd)
proc parseFor(p: var TPasParser): PNode =
var a, b, c: PNode
proc parseFor(p: var TParser): PNode =
result = newNodeP(nkForStmt, p)
getTok(p)
skipCom(p, result)
@@ -816,9 +808,9 @@ proc parseFor(p: var TPasParser): PNode =
addSon(result, createIdentNodeP(p.tok.ident, p))
getTok(p)
eat(p, pxAsgn)
a = parseExpr(p)
b = nil
c = newNodeP(nkCall, p)
var a = parseExpr(p)
var b: PNode = nil
var c = newNodeP(nkCall, p)
if p.tok.xkind == pxTo:
addSon(c, newIdentNodeP(getIdent("countup"), p))
getTok(p)
@@ -828,7 +820,7 @@ proc parseFor(p: var TPasParser): PNode =
getTok(p)
b = parseExpr(p)
else:
parMessage(p, errTokenExpected, PasTokKindToStr[pxTo])
parMessage(p, errTokenExpected, TokKindToStr(pxTo))
addSon(c, a)
addSon(c, b)
eat(p, pxDo)
@@ -836,7 +828,7 @@ proc parseFor(p: var TPasParser): PNode =
addSon(result, c)
addSon(result, parseStmt(p))
proc parseParam(p: var TPasParser): PNode =
proc parseParam(p: var TParser): PNode =
var a, v: PNode
result = newNodeP(nkIdentDefs, p)
v = nil
@@ -856,7 +848,7 @@ proc parseParam(p: var TPasParser): PNode =
of pxSymbol: a = createIdentNodeP(p.tok.ident, p)
of pxColon, pxEof, pxParRi, pxEquals: break
else:
parMessage(p, errIdentifierExpected, pasTokToStr(p.tok))
parMessage(p, errIdentifierExpected, $(p.tok))
return
getTok(p) # skip identifier
skipCom(p, a)
@@ -873,7 +865,7 @@ proc parseParam(p: var TPasParser): PNode =
else:
addSon(result, nil)
if p.tok.xkind != pxEquals:
parMessage(p, errColonOrEqualsExpected, pasTokToStr(p.tok))
parMessage(p, errColonOrEqualsExpected, $(p.tok))
if p.tok.xkind == pxEquals:
getTok(p)
skipCom(p, result)
@@ -881,7 +873,7 @@ proc parseParam(p: var TPasParser): PNode =
else:
addSon(result, nil)
proc parseParamList(p: var TPasParser): PNode =
proc parseParamList(p: var TParser): PNode =
var a: PNode
result = newNodeP(nkFormalParams, p)
addSon(result, nil) # return type
@@ -910,16 +902,16 @@ proc parseParamList(p: var TPasParser): PNode =
skipCom(p, result)
result.sons[0] = parseTypeDesc(p)
proc parseCallingConvention(p: var TPasParser): PNode =
proc parseCallingConvention(p: var TParser): PNode =
result = nil
if p.tok.xkind == pxSymbol:
case whichKeyword(p.tok.ident)
of wStdcall, wCDecl, wSafeCall, wSysCall, wInline, wFastCall:
case toLower(p.tok.ident.s)
of "stdcall", "cdecl", "safecall", "syscall", "inline", "fastcall":
result = newNodeP(nkPragma, p)
addSon(result, newIdentNodeP(p.tok.ident, p))
getTok(p)
opt(p, pxSemicolon)
of wRegister:
of "register":
result = newNodeP(nkPragma, p)
addSon(result, newIdentNodeP(getIdent("fastcall"), p))
getTok(p)
@@ -927,20 +919,20 @@ proc parseCallingConvention(p: var TPasParser): PNode =
else:
nil
proc parseRoutineSpecifiers(p: var TPasParser, noBody: var bool): PNode =
proc parseRoutineSpecifiers(p: var TParser, noBody: var bool): PNode =
var e: PNode
result = parseCallingConvention(p)
noBody = false
while p.tok.xkind == pxSymbol:
case whichKeyword(p.tok.ident)
of wAssembler, wOverload, wFar:
case toLower(p.tok.ident.s)
of "assembler", "overload", "far":
getTok(p)
opt(p, pxSemicolon)
of wForward:
of "forward":
noBody = true
getTok(p)
opt(p, pxSemicolon)
of wImportc:
of "importc":
# This is a fake for platform module. There is no ``importc``
# directive in Pascal.
if result == nil: result = newNodeP(nkPragma, p)
@@ -948,7 +940,7 @@ proc parseRoutineSpecifiers(p: var TPasParser, noBody: var bool): PNode =
noBody = true
getTok(p)
opt(p, pxSemicolon)
of wNoConv:
of "noconv":
# This is a fake for platform module. There is no ``noconv``
# directive in Pascal.
if result == nil: result = newNodeP(nkPragma, p)
@@ -956,19 +948,19 @@ proc parseRoutineSpecifiers(p: var TPasParser, noBody: var bool): PNode =
noBody = true
getTok(p)
opt(p, pxSemicolon)
of wProcVar:
of "procvar":
# This is a fake for the Nimrod compiler. There is no ``procvar``
# directive in Pascal.
if result == nil: result = newNodeP(nkPragma, p)
addSon(result, newIdentNodeP(getIdent("procvar"), p))
getTok(p)
opt(p, pxSemicolon)
of wVarargs:
of "varargs":
if result == nil: result = newNodeP(nkPragma, p)
addSon(result, newIdentNodeP(getIdent("varargs"), p))
getTok(p)
opt(p, pxSemicolon)
of wExternal:
of "external":
if result == nil: result = newNodeP(nkPragma, p)
getTok(p)
noBody = true
@@ -993,7 +985,7 @@ proc parseRoutineSpecifiers(p: var TPasParser, noBody: var bool): PNode =
if result == nil: result = newNodeP(nkPragma, p)
addSon(result, e.sons[0])
proc parseRoutineType(p: var TPasParser): PNode =
proc parseRoutineType(p: var TParser): PNode =
result = newNodeP(nkProcTy, p)
getTok(p)
skipCom(p, result)
@@ -1002,8 +994,8 @@ proc parseRoutineType(p: var TPasParser): PNode =
addSon(result, parseCallingConvention(p))
skipCom(p, result)
proc parseEnum(p: var TPasParser): PNode =
var a, b: PNode
proc parseEnum(p: var TParser): PNode =
var a: PNode
result = newNodeP(nkEnumTy, p)
getTok(p)
skipCom(p, result)
@@ -1013,14 +1005,14 @@ proc parseEnum(p: var TPasParser): PNode =
of pxEof, pxParRi: break
of pxSymbol: a = newIdentNodeP(p.tok.ident, p)
else:
parMessage(p, errIdentifierExpected, pasTokToStr(p.tok))
parMessage(p, errIdentifierExpected, $(p.tok))
break
getTok(p) # skip identifier
skipCom(p, a)
if (p.tok.xkind == pxEquals) or (p.tok.xkind == pxAsgn):
getTok(p)
skipCom(p, a)
b = a
var b = a
a = newNodeP(nkEnumFieldDef, p)
addSon(a, b)
addSon(a, parseExpr(p))
@@ -1030,7 +1022,7 @@ proc parseEnum(p: var TPasParser): PNode =
addSon(result, a)
eat(p, pxParRi)
proc identVis(p: var TPasParser): PNode =
proc identVis(p: var TParser): PNode =
# identifier with visability
var a = createIdentNodeP(p.tok.ident, p)
if p.section == seInterface:
@@ -1042,13 +1034,13 @@ proc identVis(p: var TPasParser): PNode =
getTok(p)
type
TSymbolParser = proc (p: var TPasParser): PNode
TSymbolParser = proc (p: var TParser): PNode
proc rawIdent(p: var TPasParser): PNode =
proc rawIdent(p: var TParser): PNode =
result = createIdentNodeP(p.tok.ident, p)
getTok(p)
proc parseIdentColonEquals(p: var TPasParser,
proc parseIdentColonEquals(p: var TParser,
identParser: TSymbolParser): PNode =
var a: PNode
result = newNodeP(nkIdentDefs, p)
@@ -1057,7 +1049,7 @@ proc parseIdentColonEquals(p: var TPasParser,
of pxSymbol: a = identParser(p)
of pxColon, pxEof, pxParRi, pxEquals: break
else:
parMessage(p, errIdentifierExpected, pasTokToStr(p.tok))
parMessage(p, errIdentifierExpected, $(p.tok))
return
skipCom(p, a)
if p.tok.xkind == pxComma:
@@ -1071,7 +1063,7 @@ proc parseIdentColonEquals(p: var TPasParser,
else:
addSon(result, nil)
if p.tok.xkind != pxEquals:
parMessage(p, errColonOrEqualsExpected, pasTokToStr(p.tok))
parMessage(p, errColonOrEqualsExpected, $(p.tok))
if p.tok.xkind == pxEquals:
getTok(p)
skipCom(p, result)
@@ -1082,7 +1074,7 @@ proc parseIdentColonEquals(p: var TPasParser,
getTok(p)
skipCom(p, result)
proc parseRecordCase(p: var TPasParser): PNode =
proc parseRecordCase(p: var TParser): PNode =
var a, b, c: PNode
result = newNodeP(nkRecCase, p)
getTok(p)
@@ -1123,7 +1115,7 @@ proc parseRecordCase(p: var TPasParser): PNode =
addSon(result, b)
if b.kind == nkElse: break
proc parseRecordPart(p: var TPasParser): PNode =
proc parseRecordPart(p: var TParser): PNode =
result = nil
while (p.tok.xkind != pxEof) and (p.tok.xkind != pxEnd):
if result == nil: result = newNodeP(nkRecList, p)
@@ -1137,7 +1129,7 @@ proc parseRecordPart(p: var TPasParser): PNode =
of pxComment:
skipCom(p, lastSon(result))
else:
parMessage(p, errIdentifierExpected, pasTokToStr(p.tok))
parMessage(p, errIdentifierExpected, $(p.tok))
break
proc exSymbol(n: var PNode) =
@@ -1182,7 +1174,7 @@ proc addPragmaToIdent(ident: var PNode, pragma: PNode) =
InternalError(ident.info, "addPragmaToIdent")
addSon(pragmasNode, pragma)
proc parseRecordBody(p: var TPasParser, result, definition: PNode) =
proc parseRecordBody(p: var TParser, result, definition: PNode) =
skipCom(p, result)
var a = parseRecordPart(p)
if result.kind != nkTupleTy: fixRecordDef(a)
@@ -1206,7 +1198,7 @@ proc parseRecordBody(p: var TPasParser, result, definition: PNode) =
opt(p, pxSemicolon)
skipCom(p, result)
proc parseRecordOrObject(p: var TPasParser, kind: TNodeKind,
proc parseRecordOrObject(p: var TParser, kind: TNodeKind,
definition: PNode): PNode =
result = newNodeP(kind, p)
getTok(p)
@@ -1221,7 +1213,7 @@ proc parseRecordOrObject(p: var TPasParser, kind: TNodeKind,
addSon(result, nil)
parseRecordBody(p, result, definition)
proc parseTypeDesc(p: var TPasParser, definition: PNode = nil): PNode =
proc parseTypeDesc(p: var TParser, definition: PNode = nil): PNode =
var oldcontext = p.context
p.context = conTypeDesc
if p.tok.xkind == pxPacked: getTok(p)
@@ -1273,7 +1265,7 @@ proc parseTypeDesc(p: var TPasParser, definition: PNode = nil): PNode =
of pxHat:
getTok(p)
if p.tok.xkind == pxCommand: result = parseCommand(p)
elif gCmd == cmdBoot: result = newNodeP(nkRefTy, p)
elif pfRefs in p.flags: result = newNodeP(nkRefTy, p)
else: result = newNodeP(nkPtrTy, p)
addSon(result, parseTypeDesc(p))
of pxType:
@@ -1293,7 +1285,7 @@ proc parseTypeDesc(p: var TPasParser, definition: PNode = nil): PNode =
result = a
p.context = oldcontext
proc parseTypeDef(p: var TPasParser): PNode =
proc parseTypeDef(p: var TParser): PNode =
result = newNodeP(nkTypeDef, p)
addSon(result, identVis(p))
addSon(result, nil) # generic params
@@ -1307,14 +1299,14 @@ proc parseTypeDef(p: var TPasParser): PNode =
getTok(p)
skipCom(p, result)
proc parseTypeSection(p: var TPasParser): PNode =
proc parseTypeSection(p: var TParser): PNode =
result = newNodeP(nkTypeSection, p)
getTok(p)
skipCom(p, result)
while p.tok.xkind == pxSymbol:
addSon(result, parseTypeDef(p))
proc parseConstant(p: var TPasParser): PNode =
proc parseConstant(p: var TParser): PNode =
result = newNodeP(nkConstDef, p)
addSon(result, identVis(p))
if p.tok.xkind == pxColon:
@@ -1324,7 +1316,7 @@ proc parseConstant(p: var TPasParser): PNode =
else:
addSon(result, nil)
if p.tok.xkind != pxEquals:
parMessage(p, errColonOrEqualsExpected, pasTokToStr(p.tok))
parMessage(p, errColonOrEqualsExpected, $(p.tok))
if p.tok.xkind == pxEquals:
getTok(p)
skipCom(p, result)
@@ -1335,14 +1327,14 @@ proc parseConstant(p: var TPasParser): PNode =
getTok(p)
skipCom(p, result)
proc parseConstSection(p: var TPasParser): PNode =
proc parseConstSection(p: var TParser): PNode =
result = newNodeP(nkConstSection, p)
getTok(p)
skipCom(p, result)
while p.tok.xkind == pxSymbol:
addSon(result, parseConstant(p))
proc parseVar(p: var TPasParser): PNode =
proc parseVar(p: var TParser): PNode =
result = newNodeP(nkVarSection, p)
getTok(p)
skipCom(p, result)
@@ -1350,7 +1342,7 @@ proc parseVar(p: var TPasParser): PNode =
addSon(result, parseIdentColonEquals(p, identVis))
p.lastVarSection = result
proc parseRoutine(p: var TPasParser): PNode =
proc parseRoutine(p: var TParser): PNode =
var
stmts: PNode
noBody: bool
@@ -1381,7 +1373,7 @@ proc parseRoutine(p: var TPasParser): PNode =
for i in countup(0, sonsLen(a) - 1): addSon(stmts, a.sons[i])
addSon(result, stmts)
proc fixExit(p: var TPasParser, n: PNode): bool =
proc fixExit(p: var TParser, n: PNode): bool =
result = false
if (p.tok.ident.id == getIdent("exit").id):
var length = sonsLen(n)
@@ -1396,7 +1388,7 @@ proc fixExit(p: var TPasParser, n: PNode): bool =
opt(p, pxSemicolon)
skipCom(p, a)
proc fixVarSection(p: var TPasParser, counter: PNode) =
proc fixVarSection(p: var TParser, counter: PNode) =
if p.lastVarSection == nil: return
assert(counter.kind == nkIdent)
for i in countup(0, sonsLen(p.lastVarSection) - 1):
@@ -1408,7 +1400,23 @@ proc fixVarSection(p: var TPasParser, counter: PNode) =
delSon(p.lastVarSection, i)
return
proc parseBegin(p: var TPasParser, result: PNode) =
proc exSymbols(n: PNode) =
case n.kind
of nkEmpty..nkNilLit: nil
of nkProcDef..nkIteratorDef: exSymbol(n.sons[namePos])
of nkWhenStmt, nkStmtList:
for i in countup(0, sonsLen(n) - 1): exSymbols(n.sons[i])
of nkVarSection, nkConstSection:
for i in countup(0, sonsLen(n) - 1): exSymbol(n.sons[i].sons[0])
of nkTypeSection:
for i in countup(0, sonsLen(n) - 1):
exSymbol(n.sons[i].sons[0])
if (n.sons[i].sons[2] != nil) and
(n.sons[i].sons[2].kind == nkObjectTy):
fixRecordDef(n.sons[i].sons[2])
else: nil
proc parseBegin(p: var TParser, result: PNode) =
getTok(p)
while true:
case p.tok.xkind
@@ -1423,7 +1431,7 @@ proc parseBegin(p: var TPasParser, result: PNode) =
else: addSonIfNotNil(result, parseStmt(p))
if sonsLen(result) == 0: addSon(result, newNodeP(nkNilLit, p))
proc parseStmt(p: var TPasParser): PNode =
proc parseStmt(p: var TParser): PNode =
var oldcontext = p.context
p.context = conStmt
result = nil
@@ -1489,7 +1497,7 @@ proc parseStmt(p: var TPasParser): PNode =
if result != nil: skipCom(p, result)
p.context = oldcontext
proc parseUnit(p: var TPasParser): PNode =
proc parseUnit(p: var TParser): PNode =
result = newNodeP(nkStmtList, p)
getTok(p) # read first token
while true: