123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622 |
- ;;; lang/c/cpp.scm - C preprocessor
- ;; Copyright (C) 2015-2019 Matthew R. Wette
- ;;
- ;; This library is free software; you can redistribute it and/or
- ;; modify it under the terms of the GNU Lesser General Public
- ;; License as published by the Free Software Foundation; either
- ;; version 3 of the License, or (at your option) any later version.
- ;;
- ;; This library is distributed in the hope that it will be useful,
- ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
- ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- ;; Lesser General Public License for more details.
- ;;
- ;; You should have received a copy of the GNU Lesser General Public License
- ;; along with this library; if not, see <http://www.gnu.org/licenses/>.
- ;;; Notes:
- ;; C preprocessor macro expansion and condition text parse-and-eval
- ;; ref: https://gcc.gnu.org/onlinedocs/gcc-3.0.1/cpp_3.html
- ;;; Code:
- (define-module (nyacc lang c99 cpp)
- #:export (
- cpp-line->stmt
- eval-cpp-cond-text
- expand-cpp-macro-ref
- parse-cpp-expr
- find-incl-in-dirl
- scan-arg-literal
- eval-cpp-expr)
- #:use-module (nyacc parse)
- #:use-module (nyacc lex)
- #:use-module (nyacc lang sx-util)
- #:use-module ((nyacc lang util) #:select (report-error)))
- (cond-expand
- (guile-2
- (use-modules (rnrs arithmetic bitwise))
- (use-modules (system base pmatch)))
- (else
- (use-modules (ice-9 optargs))
- (use-modules (nyacc compat18))))
- (define c99-std-defs
- '("__DATE__" "__FILE__" "__LINE__" "__STDC__" "__STDC_HOSTED__"
- "__STDC_VERSION__" "__TIME__"))
- (define (c99-std-def? str)
- (let loop ((defs c99-std-defs))
- (cond
- ((null? defs) #f)
- ((string=? (car defs) str) #t)
- (else (loop (cdr defs))))))
- (define (c99-std-val str)
- (cond
- ((string=? str "__DATE__") "M01 01 2001")
- ((string=? str "__FILE__") "(unknown)")
- ((string=? str "__LINE__") "0")
- ((string=? str "__STDC__") "1")
- ((string=? str "__STDC_HOSTED__") "0")
- ((string=? str "__STDC_VERSION__") "201701")
- ((string=? str "__TIME__") "00:00:00")
- (else #f)))
- (define inline-whitespace (list->char-set '(#\space #\tab)))
- ;;.@deffn {Procedure} skip-il-ws ch
- ;; Skip in-line whitespace
- ;; @end deffn
- (define (skip-il-ws ch)
- (cond
- ((eof-object? ch) ch)
- ((char-set-contains? inline-whitespace ch) (skip-il-ws (read-char)))
- (else ch)))
- ;; This reads the rest of the input, with ch and returns a string;
- ;; Replaces get-string-all from (ice-9 textual-ports).
- (define (read-rest ch)
- (list->string (let loop ((ch ch))
- (if (eof-object? ch) '()
- (cons ch (loop (read-char)))))))
- ;; Not sure about this. We want to turn a list of tokens into a string
- ;; with proper escapes.
- (define (esc-c-str str)
- (list->string
- (string-fold-right
- (lambda (ch chl)
- (case ch
- ((#\\ #\") (cons* #\\ ch chl))
- (else (cons ch chl))))
- '() str)))
- (define ident-like? (make-ident-like-p read-c-ident))
- ;; @deffn {Procedure} read-ellipsis ch
- ;; read ellipsis
- ;; @end deffn
- (define (read-ellipsis ch)
- (cond
- ((eof-object? ch) #f)
- ((char=? ch #\.) (read-char) (read-char) "...") ; assumes correct syntax
- (else #f)))
- ;; @deffn {Procedure} find-incl-in-dirl file dirl [next] => path
- ;; Find path to include file expression, (i.e., @code{<foo.h>} or
- ;; @code{"foo.h"}. If @code{"foo.h"} form look in current directory first.
- ;; If @var{next} (default false) is true then remove current directory from
- ;; search path.
- ;; @*Refs:
- ;; @itemize
- ;; @item https://gcc.gnu.org/onlinedocs/cpp/Search-Path.html
- ;; @item https://gcc.gnu.org/onlinedocs/cpp/Wrapper-Headers.html
- ;; @end itemize
- ;; @end deffn
- (define* (find-incl-in-dirl file dirl #:optional (next #f))
- (let* ((cid (and=> (port-filename (current-input-port)) dirname))
- (file-type (string-ref file 0)) ;; #\< or #\"
- (file-name (substring file 1 (1- (string-length file))))
- (dirl (if (and cid (char=? #\" file-type)) (cons cid dirl) dirl)))
- (let loop ((dirl dirl))
- (if (null? dirl) #f
- (if (and next (string=? (car dirl) cid))
- (loop (cdr dirl))
- (let ((p (string-append (car dirl) "/" file-name)))
- (if (access? p R_OK) p (loop (cdr dirl)))))))))
- ;; @deffn {Procedure} cpp-define
- ;; Reads CPP define from current input and generates a cooresponding sxml
- ;; expression.
- ;; @example
- ;; (define (name "ABC") (repl "123"))
- ;; OR
- ;; (define (name "ABC") (args "X" "Y") (repl "X+Y"))
- ;; @example
- ;; @end deffn
- (define (cpp-define)
- (define (p-args la) ;; parse args
- (if (eq? la #\()
- (let loop ((args '()) (la (skip-il-ws (read-char))))
- (cond
- ((eq? la #\)) (reverse args))
- ((read-c-ident la) =>
- (lambda (arg) (loop (cons arg args) (skip-il-ws (read-char)))))
- ((read-ellipsis la) =>
- (lambda (arg) (loop (cons arg args) (skip-il-ws (read-char)))))
- ((eq? la #\,) (loop args (skip-il-ws (read-char))))))
- (begin (if (char? la) (unread-char la)) #f)))
- (define (p-rest la) (read-rest la))
- (let* ((name (let loop ((ch (skip-il-ws (read-char))))
- (cond
- ((eof-object? ch) (throw 'cpp-error "bad #define"))
- ((read-c-ident ch))
- ((cpp-comm-skipper ch) (loop (skip-il-ws (read-char))))
- (else (throw 'cpp-error "bad #define")))))
- (args (or (p-args (read-char)) '()))
- (repl (p-rest (skip-il-ws (read-char)))))
- (if (pair? args)
- `(define (name ,name) (args . ,args) (repl ,repl))
- `(define (name ,name) (repl ,repl)))))
-
- ;; @deffn {Procedure} cpp-include
- ;; Parse CPP include statement.
- (define (cpp-include)
- (define (loop cl ch end-ch)
- (if (eq? ch end-ch) (reverse-list->string (cons ch cl))
- (loop (cons ch cl) (read-char) end-ch)))
- (let ((ch (skip-il-ws (read-char))))
- (cond
- ((char=? ch #\<) (loop (list #\<) (read-char) #\>))
- ((char=? ch #\") (loop (list #\") (read-char) #\"))
- ((read-c-ident ch))
- (else (throw 'cpp-error "bad include")))))
- ;; @deffn {Procedure} cpp-line->stmt line defs => (stmt-type text)
- ;; Parse a line from a CPP statement and return a parse tree.
- ;; @example
- ;; (parse-cpp-stmt "define X 123") => (define "X" "123")
- ;; (parse-cpp-stmt "if defined(A) && defined(B) && defined(C)"
- ;; => (if "defined(A) && defined(B) && defined(C)")
- ;; @end example
- ;; To evaluate the @code{if} statements use @code{parse-cpp-expr} and
- ;; @code{eval-cpp-expr}.
- ;; @end deffn
- (define (cpp-line->stmt line)
- (define (rd-ident) (read-c-ident (skip-il-ws (read-char))))
- (define (rd-num) (and=> (read-c-num (skip-il-ws (read-char))) cdr))
- (define (rd-rest) (read-rest (skip-il-ws (read-char))))
- (with-input-from-string line
- (lambda ()
- (let ((ch (skip-il-ws (read-char))))
- (cond
- ((read-c-ident ch) =>
- (lambda (cmds)
- (let ((cmd (string->symbol cmds)))
- (case cmd
- ((include) `(include ,(cpp-include)))
- ((include_next) `(include-next ,(cpp-include)))
- ((define) (cpp-define))
- ((undef) `(undef ,(rd-ident)))
- ((ifdef)
- `(if ,(string-append "defined(" (rd-ident) ")" (rd-rest))))
- ((ifndef)
- `(if ,(string-append "!defined(" (rd-ident) ")" (rd-rest))))
- ((if elif else endif line error warning pragma)
- (list cmd (rd-rest)))
- (else
- (list 'warning (simple-format #f "unknown CPP: ~S" line)))))))
- ((read-c-num ch) => (lambda (num) `(line ,num ,(rd-rest))))
- (else (error "nyacc cpp-line->stmt: missing code")))))))
-
- (include-from-path "nyacc/lang/c99/mach.d/cpp-tab.scm")
- (include-from-path "nyacc/lang/c99/mach.d/cpp-act.scm")
- (define cpp-raw-parser
- (make-lalr-parser (acons 'act-v cpp-act-v cpp-tables)))
- (define (cpp-err fmt . args)
- (apply throw 'cpp-error fmt args))
- ;; Since we want to be able to get CPP statements with comment in tact
- ;; (e.g., for passing to @code{pretty-print-c99}) we need to remove
- ;; comments when parsing CPP expressions. We convert a comm-reader
- ;; into a comm-skipper here. And from that generate a lexer generator.
- (define cpp-comm-skipper
- (let ((reader (make-comm-reader '(("/*" . "*/")))))
- (lambda (ch)
- (reader ch #f))))
- ;; generate a lexical analyzer per string
- (define gen-cpp-lexer
- (make-lexer-generator cpp-mtab
- #:comm-skipper cpp-comm-skipper
- #:chlit-reader read-c-chlit
- #:num-reader read-c-num))
- ;; @deffn {Procedure} parse-cpp-expr text => tree
- ;; Given a string returns a cpp parse tree. This is called by
- ;; @code{eval-cpp-expr}. The text will have had all CPP defined symbols
- ;; expanded already so no identifiers should appear in the text.
- ;; A @code{cpp-error} will be thrown if a parse error occurs.
- ;; @end deffn
- (define (parse-cpp-expr text)
- (with-throw-handler
- 'nyacc-error
- (lambda ()
- (with-input-from-string text
- (lambda () (cpp-raw-parser (gen-cpp-lexer)))))
- (lambda (key fmt . args)
- (apply throw 'cpp-error fmt args))))
- ;; @deffn {Procedure} eval-cpp-expr tree [options] => datum
- ;; Evaluate a tree produced from @code{parse-cpp-expr}.
- ;; Options include optional dictionary for defines and values
- ;; and @code{#:inc-dirs} for @code{has_include} etc
- ;; @end deffn
- (define* (eval-cpp-expr tree #:optional (dict '()) #:key (inc-dirs '()))
- (letrec
- ((tx (lambda (tr ix) (sx-ref tr ix)))
- (tx1 (lambda (tr) (sx-ref tr 1)))
- (ev (lambda (ex ix) (eval-expr (sx-ref ex ix))))
- (ev1 (lambda (ex) (ev ex 1))) ; eval expr in arg 1
- (ev2 (lambda (ex) (ev ex 2))) ; eval expr in arg 2
- (ev3 (lambda (ex) (ev ex 3))) ; eval expr in arg 3
- (eval-expr
- (lambda (tree)
- (case (car tree)
- ((fixed) (string->number (cnumstr->scm (tx1 tree))))
- ((char) (char->integer (string-ref (tx1 tree) 0)))
- ((defined) (if (assoc-ref dict (tx1 tree)) 1 0))
- ((has-include)
- (if (find-incl-in-dirl (tx1 tree) inc-dirs #f) 1 0))
- ((has-include-next)
- (if (find-incl-in-dirl (tx1 tree) inc-dirs #t) 1 0))
- ((pre-inc post-inc) (1+ (ev1 tree)))
- ((pre-dec post-dec) (1- (ev1 tree)))
- ((pos) (ev1 tree))
- ((neg) (- (ev1 tree)))
- ((not) (if (zero? (ev1 tree)) 1 0))
- ((mul) (* (ev1 tree) (ev2 tree)))
- ((div) (/ (ev1 tree) (ev2 tree)))
- ((mod) (modulo (ev1 tree) (ev2 tree)))
- ((add) (+ (ev1 tree) (ev2 tree)))
- ((sub) (- (ev1 tree) (ev2 tree)))
- ((lshift) (bitwise-arithmetic-shift-left (ev1 tree) (ev2 tree)))
- ((rshift) (bitwise-arithmetic-shift-right (ev1 tree) (ev2 tree)))
- ((lt) (if (< (ev1 tree) (ev2 tree)) 1 0))
- ((le) (if (<= (ev1 tree) (ev2 tree)) 1 0))
- ((gt) (if (> (ev1 tree) (ev2 tree)) 1 0))
- ((ge) (if (>= (ev1 tree) (ev2 tree)) 1 0))
- ((eq) (if (= (ev1 tree) (ev2 tree)) 1 0))
- ((ne) (if (= (ev1 tree) (ev2 tree)) 0 1))
- ((bitwise-not) (lognot (ev1 tree)))
- ((bitwise-or) (logior (ev1 tree) (ev2 tree)))
- ((bitwise-xor) (logxor (ev1 tree) (ev2 tree)))
- ((bitwise-and) (logand (ev1 tree) (ev2 tree)))
- ((or) (if (and (zero? (ev1 tree)) (zero? (ev2 tree))) 0 1))
- ((and) (if (or (zero? (ev1 tree)) (zero? (ev2 tree))) 0 1))
- ((cond-expr) (if (zero? (ev1 tree)) (ev3 tree) (ev2 tree)))
- ;; in CPP if ident is not defined it should be zero
- ((ident) (or (and=> (assoc-ref dict (tx1 tree)) string->number) 0))
- ((p-expr) (ev1 tree))
- ((cast) (ev2 tree))
- (else (error "nyacc eval-cpp-expr: incomplete implementation"))))))
- (eval-expr tree)))
- ;;.@deffn {Procedure} rtokl->string reverse-token-list => string
- ;; Convert reverse token-list to string.
- ;; @end deffn
- (define (rtokl->string tokl)
- ;; Turn reverse chl into a string and insert it into the string list stl.
- (define (add-chl chl stl)
- (if (null? chl) stl (cons (list->string chl) stl)))
- ;; Works like this: Scan through the list of tokens (key-val pairs or
- ;; lone characters). Lone characters are collected in a list (@code{chl});
- ;; pairs are converted into strings and combined with list of characters
- ;; into a list of strings. When done the list of strings is combined to
- ;; one string. (The token 'argval is expansion of argument.)
- (let loop ((stl '()) ; list of strings to reverse-append
- (chl '()) ; char list
- (nxt #f) ; next string to add after chl
- (tkl tokl)) ; input token list
- (cond
- (nxt
- (loop (cons nxt (add-chl chl stl)) '() #f tkl))
- ((null? tkl)
- (apply string-append (add-chl chl stl)))
- ((char? (car tkl))
- (loop stl (cons (car tkl) chl) nxt (cdr tkl)))
- (else
- (pmatch tkl
- ((($ident . ,rval) $dhash ($ident . ,lval) . ,rest)
- (loop stl chl nxt
- (acons '$ident (string-append lval rval) (list-tail tkl 3))))
- ((($ident . ,arg) $hash . ,rest)
- (loop stl chl (string-append "\"" arg "\"") (list-tail tkl 2)))
- ((($ident . ,iden) ($ident . ,lval) . ,rest)
- (loop stl chl iden rest))
- ((($ident . ,iden) . ,rest)
- (loop stl chl iden rest))
- ((($string . ,val) . ,rest)
- (loop stl (cons #\" chl) (esc-c-str val) (cons #\" rest)))
- ((($echo . ,val) . ,rest)
- (loop stl chl val rest))
- (($space $space . ,rest)
- (loop stl chl nxt rest))
- (($space . ,rest)
- (loop stl (cons #\space chl) nxt rest))
- ((($comm . ,val) . ,rest)
- ;; replace comment with extra trailing space
- (loop stl chl (string-append "/*" val "*/ ") rest))
- ((,asis . ,rest)
- (loop stl chl asis rest))
- (,otherwise
- (error "nyacc cpp rtokl->string, no match" tkl)))))))
- ;; We just scanned "defined", now need to scan the arg to inhibit expansion.
- ;; For example, we have scanned "defined"; we now scan "(FOO)" or "FOO", and
- ;; return "defined(FOO)" or "defined FOO".
- (define (scan-defined-arg)
- (let* ((ch (skip-il-ws (read-char))) (no-ec (not (char=? ch #\())))
- (let loop ((chl (list ch)) (ch (skip-il-ws (read-char))))
- (cond
- ((eof-object? ch)
- (if no-ec
- (list->string (cons #\space (reverse chl)))
- (cpp-err "illegal argument to `defined'")))
- ((char-set-contains? c:ir ch)
- (loop (cons ch chl) (read-char)))
- (no-ec
- (unread-char ch)
- (list->string (cons #\space (reverse chl))))
- ((char=? #\) (skip-il-ws ch))
- (reverse-list->string (cons #\) chl)))
- (else
- (cpp-err "illegal argument to `defined'"))))))
- ;; must be (\s*<xxx>\s*) OR (\s*"xxx"\s*) => ("<xxx>") OR ("\"xxx\"")
- (define (scan-arg-literal)
- (let ((ch (read-char)))
- ;; if exit, then did not defined __has_include(X)=__has_include__(X)
- (if (or (eof-object? ch) (not (char=? #\( ch)))
- (throw 'cpp-error "expedcting `('")))
- (let loop ((chl '()) (ch (skip-il-ws (read-char))))
- (cond
- ((eof-object? ch) (cpp-err "illegal argument"))
- ((char=? #\) ch)
- (let loop2 ((res '()) (chl chl))
- (cond
- ((null? chl)
- (string-append "(\"" (esc-c-str (list->string res)) "\")"))
- ((and (null? res) (char-whitespace? (car chl))) (loop2 res (cdr chl)))
- (else (loop2 (cons (car chl) res) (cdr chl))))))
- (else (loop (cons ch chl) (read-char))))))
- (define* (scan-cpp-input defs used end-tok #:key (keep-comments #t))
- ;; Works like this: scan for tokens (comments, parens, strings, char's, etc).
- ;; Tokens are collected in a (reverse ordered) list (tkl) and merged together
- ;; to a string on return using @code{rtokl->string}. Keep track of expanded
- ;; identifiers and rerun if something got expanded. Also, keep track of
- ;; ## and spaces so that we can parse ID /* foo */ ## /* bar */ 123
- ;; as well as ABC/*foo*/(123,456).
- (define (trim-spaces tkl)
- (if (and (pair? tkl) (eqv? '$space (car tkl)))
- (trim-spaces (cdr tkl))
- tkl))
- (define (finish rr tkl)
- (let* ((tkl (if end-tok (trim-spaces tkl) tkl))
- (repl (rtokl->string tkl)))
- (if (pair? rr)
- (cpp-expand-text repl defs (append rr used)) ;; re-run
- repl)))
-
- (let loop ((rr '()) ; list symbols resolved
- (tkl '()) ; token list of
- (lv 0) ; level
- (ch (skip-il-ws (read-char)))) ; next character
- (cond
- ((eof-object? ch) (finish rr tkl))
- ((and (eqv? end-tok ch) (zero? lv))
- (unread-char ch) (finish rr tkl))
- ((and end-tok (char=? #\) ch) (zero? lv))
- (unread-char ch) (finish rr tkl))
- ((char-set-contains? c:ws ch) ; whitespace
- (loop rr (cons '$space tkl) lv (skip-il-ws (read-char))))
- ((read-c-comm ch #f) => ; comment
- (lambda (comm)
- ;; Normally comments in CPP def's are replaced by a space. We allow
- ;; comments to get passed through, hoping this does not break code.
- (if keep-comments
- (loop rr (acons '$comm (cdr comm) tkl) lv (skip-il-ws (read-char)))
- (loop rr (cons '$space tkl) lv (skip-il-ws (read-char))))))
- ((read-c-ident ch) =>
- (lambda (iden)
- (cond
- ((string=? iden "defined")
- (loop rr
- (acons '$echo (string-append iden (scan-defined-arg)) tkl)
- lv (read-char)))
- ((member iden '("__has_include__" "__has_include_next__"))
- (cond
- ((scan-arg-literal) =>
- (lambda (arg)
- (loop rr (acons '$echo (string-append iden arg) tkl)
- lv (read-char))))
- (else
- (loop rr (acons '$ident iden tkl) lv (read-char)))))
- (else
- (let ((rval (expand-cpp-macro-ref iden defs used)))
- (if rval
- (loop #t (cons rval tkl) lv (read-char))
- (loop rr (acons '$ident iden tkl) lv (read-char))))))))
- ((read-c-string ch) =>
- (lambda (pair) (loop rr (cons pair tkl) lv (read-char))))
- ((char=? #\( ch) (loop rr (cons ch tkl) (1+ lv) (read-char)))
- ((char=? #\) ch) (loop rr (cons ch tkl) (1- lv) (read-char)))
- (else
- (loop rr (cons ch tkl) lv (read-char))))))
- ;; @deffn {Procedure} collect-args argl defs used => argd
- ;; Collect arguments to a macro which appears in C code. If not looking at
- ;; @code{(} return @code{#f}, else scan and eat up to closing @code{)}.
- ;; If multiple whitespace characters are skipped at the front then only
- ;; one @code{#\space} is re-inserted.
- ;; @end deffn
- (define (collect-args argl defs used)
- (let loop1 ((sp #f) (ch (read-char)))
- (cond
- ((eof-object? ch) (if sp (unread-char #\space)) #f)
- ((char-set-contains? inline-whitespace ch) (loop1 #t (read-char)))
- ((char=? #\( ch)
- (let loop2 ((argl argl) (argv '()) (ch ch))
- (cond
- ((eqv? ch #\)) (reverse argv))
- ((null? argl) (cpp-err "arg count"))
- ((and (null? (cdr argl)) (string=? (car argl) "..."))
- (let ((val (scan-cpp-input defs used #\))))
- (loop2 (cdr argl) (acons "__VA_ARGS__" val argv) (read-char))))
- ((or (char=? ch #\() (char=? ch #\,))
- (let* ((val (scan-cpp-input defs used #\,)))
- (loop2 (cdr argl) (acons (car argl) val argv) (read-char))))
- (else
- (error "nyacc cpp.scm: collect-args coding error")))))
- (else (unread-char ch) (if sp (unread-char #\space)) #f))))
- ;; @deffn {Procedure} px-cpp-ftn-repl argd repl => string
- ;; pre-expand CPP function where @var{argd} is an a-list of arg name
- ;; and replacement and repl is the defined replacement
- ;;
- ;; argd is alist of arguments and token lists
- ;; if end-tok == #f ignore levels
- ;; ident space fixed float chseq hash dhash arg
- ;; need to decide if we should use `(space ,tkl) or `((space) ,tkl)
- ;; This should replace args and execute hash and double-hash ??
- ;; @end deffn
- (define (px-cpp-ftn argd repl)
- (with-input-from-string repl
- (lambda ()
- (px-cpp-ftn-1 argd))))
- (define (px-cpp-ftn-1 argd)
- ;; Turn reverse chl into a string and insert it into the token stream.
- (define (ins-chl chl stl)
- (if (null? chl) stl (cons (reverse-list->string chl) stl)))
- (define (rem-space chl)
- (let loop ((chl chl))
- (cond
- ((null? chl) chl)
- ((char-set-contains? c:ws (car chl)) (loop (cdr chl)))
- (else chl))))
- (define (mk-string str) (string-append "\"" (esc-c-str str) "\""))
- (let loop ((stl '()) ; string list
- (chl '()) ; character list
- (nxt #f) ; next string after char list
- (ch (read-char))) ; next character
- (cond
- (nxt (loop (cons nxt (ins-chl chl stl)) '() #f ch))
- ((eof-object? ch)
- (apply string-append (reverse (ins-chl chl stl))))
- ((char-set-contains? c:ws ch)
- (loop stl (cons #\space chl) nxt (skip-il-ws (read-char))))
- ((read-c-comm ch #f) (loop stl (cons #\space chl) nxt (read-char)))
- ((read-c-string ch) =>
- (lambda (st) (loop stl chl (mk-string (cdr st)) (read-char))))
- ((char=? #\( ch) (loop stl (cons ch chl) nxt (read-char)))
- ((char=? #\) ch) (loop stl (cons ch chl) nxt (read-char)))
- ((read-c-ident ch) => ; replace if aval
- (lambda (iden)
- (loop stl chl (or (assoc-ref argd iden) iden) (read-char))))
- ((char=? #\# ch)
- (let ((ch (read-char)))
- (if (eqv? ch #\#)
- (loop stl (rem-space chl) nxt (skip-il-ws (read-char)))
- (let* ((aref (read-c-ident (skip-il-ws ch)))
- (aval (assoc-ref argd aref)))
- (if (not aref) (cpp-err "expecting arg-ref"))
- (if (not aval) (cpp-err "expecting arg-val"))
- (loop stl chl (mk-string aval) (read-char))))))
- (else (loop stl (cons ch chl) nxt (read-char))))))
-
- ;; @deffn {Procedure} cpp-expand-text text defs [used] => string
- ;; Expand the string @var{text} using the provided CPP @var{defs} a-list.
- ;; Identifiers in the list of strings @var{used} will not be expanded.
- ;; @end deffn
- (define* (cpp-expand-text text defs #:optional (used '()))
- (with-input-from-string text
- (lambda () (scan-cpp-input defs used #f))))
- ;; === exports =======================
- ;; @deffn {Procedure} eval-cpp-cond-text text [defs] => string
- ;; Evaluate CPP condition expression (text).
- ;; Undefined identifiers are replaced with @code{0}.
- ;; @end deffn
- (define* (eval-cpp-cond-text text #:optional (defs '()) #:key (inc-dirs '()))
- (with-throw-handler
- 'cpp-error
- (lambda ()
- (let* ((rhs (cpp-expand-text text defs))
- (exp (parse-cpp-expr rhs)))
- (eval-cpp-expr exp defs #:inc-dirs inc-dirs)))
- (lambda (key fmt . args)
- (report-error fmt args)
- (throw 'c99-error "CPP error"))))
- ;; @deffn {Procedure} expand-cpp-macro-ref ident defs [used] => repl|#f
- ;; Given an identifier seen in the current input, this checks for associated
- ;; definition in @var{defs} (generated from CPP defines). If found as simple
- ;; macro, the expansion is returned as a string. If @var{ident} refers
- ;; to a macro with arguments, then the arguments will be read from the
- ;; current input. The format of the @code{defs} entries are
- ;; @example
- ;; ("ABC" . "123")
- ;; ("MAX" ("X" "Y") . "((X)>(Y)?(X):(Y))")
- ;; @end example
- ;; @noindent
- ;; Note that this routine will look in the current-input so if you want to
- ;; expand text,
- ;; @end deffn
- (define* (expand-cpp-macro-ref ident defs #:optional (used '()))
- (let ((rval (assoc-ref defs ident)))
- (cond
- ((member ident used) #f)
- ((string? rval)
- (let* ((used (cons ident used))
- (repl (cpp-expand-text rval defs used)))
- (if (ident-like? repl)
- (or (expand-cpp-macro-ref repl defs used) repl)
- repl)))
- ((pair? rval)
- ;; GNU CPP manual: "A function-like macro is only expanded if its name
- ;; appears with a pair of parentheses after it. If you just write the
- ;; name, it is left alone."
- (and=> (collect-args (car rval) defs used)
- (lambda (argd)
- (let* ((used (cons ident used))
- (prep (px-cpp-ftn argd (cdr rval)))
- (repl (cpp-expand-text prep defs used)))
- (if (ident-like? repl)
- (or (expand-cpp-macro-ref repl defs used) repl)
- repl)))))
- ((c99-std-val ident) => identity)
- (else #f))))
- ;;; --- last line ---
|