1;;; -*- lisp -*- 2 3;;;; A docstring extractor for the sbcl manual. Creates 4;;;; @include-ready documentation from the docstrings of exported 5;;;; symbols of specified packages. 6 7;;;; This software is part of the SBCL software system. SBCL is in the 8;;;; public domain and is provided with absolutely no warranty. See 9;;;; the COPYING file for more information. 10;;;; 11;;;; Written by Rudi Schlatte <rudi@constantly.at>, mangled 12;;;; by Nikodemus Siivola. 13 14;;;; TODO 15;;;; * Verbatim text 16;;;; * Quotations 17;;;; * Method documentation untested 18;;;; * Method sorting, somehow 19;;;; * Index for macros & constants? 20;;;; * This is getting complicated enough that tests would be good 21;;;; * Nesting (currently only nested itemizations work) 22;;;; * doc -> internal form -> texinfo (so that non-texinfo format are also 23;;;; easily generated) 24 25;;;; FIXME: The description below is no longer complete. This 26;;;; should possibly be turned into a contrib with proper documentation. 27 28;;;; Formatting heuristics (tweaked to format SAVE-LISP-AND-DIE sanely): 29;;;; 30;;;; Formats SYMBOL as @code{symbol}, or @var{symbol} if symbol is in 31;;;; the argument list of the defun / defmacro. 32;;;; 33;;;; Lines starting with * or - that are followed by intented lines 34;;;; are marked up with @itemize. 35;;;; 36;;;; Lines containing only a SYMBOL that are followed by indented 37;;;; lines are marked up as @table @code, with the SYMBOL as the item. 38 39(eval-when (:compile-toplevel :load-toplevel :execute) 40 (require 'sb-introspect)) 41 42(defpackage :sb-texinfo 43 (:use :cl :sb-mop) 44 (:shadow #:documentation) 45 (:export #:generate-includes #:document-package) 46 (:documentation 47 "Tools to generate TexInfo documentation from docstrings.")) 48 49(in-package :sb-texinfo) 50 51;;;; various specials and parameters 52 53(defvar *texinfo-output*) 54(defvar *texinfo-variables*) 55(defvar *documentation-package*) 56(defvar *base-package*) 57 58(defparameter *undocumented-packages* '(sb-pcl sb-int sb-kernel sb-sys sb-c)) 59 60(defparameter *documentation-types* 61 '(compiler-macro 62 function 63 method-combination 64 setf 65 ;;structure ; also handled by `type' 66 type 67 variable) 68 "A list of symbols accepted as second argument of `documentation'") 69 70(defparameter *character-replacements* 71 '((#\* . "star") (#\/ . "slash") (#\+ . "plus") 72 (#\< . "lt") (#\> . "gt") 73 (#\= . "equals")) 74 "Characters and their replacement names that `alphanumize' uses. If 75the replacements contain any of the chars they're supposed to replace, 76you deserve to lose.") 77 78(defparameter *characters-to-drop* '(#\\ #\` #\') 79 "Characters that should be removed by `alphanumize'.") 80 81(defparameter *texinfo-escaped-chars* "@{}" 82 "Characters that must be escaped with #\@ for Texinfo.") 83 84(defparameter *itemize-start-characters* '(#\* #\-) 85 "Characters that might start an itemization in docstrings when 86 at the start of a line.") 87 88(defparameter *symbol-characters* "ABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890*:-+&#'" 89 "List of characters that make up symbols in a docstring.") 90 91(defparameter *symbol-delimiters* " ,.!?;") 92 93(defparameter *ordered-documentation-kinds* 94 '(package type structure condition class macro)) 95 96;;;; utilities 97 98(defun flatten (list) 99 (cond ((null list) 100 nil) 101 ((consp (car list)) 102 (nconc (flatten (car list)) (flatten (cdr list)))) 103 ((null (cdr list)) 104 (cons (car list) nil)) 105 (t 106 (cons (car list) (flatten (cdr list)))))) 107 108(defun whitespacep (char) 109 (find char #(#\tab #\space #\page))) 110 111(defun setf-name-p (name) 112 (or (symbolp name) 113 (and (listp name) (= 2 (length name)) (eq (car name) 'setf)))) 114 115(defgeneric specializer-name (specializer)) 116 117(defmethod specializer-name ((specializer eql-specializer)) 118 (list 'eql (eql-specializer-object specializer))) 119 120(defmethod specializer-name ((specializer class)) 121 (class-name specializer)) 122 123(defun ensure-class-precedence-list (class) 124 (unless (class-finalized-p class) 125 (finalize-inheritance class)) 126 (class-precedence-list class)) 127 128(defun specialized-lambda-list (method) 129 ;; courtecy of AMOP p. 61 130 (let* ((specializers (method-specializers method)) 131 (lambda-list (method-lambda-list method)) 132 (n-required (length specializers))) 133 (append (mapcar (lambda (arg specializer) 134 (if (eq specializer (find-class 't)) 135 arg 136 `(,arg ,(specializer-name specializer)))) 137 (subseq lambda-list 0 n-required) 138 specializers) 139 (subseq lambda-list n-required)))) 140 141(defun string-lines (string) 142 "Lines in STRING as a vector." 143 (coerce (with-input-from-string (s string) 144 (loop for line = (read-line s nil nil) 145 while line collect line)) 146 'vector)) 147 148(defun indentation (line) 149 "Position of first non-SPACE character in LINE." 150 (position-if-not (lambda (c) (char= c #\Space)) line)) 151 152(defun docstring (x doc-type) 153 (cl:documentation x doc-type)) 154 155(defun flatten-to-string (list) 156 (format nil "~{~A~^-~}" (flatten list))) 157 158(defun alphanumize (original) 159 "Construct a string without characters like *`' that will f-star-ck 160up filename handling. See `*character-replacements*' and 161`*characters-to-drop*' for customization." 162 (let ((name (remove-if (lambda (x) (member x *characters-to-drop*)) 163 (if (listp original) 164 (flatten-to-string original) 165 (string original)))) 166 (chars-to-replace (mapcar #'car *character-replacements*))) 167 (flet ((replacement-delimiter (index) 168 (cond ((or (< index 0) (>= index (length name))) "") 169 ((alphanumericp (char name index)) "-") 170 (t "")))) 171 (loop for index = (position-if #'(lambda (x) (member x chars-to-replace)) 172 name) 173 while index 174 do (setf name (concatenate 'string (subseq name 0 index) 175 (replacement-delimiter (1- index)) 176 (cdr (assoc (aref name index) 177 *character-replacements*)) 178 (replacement-delimiter (1+ index)) 179 (subseq name (1+ index)))))) 180 name)) 181 182;;;; generating various names 183 184(defgeneric name (thing) 185 (:documentation "Name for a documented thing. Names are either 186symbols or lists of symbols.")) 187 188(defmethod name ((symbol symbol)) 189 symbol) 190 191(defmethod name ((cons cons)) 192 cons) 193 194(defmethod name ((package package)) 195 (short-package-name package)) 196 197(defmethod name ((method method)) 198 (list 199 (generic-function-name (method-generic-function method)) 200 (method-qualifiers method) 201 (specialized-lambda-list method))) 202 203;;; Node names for DOCUMENTATION instances 204 205(defun short-name-for-symbol (symbol &optional (package *base-package*)) 206 "Given a SYMBOL, return its name if it's available in PACKAGE, 207 or PACKAGE:SYMBOL otherwise." 208 (format nil "~@[~a:~]~a" 209 (unless (eq symbol 210 (find-symbol (symbol-name symbol) 211 package)) 212 (shortest-package-name (symbol-package symbol))) 213 (symbol-name symbol))) 214 215(defgeneric name-using-kind/name (kind name doc)) 216 217(defmethod name-using-kind/name (kind (name string) doc) 218 (declare (ignore kind doc)) 219 name) 220 221(defmethod name-using-kind/name (kind (name symbol) doc) 222 (declare (ignore kind)) 223 (short-name-for-symbol name)) 224 225(defmethod name-using-kind/name (kind (name list) doc) 226 (declare (ignore kind)) 227 (assert (setf-name-p name)) 228 (let ((name (short-name-for-symbol (second name)))) 229 (format nil "(setf ~A)" name))) 230 231(defmethod name-using-kind/name ((kind (eql 'method)) name doc) 232 (format nil "~A~{ ~A~} ~A" 233 (name-using-kind/name nil (first name) doc) 234 (second name) 235 (third name))) 236 237(defun node-name (doc) 238 "Returns TexInfo node name as a string for a DOCUMENTATION instance." 239 (let ((kind (get-kind doc))) 240 (format nil "~:(~A~) ~(~A~)" kind (name-using-kind/name kind (get-name doc) doc)))) 241 242(defun shortest-package-name (package) 243 (car (sort (copy-list (cons (package-name package) (package-nicknames package))) 244 #'< :key #'length))) 245 246(defun short-package-name (package) 247 (unless (eq package *base-package*) 248 (shortest-package-name package))) 249 250 251;;; Definition titles for DOCUMENTATION instances 252 253(defgeneric title-using-kind/name (kind name doc)) 254 255(defmethod title-using-kind/name (kind (name string) doc) 256 (declare (ignore kind doc)) 257 name) 258 259(defmethod title-using-kind/name (kind (name symbol) doc) 260 (declare (ignore kind)) 261 (short-name-for-symbol name)) 262 263(defmethod title-using-kind/name (kind (name list) doc) 264 (declare (ignore kind)) 265 (assert (setf-name-p name)) 266 (format nil "(setf ~A)" (short-name-for-symbol (second name)))) 267 268(defmethod title-using-kind/name ((kind (eql 'method)) name doc) 269 (format nil "~{~A ~}~A" 270 (second name) 271 (title-using-kind/name nil (first name) doc))) 272 273(defun title-name (doc) 274 "Returns a string to be used as name of the definition." 275 (string-downcase (title-using-kind/name (get-kind doc) (get-name doc) doc))) 276 277(defun include-pathname (doc) 278 (let* ((kind (get-kind doc)) 279 (name (nstring-downcase 280 (if (eq 'package kind) 281 (format nil "package-~A" (alphanumize (get-name doc))) 282 (format nil "~A-~A-~A" 283 (case (get-kind doc) 284 ((function generic-function) "fun") 285 (structure "struct") 286 (variable "var") 287 (otherwise (symbol-name (get-kind doc)))) 288 (alphanumize (let ((*base-package* nil)) 289 (short-package-name (get-package doc)))) 290 (alphanumize (get-name doc))))))) 291 (make-pathname :name name :type "texinfo"))) 292 293;;;; documentation class and related methods 294 295(defclass documentation () 296 ((name :initarg :name :reader get-name) 297 (kind :initarg :kind :reader get-kind) 298 (string :initarg :string :reader get-string) 299 (children :initarg :children :initform nil :reader get-children) 300 (package :initform *documentation-package* :reader get-package))) 301 302(defmethod print-object ((documentation documentation) stream) 303 (print-unreadable-object (documentation stream :type t) 304 (princ (list (get-kind documentation) (get-name documentation)) stream))) 305 306(defgeneric make-documentation (x doc-type string)) 307 308(defmethod make-documentation ((x package) doc-type string) 309 (declare (ignore doc-type)) 310 (make-instance 'documentation 311 :name (name x) 312 :kind 'package 313 :string string)) 314 315(defmethod make-documentation (x (doc-type (eql 'function)) string) 316 (declare (ignore doc-type)) 317 (let* ((fdef (and (fboundp x) (fdefinition x))) 318 (name x) 319 (kind (cond ((and (symbolp x) (special-operator-p x)) 320 'special-operator) 321 ((and (symbolp x) (macro-function x)) 322 'macro) 323 ((typep fdef 'generic-function) 324 (assert (or (symbolp name) (setf-name-p name))) 325 'generic-function) 326 (fdef 327 (assert (or (symbolp name) (setf-name-p name))) 328 'function))) 329 (children (when (eq kind 'generic-function) 330 (collect-gf-documentation fdef)))) 331 (make-instance 'documentation 332 :name (name x) 333 :string string 334 :kind kind 335 :children children))) 336 337(defmethod make-documentation ((x method) doc-type string) 338 (declare (ignore doc-type)) 339 (make-instance 'documentation 340 :name (name x) 341 :kind 'method 342 :string string)) 343 344(defmethod make-documentation (x (doc-type (eql 'type)) string) 345 (make-instance 'documentation 346 :name (name x) 347 :string string 348 :kind (etypecase (find-class x nil) 349 (structure-class 'structure) 350 (standard-class 'class) 351 (sb-pcl::condition-class 'condition) 352 ((or built-in-class null) 'type)))) 353 354(defmethod make-documentation (x (doc-type (eql 'variable)) string) 355 (make-instance 'documentation 356 :name (name x) 357 :string string 358 :kind (if (constantp x) 359 'constant 360 'variable))) 361 362(defmethod make-documentation (x (doc-type (eql 'setf)) string) 363 (declare (ignore doc-type)) 364 (make-instance 'documentation 365 :name (name x) 366 :kind 'setf-expander 367 :string string)) 368 369(defmethod make-documentation (x doc-type string) 370 (make-instance 'documentation 371 :name (name x) 372 :kind doc-type 373 :string string)) 374 375(defun maybe-documentation (x doc-type) 376 "Returns a DOCUMENTATION instance for X and DOC-TYPE, or NIL if 377there is no corresponding docstring." 378 (let ((docstring (docstring x doc-type))) 379 (when docstring 380 (make-documentation x doc-type docstring)))) 381 382(defun lambda-list (doc) 383 (case (get-kind doc) 384 ((package constant variable type structure class condition nil) 385 nil) 386 (method 387 (third (get-name doc))) 388 (t 389 ;; KLUDGE: Eugh. 390 ;; 391 ;; believe it or not, the above comment was written before CSR 392 ;; came along and obfuscated this. (2005-07-04) 393 (when (symbolp (get-name doc)) 394 (labels ((clean (x &key optional key) 395 (typecase x 396 (atom x) 397 ((cons (member &optional)) 398 (cons (car x) (clean (cdr x) :optional t))) 399 ((cons (member &key)) 400 (cons (car x) (clean (cdr x) :key t))) 401 ((cons (member &whole &environment)) 402 ;; Skip these 403 (clean (cdr x) :optional optional :key key)) 404 ((cons cons) 405 (cons 406 (cond (key (if (consp (caar x)) 407 (caaar x) 408 (caar x))) 409 (optional (caar x)) 410 (t (clean (car x)))) 411 (clean (cdr x) :key key :optional optional))) 412 (cons 413 (cons 414 (cond ((or key optional) (car x)) 415 (t (clean (car x)))) 416 (clean (cdr x) :key key :optional optional)))))) 417 (clean (sb-introspect:function-lambda-list (get-name doc)))))))) 418 419(defun get-string-name (x) 420 (let ((name (get-name x))) 421 (cond ((symbolp name) 422 (symbol-name name)) 423 ((and (consp name) (eq 'setf (car name))) 424 (symbol-name (second name))) 425 ((stringp name) 426 name) 427 (t 428 (error "Don't know which symbol to use for name ~S" name))))) 429 430(defun documentation< (x y) 431 (let ((p1 (position (get-kind x) *ordered-documentation-kinds*)) 432 (p2 (position (get-kind y) *ordered-documentation-kinds*))) 433 (if (or (not (and p1 p2)) (= p1 p2)) 434 (string< (get-string-name x) (get-string-name y)) 435 (< p1 p2)))) 436 437;;;; turning text into texinfo 438 439(defun escape-for-texinfo (string &optional downcasep) 440 "Return STRING with characters in *TEXINFO-ESCAPED-CHARS* escaped 441with #\@. Optionally downcase the result." 442 (let ((result (with-output-to-string (s) 443 (loop for char across string 444 when (find char *texinfo-escaped-chars*) 445 do (write-char #\@ s) 446 do (write-char char s))))) 447 (if downcasep (nstring-downcase result) result))) 448 449(defun empty-p (line-number lines) 450 (and (< -1 line-number (length lines)) 451 (not (indentation (svref lines line-number))))) 452 453;;; line markups 454 455(defvar *not-symbols* '("ANSI" "CLHS")) 456 457(defun locate-symbols (line) 458 "Return a list of index pairs of symbol-like parts of LINE." 459 ;; This would be a good application for a regex ... 460 (let (result) 461 (flet ((grab (start end) 462 (unless (member (subseq line start end) '("ANSI" "CLHS")) 463 (push (list start end) result)))) 464 (do ((begin nil) 465 (maybe-begin t) 466 (i 0 (1+ i))) 467 ((= i (length line)) 468 ;; symbol at end of line 469 (when (and begin (or (> i (1+ begin)) 470 (not (member (char line begin) '(#\A #\I))))) 471 (grab begin i)) 472 (nreverse result)) 473 (cond 474 ((and begin (find (char line i) *symbol-delimiters*)) 475 ;; symbol end; remember it if it's not "A" or "I" 476 (when (or (> i (1+ begin)) (not (member (char line begin) '(#\A #\I)))) 477 (grab begin i)) 478 (setf begin nil 479 maybe-begin t)) 480 ((and begin (not (find (char line i) *symbol-characters*))) 481 ;; Not a symbol: abort 482 (setf begin nil)) 483 ((and maybe-begin (not begin) (find (char line i) *symbol-characters*)) 484 ;; potential symbol begin at this position 485 (setf begin i 486 maybe-begin nil)) 487 ((find (char line i) *symbol-delimiters*) 488 ;; potential symbol begin after this position 489 (setf maybe-begin t)) 490 (t 491 ;; Not reading a symbol, not at potential start of symbol 492 (setf maybe-begin nil))))))) 493 494(defun texinfo-line (line) 495 "Format symbols in LINE texinfo-style: either as code or as 496variables if the symbol in question is contained in symbols 497*TEXINFO-VARIABLES*." 498 (with-output-to-string (result) 499 (let ((last 0)) 500 (dolist (symbol/index (locate-symbols line)) 501 (write-string (subseq line last (first symbol/index)) result) 502 (let ((symbol-name (apply #'subseq line symbol/index))) 503 (format result (if (member symbol-name *texinfo-variables* 504 :test #'string=) 505 "@var{~A}" 506 "@code{~A}") 507 (string-downcase symbol-name))) 508 (setf last (second symbol/index))) 509 (write-string (subseq line last) result)))) 510 511;;; lisp sections 512 513(defun lisp-section-p (line line-number lines) 514 "Returns T if the given LINE looks like start of lisp code -- 515ie. if it starts with whitespace followed by a paren or 516semicolon, and the previous line is empty" 517 (let ((offset (indentation line))) 518 (and offset 519 (plusp offset) 520 (find (find-if-not #'whitespacep line) "(;") 521 (empty-p (1- line-number) lines)))) 522 523(defun collect-lisp-section (lines line-number) 524 (let ((lisp (loop for index = line-number then (1+ index) 525 for line = (and (< index (length lines)) (svref lines index)) 526 while (indentation line) 527 collect line))) 528 (values (length lisp) `("@lisp" ,@lisp "@end lisp")))) 529 530;;; itemized sections 531 532(defun maybe-itemize-offset (line) 533 "Return NIL or the indentation offset if LINE looks like it starts 534an item in an itemization." 535 (let* ((offset (indentation line)) 536 (char (when offset (char line offset)))) 537 (and offset 538 (member char *itemize-start-characters* :test #'char=) 539 (char= #\Space (find-if-not (lambda (c) (char= c char)) 540 line :start offset)) 541 offset))) 542 543(defun collect-maybe-itemized-section (lines starting-line) 544 ;; Return index of next line to be processed outside 545 (let ((this-offset (maybe-itemize-offset (svref lines starting-line))) 546 (result nil) 547 (lines-consumed 0)) 548 (loop for line-number from starting-line below (length lines) 549 for line = (svref lines line-number) 550 for indentation = (indentation line) 551 for offset = (maybe-itemize-offset line) 552 do (cond 553 ((not indentation) 554 ;; empty line -- inserts paragraph. 555 (push "" result) 556 (incf lines-consumed)) 557 ((and offset (> indentation this-offset)) 558 ;; nested itemization -- handle recursively 559 ;; FIXME: tables in itemizations go wrong 560 (multiple-value-bind (sub-lines-consumed sub-itemization) 561 (collect-maybe-itemized-section lines line-number) 562 (when sub-lines-consumed 563 (incf line-number (1- sub-lines-consumed)) ; +1 on next loop 564 (incf lines-consumed sub-lines-consumed) 565 (setf result (nconc (nreverse sub-itemization) result))))) 566 ((and offset (= indentation this-offset)) 567 ;; start of new item 568 (push (format nil "@item ~A" 569 (texinfo-line (subseq line (1+ offset)))) 570 result) 571 (incf lines-consumed)) 572 ((and (not offset) (> indentation this-offset)) 573 ;; continued item from previous line 574 (push (texinfo-line line) result) 575 (incf lines-consumed)) 576 (t 577 ;; end of itemization 578 (loop-finish)))) 579 ;; a single-line itemization isn't. 580 (if (> (count-if (lambda (line) (> (length line) 0)) result) 1) 581 (values lines-consumed `("@itemize" ,@(reverse result) "@end itemize")) 582 nil))) 583 584;;; table sections 585 586(defun tabulation-body-p (offset line-number lines) 587 (when (< line-number (length lines)) 588 (let ((offset2 (indentation (svref lines line-number)))) 589 (and offset2 (< offset offset2))))) 590 591(defun tabulation-p (offset line-number lines direction) 592 (let ((step (ecase direction 593 (:backwards (1- line-number)) 594 (:forwards (1+ line-number))))) 595 (when (and (plusp line-number) (< line-number (length lines))) 596 (and (eql offset (indentation (svref lines line-number))) 597 (or (when (eq direction :backwards) 598 (empty-p step lines)) 599 (tabulation-p offset step lines direction) 600 (tabulation-body-p offset step lines)))))) 601 602(defun maybe-table-offset (line-number lines) 603 "Return NIL or the indentation offset if LINE looks like it starts 604an item in a tabulation. Ie, if it is (1) indented, (2) preceded by an 605empty line, another tabulation label, or a tabulation body, (3) and 606followed another tabulation label or a tabulation body." 607 (let* ((line (svref lines line-number)) 608 (offset (indentation line)) 609 (prev (1- line-number)) 610 (next (1+ line-number))) 611 (when (and offset (plusp offset)) 612 (and (or (empty-p prev lines) 613 (tabulation-body-p offset prev lines) 614 (tabulation-p offset prev lines :backwards)) 615 (or (tabulation-body-p offset next lines) 616 (tabulation-p offset next lines :forwards)) 617 offset)))) 618 619;;; FIXME: This and itemization are very similar: could they share 620;;; some code, mayhap? 621 622(defun collect-maybe-table-section (lines starting-line) 623 ;; Return index of next line to be processed outside 624 (let ((this-offset (maybe-table-offset starting-line lines)) 625 (result nil) 626 (lines-consumed 0)) 627 (loop for line-number from starting-line below (length lines) 628 for line = (svref lines line-number) 629 for indentation = (indentation line) 630 for offset = (maybe-table-offset line-number lines) 631 do (cond 632 ((not indentation) 633 ;; empty line -- inserts paragraph. 634 (push "" result) 635 (incf lines-consumed)) 636 ((and offset (= indentation this-offset)) 637 ;; start of new item, or continuation of previous item 638 (if (and result (search "@item" (car result) :test #'char=)) 639 (push (format nil "@itemx ~A" (texinfo-line line)) 640 result) 641 (progn 642 (push "" result) 643 (push (format nil "@item ~A" (texinfo-line line)) 644 result))) 645 (incf lines-consumed)) 646 ((> indentation this-offset) 647 ;; continued item from previous line 648 (push (texinfo-line line) result) 649 (incf lines-consumed)) 650 (t 651 ;; end of itemization 652 (loop-finish)))) 653 ;; a single-line table isn't. 654 (if (> (count-if (lambda (line) (> (length line) 0)) result) 1) 655 (values lines-consumed 656 `("" "@table @emph" ,@(reverse result) "@end table" "")) 657 nil))) 658 659;;; section markup 660 661(defmacro with-maybe-section (index &rest forms) 662 `(multiple-value-bind (count collected) (progn ,@forms) 663 (when count 664 (dolist (line collected) 665 (write-line line *texinfo-output*)) 666 (incf ,index (1- count))))) 667 668(defun write-texinfo-string (string &optional lambda-list) 669 "Try to guess as much formatting for a raw docstring as possible." 670 (let ((*texinfo-variables* (flatten lambda-list)) 671 (lines (string-lines (escape-for-texinfo string nil)))) 672 (loop for line-number from 0 below (length lines) 673 for line = (svref lines line-number) 674 do (cond 675 ((with-maybe-section line-number 676 (and (lisp-section-p line line-number lines) 677 (collect-lisp-section lines line-number)))) 678 ((with-maybe-section line-number 679 (and (maybe-itemize-offset line) 680 (collect-maybe-itemized-section lines line-number)))) 681 ((with-maybe-section line-number 682 (and (maybe-table-offset line-number lines) 683 (collect-maybe-table-section lines line-number)))) 684 (t 685 (write-line (texinfo-line line) *texinfo-output*)))))) 686 687;;;; texinfo formatting tools 688 689(defun hide-superclass-p (class-name super-name) 690 (let ((super-package (symbol-package super-name))) 691 (or 692 ;; KLUDGE: We assume that we don't want to advertise internal 693 ;; classes in CP-lists, unless the symbol we're documenting is 694 ;; internal as well. 695 (and (member super-package #.'(mapcar #'find-package *undocumented-packages*)) 696 (not (eq super-package (symbol-package class-name)))) 697 ;; KLUDGE: We don't generally want to advertise SIMPLE-ERROR or 698 ;; SIMPLE-CONDITION in the CPLs of conditions that inherit them 699 ;; simply as a matter of convenience. The assumption here is that 700 ;; the inheritance is incidental unless the name of the condition 701 ;; begins with SIMPLE-. 702 (and (member super-name '(simple-error simple-condition)) 703 (let ((prefix "SIMPLE-")) 704 (mismatch prefix (string class-name) :end2 (length prefix))) 705 t ; don't return number from MISMATCH 706 )))) 707 708(defun hide-slot-p (symbol slot) 709 ;; FIXME: There is no pricipal reason to avoid the slot docs fo 710 ;; structures and conditions, but their DOCUMENTATION T doesn't 711 ;; currently work with them the way we'd like. 712 (not (and (typep (find-class symbol nil) 'standard-class) 713 (docstring slot t)))) 714 715(defun texinfo-anchor (doc) 716 (format *texinfo-output* "@anchor{~A}~%" (node-name doc))) 717 718;;; KLUDGE: &AUX *PRINT-PRETTY* here means "no linebreaks please" 719(defun texinfo-begin (doc &aux *print-pretty*) 720 (let ((kind (get-kind doc))) 721 (format *texinfo-output* "@~A {~:(~A~)} ~({~A}~@[ ~{~A~^ ~}~]~)~%" 722 (case kind 723 ((package constant variable) 724 "defvr") 725 ((structure class condition type) 726 "deftp") 727 (t 728 "deffn")) 729 (map 'string (lambda (char) (if (eql char #\-) #\Space char)) (string kind)) 730 (title-name doc) 731 ;; &foo would be amusingly bold in the pdf thanks to TeX/Texinfo 732 ;; interactions,so we escape the ampersand -- amusingly for TeX. 733 ;; sbcl.texinfo defines macros that expand @&key and friends to &key. 734 (mapcar (lambda (name) 735 (if (member name lambda-list-keywords) 736 (format nil "@~A" name) 737 name)) 738 (lambda-list doc))))) 739 740(defun texinfo-index (doc) 741 (let ((title (title-name doc))) 742 (case (get-kind doc) 743 ((structure type class condition) 744 (format *texinfo-output* "@tindex ~A~%" title)) 745 ((variable constant) 746 (format *texinfo-output* "@vindex ~A~%" title)) 747 ((compiler-macro function method-combination macro generic-function) 748 (format *texinfo-output* "@findex ~A~%" title))))) 749 750(defun texinfo-inferred-body (doc) 751 (when (member (get-kind doc) '(class structure condition)) 752 (let ((name (get-name doc))) 753 ;; class precedence list 754 (format *texinfo-output* "Class precedence list: @code{~(~{@lw{~A}~^, ~}~)}~%~%" 755 (remove-if (lambda (class) (hide-superclass-p name class)) 756 (mapcar #'class-name (ensure-class-precedence-list (find-class name))))) 757 ;; slots 758 (let ((slots (remove-if (lambda (slot) (hide-slot-p name slot)) 759 (class-direct-slots (find-class name))))) 760 (when slots 761 (format *texinfo-output* "Slots:~%@itemize~%") 762 (dolist (slot slots) 763 (format *texinfo-output* 764 "@item ~(@code{~A}~#[~:; --- ~]~ 765 ~:{~2*~@[~2:*~A~P: ~{@code{@w{~S}}~^, ~}~]~:^; ~}~)~%~%" 766 (slot-definition-name slot) 767 (remove 768 nil 769 (mapcar 770 (lambda (name things) 771 (if things 772 (list name (length things) things))) 773 '("initarg" "reader" "writer") 774 (list 775 (slot-definition-initargs slot) 776 (slot-definition-readers slot) 777 (slot-definition-writers slot))))) 778 ;; FIXME: Would be neater to handler as children 779 (write-texinfo-string (docstring slot t))) 780 (format *texinfo-output* "@end itemize~%~%")))))) 781 782(defun texinfo-body (doc) 783 (write-texinfo-string (get-string doc))) 784 785(defun texinfo-end (doc) 786 (write-line (case (get-kind doc) 787 ((package variable constant) "@end defvr") 788 ((structure type class condition) "@end deftp") 789 (t "@end deffn")) 790 *texinfo-output*)) 791 792(defun write-texinfo (doc) 793 "Writes TexInfo for a DOCUMENTATION instance to *TEXINFO-OUTPUT*." 794 (texinfo-anchor doc) 795 (texinfo-begin doc) 796 (texinfo-index doc) 797 (texinfo-inferred-body doc) 798 (texinfo-body doc) 799 (texinfo-end doc) 800 ;; FIXME: Children should be sorted one way or another 801 (mapc #'write-texinfo (get-children doc))) 802 803;;;; main logic 804 805(defun collect-gf-documentation (gf) 806 "Collects method documentation for the generic function GF" 807 (loop for method in (generic-function-methods gf) 808 for doc = (maybe-documentation method t) 809 when doc 810 collect doc)) 811 812(defun collect-name-documentation (name) 813 (loop for type in *documentation-types* 814 for doc = (maybe-documentation name type) 815 when doc 816 collect doc)) 817 818(defun collect-symbol-documentation (symbol) 819 "Collects all docs for a SYMBOL and (SETF SYMBOL), returns a list of 820the form DOC instances. See `*documentation-types*' for the possible 821values of doc-type." 822 (nconc (collect-name-documentation symbol) 823 (collect-name-documentation (list 'setf symbol)))) 824 825(defun collect-documentation (package &optional ht) 826 "Collects all documentation for all external symbols of the given 827package, as well as for the package itself." 828 (let* ((*documentation-package* (find-package package)) 829 (docs nil)) 830 (check-type package package) 831 (do-external-symbols (symbol package) 832 (unless (and ht 833 (nth-value 1 (alexandria:ensure-gethash symbol ht t))) 834 (setf (gethash symbol ht) t) 835 (setf docs (nconc (collect-symbol-documentation symbol) docs)))) 836 (let ((doc (maybe-documentation *documentation-package* t))) 837 (when doc 838 (push doc docs))) 839 docs)) 840 841(defmacro with-texinfo-file (pathname &body forms) 842 `(with-open-file (*texinfo-output* ,pathname 843 :direction :output 844 :if-does-not-exist :create 845 :if-exists :supersede) 846 ,@forms)) 847 848(defun write-ifnottex () 849 ;; We use @&key, etc to escape & from TeX in lambda lists -- so we need to 850 ;; define them for info as well. 851 ;; Texinfo > 5 doesn't allow "&" in macro names any more; 852 ;; see also https://bugs.launchpad.net/asdf/+bug/1172567 or 853 ;; ASDF commit dfa4643b212b194f2d673b6f0d9c7d4b19d823ba 854 (flet ((macro (name) 855 (let ((string (string-downcase name))) 856 (format *texinfo-output* "@macro ~A~%&~A~%@end macro~%" string string)))) 857 (macro 'allow-other-keys) 858 (macro 'optional) 859 (macro 'rest) 860 (macro 'key) 861 (macro 'body))) 862 863(defun generate-includes (directory packages &key (base-package :cl-user)) 864 "Create files in `directory' containing Texinfo markup of all 865docstrings of each exported symbol in `packages'. `directory' is 866created if necessary. If you supply a namestring that doesn't end in a 867slash, you lose. The generated files are of the form 868\"<doc-type>_<packagename>_<symbol-name>.texinfo\" and can be included 869via @include statements. Texinfo syntax-significant characters are 870escaped in symbol names, but if a docstring contains invalid Texinfo 871markup, you lose." 872 (handler-bind ((warning #'muffle-warning)) 873 (let* ((directory (merge-pathnames (pathname directory))) 874 (*base-package* (find-package base-package)) 875 (syms-seen (make-hash-table :test #'eq))) 876 (ensure-directories-exist directory) 877 (dolist (package packages) 878 (dolist (doc (collect-documentation (find-package package) syms-seen)) 879 (with-texinfo-file (merge-pathnames (include-pathname doc) directory) 880 (write-texinfo doc)))) 881 (with-texinfo-file (merge-pathnames "ifnottex.texinfo" directory) 882 (write-ifnottex)) 883 directory))) 884 885(defun document-package (package &optional filename) 886 "Create a file containing all available documentation for the 887exported symbols of `package' in Texinfo format. If `filename' is not 888supplied, a file \"<packagename>.texinfo\" is generated. 889 890The definitions can be referenced using Texinfo statements like 891@ref{<doc-type>_<packagename>_<symbol-name>.texinfo}. Texinfo 892syntax-significant characters are escaped in symbol names, but if a 893docstring contains invalid Texinfo markup, you lose." 894 (handler-bind ((warning #'muffle-warning)) 895 (let* ((package (find-package package)) 896 (filename (or filename (make-pathname 897 :name (string-downcase (short-package-name package)) 898 :type "texinfo"))) 899 (docs (sort (collect-documentation package) #'documentation<))) 900 (with-texinfo-file filename 901 (dolist (doc docs) 902 (write-texinfo doc))) 903 filename))) 904