2016-10-21 20:44:50 +00:00
|
|
|
#! /bin/sh
|
|
|
|
# -*- scheme -*-
|
|
|
|
exec ${GUILE-guile} --no-auto-compile -L $HOME/src/mes/build-aux -L build-aux -e '(@@ (mes-snarf) main)' -s "$0" ${1+"$@"}
|
|
|
|
!#
|
|
|
|
|
|
|
|
;;; Mes --- Maxwell Equations of Software
|
|
|
|
;;; Copyright © 2016 Jan Nieuwenhuizen <janneke@gnu.org>
|
|
|
|
;;;
|
|
|
|
;;; mes-snarf.scm: This file is part of Mes.
|
|
|
|
;;;
|
|
|
|
;;; Mes is free software; you can redistribute it and/or modify it
|
|
|
|
;;; under the terms of the GNU General Public License as published by
|
|
|
|
;;; the Free Software Foundation; either version 3 of the License, or (at
|
|
|
|
;;; your option) any later version.
|
|
|
|
;;;
|
|
|
|
;;; Mes 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 General Public License for more details.
|
|
|
|
;;;
|
|
|
|
;;; You should have received a copy of the GNU General Public License
|
|
|
|
;;; along with Mes. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
(define-module (mes-snarf)
|
|
|
|
#:use-module (srfi srfi-1)
|
|
|
|
#:use-module (srfi srfi-26)
|
|
|
|
#:use-module (ice-9 curried-definitions)
|
|
|
|
#:use-module (ice-9 rdelim)
|
|
|
|
#:use-module (ice-9 regex)
|
|
|
|
#:use-module (oop goops))
|
|
|
|
|
|
|
|
(define ((regexp-replace regexp replace) string)
|
|
|
|
(or (and=> (string-match regexp string)
|
|
|
|
(cut regexp-substitute #f <> 'pre replace 'post))
|
|
|
|
string))
|
|
|
|
|
|
|
|
;; (define-record-type function (make-function name formals annotation)
|
|
|
|
;; function?
|
|
|
|
;; (name .name)
|
|
|
|
;; (formals .formals)
|
|
|
|
;; (annotation .annotation))
|
|
|
|
|
|
|
|
(define-class <file> ()
|
|
|
|
(name #:accessor .name #:init-keyword #:name)
|
|
|
|
(content #:accessor .content #:init-keyword #:content))
|
|
|
|
|
|
|
|
(define-class <function> ()
|
|
|
|
(name #:accessor .name #:init-keyword #:name)
|
|
|
|
(formals #:accessor .formals #:init-keyword #:formals)
|
|
|
|
(annotation #:accessor .annotation #:init-keyword #:annotation))
|
|
|
|
|
|
|
|
(define (function-scm-name f)
|
|
|
|
(or (assoc-ref (.annotation f) 'name)
|
2016-12-23 10:31:34 +00:00
|
|
|
(let ((name ((compose
|
|
|
|
(regexp-replace "_" "-")
|
|
|
|
(regexp-replace "_" "-")
|
|
|
|
(regexp-replace "_" "-")
|
|
|
|
(regexp-replace "_" "-")
|
|
|
|
(regexp-replace "^builtin_" "")
|
|
|
|
(regexp-replace "_to_" "->")
|
|
|
|
(regexp-replace "_x$" "!")
|
|
|
|
(regexp-replace "_p$" "?"))
|
|
|
|
(.name f))))
|
|
|
|
(if (not (string-suffix? "-" name)) name
|
|
|
|
(string-append "core:" (string-drop-right name 1))))))
|
2016-10-21 20:44:50 +00:00
|
|
|
|
2016-11-21 08:28:34 +00:00
|
|
|
(define %builtin-prefix% "scm_")
|
2016-10-21 20:44:50 +00:00
|
|
|
(define (function-builtin-name f)
|
|
|
|
(string-append %builtin-prefix% (.name f)))
|
|
|
|
|
2016-11-21 08:28:34 +00:00
|
|
|
(define %cell-prefix% "cell_")
|
|
|
|
(define (function-cell-name f)
|
|
|
|
(string-append %cell-prefix% (.name f)))
|
2016-10-21 20:44:50 +00:00
|
|
|
|
2016-11-21 08:28:34 +00:00
|
|
|
(define %start 1)
|
|
|
|
(define (symbol->header s i)
|
2016-12-14 18:02:19 +00:00
|
|
|
(format #f "#define cell_~a ~a\n" s i))
|
2016-11-21 08:28:34 +00:00
|
|
|
|
|
|
|
(define (symbol->source s i)
|
|
|
|
(string-append
|
2016-12-14 18:02:19 +00:00
|
|
|
(format #f "g_free.value++;\n")
|
2016-11-19 21:31:30 +00:00
|
|
|
(format #f "g_cells[cell_~a] = scm_~a;\n\n" s s)))
|
2016-11-21 08:28:34 +00:00
|
|
|
|
2016-12-23 09:38:41 +00:00
|
|
|
(define (symbol->names s i)
|
|
|
|
(string-append
|
|
|
|
(format #f "g_cells[cell_~a].car = cstring_to_list (scm_~a.name);\n" s s)))
|
|
|
|
|
2016-11-21 08:28:34 +00:00
|
|
|
(define (function->header f i)
|
2016-11-03 20:28:05 +00:00
|
|
|
(let* ((arity (or (assoc-ref (.annotation f) 'arity)
|
|
|
|
(if (string-null? (.formals f)) 0
|
|
|
|
(length (string-split (.formals f) #\,)))))
|
|
|
|
(n (if (eq? arity 'n) -1 arity)))
|
2016-11-19 21:31:30 +00:00
|
|
|
(string-append
|
|
|
|
(format #f "SCM ~a (~a);\n" (.name f) (.formals f))
|
|
|
|
(format #f "function fun_~a = {.function~a=&~a, .arity=~a};\n" (.name f) arity (.name f) n)
|
|
|
|
(format #f "scm ~a = {FUNCTION, .name=~S, .function=0};\n" (function-builtin-name f) (function-scm-name f))
|
|
|
|
(format #f "SCM cell_~a;\n\n" (.name f)))))
|
|
|
|
|
|
|
|
(define (function->source f i)
|
|
|
|
(string-append
|
|
|
|
(format #f "~a.function = g_function;\n" (function-builtin-name f))
|
|
|
|
(format #f "functions[g_function++] = fun_~a;\n" (.name f))
|
|
|
|
(format #f "cell_~a = g_free.value++;\n" (.name f))
|
|
|
|
(format #f "g_cells[cell_~a] = ~a;\n\n" (.name f) (function-builtin-name f))))
|
|
|
|
|
|
|
|
(define (function->environment f i)
|
|
|
|
(string-append
|
2016-12-23 09:38:41 +00:00
|
|
|
(format #f "scm_~a.string = cstring_to_list (scm_~a.name);\n" (.name f) (.name f))
|
2016-12-24 00:23:50 +00:00
|
|
|
(format #f "g_cells[cell_~a].string = MAKE_STRING (scm_~a.string);\n" (.name f) (.name f))
|
2016-12-23 09:38:41 +00:00
|
|
|
(format #f "a = acons (make_symbol (scm_~a.string), ~a, a);\n" (.name f) (function-cell-name f))
|
|
|
|
;;(format #f "a = add_environment (a, ~S, ~a);\n" (function-scm-name f) (function-cell-name f))
|
|
|
|
))
|
2016-10-21 20:44:50 +00:00
|
|
|
|
|
|
|
(define (snarf-symbols string)
|
2016-11-21 08:28:34 +00:00
|
|
|
(let* ((matches (append (list-matches "\nscm scm_([a-z_0-9]+) = [{](SPECIAL)," string)
|
|
|
|
(list-matches "\nscm scm_([a-z_0-9]+) = [{](SYMBOL)," string))))
|
2016-10-21 20:44:50 +00:00
|
|
|
(map (cut match:substring <> 1) matches)))
|
|
|
|
|
|
|
|
(define (snarf-functions string)
|
|
|
|
(let* ((matches (list-matches
|
2016-11-21 08:28:34 +00:00
|
|
|
"\nSCM[ \n]?([a-z0-9_]+) [(]((SCM ?[^,)]+|, )*)[)][^\n(]*([^\n]*)"
|
2016-10-21 20:44:50 +00:00
|
|
|
string)))
|
|
|
|
(map (lambda (m)
|
|
|
|
(make <function>
|
|
|
|
#:name (match:substring m 1)
|
|
|
|
#:formals (match:substring m 2)
|
|
|
|
#:annotation (with-input-from-string (match:substring m 4) read)))
|
|
|
|
matches)))
|
|
|
|
|
Move optional type predicates to type.c.
* mes.c (char_p, macro_p, number_p, pair_p, string_p, symbol_p,
vector_p, builtin_p, boolean_p): Move to type.c
* type.c: New file.
* GNUmakefile (mes.o): Depend on type snarf output.
* module/mes/loop-0.mes (cond, map, let, or, and not, evlis-env,
apply-env, eval-expand, uquote, add-unquoters, eval,
expand-macro-env, eval-begin-env, eval-if-env, sexp:define,
env:define, env:macro): Move to mes-0.mes.
* module/mes/mes-0.mes: New file.
* module/mes/type-0.mes: New file.
* scripts/include.mes: If BOOT, also include mes-0.mes. If TYPE0,
also include type-0.mes.
2016-10-22 10:16:19 +00:00
|
|
|
(define (content? f)
|
|
|
|
((compose not string-null? .content) f))
|
|
|
|
|
2016-10-21 20:44:50 +00:00
|
|
|
(define (internal? f)
|
|
|
|
((compose (cut assoc-ref <> 'internal) .annotation) f))
|
|
|
|
|
|
|
|
(define (no-environment? f)
|
|
|
|
((compose (cut assoc-ref <> 'no-environment) .annotation) f))
|
|
|
|
|
|
|
|
(define (generate-includes file-name)
|
|
|
|
(let* ((string (with-input-from-file file-name read-string))
|
|
|
|
(functions (snarf-functions string))
|
|
|
|
(functions (delete-duplicates functions (lambda (a b) (equal? (.name a) (.name b)))))
|
|
|
|
(functions (filter (negate internal?) functions))
|
|
|
|
(symbols (snarf-symbols string))
|
|
|
|
(base-name (basename file-name ".c"))
|
|
|
|
(header (make <file>
|
2016-11-21 08:28:34 +00:00
|
|
|
#:name (string-append base-name ".h")
|
|
|
|
#:content (string-join (map function->header functions (iota (length functions) (+ %start (length symbols)))) "")))
|
|
|
|
(source (make <file>
|
|
|
|
#:name (string-append base-name ".i")
|
2016-11-19 21:31:30 +00:00
|
|
|
#:content (string-join (map function->source (filter (negate no-environment?) functions) (iota (length functions) (+ (length symbols) %start))) "")))
|
2016-10-21 20:44:50 +00:00
|
|
|
(environment (make <file>
|
|
|
|
#:name (string-append base-name ".environment.i")
|
2016-11-21 08:28:34 +00:00
|
|
|
#:content (string-join (map function->environment (filter (negate no-environment?) functions) (iota (length functions) (+ (length symbols) %start))) "")))
|
|
|
|
(symbols.h (make <file>
|
|
|
|
#:name (string-append base-name ".symbols.h")
|
|
|
|
#:content (string-join (map symbol->header symbols (iota (length symbols) %start)) "")))
|
|
|
|
(symbols.i (make <file>
|
|
|
|
#:name (string-append base-name ".symbols.i")
|
2016-12-23 09:38:41 +00:00
|
|
|
#:content (string-join (map symbol->source symbols (iota (length symbols))) "")))
|
|
|
|
(symbol-names.i (make <file>
|
|
|
|
#:name (string-append base-name ".symbol-names.i")
|
|
|
|
#:content (string-join (map symbol->names symbols (iota (length symbols))) ""))))
|
|
|
|
(list header source environment symbols.h symbols.i symbol-names.i)))
|
2016-10-21 20:44:50 +00:00
|
|
|
|
|
|
|
(define (file-write file)
|
|
|
|
(with-output-to-file (.name file) (lambda () (display (.content file)))))
|
|
|
|
|
|
|
|
(define (main args)
|
|
|
|
(let* ((files (cdr args)))
|
Move optional type predicates to type.c.
* mes.c (char_p, macro_p, number_p, pair_p, string_p, symbol_p,
vector_p, builtin_p, boolean_p): Move to type.c
* type.c: New file.
* GNUmakefile (mes.o): Depend on type snarf output.
* module/mes/loop-0.mes (cond, map, let, or, and not, evlis-env,
apply-env, eval-expand, uquote, add-unquoters, eval,
expand-macro-env, eval-begin-env, eval-if-env, sexp:define,
env:define, env:macro): Move to mes-0.mes.
* module/mes/mes-0.mes: New file.
* module/mes/type-0.mes: New file.
* scripts/include.mes: If BOOT, also include mes-0.mes. If TYPE0,
also include type-0.mes.
2016-10-22 10:16:19 +00:00
|
|
|
(map file-write (filter content? (append-map generate-includes files)))))
|
2016-10-21 20:44:50 +00:00
|
|
|
|
|
|
|
;;(define string (with-input-from-file "../mes.c" read-string))
|