ref: 3511285638220ee9837a51b88468df52da52156e
parent: bfe38a2454609a60bffe2f59340b075faeca7c4d
author: Sigrid Solveig Haflínudóttir <sigrid@ftrv.se>
date: Sun Mar 9 23:39:44 EDT 2025
add an initial implementation of defstruct References: https://todo.sr.ht/~ft/sl/48
--- a/boot/sl.boot
+++ b/boot/sl.boot
@@ -14,9 +14,12 @@
NIL NIL NIL NIL NIL NIL NIL NIL #fn("z0700}2:" #(aref)) NIL NIL NIL)
*properties* #table(*funvars* #table(*prompt* (NIL) lz-unpack ((data :to destination)
(data :size
- decompressed-bytes)) void? ((x)) >= ((a . rest)) help ((term)) length= ((lst
+ decompressed-bytes)) void? ((x)) >= ((a . rest)) defstruct ((name
+ doc options... (slot-1 DEFAULT) slot-2 (slot-3 :read-only))
+ (name (:type vector) (:named T) (:constructor NIL) (:conc-name NIL) (:predicate NIL) . slots)
+ (name doc options... (slot-1 DEFAULT) slot-2 (slot-3 :read-only))) help ((term)) length= ((lst
n)) __finish ((status)) doc-for ((term (doc NIL))) rand-u32 (NIL) = ((a . rest)) rand-u64 (NIL) car ((lst)) <= ((a . rest)) add-exit-hook ((fun)) /= ((a . rest)) lz-pack ((data
- (level 0))) rand (NIL) nan? ((x)) rand-float (NIL) void (rest) cons? ((value)) vm-stats (NIL) * ((number…)) rand-double (NIL) cdr ((lst)) + ((number…)) > ((a . rest))) *doc* #table(>= "Return T if the arguments are in non-increasing order (previous\none is greater than or equal to the next one)." void? "Return T if x is #<void>, NIL otherwise." length= "Bounded length test.\nUse this instead of (= (length lst) n), since it avoids unnecessary\nwork and always terminates." doc-for "Define documentation for a top level term.\nIf the optional doc argument is missing and the term is a function\nsignture, adds it to the documentation." car "Return the first element of a list or NIL if not available." *builtins* "VM instructions as closures." <= "Return T if the arguments are in non-decreasing order (previous\none is less than or equal to the next one)." void "Return the constant #<void> while ignoring any arguments.\n#<void> is mainly used when a function has side effects but does not\nproduce any meaningful value to return, so even though T or NIL could\nbe returned instead, in case of #<void> alone, REPL will not print\nit." rand "Return a random non-negative fixnum on its maximum range." nan? "Return T if the argument is NaN, regardless of the sign." Instructions "VM instructions mapped to their encoded byte representation." rand-double "Return a random double on [0.0, 1.0] interval." > "Return T if the arguments are in strictly decreasing order (previous\none is greater than the next one)." cdr "Return the tail of a list or NIL if not available." + "Return sum of the numbers or 0 with no arguments." __finish "A function called right before exit by the VM." lz-unpack "Return decompressed data previously compressed using lz-pack.\nEither destination for the decompressed data or the expected size of\nthe decompressed data must be specified. In the latter case a new\narray is allocated." help "Display documentation for the specified term, if available." rand-u32 "Return a random integer on [0, 2³²-1] interval." = "Return T if the arguments are equal." rand-u64 "Return a random integer on [0, 2⁶⁴-1] interval." add-exit-hook "Puts an one-argument function on top of the list of exit hooks.\nOn shutdown each exit hook is called with the exit status as a single\nargument, which is (usually) 0 on success and any other number on\nerror." /= "Return T if not all arguments are equal. Shorthand for (not (= …))." lz-pack "Return data compressed using Lempel-Ziv.\nThe data must be an array, returned value will have the same type.\nThe optional level is between 0 and 10. With level 0 a simple LZSS\nusing hashing will be performed. Levels between 1 and 9 offer a\ntrade-off between time/space and ratio. Level 10 is optimal but very\nslow." arg-counts "VM instructions mapped to their expected arguments count." rand-float "Return a random float on [0.0, 1.0] interval." *prompt* "Function called by REPL to signal the user input is required.\nDefault function prints \"#;> \"." cons? "Return T if the value is a cons cell." vm-stats "Print various VM-related information, such as the number of GC calls\nso far, heap and stack size, etc." * "Return product of the numbers or 1 with no arguments." *properties* "All properties of symbols recorded with putprop are recorded in this table."))
+ (level 0))) rand (NIL) nan? ((x)) rand-float (NIL) void (rest) cons? ((value)) vm-stats (NIL) rand-double (NIL) * ((number…)) cdr ((lst)) + ((number…)) > ((a . rest))) *doc* #table(>= "Return T if the arguments are in non-increasing order (previous\none is greater than or equal to the next one)." void? "Return T if x is #<void>, NIL otherwise." length= "Bounded length test.\nUse this instead of (= (length lst) n), since it avoids unnecessary\nwork and always terminates." doc-for "Define documentation for a top level term.\nIf the optional doc argument is missing and the term is a function\nsignture, adds it to the documentation." car "Return the first element of a list or NIL if not available." *builtins* "VM instructions as closures." <= "Return T if the arguments are in non-decreasing order (previous\none is less than or equal to the next one)." void "Return the constant #<void> while ignoring any arguments.\n#<void> is mainly used when a function has side effects but does not\nproduce any meaningful value to return, so even though T or NIL could\nbe returned instead, in case of #<void> alone, REPL will not print\nit." rand "Return a random non-negative fixnum on its maximum range." nan? "Return T if the argument is NaN, regardless of the sign." Instructions "VM instructions mapped to their encoded byte representation." rand-double "Return a random double on [0.0, 1.0] interval." > "Return T if the arguments are in strictly decreasing order (previous\none is greater than the next one)." cdr "Return the tail of a list or NIL if not available." + "Return sum of the numbers or 0 with no arguments." __finish "A function called right before exit by the VM." lz-unpack "Return decompressed data previously compressed using lz-pack.\nEither destination for the decompressed data or the expected size of\nthe decompressed data must be specified. In the latter case a new\narray is allocated." help "Display documentation for the specified term, if available." defstruct "Defines a structure type with a specific name and slots.\nThe default underlying type is a named vector, ie the first element is\nthe name of the structure type, the rest are the slot values.\n\nIf the name as the first element isn't required, \":named NIL\" should\nbe used.\n\nA list can be used instead of a vector, e.g.:\n (defstruct st :type list ...)\n\nThe option :conc-name specifies the slot accessor prefix, which\ndefaults to \"name-\".\n\nDefault predicate name (\"name?\") can be changed:\n (defstruct blah :predicate blargh? a b c)" rand-u32 "Return a random integer on [0, 2³²-1] interval." = "Return T if the arguments are equal." rand-u64 "Return a random integer on [0, 2⁶⁴-1] interval." add-exit-hook "Puts an one-argument function on top of the list of exit hooks.\nOn shutdown each exit hook is called with the exit status as a single\nargument, which is (usually) 0 on success and any other number on\nerror." /= "Return T if not all arguments are equal. Shorthand for (not (= …))." lz-pack "Return data compressed using Lempel-Ziv.\nThe data must be an array, returned value will have the same type.\nThe optional level is between 0 and 10. With level 0 a simple LZSS\nusing hashing will be performed. Levels between 1 and 9 offer a\ntrade-off between time/space and ratio. Level 10 is optimal but very\nslow." arg-counts "VM instructions mapped to their expected arguments count." rand-float "Return a random float on [0.0, 1.0] interval." *prompt* "Function called by REPL to signal the user input is required.\nDefault function prints \"#;> \"." cons? "Return T if the value is a cons cell." vm-stats "Print various VM-related information, such as the number of GC calls\nso far, heap and stack size, etc." * "Return product of the numbers or 1 with no arguments." *properties* "All properties of symbols recorded with putprop are recorded in this table."))
*syntax-environment* #table(bcode:nconst #fn("n1200r2e3:" #(aref)) doc-for #fn("\x8710002000W1000J60q?140B86;35040<;J404086;35040=863H020212287e212288e2e4e2:20212287e21e3e2:" #(void
symbol-set-doc quote)) with-input-from #fn("z12021e1220e2e1e12315163:" #(#fn(nconc)
with-bindings
@@ -34,7 +37,19 @@
#fn("n10B3500T:7060:" #(void)) letrec)) bcode:code #fn("n1200Ee3:" #(aref)) make-label #fn("n120e1:" #(gensym)) bcode:cenv #fn("n1200r3e3:" #(aref)) > #fn("z12021e12273151510e163:" #(#fn(nconc)
< #fn(copy-list) reverse!)) when #fn("z1200211Pqe4:" #(if begin)) quasiquote #fn("n1700E62:" #(bq-process)) help #fn("n17002152853W072855147350424250>170026q535247350@B0722728051524735047960:" #(getprop
*doc* princ newline #fn(for-each) #fn("n17050471A0P61:" #(newline print)) *funvars* "no help for "
- #fn(string) void)) bcode:ctable #fn("n1200Ke3:" #(aref)) with-output-to #fn("z12021e1220e2e1e12315163:" #(#fn(nconc)
+ #fn(string) void)) defstruct #fn("O10005000*///W1000J7071?14W2000J60D?24W3000J60q?34W4000J60q?44W5000J60q?54z6IIb;228;230>1_5142224?<5142586<51;360486<8=;360486=;J50486268>5127288>528<8>51292:12;525185;J>04292:02;525183;JB04292:2<052518Ae282;36040e184;J:042:02=522>502?2@e12A8C2Be22C8B2Be22D2E2F8Ee2e22G2H2BEe32F0e2e3e32I2J2Be2268E518?Me3e4e3e12K8=2L2F0e28=e3e3e12A8D2?1e12?2Fe12M8E5152e12M8@5153e3e12M7N2O8;8A8@8F8G8C8E0>88?525165:" #(#(NIL
+ NIL :named 1 :conc-name 3 :type 0 NIL NIL NIL NIL NIL NIL :predicate 4 NIL NIL NIL NIL NIL NIL
+ :constructor 2) vector #0# #fn("n17005121220A>28552485:" #(cddr #fn(for-each)
+ #fn("n17002152340q:722324A<25F2605661:" #(member
+ (:read-only) error #fn(string) "invalid option in slot " " of struct " ": "))) slot-opts)
+ #fn("n17021062:" #(map! #fn("n10B;35040<85;J404085;35040=;J604qe186RS;J9042086513=071228652@30q4232425268652512087<51390q87P@408762:" #(#fn(keyword?)
+ error "invalid slot name: " #fn(list*) #fn(symbol)
+ #fn(string) ":"))) tokw) #fn(string?) #fn(length)
+ #fn(map) #fn("n10B3500<:0:" #()) #fn(symbol) #fn(string) "?" "make-" "-" #fn(gensym)
+ #fn(nconc) begin def s and or not quote eq? aref = length when symbol-set-doc #fn(copy-list)
+ map-int #fn("n1A<70F052517092052212293865251238724252694e2e2e3279524e2e22829252694e2e32:242;96510Me37<2=85523O02>2?2@2686e22A2697e22Be6e2@B02C242;96510M25e4e4e4:" #(list-ref
+ #fn(symbol) #fn(string) def s v quote assert if eq? aref #fn(length) member :read-only error
+ string "slot " " in struct " " is :read-only" aset!)))) bcode:ctable #fn("n1200Ke3:" #(aref)) with-output-to #fn("z12021e1220e2e1e12315163:" #(#fn(nconc)
with-bindings *output-stream* #fn(copy-list))) catch #fn("n22012122e123242522e2262722e22829e2e3262:22e20e3e42;22e22<22e2e4e3e3:" #(trycatch
λ #:g347 if and cons? eq? car quote thrown-value cadr caddr raise)) let* #fn("z10H3E02021e1qe12215153e1:2021e173051e1e1220=B3H02024e10=e12215153e1@301515375051e2:" #(#fn(nconc)
λ #fn(copy-list) caar let* cadar)) letrec #fn("z1202021e12273052e122240522515154e1227605262:" #(#fn(nconc)
--- a/src/system.lsp
+++ b/src/system.lsp
@@ -845,6 +845,116 @@
(cdr strlist))
(iostream->string b))))
+;;; structs
+
+(defmacro (defstruct name (:type vector)
+ (:named T)
+ (:constructor NIL)
+ (:conc-name NIL)
+ (:predicate NIL)
+ . slots)
+ "Defines a structure type with a specific name and slots.
+The default underlying type is a named vector, ie the first element is
+the name of the structure type, the rest are the slot values.
+
+If the name as the first element isn't required, \":named NIL\" should
+be used.
+
+A list can be used instead of a vector, e.g.:
+ (defstruct st :type list ...)
+
+The option :conc-name specifies the slot accessor prefix, which
+defaults to \"name-\".
+
+Default predicate name (\"name?\") can be changed:
+ (defstruct blah :predicate blargh? a b c)"
+ (def (slot-opts slot)
+ ; check whether slot options, if any, are valid
+ (let ((opts (cddr slot)))
+ (for-each (λ (opt) (unless (member opt '(:read-only))
+ (error (string "invalid option in slot " (car slot)
+ " of struct " name
+ ": " opt))))
+ opts)
+ opts))
+ (def (tokw slots)
+ ; transform args list to keyworded variant.
+ ; eg: (a (b 1) (c :read-only)) → ((:a NIL) (:b 1) (:c NIL :read-only))
+ (map! (λ (slot) (let* {[name-cons (and (cons? slot)
+ (car slot))]
+ [name (or name-cons slot)]
+ [tail (or (and name-cons
+ (cdr slot))
+ (list nil))]}
+ (when (or (not (symbol? name))
+ (keyword? name))
+ (error "invalid slot name: " name))
+ (list* (symbol (string ":" name))
+ (if (keyword? (car tail))
+ (cons nil tail)
+ tail))))
+ slots))
+ (let* {; first element in slots may be the doc string
+ [doc (and (string? (car slots))
+ (car slots))]
+ ; if it is, rid of it
+ [slots (or (and doc (cdr slots))
+ slots)]
+ [num-slots (length slots)]
+ ; list of slot names
+ [slots-car (map (λ (f) (if (cons? f) (car f) f))
+ slots)]
+ ; slots, but with default values added (if not set)
+ ; and keywords for names
+ [slots-kw (tokw slots)]
+ ; struct's underlying type predicate (either vector? or list?)
+ [type? (symbol (string type "?"))]
+ ; struct's predicate name
+ [is? (or predicate
+ (symbol (string name "?")))]
+ ; constructor name and arguments
+ [constructor (or constructor
+ (list (symbol (string "make-" name)) slots-kw))]
+ ; should the struct name appear as the first element?
+ [named (and named (list name))]
+ ; accessor prefix
+ [access (or conc-name
+ (string name "-"))]
+ ; FIXME(sigrid): this is only until defset! becomes supported:
+ ; use a gensym have a notion of "undefined" default
+ ; value in accessors
+ [undefined (gensym)]}
+ `(begin
+ ; predicate
+ (def (,is? s)
+ (and [,type? s]
+ [or (not ',named) (eq? (aref s 0) ',name)]
+ [= (length s) ,(+ (length named) num-slots)]))
+ ; documentation string
+ (when ,doc
+ (symbol-set-doc ',name ,doc))
+ ; constructor
+ (def ,constructor
+ (,type ',@named ,@slots-car))
+ ; accessor per slot
+ ,@(map-int (λ (i) [let* {[opts (slot-opts (list-ref slots-kw i))]
+ [fld (list-ref slots-car i)]
+ [fun (symbol (string access fld))]}
+ `(def (,fun s (v ',undefined))
+ (assert (,is? s))
+ (if (eq? v ',undefined)
+ (aref s ,[+ (length named) i])
+ ,(if (member :read-only opts)
+ `(error (string "slot "
+ ',fld
+ " in struct "
+ ',name
+ " is :read-only"))
+ `(aset! s ,[+ (length named) i] v))))])
+ num-slots))))
+
+(doc-for (defstruct name doc options... (slot-1 DEFAULT) slot-2 (slot-3 :read-only)))
+
;;; toplevel
(def (macrocall? e) (and (symbol? (car e))