ref: 27bcd8512583e6cf64a74f53071111d642e67e61
dir: /parse.myr/
use std
use "types.use"
use "util.use"
use "opts.use"
use "fsel.use"
pkg bld =
const parse : (p : parser# -> bool)
;;
const failparse = {p : parser#, msg, args : ...
var buf : byte[1024]
var ap
var n
ap = std.vastart(&args)
n = std.bfmtv(buf[:], msg, ap)
std.fput(1, "%s:%i: %s", p.fname, p.line, buf[:n])
std.exit(1)
}
const parse = {p
while true
skipspace(p)
if !target(p)
break
;;
;;
skipspace(p)
if p.rest.len > 0
failparse(p, "junk in file near %s\n", p.rest[:std.min(p.rest.len, 10)])
-> false
else
-> true
;;
}
const target = {p : parser#
match word(p)
| `std.Some "bin": bintarget(p)
| `std.Some "lib": libtarget(p)
| `std.Some "sub": subtarget(p)
| `std.Some "man": mantarget(p)
| `std.Some targtype: failparse(p, "unknown targtype type %s\n", targtype)
| `std.None: -> false
;;
-> true
}
/* bintarget: myrtarget */
const bintarget = {p
p.targs = std.slpush(p.targs, `Bin myrtarget(p, "bin"))
}
/* libtarget: myrtarget */
const libtarget = {p
p.targs = std.slpush(p.targs, `Lib myrtarget(p, "lib"))
}
/* subtarget : anontarget */
const subtarget = {p
p.targs = std.slpush(p.targs, `Sub anontarget(p, "sub"))
}
/* mantarget: anontarget */
const mantarget = {p
p.targs = std.slpush(p.targs, `Man anontarget(p, "man"))
}
/*
myrtarget: name '=' inputlist ';;'
| name attrlist = inputlist ';;'
*/
const myrtarget = {p, targ
var ldscript, runtime, inst, incpath
var name, inputs, libdeps, attrs
var fsel
match word(p)
| `std.Some n: name = n
| `std.None: failparse(p, "expected target name after '%s'\n", targ)
;;
skipspace(p)
if matchc(p, '{')
match attrlist(p)
| `std.Some al: attrs = al
| `std.None: failparse(p, "invalid attr list for %s %s\n", targ, name)
;;
else
attrs = [][:]
;;
skipspace(p)
if !matchc(p, '=')
failparse(p, "expected '=' after '%s %s'\n", targ, name)
;;
match inputlist(p)
| `std.Some (wl, libs):
fsel = mkfsel()
libdeps = libs
for w in wl
fseladd(fsel, w)
;;
inputs = fselfin(fsel)
std.slfree(wl)
| `std.None: failparse(p, "expected list of file names after '%s %s'\n", targ, name)
;;
skipspace(p)
if !matchc(p, ';') || !matchc(p, ';')
failparse(p, "expected ';;' terminating input list, got %c\n", peekc(p))
;;
inst = true
ldscript = ""
runtime = ""
incpath = [][:]
for elt in attrs
match elt
| ("ldscript", lds): ldscript = std.sldup(lds)
| ("runtime", rt): runtime = std.sldup(rt)
| ("inc", path): incpath = std.slpush(incpath, std.sldup(path))
| ("noinst", val):
if val.len != 0
failparse(p, "noinst attr does not take argument\n")
;;
inst = false
| (invalid, _):
std.fatal(1, "got invalid attr '%s'\n", invalid)
;;
;;
-> std.mk([
.name=name,
.inputs=inputs,
.libdeps=libdeps,
.install=inst,
.ldscript=ldscript,
.runtime=runtime,
.incpath=incpath,
.built=false
])
}
/* anontarget: '=' namelist ';;' */
const anontarget = {p, targ
var inputs
inputs = [][:]
skipspace(p)
if !matchc(p, '=')
failparse(p, "expected '=' after '%s' target\n", targ)
;;
match wordlist(p)
| `std.None: failparse(p, "expected list of file names after '%s' target\n", targ)
| `std.Some wl: inputs = wl
;;
skipspace(p)
if !matchc(p, ';') || !matchc(p, ';')
failparse(p, "expected ';;' terminating input list\n")
;;
-> inputs
}
/*
attrlist: attrs '}'
attrs : EMPTY
| attrs attr
attr : name
| name '=' name
*/
const attrlist = {p
var al
al = [][:]
while true
match word(p)
| `std.Some k:
skipspace(p)
if matchc(p, '=')
match word(p)
| `std.Some v:
al = std.slpush(al, (k, v))
| `std.None:
failparse(p, "invalid attr in attribute list\n")
;;
else
al = std.slpush(al, (k, [][:]))
;;
| `std.None: break
;;
;;
if !matchc(p, '}')
failparse(p, "expected '}' at end of attr list\n")
;;
if al.len == 0
-> `std.None
else
-> `std.Some al
;;
}
/*
inputlist: EMPTY
| inputlist input
input : word
| "lib" word
*/
const inputlist = {p
var wl, libs
wl = [][:]
libs = [][:]
while true
match word(p)
| `std.Some "lib":
match word(p)
| `std.Some l: libs = std.slpush(libs, l)
| `std.None: failparse(p, "expected lib name after 'lib'\n")
;;
| `std.Some w: wl = std.slpush(wl, w)
| `std.None: break
;;
;;
if wl.len == 0
-> `std.None
else
-> `std.Some (wl, libs)
;;
}
/* wordlist: EMPTY | wordlist word */
const wordlist = {p
var wl
wl = [][:]
while true
match word(p)
| `std.Some w: wl = std.slpush(wl, w)
| `std.None: break
;;
;;
if wl.len == 0
-> `std.None
else
-> `std.Some wl
;;
}
/* word: /wordchar*/
const word = {p : parser#
var c, r, n
var start
skipspace(p)
r = p.rest
start = r
n = 0
while r.len > 0
c = peekc(p)
if wordchar(c)
nextc(p)
n += std.charlen(c)
else
break
;;
;;
if n > 0
-> `std.Some std.sldup(start[:n])
else
-> `std.None
;;
}
const wordchar = {c
-> std.isalnum(c) || \
c == '.' || c == '_' || c == '$' || c == '-' || \
c == '/' || c == ':' || c == '!' || c == '~' || \
c == '+'
}
const skipspace = {p : parser#
var c, r
r = p.rest
while r.len > 0
c = peekc(p)
match c
| ' ': nextc(p)
| '\t': nextc(p)
| '\n':
nextc(p)
p.line++
| '#':
while p.rest.len > 0 && peekc(p) != '\n'
nextc(p)
;;
| _:
break
;;
;;
}
const matchc = {p, c
var chr, s
if p.rest.len == 0
-> false
;;
(chr, s) = std.striter(p.rest)
if c == chr
p.rest = s
-> true
else
-> false
;;
}
const peekc = {p
-> std.decode(p.rest)
}
const nextc = {p
var c, s
(c, s) = std.striter(p.rest)
p.rest = s
-> c
}