unofficial mirror of bug-guile@gnu.org 
 help / color / mirror / Atom feed
* bug#44050: [PATCH] doc-snarf: Add support for (ice-9 optargs) define*s.
@ 2020-10-17 23:50 Leo Prikler
  0 siblings, 0 replies; only message in thread
From: Leo Prikler @ 2020-10-17 23:50 UTC (permalink / raw)
  To: 44050

* module/scripts/doc-snarf.scm (supported-languages)[scheme]: Limit
signature start to define and define*.
(peek-sexp): New variable.
(find-std-int-doc): Implement in terms of peek-sexp.
(parse-entry, make-prototype, get-symbol): Use full function definition
instead of def-line.
(snarf): Adjust accordingly.
(join-symbols): Removed variable.
(parse-defun): New variable.
---
 module/scripts/doc-snarf.scm | 141 +++++++++++++++++++++--------------
 1 file changed, 86 insertions(+), 55 deletions(-)

diff --git a/module/scripts/doc-snarf.scm b/module/scripts/doc-snarf.scm
index fa3dfb312..3dd714d9e 100644
--- a/module/scripts/doc-snarf.scm
+++ b/module/scripts/doc-snarf.scm
@@ -152,7 +152,7 @@ This procedure foos, or bars, depending on the argument @var{braz}.
      "^;;\\."
      "^;; (.*)"
      "^;;-(.*)"
-     "^\\(define"
+     "^\\(define(\\*)?( |$)"
      #t
      )))
 
@@ -178,6 +178,14 @@ This procedure foos, or bars, depending on the argument @var{braz}.
   (write-output (snarf input lang) output
                 (if texinfo? format-texinfo format-plain)))
 
