rule-syntax.scm 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. #| -*-Scheme-*-
  2. Copyright (C) 1986, 1987, 1988, 1989, 1990, 1991, 1992, 1993, 1994,
  3. 1995, 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005,
  4. 2006, 2007, 2008, 2009, 2010, 2011, 2012, 2013 Massachusetts
  5. Institute of Technology
  6. This file is part of MIT/GNU Scheme.
  7. MIT/GNU Scheme is free software; you can redistribute it and/or modify
  8. it under the terms of the GNU General Public License as published by
  9. the Free Software Foundation; either version 2 of the License, or (at
  10. your option) any later version.
  11. MIT/GNU Scheme is distributed in the hope that it will be useful, but
  12. WITHOUT ANY WARRANTY; without even the implied warranty of
  13. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  14. General Public License for more details.
  15. You should have received a copy of the GNU General Public License
  16. along with MIT/GNU Scheme; if not, write to the Free Software
  17. Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301,
  18. USA.
  19. |#
  20. ;;;; Compiler for rules.
  21. (declare (usual-integrations))
  22. ;;; Rule syntax.
  23. ;;; Rule does not apply if consequent expression returns #f.
  24. (define (rule:compile rule)
  25. (cond ((= (length rule) 3)
  26. (let* ((pattern-expression
  27. (pattern:compile (rule:pattern rule)))
  28. (predicate-expression
  29. (predicate:compile (rule:predicate rule)))
  30. (skeleton-expression
  31. (skel:compile (rule:skeleton rule))))
  32. (let ((vars (pattern:vars pattern-expression)))
  33. (let ((consequent-expression
  34. (if (eq? predicate-expression 'none)
  35. `(lambda ,vars ,skeleton-expression)
  36. `(lambda ,vars
  37. (let ((predicate-value ,predicate-expression))
  38. (and predicate-value
  39. ,skeleton-expression))))))
  40. `(rule:make ,pattern-expression
  41. ,consequent-expression)))))
  42. ((= (length rule) 2)
  43. (let* ((pattern-expression
  44. (pattern:compile (rule:pattern rule))))
  45. (let ((vars (pattern:vars pattern-expression)))
  46. (let ((consequent-expression
  47. `(lambda ,vars
  48. ,(rule:consequent rule))))
  49. `(rule:make ,pattern-expression
  50. ,consequent-expression)))))
  51. (else
  52. (error "Badly-formed rule" rule))))
  53. (define (rule:pattern rule)
  54. (car rule))
  55. (define (rule:predicate rule)
  56. (cadr rule))
  57. (define (rule:skeleton rule)
  58. (caddr rule))
  59. (define (rule:consequent rule)
  60. (cadr rule))
  61. (define (pattern:compile pattern)
  62. (define (compile pattern)
  63. (cond ((match:element? pattern)
  64. (if (match:restricted? pattern)
  65. (list '?
  66. (match:variable-name pattern)
  67. (list 'unquote (match:restriction pattern)))
  68. pattern))
  69. ((match:segment? pattern) pattern)
  70. ((match:reverse-segment? pattern) pattern)
  71. ((list? pattern) (map compile pattern))
  72. ((pair? pattern)
  73. (cons (compile (car pattern))
  74. (compile (cdr pattern))))
  75. (else pattern)))
  76. (list 'quasiquote (compile pattern)))
  77. (define (pattern:vars pattern)
  78. (let ((vars '()))
  79. (define (add-var! v)
  80. (or (memq v vars)
  81. (set! vars (cons v vars))))
  82. (define (compile pattern)
  83. (cond ((match:element? pattern)
  84. (add-var! (match:variable-name pattern)))
  85. ((match:segment? pattern)
  86. (add-var! (match:variable-name pattern)))
  87. ((match:reverse-segment? pattern)
  88. (add-var! (match:variable-name pattern)))
  89. ((list? pattern)
  90. (for-each compile pattern))
  91. ((pair? pattern)
  92. (compile (car pattern))
  93. (compile (cdr pattern)))
  94. (else 'nothing)))
  95. (compile pattern)
  96. vars))
  97. (define (match:element? pattern)
  98. (and (pair? pattern)
  99. (eq? (car pattern) '?)))
  100. (define (match:segment? pattern)
  101. (and (pair? pattern)
  102. (eq? (car pattern) '??)))
  103. (define (match:variable-name pattern)
  104. (cadr pattern))
  105. (define (match:restricted? pattern)
  106. (not (null? (cddr pattern))))
  107. (define (match:restriction pattern)
  108. (caddr pattern))
  109. (define (match:reverse-segment? pattern)
  110. (and (pair? pattern)
  111. (eq? (car pattern) '$$)))
  112. (define (predicate:compile predicate)
  113. predicate)
  114. (define none #t)
  115. (define (skel:compile skeleton)
  116. (define (compile skeleton)
  117. (cond ((skel:constant? skeleton) skeleton)
  118. ((skel:element? skeleton)
  119. (list 'unquote (skel:element-expression skeleton)))
  120. ((skel:segment? skeleton)
  121. (list 'unquote-splicing (skel:segment-expression skeleton)))
  122. ((list? skeleton)
  123. (map compile skeleton))
  124. ((pair? skeleton)
  125. (cons (compile (car skeleton))
  126. (compile (cdr skeleton))))
  127. (else
  128. (error "Unknown skeleton entry -- skel:compile"
  129. skeleton))))
  130. (list 'quasiquote (compile skeleton)))
  131. (define (skel:constant? skeleton)
  132. (not (pair? skeleton)))
  133. (define (skel:element? skeleton)
  134. (and (pair? skeleton)
  135. (eq? (car skeleton) ':)))
  136. (define (skel:element-expression skeleton)
  137. (cadr skeleton))
  138. (define (skel:segment? skeleton)
  139. (and (pair? skeleton)
  140. (eq? (car skeleton) '::)))
  141. (define (skel:segment-expression skeleton)
  142. (cadr skeleton))