record.scm 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  1. ; Copyright (c) 1993-2008 by Richard Kelsey. See file COPYING.
  2. ; Records that translate into C structs.
  3. ; Representation of record types.
  4. (define-record-type record-type :record-type
  5. (really-make-record-type name)
  6. record-type?
  7. (name record-type-name)
  8. ; FIELDS and CONSTRUCTOR-ARGS are filled in later because of circularity
  9. (fields record-type-fields set-record-type-fields!)
  10. (constructor-args ; fields passed to the constructor
  11. record-type-constructor-args set-record-type-constructor-args!))
  12. (define-record-discloser :record-type
  13. (lambda (rtype)
  14. (list 'record-type (record-type-name rtype))))
  15. ; Fields of record types.
  16. (define-record-type record-field :record-field
  17. (make-record-field record-type name type)
  18. record-field?
  19. (record-type record-field-record-type)
  20. (name record-field-name)
  21. (type record-field-type))
  22. ; Global table of record types. Since we compile to a single C file the
  23. ; record types used within a single computation must have distinct names.
  24. ; (This should really be a fluid.)
  25. (define *record-type-table* (make-symbol-table))
  26. (define (reset-record-data!)
  27. (set! *record-type-table* (make-symbol-table)))
  28. (define (get-record-type id)
  29. (cond ((table-ref *record-type-table* id)
  30. => identity)
  31. (else
  32. (error "no record type ~S" id))))
  33. (define (lookup-record-type id)
  34. (table-ref *record-type-table* id))
  35. (define (all-record-types)
  36. (table->entry-list *record-type-table*))
  37. ; Construction a record type. This gets the name, the list of fields whose
  38. ; initial values are passed to the constructor, and the field specifications.
  39. ; Each field specification consists of a name and a type.
  40. (define (make-record-type id constructor-args specs)
  41. (let ((rt (really-make-record-type id)))
  42. (if (table-ref *record-type-table* id)
  43. (user-error "multiple definitions of record type ~S" id))
  44. (table-set! *record-type-table* id rt)
  45. (set-record-type-fields! rt (map (lambda (spec)
  46. (make-record-field
  47. rt
  48. (car spec)
  49. (expand-type-spec (cadr spec))))
  50. specs))
  51. (set-record-type-constructor-args! rt
  52. (map (lambda (name)
  53. (get-record-type-field id name))
  54. constructor-args))
  55. rt))
  56. ; Return the field record for FIELD-ID in record-type TYPE-ID.
  57. (define (get-record-type-field type-id field-id)
  58. (let ((rtype (get-record-type type-id)))
  59. (cond ((any (lambda (field)
  60. (eq? field-id (record-field-name field)))
  61. (record-type-fields rtype))
  62. => identity)
  63. (else
  64. (user-error "~S is not a field of ~S" field-id rtype)))))
  65. ; The macro expander for DEFINE-RECORD-TYPE.
  66. ;
  67. ; (define-record-type <id> <type-id>
  68. ; (<constructor> . <field-names>)
  69. ; (<field-name> <type> <accessor-name> [<modifier-name>]) ...)
  70. ;
  71. ; The <type-id> is used only by Pre-Scheme-in-Scheme.
  72. (define (expand-define-record-type exp r c)
  73. (let ((id (cadr exp))
  74. (maker (cadddr exp))
  75. (fields (cddddr exp)))
  76. (let ((rt (make-record-type id (cdr maker) fields)))
  77. `(,(r 'begin)
  78. (,(r 'define) ,maker
  79. (,(r 'let) ((,(r id) (,(r 'make-record) ',id)))
  80. (,(r 'if) (,(r 'not) (,(r 'null-pointer?) ,(r id)))
  81. (,(r 'begin)
  82. . ,(map (lambda (name)
  83. `(,(r 'record-set!) ,(r id) ,name ',id ',name))
  84. (cdr maker))))
  85. ,(r id)))
  86. ,@(map (lambda (field)
  87. `(,(r 'define) (,(caddr field) ,(r id))
  88. (,(r 'record-ref) ,(r id) ',id ',(car field))))
  89. fields)
  90. ,@(map (lambda (field)
  91. `(,(r 'define) (,(cadddr field) ,(r id) ,(r 'x))
  92. (,(r 'record-set!)
  93. ,(r id) ,(r 'x)',id ',(car field))))
  94. (filter (lambda (spec)
  95. (not (null? (cdddr spec))))
  96. fields))))))
  97. ; primitives
  98. ; (make-record 'type . args)
  99. ; (record-ref thing 'type 'field)
  100. ; (record-set! thing value 'type 'field)
  101. ;
  102. ; C record creator
  103. ; global list of these things