Skip to content

Commit

Permalink
decouple parser from PNode
Browse files Browse the repository at this point in the history
The main change - introduce a `ParsedNode` type which replaces `PNode`
in the parser. This change allows for further work on decoupling `sem`
from other parts of the compiler, making it easier to implement
improvements in a way that would not rip through the whole codebase and
test suite. Right now introduced type closely mimics the `PNode`
counterpart, but this is just a temporary measure for the transition
period. This commit is a part of multi-step series - full list can be
seen in the related issue #423

- Add missing documentation for changes in the earlier commit, add more
  how-tos to the debugging section (I haven't coded in a while, so was
  especially important to write down explanations for anything I had
  trouble with)
  602367b
- Cleanup the `passes.nim` implementation a bit - despite common (at
  least seemingly shared by many of the previous author of the codebase)
  misconception longer variable names actually *do* increase
  readability. Also infamous recommendations for the "structured
  programming" also do not really mesh with proliferation of `break`
  statements in the code.
-
  • Loading branch information
haxscramper committed Sep 2, 2022
1 parent 837238f commit bb16d0a
Show file tree
Hide file tree
Showing 13 changed files with 831 additions and 435 deletions.
40 changes: 39 additions & 1 deletion compiler/ast/ast.nim
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,8 @@ import
ast_types, # Main ast type definitions
ast_idgen, # Per module Id generation
ast_query, # querying/reading the ast
ast_parsed_types, # Data types for the parsed node
lexer, # NumericalBase
],
compiler/front/[
options
Expand All @@ -29,7 +31,7 @@ import
tables # For symbol table mapping
]

export ast_types, ast_idgen, ast_query, int128
export ast_types, ast_idgen, ast_query, int128, ast_parsed_types

var ggDebug* {.deprecated.}: bool ## convenience switch for trying out things

Expand Down Expand Up @@ -604,3 +606,39 @@ proc toHumanStr*(kind: TSymKind): string =
proc toHumanStr*(kind: TTypeKind): string =
## strips leading `tk`
result = toHumanStrImpl(kind, 2)


proc setBaseFlags(n: PNode, base: NumericalBase) =
case base
of base10: discard
of base2: incl(n.flags, nfBase2)
of base8: incl(n.flags, nfBase8)
of base16: incl(n.flags, nfBase16)


proc toPNode*(parsed: ParsedNode): PNode =
result = newNodeI(parsed.kind, parsed.info)
result.comment = parsed.comment
case parsed.kind:
of nkFloatKinds:
result.floatVal = parsed.token.fNumber

of nkIntKinds - { nkCharLit }:
result.intVal = parsed.token.iNumber
result.setBaseFlags(parsed.token.base)

of nkCharLit:
result.intVal = ord(parsed.token.literal[0])

of nkStrKinds:
result.strVal = parsed.token.literal

of nkIdent:
result.ident = parsed.token.ident

else:
if parsed.isBlockArg:
result.flags.incl nfBlockArg

for sub in items(parsed):
result.add sub.toPNode()
111 changes: 111 additions & 0 deletions compiler/ast/ast_parsed_types.nim
Original file line number Diff line number Diff line change
@@ -0,0 +1,111 @@
## Data structure for the parser results

import
compiler/ast/[
ast_types, # For the node kinds
lexer # For the token type definition
]

# NOTE further refactoring considerations for the parser
#
# - store everything in tokens, do not require identifier interning for any
# purposes during the parsing stage, it must be done later, during
# conversion to a PNode. This will simplify some parts of the type
# definition.
# - remove nim"pretty" - this is an absolute joke of implementation and
# it should not be placed where it is now.

type
ParsedNode* = ref object
# NOTE next two fields are very large combined, but further plans will
# deal with that problem - current implementation is easier to write
# and it is just a transition point.
info*: TLineInfo # TODO replace line and separate token with index to
# the token, which in turn will store information
# about global positioning (tuple made up of a token
# id and a file ID)
#
# NOTE technically this is not really necessary even
# with the current implementation, but the parser
# consistently copies this information around anyway,
# so I will let it stay this way for now.
token*: Token # TODO Replace full token value with an index information
kind*: TNodeKind # NOTE/QUESTION - for now the same kind of nodes is
# reused as the main parser, to ease the transition,
# but in the future two different sets of node kinds
# might(?) be introduced.

# TODO replace `ref` object tree with begin/end ranges for the nested
# trees in the linearized structure.
sons*: seq[ParsedNode]
comment*: string # TODO this should either be a token or a sequence of
# tokens.

# HACK explicit flags in order to track down all 'extra' information
# that is collected during parsing.
isBlockArg*: bool # QUESTION add 'nkStmtListBlockArg' or similar node
# and convert it to the `nkStmtList` + `nfBlocArg`
# flags later on? Why do we need the `nfBlockArg`
# flag in the first place?

func len*(node: ParsedNode): int =
## Number of the sons of a parsed node
return node.sons.len()

# NOTE added for the sake of API similarity between PNode
func safeLen*(node: ParsedNode): int = node.len()

func `[]`*(node: ParsedNode, idx: int | BackwardsIndex): ParsedNode =
return node.sons[idx]

func `[]=`*(node: ParsedNode, idx: int | BackwardsIndex, other: ParsedNode) =
node.sons[idx] = other

iterator items*(node: ParsedNode): ParsedNode =
for item in node.sons.items():
yield item

iterator pairs*(node: ParsedNode): (int, ParsedNode) =
for idx, item in pairs(node.sons):
yield (idx, item)

proc add*(node: ParsedNode, other: ParsedNode) =
## Add new element to the sons
node.sons.add(other)

proc transitionSonsKind*(n: ParsedNode, kind: TNodeKind) =
n.kind = kind

proc transitionIntKind*(n: ParsedNode, kind: TNodeKind) =
n.kind = kind

proc transitionNoneToSym*(n: ParsedNode) =
n.kind = nkSym

func newParsedNode*(kind: TNodeKind): ParsedNode =
## Create a new parsed node without any location or token information
return ParsedNode(kind: kind)

func newParsedNode*(
kind: TNodeKind, info: TLineInfo, sons: seq[ParsedNode] = @[]): ParsedNode =
## Create a new non-leaf parsed node with a specified location
## information and sons.
return ParsedNode(kind: kind, info: info, sons: sons)

func newParsedNode*(kind: TNodeKind, info: TLineInfo, token: Token): ParsedNode =
## Create a new leaf parsed node with the specified location information
## and token kind.
return ParsedNode(kind: kind, info: info, token: token)


proc newProcNode*(
kind: TNodeKind,
info: TLineInfo,
body, params, name, pattern, genericParams,
pragmas, exceptions: ParsedNode
): ParsedNode =

result = newParsedNode(
kind,
info,
@[name, pattern, genericParams, params, pragmas, exceptions, body])
2 changes: 2 additions & 0 deletions compiler/ast/ast_query.nim
Original file line number Diff line number Diff line change
Expand Up @@ -97,6 +97,8 @@ const
callableDefs* = nkLambdaKinds + routineDefs

nkSymChoices* = {nkClosedSymChoice, nkOpenSymChoice}
nkFloatKinds* = nkFloatLiterals # QUESTION remove float literals
# altogether?
nkStrKinds* = {nkStrLit..nkTripleStrLit}
nkIntKinds* = {nkCharLit .. nkUInt64Lit}

Expand Down
Loading

0 comments on commit bb16d0a

Please sign in to comment.