+;; Read an s-expression from @var{port}, then rewind it, so that it can be
+;; read again.
+(define (peek-sexp port)
+  (let* ((pos (ftell port))
+         (sexp (read port)))
+    (seek port pos SEEK_SET)
+    sexp))
+
 ;; fixme: this comment is required to trigger standard internal
 ;; docstring snarfing...  ideally, it wouldn't be necessary.
 ;;-ttn-mod: new proc, from snarf-docs (aren't these names fun?)
@@ -185,7 +193,8 @@ This procedure foos, or bars, depending on the argument @var{braz}.
   "Unread @var{line} from @var{input-port}, then read in the entire form and
 return the standard internal docstring if found.  Return #f if not."
   (unread-string line input-port)       ; ugh
-  (let ((form (read input-port)))
+  (seek input-port -1 SEEK_CUR)         ; ugh^2
+  (let ((form (peek-sexp input-port)))
     (cond ((and (list? form)            ; (define (PROC ARGS) "DOC" ...)
                 (< 3 (length form))
                 (eq? 'define (car form))
@@ -270,9 +279,12 @@ return the standard internal docstring if found.  Return #f if not."
 		doc-strings (cons (match:substring m1 1) options) entries
 		(+ lno 1)))
 	   (m2
-            (let ((options (augmented-options line i-p options))) ; ttn-mod
-              (lp (read-line i-p) 'neutral '() '()
-                  (cons (parse-entry doc-strings options line input-file lno)
+            (let* ((options (augmented-options line i-p options)) ; ttn-mod
+                   (def (peek-sexp i-p)))
+              ;; due to the rewind in augmented-options and peek-sexp,
+              ;; we will actually see this line again, so read twice
+              (lp (begin (read-line i-p) (read-line i-p)) 'neutral '() '()
+                  (cons (parse-entry doc-strings options def input-file lno)
                         entries)
                   (+ lno 1))))
            (m3
@@ -295,9 +307,10 @@ return the standard internal docstring if found.  Return #f if not."
 		doc-strings (cons (match:substring m1 1) options) entries
 		(+ lno 1)))
 	   (m2
-            (let ((options (augmented-options line i-p options))) ; ttn-mod
-              (lp (read-line i-p) 'neutral '() '()
-                  (cons (parse-entry doc-strings options line input-file lno)
+            (let* ((options (augmented-options line i-p options)) ; ttn-mod
+                   (def (peek-sexp i-p)))
+              (lp (begin (read-line i-p) (read-line i-p)) 'neutral '() '()
+                  (cons (parse-entry doc-strings options def input-file lno)
                         entries)
                   (+ lno 1))))
 	   (m3
@@ -326,13 +339,13 @@ return the standard internal docstring if found.  Return #f if not."
 
 ;; Create a docstring entry from the docstring line list
 ;; @var{doc-strings}, the option line list @var{options} and the
-;; define line @var{def-line}
-(define (parse-entry docstrings options def-line filename line-no)
+;; definition @var{def}
+(define (parse-entry docstrings options def filename line-no)
 ;  (write-line docstrings)
   (cond
-   (def-line
-     (make-entry (get-symbol def-line)
-		 (make-prototype def-line) (reverse docstrings)
+   (def
+     (make-entry (get-symbol def)
+		 (make-prototype def) (reverse docstrings)
 		 (reverse options) filename
 		 (+ (- line-no (length docstrings) (length options)) 1)))
    ((> (length docstrings) 0)
@@ -347,48 +360,66 @@ return the standard internal docstring if found.  Return #f if not."
 
 ;; Create a string which is a procedure prototype.  The necessary
 ;; information for constructing the prototype is taken from the line
-;; @var{def-line}, which is a line starting with @code{(define...}.
-(define (make-prototype def-line)
-  (call-with-input-string
-   def-line
-   (lambda (s-p)
-     (let* ((paren (read-char s-p))
-	    (keyword (read s-p))
-	    (tmp (read s-p)))
-       (cond
-	((pair? tmp)
-	 (join-symbols tmp))
-	((symbol? tmp)
-	 (symbol->string tmp))
-	(else
-	 ""))))))
-
-(define (get-symbol def-line)
-  (call-with-input-string
-   def-line
-   (lambda (s-p)
-     (let* ((paren (read-char s-p))
-	    (keyword (read s-p))
-	    (tmp (read s-p)))
-       (cond
-	((pair? tmp)
-	 (car tmp))
-	((symbol? tmp)
-	 tmp)
-	(else
-	 'foo))))))
-
-;; Append the symbols in the string list @var{s}, separated with a
-;; space character.
-(define (join-symbols s)
-  (cond ((null? s)
-	 "")
-	((symbol? s)
-	 (string-append ". " (symbol->string s)))
-	((null? (cdr s))
-	 (symbol->string (car s)))
-	(else
-	 (string-append (symbol->string (car s)) " " (join-symbols (cdr s))))))
+;; @var{def}, which is the full function definition starting with
+;; @code{(define...}.
+(define (make-prototype def)
+  (let ((tmp (false-if-exception (cadr def))))
+    (cond
+     ((pair? tmp) (parse-defun tmp))
+     ((symbol? tmp) (symbol->string tmp))
+     (else ""))))
+
+(define (get-symbol def)
+  (let ((tmp (false-if-exception (cadr def))))
+    (cond
+     ((pair? tmp) (car tmp))
+     ((symbol? tmp) tmp)
+     (else 'foo))))
+
+;; Parse function definition @var{defun}.
+;; This parser accepts the formats
+;; @itemize
+;; @item (name . args)
+;; @item (name arg1 arg2 ...)
+;; @item (name arg1 arg2 ... [#:optional optarg...] [#:key kwarg...] . rest)
+;; @item (name arg1 arg2 ... [#:optional optarg...] [#:key kwarg...] [#:rest rest])
+;; @end itemize
+(define (parse-defun defun)
+  (define (append-arg prototype arg val optional? key?)
+    (string-append prototype
+                   " "
+                   (cond
+                    (optional? "[")
+                    (key? "[#:")
+                    (else ""))
+                   (symbol->string arg)
+                   (if val (string-append "=" (object->string val write)) "")
+                   (if (or optional? key?) "]" "")))
+  (let lp ((prototype (symbol->string (car defun)))
+           (args (cdr defun))
+           (optional? #f)
+           (key? #f))
+    (cond
+     ((null? args) prototype)
+     ((symbol? args)
+      (string-append prototype " . " (symbol->string args)))
+     (else
+      (let ((arg (car args))
+            (rest (cdr args)))
+        (cond
+         ((eq? arg #:optional) (lp prototype rest #t #f))
+         ((eq? arg #:key) (lp prototype rest #f #t))
+         ((eq? arg #:rest)
+          (lp (string-append prototype " .") rest #f #f))
+         ((symbol? arg)
+          (lp (append-arg prototype arg #f optional? key?)
+              rest optional? key?))
+         ((pair? arg)
+          (lp (append-arg prototype (car arg) (cadr arg) optional? key?)
+              rest optional? key?))
+         (else
+          (error "failed to parse ~s: cannot match ~s"
+                 defun arg))))))))
 
 ;; Write @var{entries} to @var{output-file} using @var{writer}.
 ;; @var{writer} is a proc that takes one entry.
-- 
2.28.0






^ permalink raw reply related	[flat|nested] only message in thread

only message in thread, other threads:[~2020-10-17 23:50 UTC | newest]

Thread overview: (only message) (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2020-10-17 23:50 bug#44050: [PATCH] doc-snarf: Add support for (ice-9 optargs) define*s Leo Prikler

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for read-only IMAP folder(s) and NNTP newsgroup(s).