Squashed commits
This commit is contained in:
commit
415cdc610b
9
.gitignore
vendored
Normal file
9
.gitignore
vendored
Normal file
|
@ -0,0 +1,9 @@
|
|||
*~
|
||||
\#*
|
||||
.\#*
|
||||
.DS_Store
|
||||
compiled/
|
||||
/*.css
|
||||
/*.html
|
||||
/*.js
|
||||
all-tags.log
|
42
.travis.yml
Normal file
42
.travis.yml
Normal file
|
@ -0,0 +1,42 @@
|
|||
language: c
|
||||
sudo: false
|
||||
|
||||
env:
|
||||
global:
|
||||
# RACKET_DIR is an argument to install-racket.sh
|
||||
- RACKET_DIR=~/racket
|
||||
- PATH="$RACKET_DIR/bin:$PATH"
|
||||
matrix:
|
||||
# RACKET_VERSION is an argument to install-racket.sh
|
||||
- RACKET_VERSION=6.6
|
||||
- RACKET_VERSION=6.7
|
||||
- RACKET_VERSION=6.8
|
||||
- RACKET_VERSION=RELEASE
|
||||
- RACKET_VERSION=HEAD
|
||||
|
||||
before_install:
|
||||
- curl -L https://raw.githubusercontent.com/greghendershott/travis-racket/master/install-racket.sh | bash
|
||||
- raco pkg install --deps search-auto doc-coverage cover cover-codecov # or cover-coveralls
|
||||
|
||||
install:
|
||||
- raco pkg install --deps search-auto -j 2 phc-adt/ phc-adt-doc/ phc-adt-lib/ phc-adt-test/
|
||||
|
||||
script:
|
||||
- raco test -x -p phc-adt phc-adt-doc phc-adt-lib phc-adt-test
|
||||
- raco setup --check-pkg-deps --no-zo --no-launcher --no-install --no-post-install --no-docs --pkgs phc-adt phc-adt-doc phc-adt-lib phc-adt-test
|
||||
- raco doc-coverage phc-adt
|
||||
- raco cover -s main -s test -s doc -f codecov -f html -d ~/coverage . || true
|
||||
# TODO: add an option to cover to run the "outer" module too, not just the submodules.
|
||||
# TODO: deploy the coverage info.
|
||||
#
|
||||
#
|
||||
#####################################################################################################
|
||||
- mv ~/.racket ~/.racket-local-install
|
||||
- echo
|
||||
- echo Installing from the main repository, to catch eventual dependency issues on the master branch.
|
||||
- echo
|
||||
- raco pkg install --deps search-auto -j 2 "$(basename "$TRAVIS_BUILD_DIR")"
|
||||
- raco test -x -p "$(basename "$TRAVIS_BUILD_DIR")"
|
||||
- raco setup --check-pkg-deps --no-zo --no-launcher --no-install --no-post-install --no-docs --pkgs "$(basename "$TRAVIS_BUILD_DIR")"
|
||||
- raco doc-coverage "$(basename "$TRAVIS_BUILD_DIR")"
|
||||
#####################################################################################################
|
28
LICENSE-more.md
Normal file
28
LICENSE-more.md
Normal file
|
@ -0,0 +1,28 @@
|
|||
phc-adt
|
||||
|
||||
Parts of this this software were initially written as part of a project
|
||||
at Cortus, S.A.S. which can be reached at 97 Rue de Freyr, 34000
|
||||
Montpellier, France. I got their permission to redistribute the code in
|
||||
the Public Domain.
|
||||
|
||||
|
||||
|
||||
This package is in distributed under the Creative Commons CC0 license
|
||||
https://creativecommons.org/publicdomain/zero/1.0/, as specified by
|
||||
the LICENSE.txt file.
|
||||
|
||||
|
||||
|
||||
The CC0 license is equivalent to a dedication to the Public Domain
|
||||
in most countries, but is also effective in countries which do not
|
||||
recognize explicit dedications to the Public Domain.
|
||||
|
||||
|
||||
|
||||
In order to avoid any potential licensing issues, this package is explicitly
|
||||
distributed under the Creative Commons CC0 license
|
||||
https://creativecommons.org/publicdomain/zero/1.0/, or under the GNU Lesser
|
||||
General Public License (LGPL) https://opensource.org/licenses/LGPL-3.0, or
|
||||
under the Apache License Version 2.0
|
||||
https://opensource.org/licenses/Apache-2.0, or under the MIT license
|
||||
https://opensource.org/licenses/MIT, at your option.
|
116
LICENSE.txt
Normal file
116
LICENSE.txt
Normal file
|
@ -0,0 +1,116 @@
|
|||
CC0 1.0 Universal
|
||||
|
||||
Statement of Purpose
|
||||
|
||||
The laws of most jurisdictions throughout the world automatically confer
|
||||
exclusive Copyright and Related Rights (defined below) upon the creator and
|
||||
subsequent owner(s) (each and all, an "owner") of an original work of
|
||||
authorship and/or a database (each, a "Work").
|
||||
|
||||
Certain owners wish to permanently relinquish those rights to a Work for the
|
||||
purpose of contributing to a commons of creative, cultural and scientific
|
||||
works ("Commons") that the public can reliably and without fear of later
|
||||
claims of infringement build upon, modify, incorporate in other works, reuse
|
||||
and redistribute as freely as possible in any form whatsoever and for any
|
||||
purposes, including without limitation commercial purposes. These owners may
|
||||
contribute to the Commons to promote the ideal of a free culture and the
|
||||
further production of creative, cultural and scientific works, or to gain
|
||||
reputation or greater distribution for their Work in part through the use and
|
||||
efforts of others.
|
||||
|
||||
For these and/or other purposes and motivations, and without any expectation
|
||||
of additional consideration or compensation, the person associating CC0 with a
|
||||
Work (the "Affirmer"), to the extent that he or she is an owner of Copyright
|
||||
and Related Rights in the Work, voluntarily elects to apply CC0 to the Work
|
||||
and publicly distribute the Work under its terms, with knowledge of his or her
|
||||
Copyright and Related Rights in the Work and the meaning and intended legal
|
||||
effect of CC0 on those rights.
|
||||
|
||||
1. Copyright and Related Rights. A Work made available under CC0 may be
|
||||
protected by copyright and related or neighboring rights ("Copyright and
|
||||
Related Rights"). Copyright and Related Rights include, but are not limited
|
||||
to, the following:
|
||||
|
||||
i. the right to reproduce, adapt, distribute, perform, display, communicate,
|
||||
and translate a Work;
|
||||
|
||||
ii. moral rights retained by the original author(s) and/or performer(s);
|
||||
|
||||
iii. publicity and privacy rights pertaining to a person's image or likeness
|
||||
depicted in a Work;
|
||||
|
||||
iv. rights protecting against unfair competition in regards to a Work,
|
||||
subject to the limitations in paragraph 4(a), below;
|
||||
|
||||
v. rights protecting the extraction, dissemination, use and reuse of data in
|
||||
a Work;
|
||||
|
||||
vi. database rights (such as those arising under Directive 96/9/EC of the
|
||||
European Parliament and of the Council of 11 March 1996 on the legal
|
||||
protection of databases, and under any national implementation thereof,
|
||||
including any amended or successor version of such directive); and
|
||||
|
||||
vii. other similar, equivalent or corresponding rights throughout the world
|
||||
based on applicable law or treaty, and any national implementations thereof.
|
||||
|
||||
2. Waiver. To the greatest extent permitted by, but not in contravention of,
|
||||
applicable law, Affirmer hereby overtly, fully, permanently, irrevocably and
|
||||
unconditionally waives, abandons, and surrenders all of Affirmer's Copyright
|
||||
and Related Rights and associated claims and causes of action, whether now
|
||||
known or unknown (including existing as well as future claims and causes of
|
||||
action), in the Work (i) in all territories worldwide, (ii) for the maximum
|
||||
duration provided by applicable law or treaty (including future time
|
||||
extensions), (iii) in any current or future medium and for any number of
|
||||
copies, and (iv) for any purpose whatsoever, including without limitation
|
||||
commercial, advertising or promotional purposes (the "Waiver"). Affirmer makes
|
||||
the Waiver for the benefit of each member of the public at large and to the
|
||||
detriment of Affirmer's heirs and successors, fully intending that such Waiver
|
||||
shall not be subject to revocation, rescission, cancellation, termination, or
|
||||
any other legal or equitable action to disrupt the quiet enjoyment of the Work
|
||||
by the public as contemplated by Affirmer's express Statement of Purpose.
|
||||
|
||||
3. Public License Fallback. Should any part of the Waiver for any reason be
|
||||
judged legally invalid or ineffective under applicable law, then the Waiver
|
||||
shall be preserved to the maximum extent permitted taking into account
|
||||
Affirmer's express Statement of Purpose. In addition, to the extent the Waiver
|
||||
is so judged Affirmer hereby grants to each affected person a royalty-free,
|
||||
non transferable, non sublicensable, non exclusive, irrevocable and
|
||||
unconditional license to exercise Affirmer's Copyright and Related Rights in
|
||||
the Work (i) in all territories worldwide, (ii) for the maximum duration
|
||||
provided by applicable law or treaty (including future time extensions), (iii)
|
||||
in any current or future medium and for any number of copies, and (iv) for any
|
||||
purpose whatsoever, including without limitation commercial, advertising or
|
||||
promotional purposes (the "License"). The License shall be deemed effective as
|
||||
of the date CC0 was applied by Affirmer to the Work. Should any part of the
|
||||
License for any reason be judged legally invalid or ineffective under
|
||||
applicable law, such partial invalidity or ineffectiveness shall not
|
||||
invalidate the remainder of the License, and in such case Affirmer hereby
|
||||
affirms that he or she will not (i) exercise any of his or her remaining
|
||||
Copyright and Related Rights in the Work or (ii) assert any associated claims
|
||||
and causes of action with respect to the Work, in either case contrary to
|
||||
Affirmer's express Statement of Purpose.
|
||||
|
||||
4. Limitations and Disclaimers.
|
||||
|
||||
a. No trademark or patent rights held by Affirmer are waived, abandoned,
|
||||
surrendered, licensed or otherwise affected by this document.
|
||||
|
||||
b. Affirmer offers the Work as-is and makes no representations or warranties
|
||||
of any kind concerning the Work, express, implied, statutory or otherwise,
|
||||
including without limitation warranties of title, merchantability, fitness
|
||||
for a particular purpose, non infringement, or the absence of latent or
|
||||
other defects, accuracy, or the present or absence of errors, whether or not
|
||||
discoverable, all to the greatest extent permissible under applicable law.
|
||||
|
||||
c. Affirmer disclaims responsibility for clearing rights of other persons
|
||||
that may apply to the Work or any use thereof, including without limitation
|
||||
any person's Copyright and Related Rights in the Work. Further, Affirmer
|
||||
disclaims responsibility for obtaining any necessary consents, permissions
|
||||
or other rights required for any use of the Work.
|
||||
|
||||
d. Affirmer understands and acknowledges that Creative Commons is not a
|
||||
party to this document and has no duty or obligation with respect to this
|
||||
CC0 or use of the Work.
|
||||
|
||||
For more information, please see
|
||||
<http://creativecommons.org/publicdomain/zero/1.0/>
|
137
README.md
Normal file
137
README.md
Normal file
|
@ -0,0 +1,137 @@
|
|||
[](https://travis-ci.org/jsmaniac/phc-adt)
|
||||
[](https://codecov.io/gh/jsmaniac/phc-adt)
|
||||
[](http://jsmaniac.github.io/travis-stats/#jsmaniac/phc-adt)
|
||||
[](http://docs.racket-lang.org/phc-adt/)
|
||||
[](https://github.com/jsmaniac/phc-adt/issues)
|
||||
[](https://creativecommons.org/publicdomain/zero/1.0/)
|
||||
|
||||
phc-adt
|
||||
=======
|
||||
|
||||
This library provides Algebraic Datatypes (structures and variants), with the
|
||||
features described below. It is designed to work hand-in-hand with our graph
|
||||
library (https://github.com/jsmaniac/phc), but can also be used on its own.
|
||||
|
||||
Structures
|
||||
----------
|
||||
|
||||
We define structures as an extension of Racket's syntax. Structures are
|
||||
anonymous products of values, where each value is annotated by a field
|
||||
name. The syntax used for constructing an instance of a structure is:
|
||||
|
||||
(structure [field₁ : value₁] … [fieldₙ : valueₙ])
|
||||
|
||||
The type of a structure is described using a similar syntax:
|
||||
|
||||
(structure [field₁ : type₁] … [fieldₙ : typeₙ])
|
||||
|
||||
Accessing the field `f` of an instance `i` is done using the dot operator:
|
||||
|
||||
i.f
|
||||
|
||||
The main difference with Racket's built-in `struct` is that the type does not
|
||||
need to be pre-declared to create instances of a structure, making them
|
||||
effectively anonymous. Racket also requires knowing the name of the struct
|
||||
type in order to access a field of an instance, using `(s-f i)` where `s` is
|
||||
the struct's name, `f` is the field name and `i` is the instance. When very
|
||||
similar structures are used, specifying the structure's name each time becomes
|
||||
cumbersome.
|
||||
|
||||
Unions
|
||||
------
|
||||
|
||||
Typed/Racket has built-in untagged union types, which can be constructed using
|
||||
the following notation:
|
||||
|
||||
(U type₁ … typeₙ)
|
||||
|
||||
These cannot be used as-is inside node types for our graph library
|
||||
(https://github.com/jsmaniac/phc), because there is no way at run-time to
|
||||
distinguish between the various cases of the union in the general case. For
|
||||
example, `(U (→ Integer) (→ String))` denotes the type of an union of thunks,
|
||||
one returning an `Integer` and the other a `String`, and it is not
|
||||
possible to know to which case an instance belongs without calling the thunks,
|
||||
which could have undesired side-effects.
|
||||
|
||||
The graph library's implementation relies on being able to rewrite arbitrary
|
||||
data structures, changing the type of some parts in the process. We therefore
|
||||
allow only a limited form of unions, namely those where all cases but one are
|
||||
pairs whose first element is a symbol. The first element of the pair can then
|
||||
be used to distinguish between the cases. The remaining case is used as a
|
||||
fall-back when the symbol comparison for all others have failed, and should
|
||||
not overlap with the other cases.
|
||||
|
||||
Variants
|
||||
--------
|
||||
|
||||
We define variants (tagged unions) as an alternative to Racket's untagged
|
||||
unions. In our implementation, variants are anonymous unions of
|
||||
constructors. Each constructor is a tuple of values annotated with a
|
||||
tag. Variants are usually not anonymous in other languages: the same
|
||||
constructor cannot belong to two different variants. In our implementation,
|
||||
this is not the case: two different variants can contain the same exact
|
||||
constructor. The reason for this feature is that the input and output types of
|
||||
graph transformations are very similar, and are likely to contain
|
||||
nearly-identical variants. Name collisions would be frequent if a constructor
|
||||
name was bound to a single variant. We use the following syntax to denote a
|
||||
variant type:
|
||||
|
||||
(V constructor₁ … constructorₙ)
|
||||
|
||||
Constructors are the product of a tag name and a payload, which is itself a
|
||||
product of zero or more types. Constructors types are declared as follows:
|
||||
|
||||
(constructor tag payload₁ … payloadₙ)
|
||||
|
||||
Instances of a constructor can be created using the following syntax:
|
||||
|
||||
(constructor tag v₁ … vₙ)
|
||||
|
||||
All constructors within a variant should have a distinct tag, but the same tag
|
||||
name can be used in the several variants, with an identical or different
|
||||
payload.
|
||||
|
||||
To improve encapsulation, we further define *private* constructors. These have
|
||||
a unique tag which is uncomparable to all other tags, even those using the
|
||||
same string of characters for the name. We also define *uninterned*
|
||||
constructors, which are strict subtypes of the regular constructor with the
|
||||
same name (i.e. using the same string of characters for the name). Another way
|
||||
to see these is that a *private* constructor protects both its creation
|
||||
mechanism and its matching mechanism, whereas the *uninterned* constructors
|
||||
protect only the creation mechanism, but no special permission is needed to
|
||||
match on an *uninterned* constructor.
|
||||
|
||||
When using Racket's `match` form, an instance of a variant only matches with
|
||||
its tag (or an uninterned subtype) and the same number of fields.
|
||||
|
||||
---
|
||||
|
||||
Nodes
|
||||
-----
|
||||
|
||||
Below follows a short description of graph nodes, which are provided by a
|
||||
separate library (https://github.com/jsmaniac/phc-graph, code is currently
|
||||
being refactored and migrated from https://github.com/jsmaniac/phc).
|
||||
|
||||
Graph nodes behave like tagged structures in most aspects. They are similar to
|
||||
a constructor containing a structure as its single field. An *incomplete* node
|
||||
can only be created within a *mapping* of the same graph, using a notation
|
||||
similar to the one used for structures:
|
||||
|
||||
(node-name [field₁ : value₁] … [fieldₙ : valueₙ])
|
||||
|
||||
The type of a node can be expressed similarly:
|
||||
|
||||
(node-name [field₁ : type₁] [fieldₙ : typeₙ])
|
||||
|
||||
It is not possible to access an *incomplete* node's fields. However, once the
|
||||
graph is fully constructed, the field `f` of the *promise* node `p` can be
|
||||
accessed using the same dot operator:
|
||||
|
||||
p.f
|
||||
|
||||
On the other hand, it is not possible to directly create a promise node
|
||||
without using a graph constructor.
|
||||
|
||||
**TODO:** nodes are actually rather similar to uninterned constructors
|
||||
containing a single value, which is a structure.
|
25
phc-adt-doc/info.rkt
Normal file
25
phc-adt-doc/info.rkt
Normal file
|
@ -0,0 +1,25 @@
|
|||
#lang info
|
||||
(define collection 'multi)
|
||||
(define deps '("base"))
|
||||
(define build-deps '("scribble-lib"
|
||||
"hyper-literate"
|
||||
"phc-adt-lib"
|
||||
"racket-doc"
|
||||
"typed-racket-doc"
|
||||
"typed-racket-lib"
|
||||
"scribble-enhanced"
|
||||
"scribble-math"
|
||||
"type-expander"
|
||||
"xlist"
|
||||
"alexis-util"
|
||||
"extensible-parser-specifications"
|
||||
"multi-id"
|
||||
"phc-toolkit"
|
||||
"remember"
|
||||
"threading"
|
||||
"trivial"
|
||||
"typed-struct-props"
|
||||
"datatype"))
|
||||
(define pkg-desc "Algebraic Datatypes tailored for writing compilers (documentation only)")
|
||||
(define version "1.1")
|
||||
(define pkg-authors '("Georges Dupéron"))
|
1
phc-adt-doc/phc-adt/.gitignore
vendored
Normal file
1
phc-adt-doc/phc-adt/.gitignore
vendored
Normal file
|
@ -0,0 +1 @@
|
|||
/doc/
|
4
phc-adt-doc/phc-adt/info.rkt
Normal file
4
phc-adt-doc/phc-adt/info.rkt
Normal file
|
@ -0,0 +1,4 @@
|
|||
#lang info
|
||||
(define scribblings
|
||||
'(("scribblings/phc-adt.scrbl" (multi-page) ("Data Structures"))
|
||||
("scribblings/phc-adt-implementation.scrbl" (multi-page) ("Data Structures"))))
|
124
phc-adt-doc/phc-adt/scribblings/phc-adt-choices.scrbl
Normal file
124
phc-adt-doc/phc-adt/scribblings/phc-adt-choices.scrbl
Normal file
|
@ -0,0 +1,124 @@
|
|||
#lang hyper-literate typed/racket
|
||||
@(require scribble-enhanced/doc)
|
||||
@doc-lib-setup
|
||||
|
||||
@title[#:style manual-doc-style
|
||||
#:tag "choices"
|
||||
#:tag-prefix "phc-adt/choices"]{Somewhat outdated
|
||||
overview of the implementation choices for structures, graphs and passes}
|
||||
|
||||
@(chunks-toc-prefix
|
||||
'("(lib phc-adt/scribblings/phc-adt-implementation.scrbl)"
|
||||
"phc-adt/choices"))
|
||||
|
||||
@section[#:tag "type-system|structures"]{Structures}
|
||||
|
||||
Structures are represented as lists of key/value pairs.
|
||||
@note{We need lists and can't use vectors (or hash tables) because the latter
|
||||
are mutable in @code|{typed/racket}|, and the typing system has no guarantee
|
||||
that accessing the same element twice will yield the same value (so occurence
|
||||
typing can't narrow the type in branches of conditionnals).}
|
||||
@note{Actually, we can use structs (they are immutable by default, and the
|
||||
occurrence typing knows that). There are two problems with them. The first
|
||||
problem is that we cannot have subtyping (although knowing all the structs
|
||||
used in the program means we can just filter them and use a
|
||||
@racket[(U S1 S2 …)]. The second problem is that in order to declare the
|
||||
structs, we would need to be in a define-like environment (therefore making
|
||||
anonymous structure types problematic). The second problem can be solved in
|
||||
the same way as the first: if all the structs are known in advance, we can
|
||||
pre-declare them in a shared file.}
|
||||
|
||||
@chunk[<example-simple-structure>
|
||||
(define-type abc (List (Pairof 'a Number)
|
||||
(Pairof 'b String)
|
||||
(Pairof 'c (U 'x 'y))))
|
||||
|
||||
(: make-abc (→ Number String (U 'x 'y) abc))
|
||||
(define (make-abc a b c)
|
||||
(list (cons 'a a) (cons 'b b) (cons 'c c)))
|
||||
(make-abc 1 "b" 'x)]
|
||||
|
||||
Occurrence typing works:
|
||||
|
||||
@chunk[<example-simple-structure-occurrence>
|
||||
(: f (→ abc (U 'x #f)))
|
||||
(define (f v)
|
||||
(if (eq? 'x (cdr (cddr v)))
|
||||
(cdr (cddr v))
|
||||
#f))]
|
||||
|
||||
@section{Passes, subtyping and tests}
|
||||
|
||||
Below is the definition of a function which works on
|
||||
@tc[(structure [a Number] [b String] [c Boolean])], and returns the same
|
||||
structure extended with a field @tc[[d Number]], but is only concerned with
|
||||
fields @tc[a] and @tc[c], so tests don't need to provide a value for @tc[b].
|
||||
|
||||
@chunk[<example-pass-which-extends-input>
|
||||
(: pass-calc-d (∀ (TB) (→ (List (Pairof 'a Number)
|
||||
(Pairof 'b TB)
|
||||
(Pairof 'c Boolean))
|
||||
(List (Pairof 'a Number)
|
||||
(Pairof 'b TB)
|
||||
(Pairof 'c Boolean)
|
||||
(Pairof 'd Number)))))
|
||||
(define (pass-calc-d v)
|
||||
(list (car v) ; a
|
||||
(cadr v) ; b
|
||||
(caddr v) ; c
|
||||
(cons 'd (+ (cdar v) (if (cdaddr v) 0 1)))))]
|
||||
|
||||
The example above can be called to test it with a dummy value for @tc[b]:
|
||||
|
||||
@chunk[<example-pass-which-extends-input>
|
||||
(pass-calc-d '((a . 1) (b . no-field) (c . #t)))]
|
||||
|
||||
But when called with a proper value for @tc[b], we get back the original
|
||||
string as expected, and the type is correct:
|
||||
|
||||
@chunk[<example-pass-which-extends-input>
|
||||
(ann (pass-calc-d '((a . 1) (b . "some string") (c . #t)))
|
||||
(List (Pairof 'a Number)
|
||||
(Pairof 'b String)
|
||||
(Pairof 'c Boolean)
|
||||
(Pairof 'd Number)))]
|
||||
|
||||
If the pass should be able to work on multiple graph types (each being a subtype
|
||||
containing a common subset of fields), then it should be easy to mark it as a
|
||||
@tc[case→] function. It is probably better to avoid too permissive subtyping,
|
||||
otherwise, imagine we have
|
||||
a pass which removes @tc[Addition]s and @tc[Substraction]s from an AST, and
|
||||
replaces them with a single @tc[Arithmetic] node type. If we have full duck
|
||||
typing, we could call it with @tc[Addition]s and @tc[Substraction] hidden in
|
||||
fields it does not know about, and so it would fail to replace them. Also, it
|
||||
could be called with an already-processed AST which already contains just
|
||||
@tc[Arithmetic] node types, which would be a bug most likely. Therefore,
|
||||
explicitly specifying the graph type on which the passes work seems a good
|
||||
practice. Some parts can be collapsed easily into a @tc[∀] type @tc[T], when
|
||||
we're sure there shouldn't be anything that interests us there.
|
||||
|
||||
@section{Graphs}
|
||||
|
||||
In order to be able to have cycles, while preserving the benefits of
|
||||
occurrence typing, we need to make sure that from the type system's point of
|
||||
view, accessing a successor node twice will return the same value each time.
|
||||
|
||||
The easiest way for achieving this is to wrap the to-be-created value inside a
|
||||
@tc[Promise]. Occurrence typing works on those:
|
||||
|
||||
@chunk[<test-promise-occurence-typing>
|
||||
(: test-promise-occurence (→ (Promise (U 'a 'b)) (U 'a #f)))
|
||||
(define (test-promise-occurence p)
|
||||
(if (eq? (force p) 'a)
|
||||
(force p)
|
||||
#f))]
|
||||
|
||||
@section{Conclusion}
|
||||
|
||||
@chunk[<*>
|
||||
<example-simple-structure>
|
||||
<example-simple-structure-occurrence>
|
||||
|
||||
<example-pass-which-extends-input>
|
||||
|
||||
<test-promise-occurence-typing>]
|
311
phc-adt-doc/phc-adt/scribblings/phc-adt-constructor.scrbl
Normal file
311
phc-adt-doc/phc-adt/scribblings/phc-adt-constructor.scrbl
Normal file
|
@ -0,0 +1,311 @@
|
|||
#lang scribble/manual
|
||||
|
||||
@(require racket/require
|
||||
(for-label (except-in (subtract-in typed/racket/base type-expander)
|
||||
values)
|
||||
type-expander
|
||||
phc-adt
|
||||
xlist
|
||||
(lib "phc-adt/tagged.hl.rkt")
|
||||
(lib "phc-adt/structure.hl.rkt")
|
||||
(lib "phc-adt/constructor.hl.rkt")
|
||||
(lib "phc-adt/variant.hl.rkt")
|
||||
(lib "phc-adt/tagged-supertype.hl.rkt"))
|
||||
scribble-enhanced/doc
|
||||
scribble-math
|
||||
(subtract-in scribble/struct scribble-enhanced/doc)
|
||||
scribble/decode)
|
||||
@doc-lib-setup
|
||||
|
||||
@title{Constructors}
|
||||
|
||||
@deftech{Constructors} are implemented as tagged structures, using a
|
||||
single special field: @racket[values]. The @racket[constructor] identifier and
|
||||
its derivatives therefore call @racket[tagged], using that single field. The
|
||||
identifiers described within this section provide some syntactic sugar,
|
||||
allowing constructors to contain more than one value. These values are wrapped
|
||||
in a (possibly improper) list which is stored within the tagged structure's
|
||||
@racket[values] field.
|
||||
|
||||
|
||||
@defform*[#:kind "type-expander"
|
||||
[(constructor tag-name maybe-∀ τᵢ ...)
|
||||
(constructor tag-name maybe-∀ τᵢ ... . dotted-τ-rest)
|
||||
(constructor tag-name maybe-∀ τᵢ ... #:rest τ-rest)]
|
||||
#:grammar
|
||||
[(tag-name Identifier)
|
||||
(maybe-∀ (code:line)
|
||||
(code:line #:∀ (tvarⱼ ...)))
|
||||
(τᵢ xlist-type-or-repeated-type)
|
||||
(τ-rest xlist-type-or-repeated-type)
|
||||
(dotted-τ-rest #,"like τ-rest, but must not be a syntax pair")]]{
|
||||
|
||||
Expands to the type for a constructor with the given tag name and type of
|
||||
contents. The @racket[(τᵢ ...)], @racket[(τᵢ ... . dotted-τ-rest)] or
|
||||
@racket[(τᵢ ... #:rest τ-rest)] sequence is passed unmodified to
|
||||
@racket[xlist]. Therefore, depending on the syntax used, the expanded type is
|
||||
equivalent to one of the following types:
|
||||
|
||||
@racketblock[
|
||||
(tagged tag-name maybe-∀ [values (xlist τᵢ ...)])
|
||||
(tagged tag-name maybe-∀ [values (xlist τᵢ ... . dotted-τ-rest)])
|
||||
(tagged tag-name maybe-∀ [values (xlist τᵢ ... #:rest τ-rest)])]
|
||||
|
||||
The elements may appear in any order, as long as the tag name appears before
|
||||
any element type, and as long as the element types form a contiguous
|
||||
sequence.}
|
||||
|
||||
@defform*[#:kind "syntax"
|
||||
#:link-target? #f
|
||||
#:literals (* : ! ::)
|
||||
[(constructor maybe-∀ tag-name *)
|
||||
(constructor maybe-∀ tag-name : typeᵢ ...)
|
||||
(constructor maybe-∀ tag-name ! . xlist-types)
|
||||
(constructor maybe-∀ tag-name :: . xlist-types)]
|
||||
#:grammar
|
||||
[(maybe-∀ (code:line)
|
||||
(code:line #:∀ (tvarⱼ ...)))
|
||||
(tag-name Identifier)
|
||||
(xlist-types (τᵢ ...)
|
||||
(τᵢ ... . dotted-τ-rest)
|
||||
(τᵢ ... #:rest τ-rest))
|
||||
(τᵢ xlist-type-or-repeated-type)
|
||||
(typeᵢ Type)]]{
|
||||
Expands to a builder function for a constructor with the given tag name and
|
||||
type of contents.
|
||||
|
||||
The first syntax, using @racket[*] and no types, produces a polymorphic
|
||||
builder function which accepts any number of arguments, infers their types,
|
||||
and uses the whole list of arguments as the constructor's value.
|
||||
|
||||
In the following three cases, when @racket[#:∀ (tvarⱼ ...)] is specified, a
|
||||
polymorphic builder with the @racket[tvarⱼ] type variables is produced.
|
||||
|
||||
The second syntax, using @racket[:] followed by a sequence of regular types,
|
||||
produces a builder function with one argument per type. The builder function
|
||||
aggregates all its arguments in a list, and uses that list as the
|
||||
constructor's value.
|
||||
|
||||
The second syntax, using @racket[!] followed by a sequence of types valid for
|
||||
@racket[xlist], produces a builder function which accepts a variable number of
|
||||
arguments. The builder function @racket[cast]s the whole list of arguments to
|
||||
the type @racket[(xlist . xlist-types)], which must therefore be a suitable
|
||||
argument to @racket[make-predicate]. The cast list is used as the
|
||||
constructor's value.
|
||||
|
||||
The third syntax, using @racket[::] followed by a sequence of types valid for
|
||||
@racket[xlist], produces a builder function which accepts a single value of
|
||||
type @racket[(xlist . xlist-typed)], and uses that value as the constructor's
|
||||
value.
|
||||
|
||||
Usually, the value stored within a constructor will be a list (i.e. a tuple
|
||||
in other languages), but it is possible to store a single value using
|
||||
@racket[xlist]'s rest syntax:
|
||||
|
||||
@racketblock[
|
||||
((constructor #:∀ (A) tag-name :: . A) 123)
|
||||
((constructor tag-name :: . Number) 123)
|
||||
((constructor tag-name :: #:rest (Vector Number String)) #(123 "abc"))]
|
||||
|
||||
The elements may appear in any order, as long as the tag name appears before
|
||||
any element type, and as long as the element types form a contiguous
|
||||
sequence.}
|
||||
|
||||
@defform*[#:kind "syntax"
|
||||
#:link-target? #f
|
||||
#:literals (:)
|
||||
[(constructor maybe-∀ tag-name value-maybe-typeᵢ)
|
||||
(constructor maybe-∀ tag-name value-maybe-typeᵢ . dotted-rest)
|
||||
(constructor maybe-∀ tag-name value-maybe-typeᵢ #:rest rest)]
|
||||
#:grammar
|
||||
[(maybe-∀ (code:line)
|
||||
(code:line #:∀ (tvarⱼ ...)))
|
||||
(tag-name Identifier)
|
||||
(value-maybe-typeᵢ valueᵢ
|
||||
[valueᵢ : typeᵢ]
|
||||
[: typeᵢ valueᵢ])
|
||||
(rest value-maybe-typeᵢ)
|
||||
(dotted-rest #,"like rest, but must not be a syntax pair")]]{
|
||||
|
||||
Expands to an instance of a constructor containing the given values, grouped
|
||||
inside a list.
|
||||
|
||||
When a @racket[typeᵢ] is specified, it is used to annotate the value, and is
|
||||
used as the type for that element in the resulting constructor type.
|
||||
|
||||
When @racket[#:∀ (tvarⱼ ...)] is specified, the type of values annotated with
|
||||
@racket[tvarⱼ] is inferred, and an instance of a polymorphic constructor is
|
||||
produced. A @racket[tvarⱼ] can be used within a more complex type, in which
|
||||
case only that part of the type is inferred.
|
||||
|
||||
The elements may appear in any order, as long as the tag name appears before
|
||||
any value, and as long as the values form a contiguous sequence, including the
|
||||
@racket[#:rest rest] which must appear immediately after the sequence of
|
||||
values, if specified. The @racket[dotted-rest], on the other hand, can be
|
||||
separated from the other values, so
|
||||
@racket[(constructor foo 1 [2 : A] 3 #:∀ (A) . 4)] is a valid (but awkward)
|
||||
use of @racket[constructor].
|
||||
|
||||
The type of the @racket[dotted-rest] can still be specified using
|
||||
@racket[typed/racket]'s reader abbreviation for @racket[ann], namely
|
||||
@racket[#{dotted-rest :: type}].}
|
||||
|
||||
@defform*[#:kind "match expander"
|
||||
#:link-target? #f
|
||||
#:literals (* : ! ::)
|
||||
[(constructor tag-name . xlist-pats)]
|
||||
#:grammar
|
||||
[(tag-name Identifier)
|
||||
(xlist-pats (patᵢ ...)
|
||||
(patᵢ ... . dotted-pat-rest)
|
||||
(patᵢ ... #:rest pat-rest))
|
||||
(patᵢ XList-Match-Pattern)]]{
|
||||
|
||||
Expands to a match pattern which checks whether the value is a constructor
|
||||
with the given tag name, and then matches the constructor's value against the
|
||||
match pattern @racket[(xlist . xlist-pats)]. The @racket[xlist] match expander
|
||||
in turn matches each element of a (possibly improper) list against the given
|
||||
patterns, and supports various means of specifying fixed-length, bounded and
|
||||
unbounded repetitions like "must appear between three and five times". See the
|
||||
documentation for the @racket[xlist] match expander for more details.}
|
||||
|
||||
@defform[#:kind "syntax"
|
||||
#:literals (* : ! ::)
|
||||
(constructor? tag-name . xlist-types)
|
||||
#:grammar
|
||||
[(tag-name Identifier)
|
||||
(xlist-types (τᵢ ...)
|
||||
(τᵢ ... . dotted-τ-rest)
|
||||
(τᵢ ... #:rest τ-rest))
|
||||
(τᵢ xlist-type-or-repeated-type)
|
||||
(τ-rest xlist-type-or-repeated-type)
|
||||
(dotted-τ-rest #,"like τ-rest, but must not be a syntax pair")]]{
|
||||
Expands to a predicate which returns true if and only if the following
|
||||
conditions are met:
|
||||
@itemlist[
|
||||
@item{The value is a constructor with the given tag name (i.e. a tagged
|
||||
structure with the given tag name and a single field named @racket[values],
|
||||
so nodes and untagged structures with a single field named @racket[values]
|
||||
are accepted too)}
|
||||
@item{The constructor's value (i.e. the contents of its @racket[values]
|
||||
field) is accepted by @racket[(make-predicate (xList . xlist-types))]}]}
|
||||
|
||||
@defform[(define-constructor name maybe-tag maybe-pred? maybe-∀ . type-spec)
|
||||
#:grammar
|
||||
[(name Identifier)
|
||||
(maybe-∀ (code:line)
|
||||
(code:line #:∀ (tvarⱼ ...)))
|
||||
(maybe-tag (code:line)
|
||||
(code:line #:tag tag-name))
|
||||
(tag-name Identifier)
|
||||
(maybe-pred? (code:line)
|
||||
(code:line #:? predicate-name?))
|
||||
(predicate-name? Identifier)
|
||||
(types-spec (: typeᵢ ...)
|
||||
(! . xlist-types)
|
||||
(:: . xlist-types))
|
||||
(xlist-types (τᵢ ...)
|
||||
(τᵢ ... . dotted-τ-rest)
|
||||
(τᵢ ... #:rest τ-rest))
|
||||
(τᵢ xlist-type-or-repeated-type)
|
||||
(τ-rest xlist-type-or-repeated-type)
|
||||
(dotted-τ-rest #,"like τ-rest, but must not be a syntax pair")
|
||||
(typeᵢ Type)]]{
|
||||
Defines @racket[name] as a shorthand for the type expander, match expander,
|
||||
builder function and predicate for a constructor with given
|
||||
@racket[tag-name] and content types.
|
||||
|
||||
When @racket[#:tag tag-name] is omitted, it defaults to @racket[name].
|
||||
|
||||
The predicate is bound to @racket[predicate-name?]; When
|
||||
@racket[#:? predicate-name?] is omitted, it defaults to @racket[_name?], which
|
||||
is an identifier with the same lexical context as @racket[name], with a
|
||||
@racket["?"] appended at the end.
|
||||
|
||||
The @racket[_name] and @racket[_predicate?] identifiers behave as follows:
|
||||
|
||||
@(make-blockquote
|
||||
"leftindent"
|
||||
(flow-paragraphs
|
||||
(decode-flow
|
||||
(splice-run
|
||||
@defidform[#:kind "type expander"
|
||||
#:link-target? #f
|
||||
_name]{
|
||||
Expands to the same type as @racket[(constructor tag-name typeᵢ ...)] or
|
||||
@racket[(constructor tag-name . xlist-types)] would.}))))
|
||||
|
||||
@(make-blockquote
|
||||
"leftindent"
|
||||
(flow-paragraphs
|
||||
(decode-flow
|
||||
(splice-run
|
||||
@defidform[#:link-target? #f _name]{
|
||||
Expands to the same builder function as
|
||||
@racket[(constructor tag-name types-spec)] would. The use of @racket[:],
|
||||
@racket[!] or @racket[::] before the sequence of types therefore specifies
|
||||
whether the builder function accepts a simple fixed number of arguments, a
|
||||
variable number of arguments (performing a cast), or a single argument used as
|
||||
the whole value for the constructor.}))))
|
||||
|
||||
@(make-blockquote
|
||||
"leftindent"
|
||||
(flow-paragraphs
|
||||
(decode-flow
|
||||
(splice-run
|
||||
@defform[#:kind "match expander"
|
||||
#:link-target? #f
|
||||
(_name patᵢ ...)]{
|
||||
|
||||
When using the @racket[: typeᵢ ...] form of @racket[define-constructor], the
|
||||
defined match expander expects one pattern @racket[patᵢ] per type. The
|
||||
resulting match pattern verifies that the value is a constructor with the
|
||||
given @racket[tag-name] containing a list with the correct number of elements,
|
||||
and matches each element against the corresponding @racket[patᵢ].
|
||||
|
||||
When using the @racket[(! . xlist-types)] or @racket[(:: . xlist-types)]
|
||||
forms of @racket[define-constructor], the defined match expander expects one
|
||||
pattern per (possibly repeated) xlist type. The resulting match pattern
|
||||
verifies that the value is a constructor with the given @racket[tag-name]
|
||||
containing a value accepted by
|
||||
@racket[(make-predicate (xlist . xlist-types))]. It then uses the
|
||||
@racket[split-xlist] match expander, which splits the list into one sublist
|
||||
per repeated xlist type (and a single item for each non-repeated xlist type),
|
||||
and matches each sublist or single item against the corresponding
|
||||
@racket[patᵢ]. See the documentation for @racket[split-xlist] for more details
|
||||
about this process. The resulting match pattern is therefore equivalent to:
|
||||
|
||||
@racketblock[(and (tagged? tag-name values)
|
||||
(? (make-predicate (xlist . xlist-types)))
|
||||
(split-xlist [patᵢ ...] . xlist-types))]}))))
|
||||
|
||||
@(make-blockquote
|
||||
"leftindent"
|
||||
(flow-paragraphs
|
||||
(decode-flow
|
||||
(splice-run
|
||||
@defidform[#:link-target? #f _predicate?]{
|
||||
Expands to the same predicate as
|
||||
|
||||
@racketblock[(constructor? tag-name (xlist τᵢ … . τ-rest))]
|
||||
|
||||
would, where all occurrences of @racket[tvarⱼ] type variables are replaced
|
||||
with @racket[Any].}))))
|
||||
|
||||
The elements of the grammar for @racket[define-tagged] may appear in any
|
||||
order, as long as the tag name appears before any field descriptor, and as
|
||||
long as the field descriptors form a contiguous sequence.}
|
||||
|
||||
@defidform[#:kind "type"
|
||||
ConstructorTop]{ The supertype of all @tech{constructors},
|
||||
including @tech{tagged structures}, @tech{untagged structures} and @tech{
|
||||
nodes} which only contain a single @racket[values] field.}
|
||||
|
||||
@defproc[(ConstructorTop? [v Any]) Boolean]{
|
||||
A predicate for @racket[ConstructorTop]. It accepts all @tech{constructors},
|
||||
including @tech{tagged structures}, @tech{untagged structures} and @tech{
|
||||
nodes} which contain a single @racket[values] field, and rejects any other
|
||||
value.}
|
||||
|
||||
@defproc[(constructor-values [v ConstructorTop]) T]{
|
||||
Returns the value stored within the constructor.}
|
21
phc-adt-doc/phc-adt/scribblings/phc-adt-implementation.scrbl
Normal file
21
phc-adt-doc/phc-adt/scribblings/phc-adt-implementation.scrbl
Normal file
|
@ -0,0 +1,21 @@
|
|||
#lang scribble/manual
|
||||
|
||||
@title{Algebraic Data Types for compilers: Implementation}
|
||||
@author[@author+email["Georges Dupéron" "georges.duperon@gmail.com"]]
|
||||
|
||||
This library is implemented using literate programming. The implementation
|
||||
details are presented in the following sections. The user documentation is in
|
||||
the @other-doc['(lib "phc-adt/scribblings/phc-adt.scrbl")] document.
|
||||
|
||||
@(table-of-contents)
|
||||
|
||||
@include-section[(submod (lib "phc-adt/adt.hl.rkt") doc)]
|
||||
@include-section[(submod (lib "phc-adt/tagged-structure-low-level.hl.rkt") doc)]
|
||||
@include-section[(submod (lib "phc-adt/node-low-level.hl.rkt") doc)]
|
||||
@include-section[(submod (lib "phc-adt/ctx.hl.rkt") doc)]
|
||||
@include-section[(submod (lib "phc-adt/tagged.hl.rkt") doc)]
|
||||
@include-section[(submod (lib "phc-adt/tagged-supertype.hl.rkt") doc)]
|
||||
@include-section[(submod (lib "phc-adt/structure.hl.rkt") doc)]
|
||||
@include-section[(submod (lib "phc-adt/constructor.hl.rkt") doc)]
|
||||
@include-section[(submod (lib "phc-adt/variant.hl.rkt") doc)]
|
||||
@include-section[(submod "phc-adt-choices.scrbl" doc)]
|
190
phc-adt-doc/phc-adt/scribblings/phc-adt-structure.scrbl
Normal file
190
phc-adt-doc/phc-adt/scribblings/phc-adt-structure.scrbl
Normal file
|
@ -0,0 +1,190 @@
|
|||
#lang scribble/manual
|
||||
|
||||
@(require racket/require
|
||||
(for-label (subtract-in typed/racket/base type-expander)
|
||||
type-expander
|
||||
phc-adt
|
||||
(lib "phc-adt/tagged.hl.rkt")
|
||||
(lib "phc-adt/structure.hl.rkt")
|
||||
(lib "phc-adt/constructor.hl.rkt")
|
||||
(lib "phc-adt/variant.hl.rkt")
|
||||
(lib "phc-adt/tagged-supertype.hl.rkt"))
|
||||
scribble-enhanced/doc
|
||||
scribble-math)
|
||||
@doc-lib-setup
|
||||
|
||||
@title{Untagged structures}
|
||||
|
||||
@deftech{Untagged structures} are implemented as tagged structures, using a
|
||||
special tag name: @racket[untagged]. The @racket[structure] identifier and its
|
||||
derivatives therefore simply call @racket[tagged], filling in the tag name
|
||||
with @racket[untagged].
|
||||
|
||||
@defform[#:kind "type expander"
|
||||
#:literals (:)
|
||||
(structure fields-maybe-types)
|
||||
#:grammar
|
||||
[(fields-maybe-types (code:line just-fieldᵢ ...)
|
||||
(code:line maybe-∀ field+type-descriptorᵢ ...))
|
||||
(maybe-∀ (code:line)
|
||||
(code:line #:∀ (tvarⱼ ...)))
|
||||
(just-fieldᵢ fieldᵢ
|
||||
[fieldᵢ])
|
||||
(field+type-descriptor [fieldᵢ typeᵢ]
|
||||
[fieldᵢ : typeᵢ])
|
||||
(fieldᵢ Identifier)
|
||||
(typeᵢ Type)
|
||||
(tvarⱼ Identifier)]]{
|
||||
Expands to the same type as @racket[(tagged untagged fields-maybe-types)]
|
||||
would.
|
||||
|
||||
The elements may appear in any order, as long as the field descriptors form a
|
||||
contiguous sequence.}
|
||||
|
||||
@defform*[#:kind "syntax"
|
||||
#:link-target? #f
|
||||
#:literals (:)
|
||||
((structure maybe-instance maybe-∀ fields-maybe-types)
|
||||
(structure maybe-builder maybe-∀ fields+values-maybe-types))
|
||||
#:grammar
|
||||
[(maybe-instance (code:line)
|
||||
#:instance)
|
||||
(maybe-builder (code:line)
|
||||
#:builder)
|
||||
(tag-name Identifier)
|
||||
(maybe-∀ (code:line)
|
||||
(code:line #:∀ (tvarⱼ ...)))
|
||||
(fields-maybe-types (code:line just-fieldᵢ ...)
|
||||
(code:line [fieldᵢ : typeᵢ] ...))
|
||||
(just-fieldᵢ fieldᵢ
|
||||
[fieldᵢ])
|
||||
(field+value-maybe-type (code:line [fieldᵢ valueᵢ] ...)
|
||||
(code:line field+value+typeᵢ ...))
|
||||
(field+value+typeᵢ [fieldᵢ : typeᵢ valueᵢ]
|
||||
[fieldᵢ valueᵢ : typeᵢ])
|
||||
(fieldᵢ Identifier)
|
||||
(typeᵢ Type)
|
||||
(valueᵢ Expression)]]{
|
||||
|
||||
The first syntax expands to the same instance as
|
||||
@racket[(tagged untagged maybe-instance maybe-∀ fields-maybe-types)]
|
||||
would.
|
||||
|
||||
The second syntax expands to the same builder function as
|
||||
@racket[(tagged untagged maybe-builder maybe-∀ fields+values-maybe-types)]
|
||||
would.
|
||||
|
||||
The elements may appear in any order, as long as the field descriptors form a
|
||||
contiguous sequence.}
|
||||
|
||||
@defform[#:kind "match expander"
|
||||
#:link-target? #f
|
||||
#:literals (:)
|
||||
(structure maybe-no-implicit field-maybe-patsᵢ ...)
|
||||
#:grammar
|
||||
[(maybe-no-implicit (code:line)
|
||||
(code:line #:no-implicit-bind))
|
||||
(field-maybe-patsᵢ fieldᵢ
|
||||
[fieldᵢ patᵢⱼ ...])
|
||||
(fieldᵢ Identifier)
|
||||
(patᵢⱼ #,"match pattern")]]{
|
||||
Expands to the same match pattern as
|
||||
@racket[(tagged untagged maybe-no-implicit field-maybe-patsᵢ ...)] would.
|
||||
|
||||
The elements may appear in any order, as long as the
|
||||
@racket[field-maybe-patsᵢ] form a contiguous sequence.}
|
||||
|
||||
@defform*[#:kind "syntax"
|
||||
#:literals (:)
|
||||
[(structure? fieldᵢ ...)
|
||||
(structure? [fieldᵢ : typeᵢ] ...)
|
||||
(structure? [fieldᵢ predᵢ] ...)]
|
||||
#:contracts ([tag-name Identifier]
|
||||
[fieldᵢ Identifier]
|
||||
[typeᵢ Type/Make-Predicate]
|
||||
[predᵢ (ExpressionOf (→ Any Any : typeᵢ))])]{
|
||||
|
||||
The first syntax expands to the same predicate as
|
||||
@racket[(tagged untagged fieldᵢ ...)] would.
|
||||
|
||||
The second syntax expands to the same predicate as
|
||||
@racket[(tagged untagged [fieldᵢ : typeᵢ] ...)] would.
|
||||
|
||||
The third syntax expands to the same predicate as
|
||||
@racket[(tagged untagged [fieldᵢ predᵢ] ...)] would.
|
||||
|
||||
The elements may appear in any order, as long as the field descriptors form a
|
||||
contiguous sequence.}
|
||||
|
||||
@defform[#:kind "syntax"
|
||||
#:literals (:)
|
||||
(define-structure maybe-predicate? name fields-maybe-types)
|
||||
#:grammar
|
||||
[(maybe-predicate? (code:line)
|
||||
(code:line #:? predicate-name?))
|
||||
(tag-name Identifier)
|
||||
(fields-maybe-types (code:line just-fieldᵢ ...)
|
||||
(code:line maybe-∀ field+type-descriptorᵢ ...))
|
||||
(maybe-∀ (code:line)
|
||||
(code:line #:∀ (tvarⱼ ...)))
|
||||
(just-fieldᵢ fieldᵢ
|
||||
[fieldᵢ])
|
||||
(field+type-descriptor [fieldᵢ typeᵢ]
|
||||
[fieldᵢ : typeᵢ])
|
||||
(fieldᵢ Identifier)
|
||||
(typeᵢ Type)
|
||||
(tvarⱼ Identifier)]]{
|
||||
|
||||
Defines @racket[name] and @racket[predicate?] in the same way as
|
||||
@racket[
|
||||
(define-tagged #:tag untagged maybe-predicate? name fields-maybe-types)] would.
|
||||
|
||||
The elements of the grammar for @racket[define-structure] may appear in any
|
||||
order, as long as the field descriptors form a contiguous sequence.}
|
||||
|
||||
@defform[#:kind "type expander"
|
||||
#:literals (:)
|
||||
(structure-supertype field+typeᵢ ...)
|
||||
#:grammar
|
||||
[(field+type [fieldᵢ typeᵢ]
|
||||
[fieldᵢ : typeᵢ])]]{
|
||||
Expands to the same union type as
|
||||
@racket[(tagged-supertype untagged field+typeᵢ ...)] would.}
|
||||
|
||||
@defform[#:kind "match expander"
|
||||
#:link-target? #f
|
||||
#:literals (:)
|
||||
(structure-supertype maybe-no-implicit field-maybe-patsᵢ ...)
|
||||
#:grammar
|
||||
[(maybe-no-implicit (code:line)
|
||||
(code:line #:no-implicit-bind))
|
||||
(field-maybe-patsᵢ fieldᵢ
|
||||
[fieldᵢ patᵢⱼ ...])
|
||||
(fieldᵢ Identifier)
|
||||
(patᵢⱼ #,"match pattern")]]{
|
||||
Expands to the same match pattern as
|
||||
@racket[(tagged-supertype untagged maybe-no-implicit field-maybe-patsᵢ ...)]
|
||||
would.
|
||||
|
||||
The elements may appear in any order, as long as the tag name appears before
|
||||
any field descriptor, and as long as the field descriptors form a contiguous
|
||||
sequence.}
|
||||
|
||||
@defform[(structure-supertype* …)]{
|
||||
Currently not implemented. Will be equivalent to nesting
|
||||
@racket[structure-supertype].}
|
||||
|
||||
@defidform[#:kind "type"
|
||||
StructureTop]{
|
||||
|
||||
The supertype of all @tech{untagged structures}, including @tech{tagged
|
||||
structures}, @tech{constructors} and @tech{nodes} using the tag name
|
||||
@racket[untagged]. It does not include tagged structures, constructors or
|
||||
nodes with other tag names than @racket[untagged].}
|
||||
|
||||
@defproc[(StructureTop? [v Any]) Boolean]{
|
||||
|
||||
A predicate for @racket[StructureTop]. It accepts all @tech{untagged
|
||||
structures}, including @tech{tagged structures}, @tech{constructors} and
|
||||
@tech{nodes} using the tag name @racket[untagged], and rejects any other
|
||||
value.}
|
327
phc-adt-doc/phc-adt/scribblings/phc-adt-tagged.scrbl
Normal file
327
phc-adt-doc/phc-adt/scribblings/phc-adt-tagged.scrbl
Normal file
|
@ -0,0 +1,327 @@
|
|||
#lang scribble/manual
|
||||
|
||||
@(require racket/require
|
||||
(for-label (subtract-in typed/racket/base type-expander)
|
||||
type-expander
|
||||
phc-adt
|
||||
racket/shared
|
||||
(lib "phc-adt/tagged.hl.rkt")
|
||||
(lib "phc-adt/structure.hl.rkt")
|
||||
(lib "phc-adt/constructor.hl.rkt")
|
||||
(lib "phc-adt/variant.hl.rkt")
|
||||
(lib "phc-adt/tagged-supertype.hl.rkt"))
|
||||
scribble-enhanced/doc
|
||||
scribble-math
|
||||
(subtract-in scribble/struct scribble-enhanced/doc)
|
||||
scribble/decode)
|
||||
@doc-lib-setup
|
||||
|
||||
@title{Tagged structures}
|
||||
|
||||
@deftech{Tagged structures} behave like Racket's plain @racket[struct]s, but
|
||||
do not need to be declared before they are used. They are similar to
|
||||
@tech[#:doc '(lib "scribblings/guide/guide.scrbl")]{prefab} structs in that
|
||||
aspect, but prefab structs lack field names, i.e. they behave more like
|
||||
vectors tagged with the prefab struct's name.
|
||||
|
||||
A tagged structure is identified by its tag name, and its set of field names.
|
||||
The type of a tagged structure can be expressed without having declared the
|
||||
tagged structure in advance. It is also possible to create instances of tagged
|
||||
structures without declaring them beforehand, and this applies to match
|
||||
patterns for tagged structures too. Fields can be accessed without knowing the
|
||||
structure's tag name, using @racket[(uniform-get instance field-name)].
|
||||
|
||||
These features make tagged structures particularly suited for writing
|
||||
compilers and other programs which transform large and complex data structures
|
||||
in small steps. This library is designed to work hand in hand with the
|
||||
@elem[#:style 'tt "phc-graph"] library (not available yet, but will be soon),
|
||||
which adds to tagged structures some support for safe cyclic data structures,
|
||||
and easy manipulation of those via higher-order operations. The regular tagged
|
||||
structures should normally not be used to form cyclic data structures@note{It
|
||||
is possible in theory to build cyclic data structures using @racket[shared],
|
||||
for example, but this use case is not supported by this library, and is
|
||||
unlikely to play well with Typed/Racket in any case.}. Thus, the graph library
|
||||
uses @deftech{nodes} instead, which can contain cycles, as long as the cycles
|
||||
are safely created via the graph library. Nodes also have a tag name and a set
|
||||
of fields, and each node type is a subtype of the corresponding tagged
|
||||
structure with the same name and fields.
|
||||
|
||||
@defform[#:kind "type expander"
|
||||
#:literals (:)
|
||||
(tagged tag-name fields-maybe-types)
|
||||
#:grammar
|
||||
[(tag-name Identifier)
|
||||
(fields-maybe-types (code:line just-fieldᵢ ...)
|
||||
(code:line maybe-∀ field+type-descriptorᵢ ...))
|
||||
(maybe-∀ (code:line)
|
||||
(code:line #:∀ (tvarⱼ ...)))
|
||||
(just-fieldᵢ fieldᵢ
|
||||
[fieldᵢ])
|
||||
(field+type-descriptor [fieldᵢ typeᵢ]
|
||||
[fieldᵢ : typeᵢ])
|
||||
(fieldᵢ Identifier)
|
||||
(typeᵢ Type)
|
||||
(tvarⱼ Identifier)]]{
|
||||
|
||||
Expands to the type for a tagged structure with the given tag name and
|
||||
fields. If the types for the fields are not given, then the tagged structure
|
||||
is polymorphic, with one type variable per field.
|
||||
|
||||
If @racket[#:∀ (tvarⱼ ...)] is specified, a polymorphic tagged structure is
|
||||
polymorphic, with the given type variables.
|
||||
|
||||
The elements may appear in any order, as long as the tag name appears before
|
||||
any field descriptor, and as long as the field descriptors form a contiguous
|
||||
sequence.}
|
||||
|
||||
@defform*[#:kind "syntax"
|
||||
#:link-target? #f
|
||||
#:literals (:)
|
||||
((tagged maybe-instance maybe-∀ tag-name fields-maybe-types)
|
||||
(tagged maybe-builder maybe-∀ tag-name fields+values-maybe-types))
|
||||
#:grammar
|
||||
[(maybe-instance (code:line)
|
||||
#:instance)
|
||||
(maybe-builder (code:line)
|
||||
#:builder)
|
||||
(tag-name Identifier)
|
||||
(maybe-∀ (code:line)
|
||||
(code:line #:∀ (tvarⱼ ...)))
|
||||
(fields-maybe-types (code:line just-fieldᵢ ...)
|
||||
(code:line [fieldᵢ : typeᵢ] ...))
|
||||
(just-fieldᵢ fieldᵢ
|
||||
[fieldᵢ])
|
||||
(field+value-maybe-type (code:line [fieldᵢ valueᵢ] ...)
|
||||
(code:line field+value+typeᵢ ...))
|
||||
(field+value+typeᵢ [fieldᵢ : typeᵢ valueᵢ]
|
||||
[fieldᵢ valueᵢ : typeᵢ])
|
||||
(fieldᵢ Identifier)
|
||||
(typeᵢ Type)
|
||||
(valueᵢ Expression)]]{
|
||||
When using the @racket[fields-maybe-types] syntax, this form expands to a
|
||||
lambda function which can be used to build an instance of the tagged
|
||||
structure, by passing as many values as there are fields.
|
||||
|
||||
When using the @racket[fields+values-maybe-types] syntax, this form directly
|
||||
returns an instance of the tagged structure, with the given values.
|
||||
|
||||
It is mandatory to disambiguate with either @racket[#:instance] or
|
||||
@racket[#:builder] when using @racket[tagged] with an empty list of fields
|
||||
(i.e. a structure with no fields) as it cannot be guessed from the syntax
|
||||
otherwise, so it is best to always include one or the other when writing a
|
||||
macro which expands to uses of @racket[tagged].
|
||||
|
||||
When types are specified, they are used to annotate the values when producing
|
||||
an instance, otherwise they are used as the argument types for the builder
|
||||
function.
|
||||
|
||||
When @racket[#:∀ (tvarⱼ ...)] is specified for a builder, a polymorphic
|
||||
builder is produced, with the given @racket[tvarⱼ ...] type variables.
|
||||
|
||||
When @racket[#:∀ (tvarⱼ ...)] is specified for an instance, the type of
|
||||
values annotated with @racket[tvarⱼ] is inferred, and an instance of a
|
||||
polymorphic tagged structure is produced. A @racket[tvarⱼ] can be used within
|
||||
a more complex type, in which case only that part of the type is inferred.
|
||||
|
||||
The elements may appear in any order, as long as the tag name appears before
|
||||
any field descriptor, and as long as the field descriptors form a contiguous
|
||||
sequence.}
|
||||
|
||||
@defform[#:kind "match expander"
|
||||
#:link-target? #f
|
||||
#:literals (:)
|
||||
(tagged tag-name maybe-no-implicit field-maybe-patsᵢ ...)
|
||||
#:grammar
|
||||
[(tag-name Identifier)
|
||||
(maybe-no-implicit (code:line)
|
||||
(code:line #:no-implicit-bind))
|
||||
(field-maybe-patsᵢ fieldᵢ
|
||||
[fieldᵢ patᵢⱼ ...])
|
||||
(fieldᵢ Identifier)
|
||||
(patᵢⱼ #,"match pattern")]]{
|
||||
Expands to a match pattern for a tagged structure with the given name and
|
||||
fields. The value of each @racket[fieldᵢ] is matched against all of the
|
||||
corresponding @racket[patᵢⱼ ...]. When there are not @racket[patᵢⱼ] for a
|
||||
@racket[fieldᵢ], the brackets around the field name may be omitted.
|
||||
|
||||
Unless @racket[#:no-implicit-bind] is specified, every @racket[fieldᵢ] is
|
||||
bound by the match pattern to the field's value.
|
||||
|
||||
The elements may appear in any order, as long as the tag name appears before
|
||||
any @racket[field-maybe-patsᵢ], and as long as the @racket[field-maybe-patsᵢ]
|
||||
form a contiguous sequence.}
|
||||
|
||||
@defform*[#:kind "syntax"
|
||||
#:literals (:)
|
||||
[(tagged? tag-name fieldᵢ ...)
|
||||
(tagged? tag-name [fieldᵢ : typeᵢ] ...)
|
||||
(tagged? tag-name [fieldᵢ predᵢ] ...)]
|
||||
#:contracts ([tag-name Identifier]
|
||||
[fieldᵢ Identifier]
|
||||
[typeᵢ Type/Make-Predicate]
|
||||
[predᵢ (ExpressionOf (→ Any Any : typeᵢ))])]{
|
||||
Expands to a predicate for tagged structures with the given @racket[tag] and
|
||||
@racket[field]s. If types are specified, each @racket[typeᵢ] is passed to
|
||||
@racket[make-predicate], and the resulting predicate is checked against the
|
||||
value of the corresponding @racket[fieldᵢ].
|
||||
|
||||
Each @racket[typeᵢ] must therefore be a valid type for which
|
||||
@racket[make-predicate] can generate a predicate (@racket[make-predicate]
|
||||
cannot create a predicate for some types, like function types, or any type
|
||||
which translates to a
|
||||
@tech[#:doc '(lib "scribblings/reference/reference.scrbl")]{chaperone}
|
||||
contract.
|
||||
|
||||
The last form allows the use of arbitrary predicates @racket[predᵢ] which are
|
||||
checked against the value of the corresponding @racket[fieldᵢ]. When the type
|
||||
of a given @racket[predᵢ] includes a filter asserting that it returns true if
|
||||
and only if the value is of type @racket[typeᵢ], then the predicate produced by
|
||||
@racket[tagged-predicate!] will also have that filter on the corresponding
|
||||
field. By default, any function of type @racket[(→ Any Any)] will
|
||||
implicitly have the @racket[Any] filter, which does not bring any extra
|
||||
information.
|
||||
|
||||
The elements may appear in any order, as long as the tag name appears before
|
||||
any field descriptor, and as long as the field descriptors form a contiguous
|
||||
sequence.}
|
||||
|
||||
@defform[#:kind "syntax"
|
||||
#:literals (:)
|
||||
(define-tagged name maybe-tag-name maybe-predicate? fields-maybe-types)
|
||||
#:grammar
|
||||
[(name Identifier)
|
||||
(maybe-tag-name (code:line)
|
||||
(code:line #:tag tag-name))
|
||||
(tag-name Identifier)
|
||||
(maybe-predicate? (code:line)
|
||||
(code:line #:? predicate-name?))
|
||||
(predicate-name? Identifier)
|
||||
(fields-maybe-types (code:line just-fieldᵢ ...)
|
||||
(code:line maybe-∀ field+type-descriptorᵢ ...))
|
||||
(maybe-∀ (code:line)
|
||||
(code:line #:∀ (tvarⱼ ...)))
|
||||
(just-fieldᵢ fieldᵢ
|
||||
[fieldᵢ])
|
||||
(field+type-descriptor [fieldᵢ typeᵢ]
|
||||
[fieldᵢ : typeᵢ])
|
||||
(fieldᵢ Identifier)
|
||||
(typeᵢ Type)
|
||||
(tvarⱼ Identifier)]]{
|
||||
|
||||
Defines @racket[name] as a shorthand for the type expander, match expander,
|
||||
builder function and predicate for a tagged structure with the given
|
||||
@racket[tag-name] and fields.
|
||||
|
||||
When @racket[#:tag tag-name] is omitted, it defaults to @racket[name].
|
||||
|
||||
The predicate is bound to @racket[predicate-name?]; When
|
||||
@racket[#:? predicate-name?] is omitted, it defaults to @racket[_name?], which
|
||||
is an identifier with the same lexical context as @racket[name], with a
|
||||
@racket["?"] appended at the end.
|
||||
|
||||
The @racket[_name] and @racket[_predicate?] identifiers behave as follows:
|
||||
|
||||
@(make-blockquote
|
||||
"leftindent"
|
||||
(flow-paragraphs
|
||||
(decode-flow
|
||||
(splice-run
|
||||
@defidform[#:kind "type expander"
|
||||
#:link-target? #f
|
||||
_name]{
|
||||
Expands to the same type as @racket[(tagged tag-name fields-maybe-types)]
|
||||
would.}))))
|
||||
|
||||
@(make-blockquote
|
||||
"leftindent"
|
||||
(flow-paragraphs
|
||||
(decode-flow
|
||||
(splice-run
|
||||
@defidform[#:link-target? #f _name]{
|
||||
Expands to the same builder function as
|
||||
@racket[(tagged #:builder tag-name fields-maybe-types)] would.}))))
|
||||
|
||||
@(make-blockquote
|
||||
"leftindent"
|
||||
(flow-paragraphs
|
||||
(decode-flow
|
||||
(splice-run
|
||||
@defform[#:kind "match expander"
|
||||
#:link-target? #f
|
||||
(_name patᵢ ...)]{
|
||||
Expands to the same match pattern as
|
||||
@racket[(tagged tag-name [fieldᵢ patᵢ] ...)] would.}))))
|
||||
|
||||
@(make-blockquote
|
||||
"leftindent"
|
||||
(flow-paragraphs
|
||||
(decode-flow
|
||||
(splice-run
|
||||
@defidform[#:link-target? #f _predicate?]{
|
||||
Expands to the same predicate as @racket[(tagged? just-fieldᵢ)] would. Note
|
||||
that it does not attempt to check the field values, as doing so would mean
|
||||
that all of the @racket[typeᵢ], if specified, would have to be suitable
|
||||
arguments for @racket[make-predicate].}))))
|
||||
|
||||
The elements of the grammar for @racket[define-tagged] may appear in any
|
||||
order, as long as the tag name appears before any field descriptor, and as
|
||||
long as the field descriptors form a contiguous sequence.}
|
||||
|
||||
@defform[#:kind "type expander"
|
||||
#:literals (:)
|
||||
(tagged-supertype tag-name field+typeᵢ ...)
|
||||
#:grammar
|
||||
[(tag-name Identifier)
|
||||
(field+type [fieldᵢ typeᵢ]
|
||||
[fieldᵢ : typeᵢ])]]{
|
||||
Expands to the union type of all tagged structures with the given name and a
|
||||
superset of the given fields, where each given @racket[fieldᵢ] must have the
|
||||
corresponding type @racket[typeᵢ], and the other fields have the type
|
||||
@racket[Any].}
|
||||
|
||||
@defform[#:kind "match expander"
|
||||
#:link-target? #f
|
||||
#:literals (:)
|
||||
(tagged-supertype tag-name maybe-no-implicit field-maybe-patsᵢ ...)
|
||||
#:grammar
|
||||
[(tag-name Identifier)
|
||||
(maybe-no-implicit (code:line)
|
||||
(code:line #:no-implicit-bind))
|
||||
(field-maybe-patsᵢ fieldᵢ
|
||||
[fieldᵢ patᵢⱼ ...])
|
||||
(fieldᵢ Identifier)
|
||||
(patᵢⱼ #,"match pattern")]]{
|
||||
Expands to a match pattern accepting any tagged structures with the given
|
||||
name and a superset of the given fields, where each given @racket[fieldᵢ] must
|
||||
match all the corresponding @racket[patᵢⱼ], and the other fields are matched
|
||||
against @racket[_] (i.e. they can contain any value).
|
||||
|
||||
Unless @racket[#:no-implicit-bind] is specified, every @racket[fieldᵢ] is
|
||||
bound by the match pattern to the field's value, but the other extra fields
|
||||
are not bound to any variable.
|
||||
|
||||
The elements may appear in any order, as long as the tag name appears before
|
||||
any field descriptor, and as long as the field descriptors form a contiguous
|
||||
sequence.}
|
||||
|
||||
@defform[(tagged-supertype* …)]{
|
||||
Currently not implemented. Will be equivalent to nesting
|
||||
@racket[tagged-supertype].}
|
||||
|
||||
@defidform[#:kind "type"
|
||||
TaggedTop]{
|
||||
|
||||
The supertype of all @tech{tagged structures}, including @tech{untagged
|
||||
structures}, @tech{nodes} and @tech{constructors}.}
|
||||
|
||||
@defproc[(TaggedTop? [v Any]) Boolean]{
|
||||
A predicate for @racket[TaggedTop]. It accepts all @tech{tagged structures},
|
||||
including @tech{untagged structures}, @tech{nodes} and @tech{constructors},
|
||||
and rejects any other value.}
|
||||
|
||||
@defform[(uniform-get v f)
|
||||
#:grammar
|
||||
([v Expression]
|
||||
[f Identifier])]{
|
||||
Returns the value contained within the @racket[f] field of the tagged structure
|
||||
instance @racket[v].}
|
142
phc-adt-doc/phc-adt/scribblings/phc-adt-variant.scrbl
Normal file
142
phc-adt-doc/phc-adt/scribblings/phc-adt-variant.scrbl
Normal file
|
@ -0,0 +1,142 @@
|
|||
#lang scribble/manual
|
||||
|
||||
@(require racket/require
|
||||
(for-label (subtract-in typed/racket/base type-expander)
|
||||
type-expander
|
||||
phc-adt
|
||||
(lib "phc-adt/tagged.hl.rkt")
|
||||
(lib "phc-adt/structure.hl.rkt")
|
||||
(lib "phc-adt/constructor.hl.rkt")
|
||||
(lib "phc-adt/variant.hl.rkt")
|
||||
(lib "phc-adt/tagged-supertype.hl.rkt"))
|
||||
scribble-enhanced/doc
|
||||
scribble-math
|
||||
(subtract-in scribble/struct scribble-enhanced/doc)
|
||||
scribble/decode)
|
||||
@doc-lib-setup
|
||||
|
||||
@title{Variants}
|
||||
|
||||
@deftech{Variants} behave like @racketmodname[typed/racket]'s union types
|
||||
(expressed via @racket[U]). They benefit however of special support by the
|
||||
graph library (not published yet), which needs to rewrite data structures
|
||||
in-depth during the construction of the graph, and cannot handle arbitrary data
|
||||
structures.
|
||||
|
||||
@defform[#:kind "type-expander"
|
||||
#:literals (constructor tagged)
|
||||
(variant maybe-check-overlap
|
||||
constructor-or-taggedᵢ ...
|
||||
maybe-one-other-type)
|
||||
#:grammar
|
||||
[(maybe-check-overlap (code:line)
|
||||
(code:line #:check-overlap))
|
||||
(constructor-or-tagged (tagged tag-nameᵢ . tagged-args)
|
||||
(structure . structure-args)
|
||||
(constructor tag-nameᵢ . constructor-args))
|
||||
(constructor tag-name maybe-∀ τᵢ ... #:rest τ-rest)
|
||||
(tag-nameᵢ Identifier)
|
||||
(maybe-one-other-type (code:line)
|
||||
(code:line Type))]]{
|
||||
The current implementation does not completely match this documentation, but
|
||||
the implementation should shortly be updated.
|
||||
|
||||
Expands to the union type of all the given @tech{tagged structures}, @tech{
|
||||
untagged structures} and @tech{constructors}.
|
||||
|
||||
When another type which is not a tagged structure or similar is specified,
|
||||
it is included in the union. Only one such type is allowed.
|
||||
|
||||
The types passed to @racket[variant] should not overlap. When the keyword
|
||||
@racket[#:check-overlap] is specified, @racket[variant] uses a hack to throw
|
||||
an error when two types overlap. The error message can however be unclear and
|
||||
misleading, so this feature is disabled by default.
|
||||
|
||||
The elements of the grammar for @racket[define-tagged] may appear in any
|
||||
order, as long as the @racket[constructor-or-taggedᵢ ... maybe-one-other-type]
|
||||
form a contiguous sequence (the @racket[maybe-one-other-type] may occur at any
|
||||
position within the sequence, but must be included at most once).}
|
||||
|
||||
@defidform[#:kind "type-expander"
|
||||
V]{
|
||||
An alias for the @racket[variant] type expander.
|
||||
}
|
||||
|
||||
@defform[#:kind "syntax"
|
||||
#:literals (constructor tagged)
|
||||
(variant? maybe-check-overlap
|
||||
constructor-or-taggedᵢ ...
|
||||
maybe-one-other-type)
|
||||
#:grammar
|
||||
[(maybe-check-overlap (code:line)
|
||||
(code:line #:check-overlap))
|
||||
(constructor-or-tagged (tagged tag-nameᵢ . tagged-args)
|
||||
(structure . structure-args)
|
||||
(constructor tag-nameᵢ . constructor-args))
|
||||
(constructor tag-name maybe-∀ τᵢ ... #:rest τ-rest)
|
||||
(tag-nameᵢ Identifier)
|
||||
(maybe-one-other-type (code:line)
|
||||
(code:line Type))]]{
|
||||
The current implementation does not completely match this documentation, but
|
||||
the implementation should shortly be updated.
|
||||
|
||||
Expands to a predicate for the type:
|
||||
|
||||
@racketblock[(variant maybe-check-overlap
|
||||
constructor-or-taggedᵢ ...
|
||||
maybe-one-other-type)]
|
||||
|
||||
The elements of the grammar for @racket[variant] may appear in any order, as
|
||||
long as the @racket[constructor-or-taggedᵢ ... maybe-one-other-type] form a
|
||||
contiguous sequence (the @racket[maybe-one-other-type] may occur at any
|
||||
position within the sequence, but must be included at most once).}
|
||||
|
||||
@defform[#:kind "syntax"
|
||||
#:literals (:)
|
||||
(define-variant name maybe-predicate? maybe-check-overlap cases)
|
||||
#:grammar
|
||||
[(name Identifier)
|
||||
(maybe-predicate? (code:line)
|
||||
(code:line #:? predicate-name?))
|
||||
(predicate-name? Identifier)
|
||||
(maybe-check-overlap (code:line)
|
||||
(code:line #:check-overlap))
|
||||
(cases (code:line constructor-or-taggedᵢ ... maybe-one-other-type))
|
||||
(constructor-or-tagged (tagged tag-nameᵢ . tagged-args)
|
||||
(structure . structure-args)
|
||||
(constructor tag-nameᵢ . constructor-args))
|
||||
(constructor tag-name maybe-∀ τᵢ ... #:rest τ-rest)
|
||||
(tag-nameᵢ Identifier)
|
||||
(maybe-one-other-type (code:line)
|
||||
(code:line Type))]]{
|
||||
The current implementation does not completely match this documentation, but
|
||||
the implementation should shortly be updated.
|
||||
|
||||
Defines @racket[name] as a shorthand for the type expander and predicate for
|
||||
a variant with the given cases.
|
||||
|
||||
@(make-blockquote
|
||||
"leftindent"
|
||||
(flow-paragraphs
|
||||
(decode-flow
|
||||
(splice-run
|
||||
@defidform[#:kind "type expander"
|
||||
#:link-target? #f
|
||||
_name]{
|
||||
Expands to the same type as @racket[(variant maybe-check-overlap cases)]
|
||||
would.}))))
|
||||
|
||||
@(make-blockquote
|
||||
"leftindent"
|
||||
(flow-paragraphs
|
||||
(decode-flow
|
||||
(splice-run
|
||||
@defidform[#:link-target? #f _predicate?]{
|
||||
|
||||
Expands to the same predicate as @racket[(variant? maybe-check-overlap cases)]
|
||||
would.}))))
|
||||
|
||||
The elements of the grammar for @racket[define-variant] may appear in any
|
||||
order, as long as the @racket[constructor-or-taggedᵢ ... maybe-one-other-type]
|
||||
form a contiguous sequence (the @racket[maybe-one-other-type] may occur at any
|
||||
position within the sequence, but must be included at most once).}
|
87
phc-adt-doc/phc-adt/scribblings/phc-adt.scrbl
Normal file
87
phc-adt-doc/phc-adt/scribblings/phc-adt.scrbl
Normal file
|
@ -0,0 +1,87 @@
|
|||
#lang scribble/manual
|
||||
|
||||
@(require racket/require
|
||||
(for-label typed/racket/base
|
||||
phc-adt
|
||||
(lib "phc-adt/tagged.hl.rkt")
|
||||
(lib "phc-adt/structure.hl.rkt")
|
||||
(lib "phc-adt/constructor.hl.rkt")
|
||||
(lib "phc-adt/variant.hl.rkt")
|
||||
(lib "phc-adt/tagged-supertype.hl.rkt")
|
||||
(lib "phc-adt/adt-init.rkt"))
|
||||
scribble-enhanced/doc
|
||||
scribble-math
|
||||
(subtract-in scribble/struct scribble-enhanced/doc)
|
||||
scribble/decode)
|
||||
@doc-lib-setup
|
||||
|
||||
@title{Algebraic Data Types for compilers}
|
||||
|
||||
@defmodule[phc-adt
|
||||
#:use-sources
|
||||
[(lib "phc-adt/tagged.hl.rkt")
|
||||
(lib "phc-adt/structure.hl.rkt")
|
||||
(lib "phc-adt/constructor.hl.rkt")
|
||||
(lib "phc-adt/variant.hl.rkt")
|
||||
(lib "phc-adt/tagged-supertype.hl.rkt")
|
||||
(lib "phc-adt/adt-init.rkt")]]
|
||||
|
||||
This library is implmented using literate programming. The
|
||||
implementation details are presented in
|
||||
@other-doc['(lib "phc-adt/scribblings/phc-adt-implementation.scrbl")].
|
||||
|
||||
This library defines @tech{tagged structures}, @tech{untagged structures},
|
||||
@tech{constructors} and @tech{variants}. While uses of Racket's
|
||||
@racket[struct] need the struct to be declared before they are used, the
|
||||
structures implemented by this library can be used anonymously, without
|
||||
declaring them in advance. All uses of a tagged structure with the same tag
|
||||
name and set of fields are equivalent, even if they are declared in separate
|
||||
files. It is also possible to access a field on a tagged structure instance
|
||||
without specifying the tagged structure's tag name. This is used in a separate
|
||||
library (not published yet) to implement the dotted notation commonly used in
|
||||
object-oriented languages for field accesses, @racket[instance.field].
|
||||
|
||||
This library works by saving across compilations the list of all tagged
|
||||
structures used across the program. The tag name and list of field names for
|
||||
each tagged structure is written to a file. That file is, by default, located
|
||||
in the same directory as the source file, and it is called
|
||||
@filepath{adt-pre-declarations.rkt}, but this can be changed using the optional
|
||||
argument to @racket[adt-init]. These "remembered" tagged structure definitions
|
||||
are used to pre-declare the @racket[struct]s corresponding to each tagged
|
||||
structure, so that all the code using the same
|
||||
@filepath{adt-pre-declarations.rkt} file sees the same structure definitions.
|
||||
|
||||
@defform*[[(adt-init)
|
||||
(adt-init pre-declarations-file)]
|
||||
#:contracts
|
||||
[(pre-declarations-file string?)]]{
|
||||
The @racket[(adt-init)] macro has to be called in a
|
||||
@tech[#:doc '(lib "scribblings/reference/reference.scrbl")]{module context},
|
||||
after the @racket[(require phc-adt)] but before using any of the identifiers
|
||||
provided by this library.
|
||||
|
||||
The @racket[pre-declarations-file] (which defaults to @filepath{
|
||||
adt-pre-declarations.rkt}) should be the string representation of a path,
|
||||
relative to the directory containing the file which uses @racket[adt-init].
|
||||
|
||||
The @racket[adt-init] macro creates the @racket[pre-declarations-file] if it
|
||||
does not exist, and @racket[require]s it. The @racket[pre-declarations-file]
|
||||
contains a list of remembered tagged structures (specifically, the tag name
|
||||
followed by a list of field names). It uses the @hash-lang[]
|
||||
@racketmodname[phc-adt/declarations] which pre-declares the @racket[struct]s
|
||||
corresponding to each remembered tagged structure.
|
||||
|
||||
It then makes all the identifiers defined by this library available (the
|
||||
identifiers cannot otherwise be used before calling @racket[adt-init]).}
|
||||
|
||||
@defmodulelang[phc-adt/declarations #:no-declare]{
|
||||
This @hash-lang[] is used by the @racket[_pre-declarations-file] (see
|
||||
@racket[adt-init]), and is not intended to be used in other situations.}
|
||||
|
||||
|
||||
@(table-of-contents)
|
||||
|
||||
@include-section{phc-adt-tagged.scrbl}
|
||||
@include-section{phc-adt-structure.scrbl}
|
||||
@include-section{phc-adt-constructor.scrbl}
|
||||
@include-section{phc-adt-variant.scrbl}
|
25
phc-adt-lib/info.rkt
Normal file
25
phc-adt-lib/info.rkt
Normal file
|
@ -0,0 +1,25 @@
|
|||
#lang info
|
||||
(define collection 'multi)
|
||||
(define deps '("base"
|
||||
"typed-racket-lib"
|
||||
"hyper-literate"
|
||||
"multi-id"
|
||||
"phc-toolkit"
|
||||
"remember"
|
||||
"type-expander"
|
||||
"extensible-parser-specifications"
|
||||
"alexis-util"
|
||||
"typed-struct-props"
|
||||
"match-string"
|
||||
"xlist"
|
||||
"compatibility-lib"
|
||||
"generic-bind"
|
||||
"datatype"))
|
||||
(define build-deps '("at-exp-lib"
|
||||
"sandbox-lib"
|
||||
"scribble-enhanced"
|
||||
"scribble-lib"
|
||||
"scribble-math"))
|
||||
(define pkg-desc "Algebraic Datatypes tailored for writing compilers (tests are in phc-adt-test)")
|
||||
(define version "1.1")
|
||||
(define pkg-authors '("Georges Dupéron"))
|
51
phc-adt-lib/phc-adt/adt-init.rkt
Normal file
51
phc-adt-lib/phc-adt/adt-init.rkt
Normal file
|
@ -0,0 +1,51 @@
|
|||
#lang at-exp typed/racket
|
||||
(provide adt-init)
|
||||
(require remember
|
||||
"ctx.hl.rkt"
|
||||
phc-toolkit
|
||||
(for-syntax (only-in '#%kernel [#%app #%plain-app])
|
||||
syntax/parse
|
||||
syntax/parse/experimental/template
|
||||
phc-toolkit/untyped
|
||||
racket/port
|
||||
mzlib/etc))
|
||||
|
||||
(define-for-syntax ((the-trampoline srcdir pre-declarations-filename) stx2)
|
||||
(syntax-case stx2 ()
|
||||
[(_ self2)
|
||||
#`(adt-init-2 self2
|
||||
#,srcdir
|
||||
#,pre-declarations-filename)]))
|
||||
|
||||
(define-syntax adt-init
|
||||
(syntax-parser
|
||||
[(self (~optional pre-declarations-filename
|
||||
#:defaults ([pre-declarations-filename
|
||||
#'"adt-pre-declarations.rkt"])))
|
||||
#'(begin (define-syntaxes (trampoline)
|
||||
(#%plain-app the-trampoline
|
||||
(this-expression-source-directory self)
|
||||
'pre-declarations-filename))
|
||||
(begin-for-syntax (set-adt-context #'self))
|
||||
(trampoline self))]))
|
||||
|
||||
(define-syntax/parse (adt-init-2 ctx pre-declarations-dir pre-declarations-file)
|
||||
(define pre-declarations-path
|
||||
(build-path (syntax-e #'pre-declarations-dir)
|
||||
(syntax-e #'pre-declarations-file)))
|
||||
(define pre-declarations-path-string
|
||||
(path->string pre-declarations-path))
|
||||
|
||||
;; Initialize the pre-declarations file if it is empty:
|
||||
(init-file pre-declarations-path
|
||||
"#lang s-exp phc-adt/declarations\n")
|
||||
|
||||
(remember-output-file-parameter pre-declarations-path-string)
|
||||
;(set-adt-context #'ctx)
|
||||
#`(require #,(datum->syntax #'ctx (syntax-e #'pre-declarations-file))))
|
||||
|
||||
(define-for-syntax (init-file path string-contents)
|
||||
(unless (file-exists? path)
|
||||
(with-handlers ([exn:fail:filesystem (λ (exn) (void))])
|
||||
(with-output-file [port path] #:exists 'error
|
||||
(display string-contents port)))))
|
3
phc-adt-lib/phc-adt/adt-second-step.rkt
Normal file
3
phc-adt-lib/phc-adt/adt-second-step.rkt
Normal file
|
@ -0,0 +1,3 @@
|
|||
#lang typed/racket
|
||||
(require "adt.hl.rkt")
|
||||
(provide (all-from-out "adt.hl.rkt"))
|
253
phc-adt-lib/phc-adt/adt.hl.rkt
Normal file
253
phc-adt-lib/phc-adt/adt.hl.rkt
Normal file
|
@ -0,0 +1,253 @@
|
|||
#lang hyper-literate typed/racket/base
|
||||
@(require scribble-enhanced/doc
|
||||
scribble-math
|
||||
racket/sandbox
|
||||
scribble/example
|
||||
(for-label typed/racket/base
|
||||
phc-toolkit
|
||||
phc-toolkit/untyped-only
|
||||
datatype
|
||||
"ctx.hl.rkt"
|
||||
"tagged.hl.rkt"
|
||||
"tagged-supertype.hl.rkt"
|
||||
"structure.hl.rkt"
|
||||
"constructor.hl.rkt"
|
||||
"variant.hl.rkt"
|
||||
"adt-init.rkt"))
|
||||
@doc-lib-setup
|
||||
|
||||
@title[#:style (with-html5 manual-doc-style)
|
||||
#:tag "adt"
|
||||
#:tag-prefix "phc-adt/adt"]{Algebraic Data Types}
|
||||
|
||||
@(chunks-toc-prefix
|
||||
'("(lib phc-adt/scribblings/phc-adt-implementation.scrbl)"
|
||||
"phc-adt/adt"))
|
||||
|
||||
@(table-of-contents)
|
||||
|
||||
@section{A note on polysemy}
|
||||
|
||||
The name ``constructor'' usually designates two things:
|
||||
@itemlist[
|
||||
@item{A tagged value, like the ones created or accessed using the
|
||||
@racket[constructor] macro defined in
|
||||
@secref["constructor" #:tag-prefixes '("phc-adt/constructor")]}
|
||||
@item{A constructor function or macro for some kind of data structure, which
|
||||
is the function or macro used to create instances of that data structure.}]
|
||||
|
||||
Since this could lead to ambiguities, we clarify by saying ``constructor'' in
|
||||
the former case, and ``builder'' or ``builder function'' in the latter case.
|
||||
|
||||
@section[#:tag "ADT|introduction"]{Introduction}
|
||||
|
||||
We define variants (tagged unions), with the following constraints:
|
||||
|
||||
@; TODO: put a short usage example here
|
||||
|
||||
@itemlist[
|
||||
@item{A constructor is described by a tag name, followed by zero or more
|
||||
values. Likewise, a tagged structure is described by a tag name, followed by
|
||||
zero or more field names, each field name being mapped to a value.}
|
||||
@item{Two different variants can contain the same constructor or tagged
|
||||
structure, and it is not possible to create an instance of that constructor or
|
||||
tagged structure that would belong to one variant but not the other.}
|
||||
@item{Constructors and tagged structures are "anonymous": it is not necessary
|
||||
to declare a constructor or tagged structure before creating instances of it,
|
||||
expressing its type, or using it as a match pattern.}
|
||||
@item{Constructors types and tagged structures types are "interned": two
|
||||
constructors with the same tag name have the same type, even if they are used
|
||||
in different files. The same applies to two tagged structures with the same
|
||||
tag name and field names: even if they are used in different files, they have
|
||||
the same type.}]
|
||||
|
||||
The @racketmodname[datatype] package by Andrew Kent also
|
||||
implements Algebraic Data Types. The main differences are that unlike our
|
||||
library, data structures have to be declared before they are used (they are
|
||||
not "anonymous"), and a given constructor name cannot be shared by multiple
|
||||
unions, as can be seen in the example below where the second
|
||||
@tc[define-datatype] throws an error:
|
||||
|
||||
@(define tr-evaluator
|
||||
(parameterize ([sandbox-output 'string]
|
||||
[sandbox-error-output 'string])
|
||||
(make-evaluator 'typed/racket)))
|
||||
@examples[#:eval tr-evaluator
|
||||
(require datatype)
|
||||
|
||||
(define-datatype Expr
|
||||
[Var (Symbol)]
|
||||
[Lambda (Symbol Expr)]
|
||||
[App (Expr Expr)])
|
||||
|
||||
;; define-datatype: variant type #<syntax:11:3 Var> already bound
|
||||
;; in: Simple-Expr
|
||||
(eval:error
|
||||
(define-datatype Simple-Expr
|
||||
[Var (Symbol)]
|
||||
[Lambda (Symbol Expr)]))]
|
||||
|
||||
@section{Remembered data types and pre-declarations}
|
||||
|
||||
This library works by remembering all the constructors and all the tagged
|
||||
structures across compilations. More precisely, each constructor's tag name is
|
||||
written to a file named @filepath{adt-pre-declarations.rkt} in the same
|
||||
directory as the user code. The tag name and list of fields of each tagged
|
||||
structure is also written in the same file.
|
||||
|
||||
The generated @racket{adt-pre-declarations.rkt} file declares a
|
||||
@racket[struct] for each tagged structure and constructor, so that all user
|
||||
files which @racket[require] the same @racket{adt-pre-declarations.rkt} will
|
||||
share the same @racket[struct] definitions.
|
||||
|
||||
User files which make use of the @racketmodname[phc-adt] should include a call
|
||||
to @racket[adt-init] before using anything else. The @racket[adt-init] macro
|
||||
@racket[require]s the @filepath{adt-pre-declarations.rkt} file, and records
|
||||
the lexical context of that @racket[require], so that the other macros
|
||||
implemented by this library can fetch the pre-declared @racket[struct] types
|
||||
from the correct lexical scope. The @filepath{ctx.hl.rkt} file takes care of
|
||||
recording that lexical scope, while @filepath{adt-init.rkt} performs the
|
||||
initialisation sequence (creating the @filepath{adt-pre-declarations.rkt} file
|
||||
if it does not exist, loading the pre-declared @racket[struct] types from
|
||||
@filepath{adt-pre-declarations.rkt}, and using a utility from
|
||||
@filepath{ctx.hl.rkt} to record the lexical context).
|
||||
|
||||
@chunk[<require-modules>
|
||||
(require "ctx.hl.rkt")]
|
||||
|
||||
@section{The initialisation process}
|
||||
|
||||
The initialisation process can be somewhat complex: the directives
|
||||
@racket[(remember-output-file "adt-pre-declarations.rkt")],
|
||||
@racket[(set-adt-context)] and @racket[(require "adt-pre-declarations.rkt")]
|
||||
have to be inserted in the right order, and the file
|
||||
@filepath{adt-pre-declarations.rkt} has to be initialised with the appropriate
|
||||
contents when it does not exist. The @racket[adt-init] macro defined in
|
||||
@filepath{"adt-init.rkt"} takes care of these steps.
|
||||
|
||||
@chunk[<require-modules>
|
||||
(require "adt-init.rkt")]
|
||||
|
||||
The generated @filepath{adt-pre-declarations.rkt} file will call the
|
||||
@racket[pre-declare-all-tagged-structure-structs] macro defined in
|
||||
@filepath{tagged-structure-low-level.hl.rkt}.
|
||||
|
||||
@section{Tagged structures, untagged structures, constructors, and variants}
|
||||
|
||||
We first define a low-level interface for tagged structures in the
|
||||
@filepath{tagged-structure-low-level.hl.rkt} file. This low-level interface
|
||||
includes for-syntax functions for expressing the type of tagged structures,
|
||||
creating builder functions for them, as well as match patterns. It also includes
|
||||
means to access the value of a given field on any tagged structure which
|
||||
contains that field. The @filepath{tagged.hl.rkt} file provides syntactic sugar
|
||||
for this low-level interface, and defines the @racket[tagged] identifier, which
|
||||
acts as a type expander, match expander and macro. The macro can be used to
|
||||
create builder functions which return instances of tagged structures, or to
|
||||
directly create such instances.
|
||||
|
||||
@chunk[<require-modules>
|
||||
(require "tagged.hl.rkt")]
|
||||
|
||||
The @filepath{"tagged-supertype.hl.rkt"} file defines a few operations
|
||||
implementing some form of "static duck typing": As a type expander,
|
||||
@racket[(tagged-supertype fieldᵢ …)] expands to the union type of all tagged
|
||||
structures containing a superset of the given set of fields. As a match
|
||||
expander, @racket[(tagged-supertype [fieldᵢ patᵢⱼ …] …)] expands to a match
|
||||
pattern which accepts any tagged structure with a superset of the given set of
|
||||
fields, as long as the value of each @racket[fieldᵢ] matches against all of the
|
||||
corresponding @racket[patᵢⱼ …].
|
||||
|
||||
@chunk[<require-modules>
|
||||
(require "tagged-supertype.hl.rkt")]
|
||||
|
||||
We then define untagged structures, which are tagged structures with the
|
||||
@racket[untagged] tag name. Untagged structures can be used conveniently when
|
||||
the tag name is not important and the goal is simply to map a set of field names
|
||||
to values. The @filepath{structure.hl.rkt} file defines the @tc[structure]
|
||||
type expander, match expander and macro. The @tc[structure] identifier acts as
|
||||
a simple wrapper around @racket[tagged] which supplies @racket[untagged] as the
|
||||
tag name.
|
||||
|
||||
@chunk[<require-modules>
|
||||
(require "structure.hl.rkt")]
|
||||
|
||||
Constructors are defined as tagged structures containing a single field, called
|
||||
@racket[values]. The @racket[constructor] macro, defined in
|
||||
@filepath{"constructor.hl.rkt"} accepts a rich syntax for creating constructor
|
||||
instances containing multiple values, associated with the tag name. The values
|
||||
are aggregated in a list, which is stored within the @racket[values] field of
|
||||
the tagged structure used to implement the constructor. The @racket[constructor]
|
||||
identifier is therefore nothing more than syntactic sugar for @racket[tagged].
|
||||
It relies on the @racketmodname[xlist] library, which provides a rich syntax for
|
||||
expressing the complex list types, as well as the corresponding match pattern.
|
||||
|
||||
@chunk[<require-modules>
|
||||
(require "constructor.hl.rkt")]
|
||||
|
||||
For convenience, we write a @tc[variant] form, which is a thin wrapper against
|
||||
the union type of several constructors and tagged structures,
|
||||
@tc[(U constructor-or-tagged …)].
|
||||
|
||||
@chunk[<require-modules>
|
||||
(require "variant.hl.rkt")]
|
||||
|
||||
Finally, we directly include the row polymorphism features from
|
||||
@filepath{tagged-structure-low-level.hl.rkt}:
|
||||
|
||||
@chunk[<require-modules>
|
||||
(require "tagged-structure-low-level.hl.rkt")]
|
||||
|
||||
@;{Finally, we define a @tc[uniform-get] form, which can
|
||||
operate on @tc[tagged] structures. We also wrap the plain
|
||||
@tc[structure] form so that it instead returns a tagged
|
||||
structure, using a common tag for all plain structures. This
|
||||
allows us to rely on the invariant that @tc[uniform-get]
|
||||
always operates on data with the same shape (a constructor
|
||||
whose single value is a promise for a structure)@note{This
|
||||
avoids the risk of combinatorial explosion for the input
|
||||
type of @racket[uniform-get], when accessing a deeply nested
|
||||
field: allowing
|
||||
@racket[(U structure
|
||||
(constructor structure)
|
||||
(constructor (Promise structure)))]
|
||||
would result in a type of size @${n⁴}, with @${n} the depth
|
||||
of the accessed field.}}
|
||||
|
||||
@chunk[<*>
|
||||
(begin <require-modules>)
|
||||
(provide adt-init
|
||||
tagged
|
||||
tagged?
|
||||
define-tagged
|
||||
TaggedTop
|
||||
TaggedTop?
|
||||
tagged-supertype
|
||||
tagged-supertype*
|
||||
|
||||
structure
|
||||
structure?
|
||||
define-structure
|
||||
StructureTop
|
||||
StructureTop?
|
||||
structure-supertype
|
||||
|
||||
constructor
|
||||
constructor?
|
||||
define-constructor
|
||||
ConstructorTop
|
||||
ConstructorTop?
|
||||
|
||||
variant
|
||||
define-variant
|
||||
|
||||
constructor-values
|
||||
uniform-get
|
||||
|
||||
split
|
||||
#;(for-syntax split/type)
|
||||
merge
|
||||
#;(for-syntax merge/type)
|
||||
with+
|
||||
with!
|
||||
with!!
|
||||
#;(for-syntax tagged-struct-id?))]
|
24
phc-adt-lib/phc-adt/check-no-overlap.rkt
Normal file
24
phc-adt-lib/phc-adt/check-no-overlap.rkt
Normal file
|
@ -0,0 +1,24 @@
|
|||
#lang typed/racket
|
||||
(struct (A) st ([x : A]))
|
||||
(require phc-toolkit
|
||||
racket/format)
|
||||
|
||||
(struct variant-no-overlap ())
|
||||
(define-syntax/parse (check-no-overlap τ₁ τ₂)
|
||||
#`(let ()
|
||||
(λ ([x : (U (∩ τ₁ τ₂) variant-no-overlap)])
|
||||
(cond
|
||||
[(variant-no-overlap? x) #t]
|
||||
[(typecheck-fail τ₁
|
||||
#,(format "The types ~a and ~a seem to overlap"
|
||||
(syntax->datum #'τ₁)
|
||||
(syntax->datum #'τ₂)))]))
|
||||
(void)))
|
||||
|
||||
|
||||
(check-no-overlap (st Number) (st String))
|
||||
(check-no-overlap (st Negative-Integer) (st Byte))
|
||||
(check-not-tc
|
||||
#:message-regexp
|
||||
#rx"The types \\(st Nonpositive-Integer\\) and \\(st Byte\\) seem to overlap"
|
||||
(check-no-overlap (st Nonpositive-Integer) (st Byte)))
|
673
phc-adt-lib/phc-adt/constructor.hl.rkt
Normal file
673
phc-adt-lib/phc-adt/constructor.hl.rkt
Normal file
|
@ -0,0 +1,673 @@
|
|||
#lang hyper-literate typed/racket/base #:no-require-lang #:no-auto-require
|
||||
@(require racket/require
|
||||
scribble-math
|
||||
scribble-enhanced/doc
|
||||
(subtract-in scribble/core scribble-enhanced/doc)
|
||||
xlist/scribble-enhanced
|
||||
(for-label (lib "phc-adt/tagged-structure-low-level.hl.rkt")
|
||||
(lib "phc-adt/node-low-level.hl.rkt")
|
||||
(lib "phc-adt/tagged.hl.rkt")
|
||||
xlist
|
||||
racket/list
|
||||
(subtract-in racket/set type-expander)
|
||||
syntax/parse
|
||||
syntax/parse/experimental/template
|
||||
(subtract-in racket/syntax phc-toolkit)
|
||||
phc-toolkit/untyped-only
|
||||
(except-in (subtract-in typed/racket/base type-expander)
|
||||
values)
|
||||
(except-in phc-toolkit ?)
|
||||
multi-id
|
||||
type-expander
|
||||
type-expander/expander))
|
||||
@doc-lib-setup
|
||||
|
||||
@(unless-preexpanding
|
||||
(require (for-label (submod ".."))))
|
||||
|
||||
@title[#:style (with-html5 manual-doc-style)
|
||||
#:tag "constructor"
|
||||
#:tag-prefix "phc-adt/constructor"]{User API for constructors}
|
||||
|
||||
@(chunks-toc-prefix
|
||||
'("(lib phc-adt/scribblings/phc-adt-implementation.scrbl)"
|
||||
"phc-adt/constructor"))
|
||||
|
||||
@(table-of-contents)
|
||||
|
||||
@section{Introduction}
|
||||
|
||||
This file defines @tc[constructor], a form which allows tagging values, so that
|
||||
two otherwise identical values can be distinguished by the constructors used to
|
||||
wrap them. Coupled with the variants defined by this library, it implements a
|
||||
slight variation on the constructors and variants commonly found in other
|
||||
languages. The @tc[constructor] form is effectively a wrapper around @tc[tagged]
|
||||
structures, which stores all values within a single field named @tc[values].
|
||||
|
||||
The constructors defined in this library are "interned", meaning that two
|
||||
constructors in different files will be the same if they use same tag name. In
|
||||
other words, the tag of a constructor works in the same way as a symbol in
|
||||
Racket: unless otherwise specified, the same string of characters will always
|
||||
produce the same symbol, even across modules. The same goes for constructors:
|
||||
the same constructor name will always refer to the same type.
|
||||
|
||||
@section{The polyvalent identifier @racket[constructor]:
|
||||
type, match, builder and instance}
|
||||
|
||||
We define the @tc[constructor] macro which acts as a type, a match expander, and
|
||||
a constructor function (which can be called to create a tagged value, i.e. a
|
||||
constructor instance). It can also be directly given a value to directly produce
|
||||
a tagged value, i.e. a constructor instance.
|
||||
|
||||
@chunk[<constructor>
|
||||
(define-multi-id constructor
|
||||
#:type-expander (make-rest-transformer <type-expander>)
|
||||
#:match-expander (make-rest-transformer <match-expander>)
|
||||
#:call (make-rest-transformer <call-expander>))]
|
||||
|
||||
The @tc[constructor?] macro returns a predicate for the
|
||||
given constructor name, or checks if a value is an instance
|
||||
of the given constructor name. This form is implemented in
|
||||
@racket[<predicate>] below.
|
||||
|
||||
@chunk[<constructor?>
|
||||
(define-syntax constructor? (make-rest-transformer <predicate>))]
|
||||
|
||||
@section{Type-expander}
|
||||
|
||||
@chunk[#:save-as constructor-type-types-mixin <constructor-type-types-mixin>
|
||||
(define-eh-alternative-mixin types-mixin
|
||||
(pattern
|
||||
(~maybe/empty (~after name-order-point <name-after-field-error>
|
||||
τᵢ:type … {~lift-rest τ-rest}))))]
|
||||
|
||||
|
||||
@chunk[#:save-as name-after-field-error <name-after-field-error>
|
||||
"The name must appear before any value or type"]
|
||||
|
||||
@chunk[#:save-as name-id-mixin <name-id-mixin>
|
||||
(define-eh-alternative-mixin name-id-mixin
|
||||
(pattern
|
||||
(~once (~order-point name-order-point name:id))))]
|
||||
|
||||
@chunk[#:save-as ∀-mixin <∀-mixin>
|
||||
(define-eh-alternative-mixin ∀-mixin
|
||||
(pattern {~optional (~seq #:∀ ({~named-seq tvarᵢ :id …})
|
||||
(~global-or tvars?)
|
||||
#;(~global-or [no-types? #f])
|
||||
#;<∀-fail-no-types>)}))]
|
||||
|
||||
@; TODO: this depends on the order in which mixins are included, because
|
||||
@; no-types? may be declared by a mixin included later on.
|
||||
@chunk[#:save-as ∀-fail-no-types <∀-fail-no-types>
|
||||
#|
|
||||
{~post-fail (string-append "Expected [field:id type:expr] … or"
|
||||
" [field:id : type:expr] … because #:∀ is"
|
||||
" used")
|
||||
#:when (attribute no-types?)}
|
||||
|#]
|
||||
|
||||
The type-expander for @tc[constructor] expects:
|
||||
|
||||
@(require scribble/decode)
|
||||
|
||||
@itemlist[
|
||||
@item{The constructor's tag name, as defined for the tagged call expander in
|
||||
@(make-link-element
|
||||
#f
|
||||
(racket <name-id-mixin>)
|
||||
`(elem (prefixable "(lib phc-adt/scribblings/phc-adt-implementation.scrbl)"
|
||||
"phc-adt/tagged"
|
||||
"chunk:<name-id-mixin>:1:1"))):
|
||||
|
||||
@(name-id-mixin)}
|
||||
@item{An optional list of type variables, as defined for the tagged call
|
||||
expander in
|
||||
@(make-link-element
|
||||
#f
|
||||
(racket <∀-mixin>)
|
||||
`(elem (prefixable "(lib phc-adt/scribblings/phc-adt-implementation.scrbl)"
|
||||
"phc-adt/tagged"
|
||||
"chunk:<∀-mixin>:1:1"))):
|
||||
|
||||
@(∀-mixin)}
|
||||
@item{An optional list of types:
|
||||
|
||||
@(constructor-type-types-mixin)}]
|
||||
|
||||
The three elements can appear in any order, with one constraint: the name must
|
||||
appear before the first type. Not only does it make more sense semantically,
|
||||
but it also avoids ambiguities when some of the types are plain type
|
||||
identifiers.
|
||||
|
||||
@(name-after-field-error)
|
||||
|
||||
@chunk[<constructor-type-args-mixin>
|
||||
(define-eh-alternative-mixin constructor-type-seq-args-mixin
|
||||
#:define-syntax-class constructor-type-seq-args-syntax-class
|
||||
(pattern {~mixin name-id-mixin})
|
||||
(pattern {~mixin types-mixin})
|
||||
(pattern {~mixin ∀-mixin}))]
|
||||
|
||||
The type expander handles two cases: when type variables are present, it uses
|
||||
the low-level function @racket[tagged-∀-type!], otherwise it uses the low-level
|
||||
function @racket[tagged-type!]. The constructor contains a (possibly improper)
|
||||
list of values. The type of that list is expressed using the syntax of the
|
||||
@racketmodname[xlist] library.
|
||||
|
||||
@chunk[<type-expander>
|
||||
(λ/syntax-parse :constructor-type-seq-args-syntax-class
|
||||
(if (attribute tvars?)
|
||||
(tagged-∀-type! #'((tvarᵢ …) name [values (xlist τᵢ … . τ-rest)]))
|
||||
(tagged-type! #'(name [values (xlist τᵢ … . τ-rest)]))))]
|
||||
|
||||
@section{Match-expander}
|
||||
|
||||
@CHUNK[<match-expander>
|
||||
(syntax-parser
|
||||
[(name:id . pats)
|
||||
(tagged-match! #'(name [values (xlist . pats)]))])]
|
||||
|
||||
The match expander simply matches the given patterns against the constructor's
|
||||
single field, @racket[values]. The patterns will usually match one value each,
|
||||
but the @racket[xlist] pattern expander allows a more flexible syntax than the
|
||||
regular @racket[list] match pattern.
|
||||
|
||||
@section{Predicate}
|
||||
|
||||
The @racket[constructor?] macro expands to a predicate and accepts the same
|
||||
syntax as for the type expander, without polymorphic variables. Additionally the
|
||||
resulting type as expanded by @racket[xlist] must be a suitable argument to
|
||||
@racket[make-predicate].
|
||||
|
||||
@CHUNK[<predicate>
|
||||
(λ/syntax-parse (name:id . types)
|
||||
(tagged-predicate! #'(name [values (xList . types)])))]
|
||||
|
||||
@section{Instance creation}
|
||||
|
||||
The @racket[constructor] macro can return a builder function or an instance. It
|
||||
accepts the following syntaxes:
|
||||
|
||||
@chunk[#:save-as value-maybe-type <value-maybe-type>
|
||||
(define-syntax-class value-maybe-type
|
||||
(pattern [vᵢ :colon τᵢ:type] #:with aᵢ #'τᵢ #:with (tvarₖ …) #'())
|
||||
(pattern [:colon τᵢ:type vᵢ] #:with aᵢ #'τᵢ #:with (tvarₖ …) #'())
|
||||
(pattern vᵢ:literal-value
|
||||
#:with τᵢ #'vᵢ.type
|
||||
#:with aᵢ #'vᵢ.type
|
||||
#:with (tvarₖ …) #'())
|
||||
(pattern (~and vᵢ (~not #:rest))
|
||||
#:with τᵢ (gensym 'τ)
|
||||
#:attr aᵢ #f
|
||||
#:with (tvarₖ …) #'(τᵢ)))]
|
||||
|
||||
@CHUNK[#:save-as literal-value <literal-value>
|
||||
(define-syntax-class literal-value
|
||||
(pattern n:number #:with type #'n)
|
||||
(pattern s:str #:with type #'s)
|
||||
(pattern b:boolean #:with type #'b)
|
||||
(pattern c:char #:with type #'Char)
|
||||
(pattern ((~literal quote) v) #:with type (replace-chars #'v))
|
||||
(pattern v
|
||||
#:when (vector? (syntax-e #'v))
|
||||
#:with type (replace-chars #'v)))]
|
||||
|
||||
@chunk[#:save-as replace-chars <replace-chars>
|
||||
;https://github.com/racket/typed-racket/issues/434
|
||||
(define (replace-chars t)
|
||||
(cond [(syntax? t) (datum->syntax t
|
||||
(replace-chars (syntax-e t))
|
||||
t
|
||||
t)]
|
||||
[(pair? t) (list 'Pairof
|
||||
(replace-chars (car t))
|
||||
(replace-chars (cdr t)))]
|
||||
[(char? t) 'Char]
|
||||
[(vector? t) (cons 'Vector (map replace-chars
|
||||
(vector->list t)))]
|
||||
[(null? t) 'Null]
|
||||
[(number? t) t]
|
||||
[(string? t) t]
|
||||
[(boolean? t) t]
|
||||
(code:comment "Hope for the best.")
|
||||
(code:comment "We really should use a ∀ tvar instead.")
|
||||
[else (list 'quote t)]))]
|
||||
|
||||
@chunk[#:save-as infer-pat <infer-pat>
|
||||
(~after name-order-point <name-after-field-error>
|
||||
{~literal *})]
|
||||
|
||||
@CHUNK[#:save-as call-expander-infer-case <call-expander-cases>
|
||||
[(~no-order {~mixin ∀-mixin} {~mixin name-id-mixin} {~once <infer-pat>})
|
||||
#`(… (λ #:∀ (A ...) [l : A ... A]
|
||||
(#,(tagged-builder! #'(… (name [values (List A ... A)])))
|
||||
l)))]]
|
||||
|
||||
@chunk[#:save-as colon-pat <colon-pat>
|
||||
(~after name-order-point <name-after-field-error>
|
||||
:colon τᵢ …
|
||||
{~lift-rest {~and τ-rest ()}})]
|
||||
|
||||
@CHUNK[#:save-as call-expander-:-case <call-expander-cases>
|
||||
[(~no-order {~mixin ∀-mixin} {~mixin name-id-mixin} {~once <colon-pat>})
|
||||
(define-temp-ids "~a/arg" (τᵢ …))
|
||||
#`(λ #,@(when-attr tvars? #'(#:∀ (tvarᵢ …))) ([τᵢ/arg : τᵢ] …)
|
||||
(#,(tagged-builder! #'(name [values (List τᵢ …)]))
|
||||
(list τᵢ/arg …)))]]
|
||||
|
||||
@chunk[#:save-as !-pat <!-pat>
|
||||
(~after name-order-point <name-after-field-error>
|
||||
{~datum !} τᵢ … {~lift-rest τ-rest})]
|
||||
|
||||
@CHUNK[#:save-as call-expander-!-case <call-expander-cases>
|
||||
[(~no-order {~mixin ∀-mixin} {~mixin name-id-mixin} {~once <!-pat>})
|
||||
#`(λ [l : Any *]
|
||||
(#,(tagged-builder! #'(name [values (xList τᵢ … . τ-rest)]))
|
||||
(cast l (xlist τᵢ … . τ-rest))))]]
|
||||
|
||||
@chunk[#:save-as dcolon-pat <dcolon-pat>
|
||||
(~after name-order-point <name-after-field-error>
|
||||
{~datum ::} τᵢ … {~lift-rest τ-rest})]
|
||||
|
||||
@CHUNK[#:save-as call-expander-::-case <call-expander-cases>
|
||||
[(~no-order {~mixin ∀-mixin} {~mixin name-id-mixin} {~once <dcolon-pat>})
|
||||
(if (attribute tvars?)
|
||||
(tagged-builder! #'(name
|
||||
[values (xlist τᵢ … . τ-rest)]))
|
||||
(tagged-∀-builder! #'((tvarᵢ …)
|
||||
name
|
||||
[values (xList τᵢ … . τ-rest)])))]]
|
||||
|
||||
@CHUNK[#:save-as call-expander-values-case <call-expander-cases>
|
||||
[(~no-order {~mixin ∀-mixin}
|
||||
{~mixin name-id-mixin}
|
||||
(~maybe/empty
|
||||
(~after name-order-point <name-after-field-error>
|
||||
:value-maybe-type …
|
||||
<call-expander-rest>)))
|
||||
(define-temp-ids "~a/arg" (τᵢ …))
|
||||
(quasitemplate
|
||||
(#,(tagged-∀-builder! #'((tvarᵢ … tvarₖ … … tvar-rest …)
|
||||
name
|
||||
[values (xlist τᵢ … #:rest x-τ-rest)]))
|
||||
(list* {?? (ann vᵢ aᵢ) vᵢ}
|
||||
…
|
||||
{?? (ann v-rest a-rest) v-rest})))]]
|
||||
|
||||
@CHUNK[#:save-as call-expander-rest <call-expander-rest>
|
||||
(~either <call-expander-rest-keyword>
|
||||
<call-expander-empty-rest>
|
||||
<call-expander-dotted-rest>)]
|
||||
|
||||
@(define comment1 "pattern for the value, infers type for literals")
|
||||
@(define-for-syntax comment1 "pattern for the value, infers type for literals")
|
||||
@chunk[#:save-as call-expander-rest-keyword <call-expander-rest-keyword>
|
||||
(~as-rest #:rest
|
||||
(code:comment #,comment1)
|
||||
(~either (~and v-rest:literal-value
|
||||
{~with a-rest #'v-rest.type})
|
||||
(~and v-rest
|
||||
{~attr a-rest #f}))
|
||||
(~either (~and {~seq}
|
||||
{~with x-τ-rest (gensym 'x-τ-rest)}
|
||||
{~with (tvar-rest …) #'(x-τ-rest)})
|
||||
(~and (~seq :colon x-τ-rest)
|
||||
{~with (tvar-rest …) #'()})))]
|
||||
|
||||
@CHUNK[#:save-as call-expander-empty-rest <call-expander-empty-rest>
|
||||
(~seq
|
||||
(~lift-rest
|
||||
(~and ()
|
||||
{~with v-rest #'null}
|
||||
{~with a-rest #'Null}
|
||||
{~with x-τ-rest #'Null}
|
||||
{~with (tvar-rest …) #'()})))]
|
||||
|
||||
@CHUNK[#:save-as call-expander-dotted-rest <call-expander-dotted-rest>
|
||||
(~seq
|
||||
(~lift-rest
|
||||
(~either (~and v-rest:type-label
|
||||
(~with x-τ-rest #'v-rest.type)
|
||||
{~with a-rest #'v-rest.type}
|
||||
(~with (tvar-rest …) #'()))
|
||||
(~and v-rest:literal-value
|
||||
(~with x-τ-rest #'v-rest.type)
|
||||
{~with a-rest #'v-rest.type}
|
||||
(~with (tvar-rest …) #'()))
|
||||
(~and v-rest
|
||||
(~with x-τ-rest (gensym 'x-τ-rest))
|
||||
{~attr a-rest #f}
|
||||
(~with (tvar-rest …) #'(x-τ-rest))))))]
|
||||
|
||||
@CHUNK[#:save-as type-label-syntax-class <type-label-syntax-class>
|
||||
(define-syntax-class type-label
|
||||
#:attributes (type raw-type)
|
||||
(pattern v
|
||||
#:attr raw-type (syntax-property #'v-rest 'type-label)
|
||||
#:when (attribute raw-type)
|
||||
#:attr type (datum->syntax #'v-rest
|
||||
(attribute raw-type)
|
||||
#'v-rest)))]
|
||||
|
||||
@itemlist[
|
||||
@item{@racket[(constructor name *)], which returns a polymorphic builder
|
||||
function that infers the type of its arguments. All arguments are aggregated
|
||||
into a list with the inferred type for each element, and that list is used as
|
||||
the constructor's value.
|
||||
|
||||
@(infer-pat)
|
||||
|
||||
@(call-expander-infer-case)}
|
||||
@item{@racket[(constructor : τᵢ …)], which returns a builder function. This
|
||||
does not support the extended @racket[xlist] syntax, as Typed/Racket's
|
||||
function types are not expressive enough to support it.
|
||||
|
||||
@(colon-pat)
|
||||
|
||||
@(call-expander-:-case)}
|
||||
@item{@racket[(constructor ! . _xlist-type)], which returns a builder function
|
||||
expecting the values as a rest argument, and casts the list at runtime. The
|
||||
@racket[_xlist-type] must be a valid sequence of types for the type form of
|
||||
@racket[xlist], and the result must be a suitable argument to
|
||||
@racket[make-predicate].
|
||||
|
||||
@(!-pat)
|
||||
|
||||
@(call-expander-!-case)}
|
||||
@item{@racket[(constructor :: . _xlist-type)], which returns a builder function
|
||||
expecting the whole list of values as a single argument, and returns the
|
||||
constructor instance containing that list. The @racket[_xlist-type] must be a
|
||||
valid sequence of types for the type form of @racket[xlist].
|
||||
|
||||
@(dcolon-pat)
|
||||
|
||||
@(call-expander-::-case)}
|
||||
@item{@racket[(constructor _value-maybe-typeᵢ … . rest)], which returns an
|
||||
instance containing a (possibly improper) list with the given values and
|
||||
@racket[rest] as the tail of the list. If @racket[rest] is @racket[()], then
|
||||
the result is a proper list.
|
||||
|
||||
@;@(constructor-value-mixin)
|
||||
@(call-expander-values-case)
|
||||
|
||||
Each @racket[_value-maybe-typeᵢ] may be one of:
|
||||
@itemlist[
|
||||
@item{@racket[[valᵢ : τᵢ]]}
|
||||
@item{@racket[[: τᵢ valᵢ]]}
|
||||
@item{@racket[valᵢ]}]
|
||||
|
||||
@(value-maybe-type)
|
||||
|
||||
Literals are specially recognised so that their type is preserved with as much
|
||||
precision as possible:
|
||||
|
||||
@(literal-value)
|
||||
|
||||
As noted in Typed/Racket bug
|
||||
@hyperlink["https://github.com/racket/typed-racket/issues/434"]{#434}, literal
|
||||
characters are not currently recognised as belonging to their own singleton
|
||||
type. We therefore rewrite the type for quoted data to turn literal characters
|
||||
into the @racket[Char] type:
|
||||
|
||||
@(replace-chars)
|
||||
|
||||
Optionally, a rest element may be specified using the following syntax:
|
||||
@(call-expander-rest)
|
||||
|
||||
@(call-expander-rest-keyword)
|
||||
@(call-expander-empty-rest)
|
||||
@(call-expander-dotted-rest)
|
||||
|
||||
The last case depends on the @racket[type-label?] syntax class to recognise
|
||||
uses of the @elem[#:style 'tt "#{val : type}"] type annotation syntax from
|
||||
@racketmodname[typed/racket]. Typed/Racket enables that reader extension,
|
||||
which embeds the type into the value as a syntax property for later use by the
|
||||
type checker
|
||||
|
||||
@(type-label-syntax-class)}]
|
||||
|
||||
All four forms accept a @racket[#:∀ (tvarᵢ …)] specification, and the fourth
|
||||
injects a @racket[tvarᵢ] type variable for values for which no type is given.
|
||||
|
||||
@CHUNK[<call-expander>
|
||||
(syntax-parser
|
||||
<call-expander-cases>)]
|
||||
|
||||
@section{Defining shorthands for constructors with @racket[define-constructor]}
|
||||
|
||||
The @racket[define-constructor] macro binds an identifier to a type-expander,
|
||||
match-expander and call-expander for the constructor with the same name. It
|
||||
also defines a predicate for that constructor type.
|
||||
|
||||
@;; Copied over from tagged.hl.rkt without any change.
|
||||
|
||||
@chunk[#:save-as tag-kw-mixin <tag-kw-mixin>
|
||||
(define-eh-alternative-mixin tag-kw-mixin
|
||||
(pattern {~optional {~seq #:tag explicit-tag <default-tag-name>}}))]
|
||||
|
||||
@chunk[#:save-as tag-kw-mixin-default <default-tag-name>
|
||||
{~post-check
|
||||
{~bind [tag-name (or (attribute explicit-tag)
|
||||
#'name)]}}]
|
||||
|
||||
@chunk[#:save-as predicate?-mixin <predicate?-mixin>
|
||||
(define-eh-alternative-mixin predicate?-mixin
|
||||
(pattern {~optional {~seq #:? predicate? <default-name?>}}))]
|
||||
|
||||
@chunk[#:save-as predicate?-mixin-default <default-name?>
|
||||
{~post-check
|
||||
{~bind [name? (or (attribute predicate?)
|
||||
(format-id/record #'name "~a?" #'name))]}}]
|
||||
|
||||
Like @tc[define-tagged], the @tc[constructor] macro expects:
|
||||
|
||||
@itemlist[
|
||||
@item{The tagged structure's tag name, as defined for the call expander in
|
||||
@racket[<name-id-mixin>]}
|
||||
@item{An optional list of type variables, as defined for the call expander in
|
||||
@racket[<∀-mixin>]}
|
||||
@item{Optionally, the tag name to be used, specified with
|
||||
@racket[#:tag tag-name] as for @racket[define-tagged] in
|
||||
@secref["Defining_shorthands_with_define-tagged"
|
||||
#:tag-prefixes '("phc-adt/tagged")]:
|
||||
|
||||
@(tag-kw-mixin)
|
||||
|
||||
The tag name defaults to @racket[_name], i.e. the identifier currently being
|
||||
defined.
|
||||
|
||||
@(tag-kw-mixin-default)}
|
||||
@item{Optionally, a name for the predicate, specified with
|
||||
@racket[#:? predicate-name?] as for @racket[define-tagged] in
|
||||
@secref["Defining_shorthands_with_define-tagged"
|
||||
#:tag-prefixes '("phc-adt/tagged")]:
|
||||
|
||||
@(predicate?-mixin)
|
||||
|
||||
The predicate name defaults to @racket[_name?], where @racket[_name] is the
|
||||
identifier currently being defined.
|
||||
|
||||
@(predicate?-mixin-default)}]
|
||||
|
||||
Unlike @tc[define-tagged], which also expects a list of field names possibly
|
||||
annotated with a type, the @tc[constructor] macro instead expects a
|
||||
description of the list of values it contains. Three syntaxes are accepted:
|
||||
|
||||
@itemlist[
|
||||
@item{@(colon-pat)}
|
||||
@item{@(!-pat)}
|
||||
@item{@(dcolon-pat)}]
|
||||
|
||||
These syntaxes control how the call expander for the defined @racket[_name]
|
||||
works, and have the same meaning as in the call expander for
|
||||
@racket[constructor] (@racket[xlist], @racket[cast] and single-argument
|
||||
@racket[xlist]).
|
||||
|
||||
@chunk[<define-constructor>
|
||||
(define-syntax define-constructor
|
||||
(syntax-parser-with-arrows
|
||||
[(_ . (~no-order {~mixin name-id-mixin}
|
||||
{~mixin ∀-mixin}
|
||||
{~mixin tag-kw-mixin}
|
||||
{~mixin predicate?-mixin}
|
||||
(~once
|
||||
(~and (~seq type-decls …)
|
||||
(~either <colon-pat>
|
||||
<!-pat>
|
||||
<dcolon-pat>)))))
|
||||
#:with tvarᵢ→Any (stx-map (const #'Any) #'(tvarᵢ …))
|
||||
<normalize-type/define>
|
||||
(quasisyntax/top-loc stx
|
||||
(begin
|
||||
<multi-id/define>
|
||||
<predicate/define>))]))]
|
||||
|
||||
@chunk[<multi-id/define>
|
||||
(define-multi-id name
|
||||
#:type-expander (make-id+call-transformer <type-expander/define>)
|
||||
#:match-expander (make-rest-transformer <match-expander/define>)
|
||||
#:else <call-expander/define>)]
|
||||
|
||||
@; exact copy-paste from the type expander: TODO: factor it out.
|
||||
@CHUNK[<type-expander/define>
|
||||
#'(constructor tag-name
|
||||
#,@(when-attr tvars? #'(#:∀ (tvarᵢ …)))
|
||||
τᵢ … . τ-rest)]
|
||||
|
||||
@CHUNK[<call-expander/define>
|
||||
#'(constructor tag-name
|
||||
#,@(when-attr tvars? #'(#:∀ (tvarᵢ …)))
|
||||
type-decls …)]
|
||||
|
||||
In order to attach patterns to the @racket[xlist] type, pre-process the types
|
||||
using @racket[normalize-xlist-type].
|
||||
|
||||
@chunk[<normalize-type/define>
|
||||
#:with <with-normalize> (normalize-xlist-type #'(τᵢ … . τ-rest) stx)]
|
||||
|
||||
Once normalized, the types for the @racket[xlist] are all of the form
|
||||
@racket[τᵢ ^ {repeat …}], except for the rest type, which is always present
|
||||
including when it is @racket[Null], and is specified using
|
||||
@racket[#:rest rest-type].
|
||||
|
||||
@chunk[<with-normalize>
|
||||
({~seq normalized-τᵢ {~literal ^} (normalized-repeat …)} …
|
||||
#:rest normalized-rest)]
|
||||
|
||||
We then define an argument for the pattern expander corresponding to each type
|
||||
within the normalized sequence:
|
||||
|
||||
@chunk[<normalize-type/define>
|
||||
(define-temp-ids "~a/pat" (normalized-τᵢ …))]
|
||||
|
||||
The match expander expects these patterns and a rest pattern:
|
||||
|
||||
@CHUNK[<match-expander/define>
|
||||
(syntax-parser
|
||||
[({~var normalized-τᵢ/pat} … . {~either <match-rest-signature/define>})
|
||||
#'#,(tagged-match! #'(name [values <match-xlist/define>]))])]
|
||||
|
||||
The rest pattern can be specified either using a dotted notation if it is a
|
||||
single term, using @racket[#:rest pat-rest], or can be omitted in which case
|
||||
it defaults to matching @racket[null]. The following syntaxes are therefore
|
||||
accepted:
|
||||
|
||||
@chunk[<match-rest-signature/define>
|
||||
(#:rest pat-rest)
|
||||
(~and () {~bind [pat-rest #'(? null?)]})
|
||||
pat-rest:not-stx-pair]
|
||||
|
||||
The match expander produces an @racket[xlist] pattern using the given patterns
|
||||
and the optional rest pattern. The given patterns are repeated as within the
|
||||
type specification.
|
||||
|
||||
@chunk[<match-xlist/define>
|
||||
(and (? (make-predicate (xlist τᵢ … . τ-rest)))
|
||||
(split-xlist (list normalized-τᵢ/pat … pat-rest)
|
||||
τᵢ … . τ-rest))]
|
||||
|
||||
@CHUNK[<predicate/define>
|
||||
(define name?
|
||||
#,(if (attribute tvars?)
|
||||
(tagged-predicate!
|
||||
#'(tag-name [values ((xlist τᵢ … . τ-rest) tvarᵢ→Any)]))
|
||||
(tagged-predicate!
|
||||
#'(tag-name [values (xlist τᵢ … . τ-rest)]))))]
|
||||
|
||||
@; TODO: add a #:predicate-type option.
|
||||
|
||||
@section{Miscellanea}
|
||||
|
||||
@chunk[<constructor-values>
|
||||
(define-syntax constructor-values
|
||||
(make-id+call-transformer-delayed
|
||||
(λ () #'(λ-tagged-get-field values))))]
|
||||
|
||||
@CHUNK[<ConstructorTop?>
|
||||
(define-syntax ConstructorTop?
|
||||
(make-id+call-transformer-delayed
|
||||
(λ ()
|
||||
#`(struct-predicate
|
||||
#,(check-remembered-common!
|
||||
#'(always-remembered values))))))]
|
||||
|
||||
@CHUNK[<ConstructorTop>
|
||||
(define-type-expander (ConstructorTop stx)
|
||||
(syntax-case stx ()
|
||||
[id
|
||||
(identifier? #'id)
|
||||
#'((check-remembered-common!
|
||||
#'(always-remembered values))
|
||||
Any)]))]
|
||||
|
||||
@section{Putting it all together}
|
||||
|
||||
@chunk[<*>
|
||||
(require phc-toolkit
|
||||
"tagged.hl.rkt"
|
||||
"tagged-structure-low-level.hl.rkt"
|
||||
(only-in match-string [append match-append])
|
||||
type-expander
|
||||
xlist
|
||||
multi-id
|
||||
(for-syntax racket/base
|
||||
syntax/parse
|
||||
syntax/parse/experimental/template
|
||||
racket/contract
|
||||
racket/syntax
|
||||
racket/string
|
||||
racket/function
|
||||
racket/list
|
||||
type-expander/expander
|
||||
phc-toolkit/untyped
|
||||
extensible-parser-specifications))
|
||||
|
||||
(provide constructor-values
|
||||
constructor
|
||||
constructor?
|
||||
ConstructorTop
|
||||
ConstructorTop?
|
||||
define-constructor
|
||||
(for-syntax constructor-type-seq-args-syntax-class))
|
||||
|
||||
(begin-for-syntax
|
||||
(define-syntax-class not-stx-pair
|
||||
(pattern {~not (_ . _)}))
|
||||
<type-label-syntax-class>
|
||||
<name-id-mixin>
|
||||
<∀-mixin>
|
||||
<constructor-type-types-mixin>
|
||||
<constructor-type-args-mixin>
|
||||
<tag-kw-mixin>
|
||||
<predicate?-mixin>
|
||||
<replace-chars>
|
||||
<literal-value>
|
||||
<value-maybe-type>)
|
||||
|
||||
<constructor>
|
||||
<constructor?>
|
||||
<ConstructorTop>
|
||||
<ConstructorTop?>
|
||||
<define-constructor>
|
||||
<constructor-values>]
|
137
phc-adt-lib/phc-adt/ctx.hl.rkt
Normal file
137
phc-adt-lib/phc-adt/ctx.hl.rkt
Normal file
|
@ -0,0 +1,137 @@
|
|||
#lang hyper-literate typed/racket/base #:no-require-lang
|
||||
@(require scribble-enhanced/doc
|
||||
racket/require
|
||||
hyper-literate
|
||||
(for-label racket/format
|
||||
racket/list
|
||||
(subtract-in racket/contract typed/racket/base)
|
||||
phc-toolkit
|
||||
phc-toolkit/untyped-only
|
||||
remember
|
||||
(subtract-in typed/racket/base type-expander)
|
||||
type-expander))
|
||||
@doc-lib-setup
|
||||
|
||||
@title[#:style manual-doc-style
|
||||
#:tag "ctx"
|
||||
#:tag-prefix "phc-adt/ctx"
|
||||
]{Implementation of ADTs: syntax scopes}
|
||||
|
||||
@(chunks-toc-prefix
|
||||
'("(lib phc-adt/scribblings/phc-adt-implementation.scrbl)"
|
||||
"phc-adt/ctx"))
|
||||
|
||||
|
||||
Due to TR bug #399, structs declared by a macro do not work
|
||||
if the macro itself is declared in a separate module. This
|
||||
seems to be due to the extra scope added as pieces of syntax
|
||||
cross the module boundary. There is unfortunately no
|
||||
equivalent to @racket[syntax-local-introduce] that could be
|
||||
used to flip this module scope.
|
||||
|
||||
We therefore require the user to call
|
||||
@racket[(set-adt-context)] at the beginning of the file.
|
||||
This macro stores the scopes present where it was called in
|
||||
a mutable for-syntax variable:
|
||||
|
||||
@chunk[<adt-context>
|
||||
(define-for-syntax mutable-adt-context (box #f))]
|
||||
|
||||
These scopes are later used as the context for struct
|
||||
identifiers:
|
||||
|
||||
@chunk[<ctx-introduce>
|
||||
(define-for-syntax (ctx-introduce id)
|
||||
(unless (unbox mutable-adt-context)
|
||||
(raise-syntax-error 'adt
|
||||
(~a "(adt-init) must be called in the"
|
||||
" file (or REPL). ") id))
|
||||
(struct-identifier-fresh-introducer
|
||||
(replace-context (syntax-local-introduce
|
||||
(unbox mutable-adt-context))
|
||||
id)))]
|
||||
|
||||
The @racket[(set-adt-context)] macro should be called at
|
||||
the beginning of the file or typed in the REPL before using
|
||||
structures. It simply stores the syntax used to call it in
|
||||
@racket[mutable-adt-context].
|
||||
|
||||
@chunk[<adt-context>
|
||||
(define-for-syntax (set-adt-context ctx)
|
||||
(set-box! mutable-adt-context ctx))
|
||||
|
||||
(define-syntax (set-adt-context-macro stx)
|
||||
(syntax-case stx ()
|
||||
[(_ ctx)
|
||||
(begin (set-box! mutable-adt-context #'ctx)
|
||||
#'(void))]))]
|
||||
|
||||
For debugging purposes, we provide a macro and a for-syntax
|
||||
function which show the current ADT context (i.e. the list of
|
||||
scopes).
|
||||
|
||||
@chunk[<adt-context>
|
||||
(define-for-syntax (debug-show-adt-context)
|
||||
(displayln
|
||||
(hash-ref (syntax-debug-info (unbox mutable-adt-context))
|
||||
'context)))
|
||||
(define-syntax (debug-show-adt-context-macro stx)
|
||||
(debug-show-adt-context)
|
||||
#'(define dummy (void)))]
|
||||
|
||||
The @tc[struct] identifiers are introduced in a fresh scope
|
||||
@note{Due to TR bug #399, this feature is temporarily
|
||||
disabled, until the bug is fixed.}, so that they do not
|
||||
conflict with any other user value.
|
||||
|
||||
@chunk[<fresh-introducer>
|
||||
(define-for-syntax struct-identifier-fresh-introducer
|
||||
(λ (x) x) #;(make-syntax-introducer))]
|
||||
|
||||
We provide two ways of checking whether @racket[set-adt-context] was called:
|
||||
@racket[(adt-context?)] returns a boolean, while @racket[(check-adt-context)]
|
||||
raises an error when @racket[set-adt-context] has not been called.
|
||||
|
||||
@chunk[<adt-context?>
|
||||
(define-for-syntax (adt-context?)
|
||||
(true? (unbox mutable-adt-context)))]
|
||||
|
||||
@chunk[<check-adt-context>
|
||||
(define-for-syntax (check-adt-context)
|
||||
(unless (adt-context?)
|
||||
(raise-syntax-error 'phc-adt
|
||||
(string-append
|
||||
"adt-init must be called before"
|
||||
" using the features in phc-adt"))))]
|
||||
|
||||
@section{Putting it all together}
|
||||
|
||||
@chunk[<*>
|
||||
(begin
|
||||
(require (for-syntax racket/base
|
||||
racket/syntax
|
||||
racket/set
|
||||
racket/list
|
||||
racket/format
|
||||
phc-toolkit/untyped
|
||||
syntax/strip-context)
|
||||
racket/require-syntax
|
||||
type-expander
|
||||
phc-toolkit
|
||||
remember))
|
||||
|
||||
(provide (for-syntax set-adt-context)
|
||||
set-adt-context-macro
|
||||
debug-show-adt-context-macro)
|
||||
|
||||
(begin-for-syntax
|
||||
(provide debug-show-adt-context
|
||||
adt-context?
|
||||
check-adt-context
|
||||
ctx-introduce))
|
||||
|
||||
<adt-context>
|
||||
<fresh-introducer>
|
||||
<ctx-introduce>
|
||||
<adt-context?>
|
||||
<check-adt-context>]
|
14
phc-adt-lib/phc-adt/declarations.rkt
Normal file
14
phc-adt-lib/phc-adt/declarations.rkt
Normal file
|
@ -0,0 +1,14 @@
|
|||
#lang racket
|
||||
|
||||
(require (prefix-in tr: typed/racket)
|
||||
remember
|
||||
(submod "tagged-structure-low-level.hl.rkt" pre-declare)
|
||||
"ctx.hl.rkt")
|
||||
(provide remembered!
|
||||
(rename-out [new-#%module-begin #%module-begin]))
|
||||
|
||||
(define-syntax-rule (new-#%module-begin . body)
|
||||
(tr:#%module-begin
|
||||
(tr:begin . body)
|
||||
(set-adt-context-macro here)
|
||||
(pre-declare-all-tagged-structure-structs)))
|
3
phc-adt-lib/phc-adt/main.rkt
Normal file
3
phc-adt-lib/phc-adt/main.rkt
Normal file
|
@ -0,0 +1,3 @@
|
|||
#lang typed/racket
|
||||
(require "adt.hl.rkt")
|
||||
(provide (all-from-out "adt.hl.rkt"))
|
413
phc-adt-lib/phc-adt/node-low-level.hl.rkt
Normal file
413
phc-adt-lib/phc-adt/node-low-level.hl.rkt
Normal file
|
@ -0,0 +1,413 @@
|
|||
#lang hyper-literate typed/racket/base #:no-require-lang #:no-auto-require
|
||||
@(require scribble-enhanced/doc
|
||||
racket/require
|
||||
hyper-literate
|
||||
(for-label racket/format
|
||||
racket/promise
|
||||
racket/list
|
||||
type-expander
|
||||
(except-in (subtract-in typed/racket/base type-expander)
|
||||
values)
|
||||
(only-in racket/base values)
|
||||
(subtract-in racket/contract typed/racket/base)
|
||||
phc-toolkit
|
||||
phc-toolkit/untyped-only
|
||||
remember))
|
||||
@(unless-preexpanding
|
||||
(require (for-label (submod ".."))))
|
||||
@doc-lib-setup
|
||||
|
||||
@title[#:style manual-doc-style
|
||||
#:tag "node-low-level"
|
||||
#:tag-prefix "phc-adt/node-low-level"
|
||||
]{Implementation of nodes: printing and equality}
|
||||
|
||||
@(chunks-toc-prefix
|
||||
'("(lib phc-adt/scribblings/phc-adt-implementation.scrbl)"
|
||||
"phc-adt/node-low-level"))
|
||||
|
||||
This section discusses the implementation of @tc[prop:custom-write] and
|
||||
@tc[prop:equal+hash] for nodes.
|
||||
|
||||
@(table-of-contents)
|
||||
|
||||
@section{Printing nodes}
|
||||
|
||||
To avoid printing large and confusing swathes of data when a node is displayed,
|
||||
we only print its constituents up to a certain depth. The
|
||||
@tech[#:doc '(lib "scribblings/guide/guide.scrbl")]{parameter}
|
||||
@tc[write-node-depth] controls the depth for printing nested nodes.
|
||||
|
||||
@CHUNK[<write-node-depth>
|
||||
(define write-node-depth (make-parameter 1))]
|
||||
|
||||
The @tc[make-node-writer] macro expands to a procedure which prints a node with
|
||||
the given name and fields. If the @racket[write-node-depth] is @racket[0], then
|
||||
the contents of the node are elided, and only its name is printed, so that the
|
||||
resulting printed representation is @racket["(node name …)"] with an actual
|
||||
ellipsis character.
|
||||
|
||||
@CHUNK[<node-custom-write>
|
||||
(define-syntax/parse (make-node-writer pid name fieldᵢ …)
|
||||
#'(λ (self out mode)
|
||||
(if (> (write-node-depth) 0)
|
||||
(parameterize ([write-node-depth (sub1 (write-node-depth))])
|
||||
(fprintf out
|
||||
"(node ~a ~a)"
|
||||
'name
|
||||
(string-join (list <format-field> …) " ")))
|
||||
(fprintf out "(node ~a …)" 'name))))]
|
||||
|
||||
Each field is formatted as @tc[[fieldᵢ valueᵢ]]. Copy-pasting the whole printed
|
||||
form will not form a valid expression which would be @tc[equal?] to the
|
||||
original. This limitation is deliberate: a node will often refer to many other
|
||||
nodes, and a stand-alone representation of such a node would result in a very
|
||||
large printed form. Instead, the user should call the @tc[serialize-graph]
|
||||
macro, which will produce a complete, canonical @note{The representation is
|
||||
canonical so long as unordered sets or hash tables are not used as part of the
|
||||
node's contents. In that case, the printed form is canonical modulo the order
|
||||
of elements within the set or hash table. Once executed, it will nevertheless
|
||||
produce a node which is @racket[equal?] to the original.} and self-contained
|
||||
representation of the node.
|
||||
|
||||
@chunk[<format-field>
|
||||
(format "[~a ~a]" 'fieldᵢ (force ((struct-accessor pid fieldᵢ) self)))]
|
||||
|
||||
@section{Comparing and hashing nodes}
|
||||
|
||||
Nodes are represented like tagged structures, but contain an extra @tc[raw]
|
||||
field. The @tc[raw] field contains a low-level representation of the node, which
|
||||
is used to implement node equality. The low-level representation uses the
|
||||
@tc[raw-node] Racket @racket[struct]. It contains two fields, @tc[database] and
|
||||
@tc[index]. The first is the database of nodes, as created by the graph
|
||||
construction macro. It contains one vector of nodes per node type. The second is
|
||||
a logical pointer into that database, consisting of the node's type's name,
|
||||
represented as a symbol, and an offset within the corresponding vector,
|
||||
represented as an @tc[Index].
|
||||
|
||||
@chunk[<raw-node>
|
||||
(struct/props (D I) raw-node ([database : D] [index : I]) #:transparent
|
||||
<raw-node-equality>)]
|
||||
|
||||
A regular with-promises node can have several in-memory representations which
|
||||
are not pointer-equal. This is due to the fact that the contents of node fields
|
||||
are wrapped with promises, and accessing the node via two distinct paths will
|
||||
yield two copies, each with fresh promises. We therefore use the @tc[raw-node]
|
||||
as a proxy for pointer equality: we know for sure that two nodes are exactly the
|
||||
same if the @tc[database] and @tc[index] is the same for both nodes.
|
||||
|
||||
@chunk[<raw-node-equality>
|
||||
#:property prop:equal+hash
|
||||
(list (λ (a b r)
|
||||
(and (raw-node? a)
|
||||
(raw-node? b)
|
||||
(eq? (raw-node-database a) (raw-node-database b))
|
||||
(equal? (raw-node-index a) (raw-node-index b))))
|
||||
(λ (a r)
|
||||
(fxxor (eq-hash-code (raw-node-database a))
|
||||
(r (raw-node-index a))))
|
||||
(λ (a r)
|
||||
(fxxor (eq-hash-code (raw-node-database a))
|
||||
(r (raw-node-index a)))))]
|
||||
|
||||
The following function can then be used to test if two nodes are the same, based
|
||||
on the contents of their @tc[raw] field:
|
||||
|
||||
@chunk[<same-node?>
|
||||
(define (same-node? a b)
|
||||
(and ((struct-predicate node-id) a)
|
||||
((struct-predicate node-id) b)
|
||||
(equal? ((struct-accessor node-id raw) a)
|
||||
((struct-accessor node-id raw) b))))]
|
||||
|
||||
To detect cycles within the graph while implementing node equality, we use a
|
||||
global hash table tracking which nodes have already been visited.
|
||||
|
||||
@chunk[<seen-hash-table>
|
||||
(define seen-nodes
|
||||
: (Parameterof (U #f (HashTable (raw-node Any Any) Any)))
|
||||
(make-parameter #f))]
|
||||
|
||||
The current implementation uses a mutable hash table. It is only initialised
|
||||
when @tc[equal?] starts comparing two nodes, so that references to nodes are not
|
||||
kept once @tc[equal?] finished running. However, in theory, an immutable hash
|
||||
table could be threaded through all the recursive calls to @tc[equal?].
|
||||
Unfortunately, the recursive equality function supplied by Racket when
|
||||
implementing @tc[prop:equal+hash] does not accept an extra parameter to thread
|
||||
state throughout the recursion. It would therefore be necessary to re-implement
|
||||
the algorithm used by Racket's @tc[equal?] as described by
|
||||
@cite[adams2008scheme-equality] tailored to the comparison of data structures
|
||||
with high-level logical cycles. To be correct, such a re-implementation would
|
||||
however need to access the @tc[prop:equal+hash] property of other structs, but
|
||||
Racket provides no public predicate or accessor for that property. Therefore,
|
||||
although it would, in theory, be possible to implement node equality without
|
||||
mutable state, Racket's library does not offer the primitives needed to build
|
||||
it. We therefore settle on using a global, mutable hash table, which will exist
|
||||
only during the execution of @tc[equal?].
|
||||
|
||||
@chunk[<node-equal+hash>
|
||||
(define-syntax/parse
|
||||
(make-node-comparer common-id node-id name fieldᵢ …)
|
||||
(define-temp-ids "~a/τ" (fieldᵢ …))
|
||||
#'(let ()
|
||||
<same-node?>
|
||||
<find-in-table>
|
||||
<node-hash>
|
||||
(list <node-equal?>
|
||||
<node-equal-hash-code>
|
||||
<node-equal-secondary-hash-code>)))]
|
||||
|
||||
@subsection{Hashing nodes}
|
||||
|
||||
@tc[equal-hash-code] and @tc[equal-secondary-hash-code] are implemented via
|
||||
a single function @tc[node-hash], the only difference being the function used to
|
||||
recursively compute the hash of sub-elements.
|
||||
|
||||
@chunk[<node-equal-hash-code>
|
||||
(λ (a rec-equal-hash-code)
|
||||
(node-hash a rec-equal-hash-code))]
|
||||
|
||||
@chunk[<node-equal-secondary-hash-code>
|
||||
(λ (a rec-equal-secondary-hash-code)
|
||||
(node-hash a rec-equal-secondary-hash-code))]
|
||||
|
||||
It would be desirable to implement hashing in the following way: if the current
|
||||
node is already present in a hash table of seen nodes, but is not @tc[eq?] to
|
||||
that copy, then the racket hash function is called on the already-seen node.
|
||||
Otherwise, if the node has never been seen, or if it is @tc[eq?] to the seen
|
||||
node, the hash code is computed.
|
||||
|
||||
The problem with this approach is that it introduces an intermediate recursive
|
||||
call to Racket's hashing function. When Racket's hashing function is applied to
|
||||
a structure with the @tc[prop:equal+hash] property, it does @emph{not}
|
||||
return the result of the struct's hash implementation unmodified.
|
||||
|
||||
For example, the code below implements a struct @tc[s] with no fields, which
|
||||
computes its hash code by recursively calling Racket's hashing function on other
|
||||
(unique) instances of @tc[s], and returns the constant @tc[1] at a certain
|
||||
depth. The custom hashing function does not alter in any way the result returned
|
||||
by Racket's hashing function, however we can see that the hash for the same
|
||||
instance of @tc[s] depends on the number of recursive calls to Racket's hashing
|
||||
function @tc[r]. This simple experiment seems to suggest that Racket adds
|
||||
@tc[50] at each step, but this is not something that can be relied upon.
|
||||
|
||||
@(require scribble/eval)
|
||||
@defs+int[
|
||||
{(define recursion-depth (make-parameter #f))
|
||||
(struct s (x) #:transparent
|
||||
#:property prop:equal+hash
|
||||
(list (λ (a b r) (error "Not implemented"))
|
||||
(λ (a r)
|
||||
(if (= 0 (recursion-depth))
|
||||
1
|
||||
(parameterize ([recursion-depth (sub1 (recursion-depth))])
|
||||
(r (s (gensym))))))
|
||||
(λ (a r) (error "Not implemented"))))
|
||||
(define s-instance (s 'x))}
|
||||
(parameterize ([recursion-depth 0])
|
||||
(equal-hash-code s-instance))
|
||||
(parameterize ([recursion-depth 1])
|
||||
(equal-hash-code s-instance))
|
||||
(parameterize ([recursion-depth 2])
|
||||
(equal-hash-code s-instance))]
|
||||
|
||||
Since the order of traversal of the nodes is not fixed in the presence of sets
|
||||
and hash tables, we need to make sure that the recursion depth at which a node's
|
||||
hash is computed is constant. We achieve this by @emph{always} calling Racket's
|
||||
hash function on the already-seen node from the hash table, even if was inserted
|
||||
just now. To distinguish between the current node and the already-seen node from
|
||||
the hash table, we remove the contents of the node's @tc[raw] field, and replace
|
||||
them with a special marker.
|
||||
|
||||
@chunk[<node-hash>
|
||||
(: node-hash (∀ (fieldᵢ/τ …)
|
||||
(→ (node-id fieldᵢ/τ … Any Any) (→ Any Fixnum) Fixnum)))
|
||||
(define (node-hash nd racket-recur-hash)
|
||||
(if (eq? (raw-node-database ((struct-accessor node-id raw) nd))
|
||||
'unique-copy)
|
||||
<compute-hash>
|
||||
<hash-init-table-and-recur>))]
|
||||
|
||||
When the node's @tc[raw] field does not indicate @tc['unique-copy], we first
|
||||
initialise the hash table if needed, then recursively call
|
||||
@tc[racket-recur-hash] on the unique copy of the node:
|
||||
|
||||
@chunk[<hash-init-table-and-recur>
|
||||
(let ([seen-table (or (seen-nodes)
|
||||
((inst make-hash (raw-node Any Any) Any)))])
|
||||
(parameterize ([seen-nodes seen-table])
|
||||
(racket-recur-hash (find-in-table seen-table nd))))]
|
||||
|
||||
To obtain the unique copy of the node, we look it up in the hash table, creating
|
||||
it and adding it to the hash table if the current node is not already present
|
||||
there:
|
||||
|
||||
@chunk[<find-in-table>
|
||||
(: find-in-table (∀ (fieldᵢ/τ …)
|
||||
(→ (HashTable (raw-node Any Any) Any)
|
||||
(node-id fieldᵢ/τ … Any Any)
|
||||
Any)))
|
||||
(define (find-in-table seen-table nd)
|
||||
(hash-ref! seen-table
|
||||
((struct-accessor node-id raw) nd)
|
||||
(λ () <make-unique-copy-node>)))]
|
||||
|
||||
To create a unique copy of the node, we create a new instance of the node's
|
||||
struct, and copy over all the fields except for the @tc[raw] field, whose value
|
||||
becomes @tc['unique-copy].
|
||||
|
||||
@chunk[<make-unique-copy-node>
|
||||
((struct-constructor node-id) ((struct-accessor common-id fieldᵢ) nd)
|
||||
…
|
||||
(raw-node 'unique-copy 'unique-copy))]
|
||||
|
||||
The hash code is finally computed by combining the hash code for each field's
|
||||
contents (after forcing it). The node's tag name is also hashed, and added to
|
||||
the mix.
|
||||
|
||||
@chunk[<compute-hash>
|
||||
(combine-hash-codes
|
||||
(racket-recur-hash 'name)
|
||||
(racket-recur-hash (force ((struct-accessor common-id fieldᵢ) nd)))
|
||||
…)]
|
||||
|
||||
To combine hash codes, we simply compute their @elem[#:style 'tt]{xor}. Later
|
||||
versions of this library may use more sophisticated mechanisms.
|
||||
|
||||
@chunk[<combine-hash-codes>
|
||||
(: combine-hash-codes (→ Fixnum * Fixnum))
|
||||
(define (combine-hash-codes . fixnums)
|
||||
(apply fxxor fixnums))]
|
||||
|
||||
@subsection{Caching node equality}
|
||||
|
||||
We provide a mechanism at run-time to cache the result of equality tests
|
||||
within a limited dynamic scope. The graph generation procedure can coalesce
|
||||
nodes which are @racket[equal?], which means that it needs to perform a
|
||||
significant number of equality comparisons, and can therefore benefit from
|
||||
caching the result of inner equality tests during the execution of the
|
||||
coalescing operation.
|
||||
|
||||
@chunk[<equality-cache>
|
||||
(define equality-cache
|
||||
: (Parameterof (U #f (HashTable (Pairof (raw-node Any Any)
|
||||
(raw-node Any Any))
|
||||
Boolean)))
|
||||
(make-parameter #f))]
|
||||
|
||||
The @racket[with-node-equality-cache] form executes its body while enabling
|
||||
caching of the result of direct and recursive calls to @racket[equal?] on
|
||||
nodes.
|
||||
|
||||
@chunk[<with-node-equality-cache>
|
||||
(define-syntax-rule (with-node-equality-cache . body)
|
||||
(parameterize ([equality-cache (or (equality-cache)
|
||||
<make-equality-cache>)])
|
||||
. body))]
|
||||
|
||||
If necessary, a new equality cache is created, unless
|
||||
@racket[with-node-equality-cache] is used within the dynamic extent of another
|
||||
use of itself.
|
||||
|
||||
@chunk[<make-equality-cache>
|
||||
((inst make-hash (Pairof (raw-node Any Any) (raw-node Any Any)) Any))]
|
||||
|
||||
When comparing two nodes, we first check whether an equality cache is
|
||||
installed. If so, we attempt to query the cache, and memoize the result of the
|
||||
comparison when the pair of values is not already in the cache.
|
||||
|
||||
@chunk[<memoize-equality>
|
||||
(λ (result-thunk)
|
||||
(let ([e-cache (equality-cache)])
|
||||
(if e-cache
|
||||
(cond
|
||||
[(hash-has-key? e-cache (cons a-raw b-raw))
|
||||
(hash-ref e-cache (cons a-raw b-raw))]
|
||||
[(hash-has-key? e-cache (cons b-raw a-raw))
|
||||
(hash-ref e-cache (cons b-raw a-raw))]
|
||||
[else
|
||||
(let ([result (result-thunk)])
|
||||
(hash-set! e-cache (cons a-raw b-raw) result)
|
||||
result)])
|
||||
(result-thunk))))]
|
||||
|
||||
@subsection{Comparing nodes for equality}
|
||||
|
||||
We implement equality following the same architecture as for hash codes, but
|
||||
check that both nodes are already unique copies. In addition, the implementation
|
||||
of @tc[equal?] checks that both values are of the node's type.
|
||||
|
||||
@chunk[<node-equal?>
|
||||
(λ (a b racket-recur-equal?)
|
||||
(and ((struct-predicate node-id) a)
|
||||
((struct-predicate node-id) b)
|
||||
(let ([a-raw ((struct-accessor node-id raw) a)]
|
||||
[b-raw ((struct-accessor node-id raw) b)])
|
||||
(if (and (eq? (raw-node-database a-raw) 'unique-copy)
|
||||
(eq? (raw-node-database b-raw) 'unique-copy))
|
||||
<compare>
|
||||
(or (same-node? a b)
|
||||
(<memoize-equality>
|
||||
(λ () <equality-init-table-and-recur>)))))))]
|
||||
|
||||
When either or both of the node's @tc[raw] field do not indicate
|
||||
@tc['unique-copy], we first initialise the hash table if needed, then
|
||||
recursively call @tc[racket-recur-hash] on the unique copy of each node:
|
||||
|
||||
@chunk[<equality-init-table-and-recur>
|
||||
(let ([seen-table (or (seen-nodes)
|
||||
((inst make-hash (raw-node Any Any) Any)))])
|
||||
(parameterize ([seen-nodes seen-table])
|
||||
(racket-recur-equal? (find-in-table seen-table a)
|
||||
(find-in-table seen-table b))))]
|
||||
|
||||
The nodes are compared pointwise, checking each pair of fields for equality,
|
||||
after forcing both:
|
||||
|
||||
@chunk[<compare>
|
||||
(and (racket-recur-equal? (force ((struct-accessor common-id fieldᵢ) a))
|
||||
(force ((struct-accessor common-id fieldᵢ) b)))
|
||||
…)]
|
||||
|
||||
@chunk[<*>
|
||||
(require racket/promise
|
||||
racket/string
|
||||
racket/require
|
||||
phc-toolkit
|
||||
remember
|
||||
typed-struct-props
|
||||
(for-syntax racket/base
|
||||
racket/syntax
|
||||
racket/list
|
||||
racket/set
|
||||
racket/format
|
||||
(subtract-in syntax/stx phc-toolkit/untyped)
|
||||
syntax/parse
|
||||
phc-toolkit/untyped))
|
||||
|
||||
(provide make-node-comparer
|
||||
make-node-writer
|
||||
raw-node
|
||||
write-node-depth
|
||||
with-node-equality-cache)
|
||||
|
||||
<equality-cache>
|
||||
<with-node-equality-cache>
|
||||
<seen-hash-table>
|
||||
<write-node-depth>
|
||||
<node-custom-write>
|
||||
<raw-node>
|
||||
<combine-hash-codes>
|
||||
<node-equal+hash>]
|
||||
|
||||
@define[adams2008scheme-equality
|
||||
(string-append "Efficient nondestructive equality checking for trees"
|
||||
" and graphs, Adams and Dybvig, 2008")]
|
||||
@bibliography[
|
||||
@bib-entry[#:key adams2008scheme-equality
|
||||
#:title @list{Efficient nondestructive equality checking for trees
|
||||
and graphs in @emph{ACM Sigplan Notices} (Vol. 43, No. 9)
|
||||
pp. 179–188}
|
||||
#:date "2008"
|
||||
#:author "Michael D. Adams and R. Kent Dybvig"
|
||||
#:url "http://www.cs.indiana.edu/~dyb/pubs/equal.pdf"]]
|
124
phc-adt-lib/phc-adt/structure.hl.rkt
Normal file
124
phc-adt-lib/phc-adt/structure.hl.rkt
Normal file
|
@ -0,0 +1,124 @@
|
|||
#lang hyper-literate typed/racket/base #:no-require-lang #:no-auto-require
|
||||
@(require racket/require
|
||||
scribble-math
|
||||
scribble-enhanced/doc
|
||||
(for-label phc-toolkit
|
||||
(lib "phc-adt/tagged.hl.rkt")
|
||||
multi-id
|
||||
racket/base
|
||||
phc-toolkit/untyped-only))
|
||||
@doc-lib-setup
|
||||
|
||||
@(unless-preexpanding
|
||||
(require (for-label (submod ".."))))
|
||||
|
||||
@title[#:style (with-html5 manual-doc-style)
|
||||
#:tag "structure"
|
||||
#:tag-prefix "phc-adt/structure"
|
||||
]{User API for untagged structures}
|
||||
|
||||
@(chunks-toc-prefix
|
||||
'("(lib phc-adt/scribblings/phc-adt-implementation.scrbl)"
|
||||
"phc-adt/structure"))
|
||||
|
||||
@(table-of-contents)
|
||||
|
||||
@section{Introduction}
|
||||
|
||||
Untagged structures are implemented exactly like @racket[tagged] structures,
|
||||
except that they always use the @racket[untagged] tag name.
|
||||
|
||||
@chunk[<structure>
|
||||
(define-multi-id structure
|
||||
#:type-expander <expand-to-tagged>
|
||||
#:match-expander <expand-to-tagged>
|
||||
#:call <expand-to-tagged>)]
|
||||
|
||||
All three cases simply expand to
|
||||
@racket[(tagged untagged . _original-arguments)].
|
||||
|
||||
@chunk[<expand-to-tagged>
|
||||
(λ/syntax-case (_ . _original-arguments) ()
|
||||
(syntax/top-loc stx
|
||||
(tagged untagged . _original-arguments)))]
|
||||
|
||||
The @racket[structure?] predicate is implemented in the same way:
|
||||
|
||||
@chunk[<structure?>
|
||||
(define-syntax structure?
|
||||
(λ/syntax-case (_ . _original-arguments) ()
|
||||
(syntax/top-loc stx
|
||||
(tagged? untagged . _original-arguments))))]
|
||||
|
||||
@section{Defining untagged structures with @racket[define-structure]}
|
||||
|
||||
The @racket[define-structure] expands to the
|
||||
@racket[(define-tagged #:tag untagged . _original-arguments)], which uses
|
||||
@racket[define-tagged] but forces the tag name to be @racket[untagged].
|
||||
|
||||
@chunk[<define-structure>
|
||||
(define-syntax/case (define-structure . _original-arguments) ()
|
||||
(syntax/top-loc stx
|
||||
(define-tagged #:tag untagged . _original-arguments)))]
|
||||
|
||||
@section{Implementation of @racket[StructureTop] and @racket[StructureTop?]}
|
||||
|
||||
The @racket[StructureTop?] predicate is defined in terms of
|
||||
@racket[tagged-any-fields-predicate]:
|
||||
|
||||
@CHUNK[<StructureTop?>
|
||||
(define-syntax StructureTop?
|
||||
(make-id+call-transformer-delayed
|
||||
(λ () (tagged-any-fields-predicate #'untagged))))]
|
||||
|
||||
Similarly, the @racket[StructureTop] type is defined using
|
||||
@racket[tagged-any-fields-type]:
|
||||
|
||||
@CHUNK[<StructureTop>
|
||||
(define-type-expander (StructureTop stx)
|
||||
(syntax-case stx ()
|
||||
[id
|
||||
(identifier? #'id)
|
||||
(tagged-any-fields-type #'untagged)]))]
|
||||
|
||||
@section{Supertypes for structures}
|
||||
|
||||
Like the @racket[structure] and @racket[structure?] identifiers,
|
||||
@racket[structure-supertype] is defined in terms of its tagged structure
|
||||
counterpart, @racket[tagged-supertype]:
|
||||
|
||||
@chunk[<structure-supertype>
|
||||
(define-multi-id structure-supertype
|
||||
#:type-expander <expand-to-tagged-supertype>
|
||||
#:match-expander <expand-to-tagged-supertype>)]
|
||||
|
||||
@chunk[<expand-to-tagged-supertype>
|
||||
(λ/syntax-case (_ . _original-arguments) ()
|
||||
(syntax/top-loc stx
|
||||
(tagged-supertype untagged . _original-arguments)))]
|
||||
|
||||
@section{Putting it all together}
|
||||
|
||||
@chunk[<*>
|
||||
(require phc-toolkit
|
||||
"tagged.hl.rkt"
|
||||
"tagged-structure-low-level.hl.rkt"
|
||||
"tagged-supertype.hl.rkt"
|
||||
multi-id
|
||||
type-expander
|
||||
(for-syntax racket/base
|
||||
phc-toolkit/untyped))
|
||||
|
||||
(provide structure
|
||||
structure?
|
||||
define-structure
|
||||
StructureTop
|
||||
StructureTop?
|
||||
structure-supertype)
|
||||
|
||||
<structure>
|
||||
<structure?>
|
||||
<define-structure>
|
||||
<StructureTop>
|
||||
<StructureTop?>
|
||||
<structure-supertype>]
|
1790
phc-adt-lib/phc-adt/tagged-structure-low-level.hl.rkt
Normal file
1790
phc-adt-lib/phc-adt/tagged-structure-low-level.hl.rkt
Normal file
File diff suppressed because it is too large
Load Diff
153
phc-adt-lib/phc-adt/tagged-supertype.hl.rkt
Normal file
153
phc-adt-lib/phc-adt/tagged-supertype.hl.rkt
Normal file
|
@ -0,0 +1,153 @@
|
|||
#lang hyper-literate typed/racket/base #:no-require-lang #:no-auto-require
|
||||
@(require scribble-enhanced/doc
|
||||
scribble-math
|
||||
racket/require
|
||||
hyper-literate
|
||||
(for-label (lib "phc-adt/tagged-structure-low-level.hl.rkt")
|
||||
(lib "phc-adt/tagged.hl.rkt")
|
||||
extensible-parser-specifications
|
||||
racket/format
|
||||
phc-toolkit
|
||||
phc-toolkit/untyped-only
|
||||
remember
|
||||
syntax/parse
|
||||
syntax/parse/experimental/template
|
||||
(subtract-in typed/racket/base type-expander)
|
||||
type-expander
|
||||
type-expander/expander
|
||||
multi-id))
|
||||
@doc-lib-setup
|
||||
|
||||
@title[#:style manual-doc-style
|
||||
#:tag "tagged-supertype"
|
||||
#:tag-prefix "phc-adt/tagged-supertype"
|
||||
]{Supertypes of tagged structures}
|
||||
|
||||
@(chunks-toc-prefix
|
||||
'("(lib phc-adt/scribblings/phc-adt-implementation.scrbl)"
|
||||
"phc-adt/tagged-supertype"))
|
||||
|
||||
|
||||
@section{type-expander}
|
||||
|
||||
@chunk[<tagged-supertype>
|
||||
(define-multi-id tagged-supertype
|
||||
#:type-expander <tagged-supertype-type-expander>
|
||||
#:match-expander <tagged-supertype-match-expander>)]
|
||||
|
||||
As a type, @racket[tagged-supertype] accepts two syntaxes. With the first
|
||||
one, the type of each field is specified, and the second returns a parametric
|
||||
structure:
|
||||
|
||||
@chunk[<tagged-supertype-type-expander-signature-types>
|
||||
(_ name:id [field:id (~optional :colon) type:expr] …)]
|
||||
|
||||
@chunk[<tagged-supertype-type-expander-signature-infer>
|
||||
(_ name (~either [field:id] field:id) …)]
|
||||
|
||||
The type uses the @racket[structure] type-expander, and
|
||||
expands to the union of all structures which contain a
|
||||
superset of the given set of fields. It uses the specified
|
||||
type for the given fields, and defaults to @racket[Any] for the
|
||||
other extra fields.
|
||||
|
||||
@chunk[<tagged-supertype-type-expander-impl-types>
|
||||
(has-fields/type #'([field type] …))]
|
||||
|
||||
The second syntax builds upon the first, and produces a
|
||||
parametric type, with a @racket[∀] type argument for each
|
||||
specified field (other fields still falling back to
|
||||
@racket[Any]).
|
||||
|
||||
@CHUNK[<tagged-supertype-type-expander-impl-infer>
|
||||
(define-temp-ids "~a/τ" (field …))
|
||||
#`(∀ (field/τ …)
|
||||
#,(has-fields/type #'([field field/τ] …)))]
|
||||
|
||||
The type-expander finally calls either case depending on the
|
||||
syntax used.
|
||||
|
||||
@chunk[<tagged-supertype-type-expander>
|
||||
(λ (stx)
|
||||
(syntax-parse stx
|
||||
[<tagged-supertype-type-expander-signature-types>
|
||||
<tagged-supertype-type-expander-impl-types>]
|
||||
[<tagged-supertype-type-expander-signature-infer>
|
||||
<tagged-supertype-type-expander-impl-infer>]))]
|
||||
|
||||
@section{Match}
|
||||
|
||||
The match-expander for tagged-supertype accepts all
|
||||
structures which contain a superset of the given set of fields:
|
||||
|
||||
@chunk[<tagged-supertype-match-expander>
|
||||
(λ/syntax-parse (_ . :tagged-match-args-syntax-class)
|
||||
(define/with-syntax ([common . (all-field …)] …)
|
||||
(has-fields/common #'(fieldᵢ …)))
|
||||
(define/with-syntax ((maybe-fieldᵢ …) …)
|
||||
(if (attribute no-implicit)
|
||||
(map (const #'()) #'(fieldᵢ …))
|
||||
#'((fieldᵢ) …)))
|
||||
(define/with-syntax ((maybe-pats …) …)
|
||||
(quasitemplate ((<maybe-pat…> …) …)))
|
||||
#`(or (tagged name #:no-implicit-bind [all-field . maybe-pats] …) …))]
|
||||
|
||||
@chunk[<tagged-anytag-match>
|
||||
(define-match-expander tagged-anytag-match
|
||||
(λ/syntax-case ([fieldᵢ patᵢⱼ …] …) ()
|
||||
(tagged-anytag-match! #'([fieldᵢ (and patᵢⱼ …)] …))))]
|
||||
|
||||
Each field that was passed to @racket[tagged-supertype]
|
||||
additionally matches against the given @racket[pat …], and
|
||||
other fields do not use any extra pattern.
|
||||
|
||||
@chunk[<maybe-pat…>
|
||||
(!cdr-assoc #:default []
|
||||
all-field
|
||||
[fieldᵢ . [maybe-fieldᵢ … patᵢⱼ …]]
|
||||
…)]
|
||||
|
||||
@section{Nested supertype}
|
||||
|
||||
The @racket[(tagged-supertype* f₁ f₂ … fₙ T)] type describes any structure
|
||||
containing a field @racket[f₁], whose type is any structure containing a field
|
||||
@racket[f₂] etc. The last field's type is given by @racket[T].
|
||||
|
||||
@chunk[<tagged-supertype*>
|
||||
(define-multi-id tagged-supertype*
|
||||
#:type-expander
|
||||
(λ (stx)
|
||||
(error (string-append "tagged-supertype* is currently broken (needs"
|
||||
" to ignore the tag name, since it doe not"
|
||||
" have a tag at each step."))
|
||||
(syntax-parse stx
|
||||
[(_ T:expr)
|
||||
#`T]
|
||||
[(_ T:expr field:id other-fields:id …)
|
||||
#`(tagged-supertype
|
||||
[field (tagged-supertype* T other-fields …)])]))
|
||||
(code:comment
|
||||
"#:match-expander <tagged-supertype-match-expander> ; TODO"))]
|
||||
|
||||
@section{Conclusion}
|
||||
|
||||
@chunk[<*>
|
||||
(require (for-syntax racket/base
|
||||
racket/function
|
||||
racket/syntax
|
||||
syntax/parse
|
||||
syntax/parse/experimental/template
|
||||
phc-toolkit/untyped
|
||||
type-expander/expander)
|
||||
phc-toolkit
|
||||
multi-id
|
||||
type-expander
|
||||
"tagged-structure-low-level.hl.rkt"
|
||||
"tagged.hl.rkt")
|
||||
|
||||
(provide tagged-supertype
|
||||
tagged-supertype*)
|
||||
|
||||
<tagged-anytag-match>
|
||||
<tagged-supertype>
|
||||
<tagged-supertype*>]
|
718
phc-adt-lib/phc-adt/tagged.hl.rkt
Normal file
718
phc-adt-lib/phc-adt/tagged.hl.rkt
Normal file
|
@ -0,0 +1,718 @@
|
|||
#lang hyper-literate typed/racket/base #:no-require-lang #:no-auto-require
|
||||
@(require racket/require
|
||||
scribble-enhanced/doc
|
||||
scribble-math
|
||||
hyper-literate
|
||||
(for-label (lib "phc-adt/tagged-structure-low-level.hl.rkt")
|
||||
(lib "phc-adt/node-low-level.hl.rkt")
|
||||
extensible-parser-specifications
|
||||
racket/format
|
||||
phc-toolkit
|
||||
phc-toolkit/untyped-only
|
||||
remember
|
||||
syntax/parse
|
||||
(subtract-in typed/racket/base type-expander)
|
||||
type-expander
|
||||
multi-id))
|
||||
|
||||
@(unless-preexpanding
|
||||
(require (for-label (submod ".."))))
|
||||
|
||||
@doc-lib-setup
|
||||
|
||||
@title[#:style manual-doc-style
|
||||
#:tag "tagged"
|
||||
#:tag-prefix "phc-adt/tagged"]{User API for tagged structures}
|
||||
|
||||
@(chunks-toc-prefix
|
||||
'("(lib phc-adt/scribblings/phc-adt-implementation.scrbl)"
|
||||
"phc-adt/tagged"))
|
||||
|
||||
@(table-of-contents)
|
||||
|
||||
@section{Overview of the implementation of structures}
|
||||
|
||||
Tagged structures are represented using regular Racket @tc[struct]s,
|
||||
see @secref["choices" #:tag-prefixes '("phc-adt/choices")]
|
||||
for a somewhat outdated discussion of alternative possibilities.
|
||||
|
||||
The ADT type system implemented by this library needs to know about all declared
|
||||
structures across the program, so that fields can be accessed anonymously, e.g.
|
||||
@tc[(get _instance _f)] instead of the traditional @tc[(_s-f _instance)] (where
|
||||
@tc[_s] is the struct's identifier, and @tc[f] is the field name). This allows
|
||||
the accessor @racket[get] to work on all structures which contain a field with
|
||||
the given name (and therefore plays well with
|
||||
@seclink["Union_Types" #:doc '(lib "typed-racket/scribblings/ts-guide.scrbl")]{
|
||||
unions} of structures which share some fields). It also avoids the need to
|
||||
specify the struct which declared a field in order to accessing it. A separate
|
||||
library adds a coat of syntactic sugar to enable the notation
|
||||
@racket[instance.f1.f2.f3], following the well-known convention often used by
|
||||
object-oriented languages.
|
||||
|
||||
The section @secref["tagged-low-level"
|
||||
#:tag-prefixes '("phc-adt/tagged-low-level")]
|
||||
describes the low-level implementation of tagged structures, including how all
|
||||
declared structures are remembered across compilations, and the implementation
|
||||
of the basic operations on them: constructor, predicate, accessor, type,
|
||||
match expander, and row polymorphism (also known as static duck typing). The
|
||||
implementation of row polymorphism works by querying the list of all known
|
||||
tagged structures and returns those containing a given set of fields.
|
||||
|
||||
@section{A polyvalent identifier: type, match, constructor and instance}
|
||||
|
||||
The @tc[tagged] identifier can be used to describe a tagged structure type, a
|
||||
match pattern, a tagged structure builder function, or to directly create an
|
||||
instance. The last two cases are handled by the @tc[<call-expander>].
|
||||
|
||||
@chunk[<tagged>
|
||||
(define-multi-id tagged
|
||||
#:type-expander tagged-type-expander
|
||||
#:match-expander tagged-match-expander
|
||||
#:call tagged-call-expander)]
|
||||
|
||||
@section{The @racket[tagged] call expander (builder and instance)}
|
||||
|
||||
@chunk[#:save-as parse-field-instance… |<call [fieldᵢ] …+>|
|
||||
(~seq {~either [fieldᵢ:id] fieldᵢ:id} …+
|
||||
{~global-or builder?}
|
||||
{~global-or no-types?}
|
||||
{~post-fail <no-values-error> #:when (attribute instance?)})]
|
||||
|
||||
@chunk[#:save-as parse-field… |<[fieldᵢ] …+>|
|
||||
(~seq {~either [fieldᵢ:id] fieldᵢ:id} …+
|
||||
{~global-or no-types?})]
|
||||
|
||||
@chunk[#:save-as parse-field-colon-type… |<call [fieldᵢ : τᵢ] …+>|
|
||||
(~seq [fieldᵢ:id C:colon τᵢ:expr] …+
|
||||
{~global-or builder?}
|
||||
{~global-or types?}
|
||||
{~post-fail <no-values-error> #:when (attribute instance?)})]
|
||||
|
||||
@chunk[#:save-as parse-field-type… |<[fieldᵢ τᵢ] …+>|
|
||||
(~seq [fieldᵢ:id {~optional C:colon} τᵢ:expr] …+
|
||||
{~global-or types?})]
|
||||
|
||||
@chunk[#:save-as parse-field-pats… |<[fieldᵢ patᵢⱼ …] …+>|
|
||||
(~seq (~either {~and fieldᵢ:id {~bind [(patᵢⱼ 1) (list)]}}
|
||||
[fieldᵢ:id patᵢⱼ:expr …])
|
||||
…+)]
|
||||
|
||||
@chunk[#:save-as parse-field-value… |<call [fieldᵢ valueᵢ] …+>|
|
||||
(~seq [fieldᵢ:id valueᵢ:expr] …+
|
||||
{~global-or instance?}
|
||||
{~global-or no-types?}
|
||||
{~post-fail <values-error> #:when (attribute builder?)})]
|
||||
|
||||
@chunk[#:save-as parse-field-value-type… |<call [fieldᵢ valueᵢ : τᵢ] …+>|
|
||||
(~seq (~either [fieldᵢ:id valueᵢ:expr C:colon τᵢ:expr]
|
||||
[fieldᵢ:id C:colon τᵢ:expr valueᵢ:expr])
|
||||
…+
|
||||
{~global-or instance?}
|
||||
{~global-or types?}
|
||||
{~post-fail <values-error> #:when (attribute builder?)})]
|
||||
|
||||
@chunk[#:save-as no-values-error <no-values-error>
|
||||
(~a "The #:instance keyword implies the use of [field value],"
|
||||
" [field : type value] or [field value : type].")]
|
||||
|
||||
@chunk[#:save-as values-error <values-error>
|
||||
(~a "The #:builder keyword implies the use of [field], field"
|
||||
" or [field : type].")]
|
||||
|
||||
@chunk[#:save-as empty-error <empty-err>
|
||||
(~a "If no fields are specified, then either #:builder or #:instance"
|
||||
" must be present")]
|
||||
|
||||
When called like a macro, @tc[tagged] accepts several syntaxes:
|
||||
@itemlist[
|
||||
@item{@racket[(tagged name [fieldᵢ] …+)] or @racket[(tagged name fieldᵢ …+)],
|
||||
which return a builder function which has a type argument @racket[τᵢ]
|
||||
corresponding to each @racket[fieldᵢ]'s type.
|
||||
|
||||
@(parse-field-instance…)
|
||||
|
||||
This clause implies the creation of a builder function, so if
|
||||
@racket[#:instance] is specified, the following error is raised:
|
||||
|
||||
@(no-values-error)}
|
||||
@item{@racket[(tagged name [fieldᵢ valueᵢ] …+)], which returns an instance,
|
||||
inferring the type of the fields
|
||||
|
||||
@(parse-field-colon-type…)
|
||||
|
||||
This clause implies the creation of an instance, so if @racket[#:builder] is
|
||||
specified, the following error is raised:
|
||||
|
||||
@(values-error)}
|
||||
@item{@racket[(tagged name [fieldᵢ : τᵢ] …+)], which returns a constructor with
|
||||
the given types
|
||||
|
||||
@(parse-field-value…)
|
||||
|
||||
This clause implies the creation of a builder function, so if
|
||||
@racket[#:instance] is specified, the following error is raised:
|
||||
|
||||
@(no-values-error)}
|
||||
@item{@racket[(tagged name [fieldᵢ valueᵢ : τᵢ] …+)], which returns an instance
|
||||
using the given types
|
||||
|
||||
@(parse-field-value-type…)
|
||||
|
||||
This clause implies the creation of an instance, so if @racket[#:builder] is
|
||||
specified, the following error is raised:
|
||||
|
||||
@(values-error)}]
|
||||
|
||||
@subsection{Call to @racket[tagged] with no fields:
|
||||
instance or constructor?}
|
||||
|
||||
A call to @tc[(tagged)] with no field is ambiguous: it could return a
|
||||
constructor function for the structure with no fields, or an instance of that
|
||||
structure.
|
||||
|
||||
@racketblock[(tagged)]
|
||||
|
||||
We tried to make a hybrid object using @tc[define-struct/exec] which would be an
|
||||
instance of the structure with no fields, but could also be called as a function
|
||||
(which would return itself). Unfortunately, this use case is not yet fully
|
||||
supported by Typed/Racket: the instance cannot be annotated as a function type,
|
||||
or passed as an argument to a higher-order function (Typed/Racket issue
|
||||
@hyperlink["https://github.com/racket/typed-racket/issues/430"]{#430}). This can
|
||||
be circumvented by using unsafe operations to give the instance its proper type
|
||||
@tc[(Rec R (∩ (→ R) struct-type))], but Typed/Racket does not consider this type
|
||||
applicable, and an annotation is required each time the instance is used as a
|
||||
builder function (Typed/Racket issue
|
||||
@hyperlink["https://github.com/racket/typed-racket/issues/431"]{#431}.
|
||||
|
||||
We therefore added two optional keywords, @tc[#:instance] and @tc[#:builder],
|
||||
which can be used to disambiguate. They can also be used when fields
|
||||
respectively with or without values are specified, so that macros don't need to
|
||||
handle the empty structure as a special case.
|
||||
|
||||
@subsection{Signature for the @racket[tagged] call expander}
|
||||
|
||||
@chunk[#:save-as disambiguate-mixin <tagged-call-instance-or-builder-mixin>
|
||||
(define-eh-alternative-mixin tagged-call-instance-or-builder-mixin
|
||||
(pattern
|
||||
(~optional (~and instance-or-builder
|
||||
(~or {~global-or instance? #:instance}
|
||||
{~global-or builder? #:builder}))
|
||||
#:name "either #:instance or #:builder")))]
|
||||
|
||||
@chunk[#:save-as fields-mixin <tagged-call-fields-mixin>
|
||||
(define-eh-alternative-mixin tagged-call-fields-mixin
|
||||
(pattern
|
||||
(~optional/else
|
||||
(~try-after name-order-point <name-after-field-error>
|
||||
(~or |<call [fieldᵢ] …+>|
|
||||
|<call [fieldᵢ : τᵢ] …+>|
|
||||
|<call [fieldᵢ valueᵢ] …+>|
|
||||
|<call [fieldᵢ valueᵢ : τᵢ] …+>|))
|
||||
#:defaults ([(fieldᵢ 1) (list)]
|
||||
[(valueᵢ 1) (list)]
|
||||
[(τᵢ 1) (list)])
|
||||
#:else-post-fail <empty-err> #:unless (or (attribute builder?)
|
||||
(attribute instance?))
|
||||
#:name (~a "field or [field] or [field : type] for #:builder,"
|
||||
" [field value] or [field : type value]"
|
||||
" or [field value : type] for #:instance"))))]
|
||||
|
||||
@chunk[#:save-as ∀-mixin <∀-mixin>
|
||||
(define-eh-alternative-mixin ∀-mixin
|
||||
(pattern {~optional (~seq #:∀ ({~named-seq tvarᵢ :id …})
|
||||
(~global-or tvars?))}))]
|
||||
|
||||
@chunk[#:save-as name-id-mixin <name-id-mixin>
|
||||
(define-eh-alternative-mixin name-id-mixin
|
||||
(pattern
|
||||
(~once (~order-point name-order-point name:id))))]
|
||||
|
||||
@chunk[#:save-as name-after-field-error <name-after-field-error>
|
||||
"the name must appear before any field"]
|
||||
|
||||
When called as a macro, @tc[tagged] expects:
|
||||
|
||||
@itemlist[
|
||||
@item{The tagged structure's tag name:
|
||||
|
||||
@(name-id-mixin)}
|
||||
@item{An optional list of type variables, preceded by @racket[#:∀]:
|
||||
|
||||
@(∀-mixin)}
|
||||
@item{Either of the @racket[#:builder] or @racket[#:instance] options, or none:
|
||||
|
||||
@(disambiguate-mixin)}
|
||||
@item{An optional list of fields, possibly annotated with a type, and possibly
|
||||
associated to a value:
|
||||
|
||||
@(fields-mixin)
|
||||
|
||||
When no fields are specified, the following error is raised unless either
|
||||
@racket[#:builder] or @racket[#:instance] is specified.
|
||||
|
||||
@(empty-error)}]
|
||||
|
||||
The four elements can appear in any order, with one constraint: the name must
|
||||
appear before the first field descriptor. Not only does it make more sense
|
||||
semantically, but it also avoids ambiguities when the list of field names is
|
||||
just a list of plain identifiers (without any type or value).
|
||||
|
||||
@(name-after-field-error)
|
||||
|
||||
We can now combine all four mixins.
|
||||
|
||||
@chunk[<tagged-call-args-mixin>
|
||||
(define-eh-alternative-mixin tagged-call-args-mixin
|
||||
#:define-splicing-syntax-class tagged-call-args-syntax-class
|
||||
(pattern {~mixin name-id-mixin})
|
||||
(pattern {~mixin tagged-call-instance-or-builder-mixin})
|
||||
(pattern {~mixin tagged-call-fields-mixin})
|
||||
(pattern {~mixin ∀-mixin}))]
|
||||
|
||||
@subsection{Implementation}
|
||||
|
||||
The call expander uses the low-level functions @tc[tagged-builder!],
|
||||
@tc[tagged-∀-builder!] and @tc[tagged-infer-builder!] implemented in
|
||||
@secref["Creating_instances_of_a_tagged_structure"
|
||||
#:tag-prefixes '("phc-adt/tagged-low-level")].
|
||||
The first returns the syntax for a builder function for the given tagged
|
||||
structure. The second returns the syntax for a polymorphic builder function for
|
||||
the given tagged structure, using the given type variables which are bound
|
||||
inside the field type declarations. The last returns the syntax for a
|
||||
polymorphic builder function for the given tagged structure, with one type
|
||||
parameter per field, which allows the type of each field to be inferred.
|
||||
|
||||
@chunk[<call-expander>
|
||||
(define/syntax-parse+simple
|
||||
(tagged-call-expander :tagged-call-args-syntax-class)
|
||||
<call-expander-cases>)]
|
||||
|
||||
If type variables are present, then @tc[tagged-∀-builder!] is used. Otherwise,
|
||||
if types are specified, then @tc[tagged-builder!] is used, otherwise
|
||||
@tc[tagged-infer-builder!] is used.
|
||||
|
||||
@chunk[<call-expander-cases>
|
||||
(define/with-syntax f
|
||||
(if (attribute tvars?)
|
||||
(tagged-∀-builder! #'((tvarᵢ …) name [fieldᵢ : τᵢ] …))
|
||||
(if (attribute types?)
|
||||
(tagged-builder! #'(name [fieldᵢ τᵢ] …))
|
||||
(tagged-infer-builder! #'(name fieldᵢ …)))))]
|
||||
|
||||
If the @tc[#:instance] keyword was specified, or if values are specified for
|
||||
each field, the builder function is immediately called with those values, in
|
||||
order to produce an instance of the tagged structure. Otherwise, the builder
|
||||
function itself is produced.
|
||||
|
||||
@chunk[<call-expander-cases>
|
||||
(if (attribute instance?)
|
||||
#'(f valueᵢ …)
|
||||
#'f)]
|
||||
|
||||
@section{Type expander}
|
||||
|
||||
@chunk[#:save-as type-fields-mixin <tagged-type-fields-mixin>
|
||||
(define-eh-alternative-mixin tagged-type-fields-mixin
|
||||
(pattern
|
||||
(~optional
|
||||
(~try-after name-order-point <name-after-field-error>
|
||||
(~named-seq field-declarations
|
||||
(~or |<[fieldᵢ] …+>|
|
||||
|<[fieldᵢ τᵢ] …+>|)))
|
||||
#:defaults ([(fieldᵢ 1) (list)]
|
||||
[(τᵢ 1) (list)])
|
||||
#:name "field or [field] or [field type] or [field : type]")))]
|
||||
|
||||
When used as a type expander, @tc[tagged] expects:
|
||||
|
||||
@itemlist[
|
||||
@item{The tagged structure's tag name, as defined for the call expander in
|
||||
@racket[<name-id-mixin>]}
|
||||
@item{An optional list of type variables, as defined for the call expander in
|
||||
@racket[<∀-mixin>]}
|
||||
@item{An optional list of fields, possibly annotated with a type:
|
||||
|
||||
@(type-fields-mixin)
|
||||
|
||||
The main difference with the allowed field specifications for the call
|
||||
expander are that values are not allowed. Furthermore, the @racket[:] between
|
||||
a field and its type is optional:
|
||||
|
||||
@(parse-field-type…)
|
||||
|
||||
Furthermore, the @racket[instance?] and @racket[builder?] attributes are not
|
||||
meaningful for the type expander.
|
||||
|
||||
@(parse-field…)}]
|
||||
|
||||
The three elements can appear in any order, with the same constraint as for the
|
||||
call expander: the name must appear before the first field descriptor.
|
||||
|
||||
@(name-after-field-error)
|
||||
|
||||
@chunk[<tagged-type-args-mixin>
|
||||
(define-eh-alternative-mixin tagged-type-args-mixin
|
||||
#:define-splicing-syntax-class tagged-type-args-syntax-class
|
||||
(pattern {~mixin name-id-mixin})
|
||||
(pattern {~mixin tagged-type-fields-mixin})
|
||||
(pattern {~mixin ∀-mixin}))]
|
||||
|
||||
The type expander uses the low-level functions @tc[tagged-type!],
|
||||
@tc[tagged-∀-type!] and @tc[tagged-infer-type!] implemented in
|
||||
@secref["Type_of_a_tagged_structure"
|
||||
#:tag-prefixes '("phc-adt/tagged-low-level")]. The first
|
||||
returns the syntax for the type for the given tagged structure. The second
|
||||
returns the syntax for a polymorphic type for the given tagged structure, using
|
||||
the given type variables which are bound inside the field type declarations. The
|
||||
last returns the syntax for a polymorphic type for the given tagged structure,
|
||||
with one type parameter per field, which allows the type of each field to be
|
||||
inferred or filled in later.
|
||||
|
||||
@chunk[<type-expander>
|
||||
(define/syntax-parse+simple
|
||||
(tagged-type-expander :tagged-type-args-syntax-class)
|
||||
<type-expander-cases>)]
|
||||
|
||||
If type variables are present, then @tc[tagged-∀-type!] is used. Otherwise,
|
||||
if types are specified, then @tc[tagged-type!] is used, otherwise
|
||||
@tc[tagged-infer-type!] is used.
|
||||
|
||||
@chunk[<type-expander-cases>
|
||||
(if (attribute tvars?)
|
||||
(tagged-∀-type! #'((tvarᵢ …) name [fieldᵢ : τᵢ] …))
|
||||
(if (attribute types?)
|
||||
(tagged-type! #'(name [fieldᵢ τᵢ] …))
|
||||
(tagged-infer-type! #'(name fieldᵢ …))))]
|
||||
|
||||
@subsection{The @racket[TaggedTop] type}
|
||||
|
||||
The @tc[TaggedTop] type is extracted from the low-level @tc[TaggedTop-struct]
|
||||
identifier (which is a struct identifier). The @tc[TaggedTop] type includes not
|
||||
only tagged structures, but also nodes.
|
||||
|
||||
@chunk[<TaggedTop>
|
||||
(define-type TaggedTop TaggedTop-struct)]
|
||||
|
||||
Additionally, the @racket[TaggedTop?] predicate is simply aliased from the
|
||||
low-level @racket[TaggedTop-struct?].
|
||||
|
||||
@; Do not use rename-out, as it confuses scribble (two documentations for one
|
||||
@; identifier: the user-level documentation of the TaggedTop? function, and
|
||||
@; the low-level documentation of the TaggedTop-struct struct.
|
||||
@chunk[<TaggedTop>
|
||||
(define TaggedTop? TaggedTop-struct?)]
|
||||
|
||||
@section{Match expander}
|
||||
|
||||
@chunk[#:save-as match-fields-mixin <tagged-match-fields-mixin>
|
||||
(define-eh-alternative-mixin tagged-match-fields-mixin
|
||||
(pattern
|
||||
(~maybe/empty
|
||||
(~try-after name-order-point <name-after-field-error>
|
||||
|<[fieldᵢ patᵢⱼ …] …+>|)
|
||||
#:name (~a "field or [field pat …]"))))]
|
||||
|
||||
@chunk[#:save-as no-implicit-mixin <tagged-match-no-implicit-bind-mixin>
|
||||
(define-eh-alternative-mixin tagged-match-no-implicit-bind-mixin
|
||||
(pattern (~optional (~global-or no-implicit #:no-implicit-bind))))]
|
||||
|
||||
When used as a match expander, @tc[tagged] expects:
|
||||
|
||||
@itemlist[
|
||||
@item{The tagged structure's tag name, as defined for the call expander in
|
||||
@racket[<name-id-mixin>]}
|
||||
@item{The @racket[#:no-implicit-bind], which specified that the field name
|
||||
should not automatically be bound by the match pattern to the field's
|
||||
contents:
|
||||
|
||||
@(no-implicit-mixin)}
|
||||
@item{A (possibly empty) list of fields, each associated with zero or more
|
||||
patterns:
|
||||
|
||||
@(match-fields-mixin)
|
||||
|
||||
The main differences with the allowed field specifications for the call
|
||||
expander are that values and types are not allowed, but instead the field name
|
||||
may be followed by match patterns:
|
||||
|
||||
@(parse-field-pats…)}]
|
||||
|
||||
The three elements can appear in any order, with the same constraint as for the
|
||||
call expander: the name must appear before the first field descriptor.
|
||||
|
||||
@(name-after-field-error)
|
||||
|
||||
@chunk[<tagged-match-args-mixin>
|
||||
(define-eh-alternative-mixin tagged-match-args-mixin
|
||||
#:define-syntax-class tagged-match-args-syntax-class
|
||||
(pattern {~mixin name-id-mixin})
|
||||
(pattern {~mixin tagged-match-no-implicit-bind-mixin})
|
||||
(pattern {~mixin tagged-match-fields-mixin}))]
|
||||
|
||||
The match expander uses the low-level function @tc[tagged-match!] implemented in
|
||||
@secref["Type_of_a_tagged_structure"
|
||||
#:tag-prefixes '("phc-adt/tagged-low-level")]. It returns
|
||||
the syntax for a match pattern for the given tagged structure. The resulting
|
||||
match pattern checks that the value is an instance of a tagged structure with
|
||||
the given name and fields, and matches the value of each field against the
|
||||
corresponding pattern.
|
||||
|
||||
@chunk[<match-expander>
|
||||
(define/syntax-parse+simple
|
||||
(tagged-match-expander . :tagged-match-args-syntax-class)
|
||||
<match-expander-body>)]
|
||||
|
||||
Unless @racket[#:no-implicit-bind] is specified, we include the field name as
|
||||
part of the pattern, so that field names are bound to the field's contents.
|
||||
|
||||
@chunk[<match-expander-body>
|
||||
(if (attribute no-implicit)
|
||||
(tagged-match! #'(name [fieldᵢ (and patᵢⱼ …)] …))
|
||||
(tagged-match! #'(name [fieldᵢ (and fieldᵢ patᵢⱼ …)] …)))]
|
||||
|
||||
@section{Predicates for tagged structures}
|
||||
|
||||
@chunk[<tagged?>
|
||||
(define-syntax tagged?
|
||||
(syntax-parser
|
||||
[(_ name fieldᵢ:id …)
|
||||
(tagged-any-predicate! #'(name fieldᵢ …))]
|
||||
[(_ name [fieldᵢ:id :colon τᵢ:type] …)
|
||||
(tagged-predicate! #'(name [fieldᵢ τᵢ] …))]
|
||||
[(_ name [fieldᵢ:id predᵢ:type] …)
|
||||
(tagged-pred-predicate! #'(name [fieldᵢ predᵢ] …))]))]
|
||||
|
||||
@subsection{The @racket[TaggedTop?] predicate}
|
||||
|
||||
The @tc[TaggedTop?] predicate is simply re-provided. It is initially defined in
|
||||
@secref["Common_ancestor_to_all_tagged_structures__TaggedTop-struct"
|
||||
#:tag-prefixes '("phc-adt/tagged-low-level")].
|
||||
|
||||
@chunk[|<provide TaggedTop?>|
|
||||
(provide (rename-out [TaggedTop-struct? TaggedTop?]))]
|
||||
|
||||
@section{Defining shorthands with @racket[define-tagged]}
|
||||
|
||||
The @tc[define-tagged] macro can be used to bind to an
|
||||
identifier the type expander, match expander, predicate and
|
||||
constructor function for a given tagged structure.
|
||||
|
||||
@chunk[#:save-as tag-kw-mixin <tag-kw-mixin>
|
||||
(define-eh-alternative-mixin tag-kw-mixin
|
||||
(pattern {~optional {~seq #:tag explicit-tag <default-tag-name>}}))]
|
||||
|
||||
@chunk[#:save-as tag-kw-mixin-default <default-tag-name>
|
||||
{~post-check
|
||||
{~bind [tag-name (or (attribute explicit-tag)
|
||||
#'name)]}}]
|
||||
|
||||
@chunk[#:save-as predicate?-mixin <predicate?-mixin>
|
||||
(define-eh-alternative-mixin predicate?-mixin
|
||||
(pattern {~optional {~seq #:? predicate? <default-name?>}}))]
|
||||
|
||||
@chunk[#:save-as predicate?-mixin-default <default-name?>
|
||||
{~post-check
|
||||
{~bind [name? (or (attribute predicate?)
|
||||
(format-id/record #'name "~a?" #'name))]}}]
|
||||
|
||||
The @tc[define-tagged] macro expects:
|
||||
|
||||
@itemlist[
|
||||
@item{The tagged structure's tag name, as defined for the call expander in
|
||||
@racket[<name-id-mixin>]}
|
||||
@item{An optional list of type variables, as defined for the call expander in
|
||||
@racket[<∀-mixin>]}
|
||||
@item{A possibly empty list of fields, possibly annotated with a type, as
|
||||
defined for the type expander in @racket[<tagged-type-fields-mixin>]}
|
||||
@item{Optionally, the tag name to be used, specified with
|
||||
@racket[#:tag tag-name]:
|
||||
|
||||
@(tag-kw-mixin)
|
||||
|
||||
The tag name defaults to @racket[_name], i.e. the identifier currently being
|
||||
defined.
|
||||
|
||||
@(tag-kw-mixin-default)}
|
||||
@item{Optionally, a name for the predicate, specified with
|
||||
@racket[#:? predicate-name?]:
|
||||
|
||||
@(predicate?-mixin)
|
||||
|
||||
The predicate name defaults to @racket[_name?], where @racket[_name] is the
|
||||
identifier currently being defined.
|
||||
|
||||
@(predicate?-mixin-default)}]
|
||||
|
||||
The five elements can appear in any order, with the same constraint as for the
|
||||
call expander: the name must appear before the first field descriptor.
|
||||
|
||||
@chunk[<define-tagged-args-mixin>
|
||||
(define-eh-alternative-mixin define-tagged-args-mixin
|
||||
#:define-splicing-syntax-class define-tagged-args-syntax-class
|
||||
(pattern (~or {~mixin name-id-mixin}
|
||||
{~mixin tag-kw-mixin}
|
||||
{~mixin tagged-type-fields-mixin}
|
||||
{~mixin predicate?-mixin}
|
||||
{~mixin ∀-mixin})))]
|
||||
|
||||
The @tc[define-tagged] macro is then implemented using @racket[define-multi-id]:
|
||||
|
||||
@CHUNK[<define-tagged>
|
||||
(define-syntax/parse+simple
|
||||
(define-tagged :define-tagged-args-syntax-class)
|
||||
(define-temp-ids "~a/pat" (fieldᵢ …))
|
||||
(quasisyntax/top-loc stx
|
||||
(begin
|
||||
(define-multi-id name
|
||||
#:type-expander (make-id+call-transformer
|
||||
#'<type-expander/define>)
|
||||
#:match-expander <match-expander/define>
|
||||
#:else <else-expander/define>)
|
||||
(define name? <predicate/define>))))]
|
||||
|
||||
The type expander handles the same three cases as for @tc[tagged]: with type
|
||||
variables, with a type for each field, or inferred.
|
||||
|
||||
@CHUNK[<type-expander/define>
|
||||
#,(if (attribute tvars?)
|
||||
(tagged-∀-type! #'((tvarᵢ …) tag-name [fieldᵢ τᵢ] …))
|
||||
(if (attribute types?)
|
||||
(tagged-type! #'(tag-name [fieldᵢ τᵢ] …))
|
||||
(tagged-infer-type! #'(tag-name fieldᵢ …))))]
|
||||
|
||||
The match expander is a short form of the one implemented for @tc[tagged], as it
|
||||
takes only one positional pattern per field.
|
||||
|
||||
@chunk[<match-expander/define>
|
||||
(λ (stx2)
|
||||
(syntax-case stx2 ()
|
||||
[(_ fieldᵢ/pat …)
|
||||
(tagged-match! #'(tag-name [fieldᵢ fieldᵢ/pat] …))]
|
||||
(code:comment "Todo: implement a \"rest\" pattern")))]
|
||||
|
||||
Otherwise, when @racket[_name] is called as a function, or used as an identifier
|
||||
on its own, we produce a builder function. When @racket[_name] is called as a
|
||||
function, the builder function is applied immediately to the arguments,
|
||||
otherwise the builder function itself is used. The same three cases as for
|
||||
@tc[tagged] are handled: with type variables, with a type for each field, or
|
||||
inferred.
|
||||
|
||||
@CHUNK[<else-expander/define>
|
||||
#'#,(if (attribute tvars?)
|
||||
(tagged-∀-builder!
|
||||
#'((tvarᵢ …) tag-name [fieldᵢ τᵢ] …))
|
||||
(if (attribute types?)
|
||||
(tagged-builder! #'(tag-name [fieldᵢ τᵢ] …))
|
||||
(tagged-infer-builder! #'(tag-name fieldᵢ …))))]
|
||||
|
||||
Finally, we define the predicate @racket[name?]. Contrarily to @racket[tagged?],
|
||||
it does not take into account the field types, as we have no guarantee that
|
||||
Typed/Racket's @racket[make-predicate] works for those. Instead, @racket[name?]
|
||||
recognises any instance of a tagged structure with the given tag name and
|
||||
fields. If a more accurate predicate is desired, it can easily be implemented
|
||||
using @racket[tagged?].
|
||||
|
||||
@CHUNK[<predicate/define>
|
||||
#,(tagged-any-predicate! #'(tag-name fieldᵢ …))]
|
||||
|
||||
@section{Implementation of @racket[uniform-get]}
|
||||
|
||||
@racket[uniform-get] operates on tagged structures. It retrieves the desired
|
||||
field from the structure, and forces it to obtain the actual value.
|
||||
|
||||
It is implemented as @racket[tagged-get-field] in
|
||||
@secref["Accessing_fields_of_tagged_structures"
|
||||
#:tag-prefixes '("phc-adt/tagged-low-level")], and is
|
||||
simply re-provided here.
|
||||
|
||||
@section{Putting it all together}
|
||||
|
||||
@chunk[<*>
|
||||
(require (for-syntax racket/base
|
||||
racket/syntax
|
||||
syntax/parse
|
||||
phc-toolkit/untyped
|
||||
syntax/strip-context
|
||||
racket/function
|
||||
extensible-parser-specifications
|
||||
racket/format
|
||||
type-expander/expander)
|
||||
phc-toolkit
|
||||
multi-id
|
||||
type-expander
|
||||
racket/promise
|
||||
"tagged-structure-low-level.hl.rkt"
|
||||
racket/format)
|
||||
|
||||
@; Do not use rename-out, as it confuses scribble (two documentations for
|
||||
@; one identifier: the user-level documentation of uniform-get, and the
|
||||
@; low-level documentation of tagged-get-field.
|
||||
(define-syntax uniform-get
|
||||
(make-rename-transformer #'tagged-get-field))
|
||||
(define-syntax λuniform-get
|
||||
(make-rename-transformer #'λ-tagged-get-field))
|
||||
(provide uniform-get
|
||||
λuniform-get
|
||||
tagged
|
||||
tagged?
|
||||
define-tagged
|
||||
TaggedTop
|
||||
TaggedTop?
|
||||
|
||||
(for-syntax tagged-call-args-syntax-class
|
||||
tagged-call-expander-forward-attributes
|
||||
tagged-call-expander
|
||||
|
||||
tagged-type-args-syntax-class
|
||||
tagged-type-expander-forward-attributes
|
||||
tagged-type-expander
|
||||
|
||||
tagged-match-args-syntax-class
|
||||
tagged-match-expander-forward-attributes
|
||||
tagged-match-expander
|
||||
|
||||
define-tagged-args-syntax-class
|
||||
define-tagged-forward-attributes))
|
||||
|
||||
(begin-for-syntax
|
||||
<∀-mixin>
|
||||
<name-id-mixin>
|
||||
<tagged-call-instance-or-builder-mixin>
|
||||
<tagged-call-fields-mixin>
|
||||
<tagged-call-args-mixin>
|
||||
<tagged-type-fields-mixin>
|
||||
<tagged-type-args-mixin>
|
||||
<tagged-match-fields-mixin>
|
||||
<tagged-match-no-implicit-bind-mixin>
|
||||
<tagged-match-args-mixin>
|
||||
|
||||
<predicate?-mixin>
|
||||
<tag-kw-mixin>
|
||||
<define-tagged-args-mixin>)
|
||||
|
||||
(begin-for-syntax
|
||||
<call-expander>
|
||||
<type-expander>
|
||||
<match-expander>)
|
||||
<tagged>
|
||||
<tagged?>
|
||||
<TaggedTop>
|
||||
<define-tagged>]
|
||||
|
||||
@;tagged-call-instance-or-builder-mixin
|
||||
@;tagged-call-fields-mixin
|
||||
@;tagged-call-args-mixin
|
||||
|
||||
@;tagged-type-fields-mixin
|
||||
@;tagged-type-args-mixin
|
||||
|
||||
@;tagged-match-fields-mixin
|
||||
@;tagged-match-no-implicit-bind-mixin
|
||||
@;tagged-match-args-mixin
|
||||
|
||||
@;tag-kw-mixin
|
||||
@;predicate?-mixin
|
||||
@;define-tagged-args-mixin
|
||||
|
||||
@;name-id-mixin
|
||||
@;∀-mixin
|
147
phc-adt-lib/phc-adt/variant.hl.rkt
Normal file
147
phc-adt-lib/phc-adt/variant.hl.rkt
Normal file
|
@ -0,0 +1,147 @@
|
|||
#lang hyper-literate typed/racket/base #:no-require-lang #:no-auto-require
|
||||
@(require scribble-enhanced/doc)
|
||||
@doc-lib-setup
|
||||
|
||||
@require[racket/require
|
||||
@for-label[(subtract-in typed/racket/base type-expander)
|
||||
racket/list
|
||||
syntax/parse
|
||||
syntax/parse/experimental/template
|
||||
(subtract-in racket/syntax phc-toolkit)
|
||||
phc-toolkit/untyped-only
|
||||
type-expander/expander
|
||||
phc-toolkit
|
||||
multi-id
|
||||
type-expander
|
||||
"constructor.hl.rkt"
|
||||
"structure.hl.rkt"]]
|
||||
|
||||
@title[#:style manual-doc-style
|
||||
#:tag "variant"
|
||||
#:tag-prefix "phc-adt/variant"]{User API for variants}
|
||||
|
||||
@(chunks-toc-prefix
|
||||
'("(lib phc-adt/scribblings/phc-adt-implementation.scrbl)"
|
||||
"phc-adt/variant"))
|
||||
|
||||
@(table-of-contents)
|
||||
|
||||
@section{Introduction}
|
||||
|
||||
For convenience, we write a @tc[variant] form, which is a
|
||||
thin wrapper against @tc[(U (~or constructor tagged) …)].
|
||||
|
||||
@section{Implementation of @racket[variant]}
|
||||
|
||||
In @tc[define-variant], we only define the type (which is
|
||||
the union of all the possible constructors. We do not bind
|
||||
identifiers for the constructors, for two reasons: the same
|
||||
@tc[constructor]s could appear in several variants, so we
|
||||
would define them twice, and it is likely that a constructor
|
||||
will have the same identifier as an existing variable or
|
||||
function.
|
||||
|
||||
@chunk[<constructor-or-tagged-stx-class>
|
||||
(begin-for-syntax
|
||||
(define-syntax-class constructor-or-tagged
|
||||
(pattern [constructor-name:id . (~or ([field:id C:colon type:expr] …)
|
||||
(type:expr …))])))]
|
||||
|
||||
@chunk[<variant>
|
||||
(define-type-expander (variant stx)
|
||||
(syntax-parse stx
|
||||
[(_ :constructor-or-tagged …)
|
||||
(template
|
||||
(U (?? (tagged constructor-name [field C type] …)
|
||||
(constructor constructor-name type …))
|
||||
…))]))]
|
||||
|
||||
@section{Predicate}
|
||||
|
||||
@chunk[<variant?>
|
||||
(define-syntax/parse (variant? :constructor-or-tagged …)
|
||||
(template
|
||||
(λ (v) (or (?? ((tagged? constructor-name field …) v)
|
||||
(constructor? constructor-name v))
|
||||
…))))]
|
||||
|
||||
@section{@racket[define-variant]}
|
||||
|
||||
@chunk[<define-variant>
|
||||
(define-syntax/parse
|
||||
(define-variant variant-name
|
||||
(~optkw #:debug)
|
||||
(~maybe #:? name?)
|
||||
(~maybe #:match variant-match)
|
||||
(~and constructor-or-tagged :constructor-or-tagged) …)
|
||||
(define/with-syntax default-name? (format-id #'name "~a?" #'name))
|
||||
(define/with-syntax default-match (format-id #'name "~a-match" #'name))
|
||||
(define-temp-ids "pat" ((type …) …))
|
||||
(define-temp-ids "match-body" (constructor-name …))
|
||||
(template
|
||||
(begin
|
||||
(define-type variant-name
|
||||
(variant [constructor-name (?? (?@ [field C type] …)
|
||||
(?@ type …))]
|
||||
…))
|
||||
(define-syntax (?? variant-match default-match)
|
||||
(syntax-rules (constructor-name … (?? (?@ field …)) …)
|
||||
[(_ v
|
||||
[(constructor-name (?? (?@ [field pat] …)
|
||||
(pat …)))
|
||||
. match-body]
|
||||
…)
|
||||
(match v
|
||||
(?? [(tagged constructor-name [field pat] …) . match-body]
|
||||
[(constructor constructor-name pat …) . match-body])
|
||||
…)]))
|
||||
(define-multi-id (?? name? default-name?)
|
||||
#:else
|
||||
#'(variant? constructor-or-tagged …))))
|
||||
#|
|
||||
(if (andmap (λ (t) (check-remember-all 'variant t))
|
||||
(syntax->list #'(tag …)))
|
||||
(let ()
|
||||
(define/with-syntax (stx-name …)
|
||||
(stx-map (λ (t)
|
||||
(cdr (assoc (syntax->datum (datum->syntax #f t))
|
||||
tag-name→stx-name/alist)))
|
||||
#'(tag …)))
|
||||
(quasitemplate
|
||||
(begin
|
||||
(define-type name (U (constructor tag type …) …))
|
||||
(: (?? name? default-name?)
|
||||
(→ Any Boolean :
|
||||
#:+ (or (stx-name Any) …)
|
||||
#:- (and (! (stx-name Any)) …)))
|
||||
(define ((?? name? default-name?) x)
|
||||
(or (Tagged-predicate? tag x) …)))))
|
||||
(stx-map (λ (t)
|
||||
(remember-all-errors2 (syntax/loc t #'please-recompile)
|
||||
t))
|
||||
#'(tag …)))|#)]
|
||||
|
||||
@section{Conclusion}
|
||||
|
||||
@chunk[<*>
|
||||
(require (for-syntax racket/base
|
||||
racket/list
|
||||
syntax/parse
|
||||
syntax/parse/experimental/template
|
||||
racket/syntax
|
||||
phc-toolkit/untyped
|
||||
type-expander/expander)
|
||||
phc-toolkit
|
||||
multi-id
|
||||
type-expander
|
||||
"constructor.hl.rkt"
|
||||
"structure.hl.rkt")
|
||||
|
||||
(provide variant
|
||||
variant?
|
||||
define-variant)
|
||||
|
||||
<constructor-or-tagged-stx-class>
|
||||
<variant>
|
||||
<variant?>
|
||||
<define-variant>]
|
14
phc-adt-test/info.rkt
Normal file
14
phc-adt-test/info.rkt
Normal file
|
@ -0,0 +1,14 @@
|
|||
#lang info
|
||||
(define collection 'multi)
|
||||
(define deps '("base"
|
||||
"phc-adt-lib"
|
||||
"rackunit-lib"
|
||||
"typed-racket-lib"
|
||||
"typed-racket-more"
|
||||
"multi-id"
|
||||
"phc-toolkit"
|
||||
"type-expander"))
|
||||
(define build-deps '())
|
||||
(define pkg-desc "Tests for the phc-adt library")
|
||||
(define version "1.1")
|
||||
(define pkg-authors '("Georges Dupéron"))
|
54
phc-adt-test/phc-adt/test/adt-pre-declarations.rkt
Normal file
54
phc-adt-test/phc-adt/test/adt-pre-declarations.rkt
Normal file
|
@ -0,0 +1,54 @@
|
|||
#lang s-exp phc-adt/declarations
|
||||
(remembered! tagged-structure (untagged a b))
|
||||
(remembered! tagged-structure (st2 a b))
|
||||
(remembered! tagged-structure (t a b c))
|
||||
(remembered! tagged-structure (tabc a b c))
|
||||
(remembered! tagged-structure (tag1 values))
|
||||
(remembered! tagged-structure (a values))
|
||||
(remembered! tagged-structure (b values))
|
||||
(remembered! tagged-structure (c values))
|
||||
(remembered! tagged-structure (d values))
|
||||
(remembered! tagged-structure (e values))
|
||||
(remembered! tagged-structure (F values))
|
||||
(remembered! tagged-structure (tag0 values))
|
||||
(remembered! tagged-structure (tag2 values))
|
||||
(remembered! tagged-structure (tag3 values))
|
||||
(remembered! tagged-structure (tagged-s1))
|
||||
(remembered! tagged-structure (tagged-s2 f g))
|
||||
(remembered! tagged-structure (tagged-s3 f g))
|
||||
(remembered! tagged-structure (tagged-s4 f g))
|
||||
(remembered! tagged-structure (c1 values))
|
||||
(remembered! tagged-structure (c2 values))
|
||||
(remembered! tagged-structure (c3 values))
|
||||
(remembered! tagged-structure (txyz a b))
|
||||
(remembered! tagged-structure (cxyz values))
|
||||
(remembered! tagged-structure (txyz x y))
|
||||
(remembered! tagged-structure (untagged))
|
||||
(remembered! tagged-structure (empty-tg))
|
||||
(remembered! tagged-structure (empty-ct values))
|
||||
(remembered! tagged-structure (empty-ct-t2 values))
|
||||
(remembered! tagged-structure (empty-ct-t3 values))
|
||||
(remembered! tagged-structure (empty-ct-2 values))
|
||||
(remembered! tagged-structure (empty-ct-3 values))
|
||||
(remembered! tagged-structure (untagged test-fa test-fb))
|
||||
(remembered! tagged-structure (untagged test-fa test-fc))
|
||||
(remembered! tagged-structure (untagged test-fa test-fd))
|
||||
(remembered! tagged-structure (untagged a c))
|
||||
(remembered! tagged-structure (untagged a b c d))
|
||||
(remembered! tagged-structure (untagged a b c y))
|
||||
(remembered! tagged-structure (untagged a b c))
|
||||
(remembered! tagged-structure (untagged a))
|
||||
(remembered! tagged-structure (t0 tagged-fa tagged-fb tagged-fc))
|
||||
(remembered! tagged-structure (t0 tagged-fx tagged-fy tagged-fz))
|
||||
(remembered! tagged-structure (a a b))
|
||||
(remembered! tagged-structure (b a b))
|
||||
(remembered! tagged-structure (c a b))
|
||||
(remembered! tagged-structure (ma faa fab fav))
|
||||
(remembered! tagged-structure (mb fba fbv))
|
||||
(remembered! tagged-structure (t1 x y))
|
||||
(remembered! tagged-structure (foo x y z))
|
||||
(remembered! tagged-structure (x values))
|
||||
(remembered! tagged-structure (y values))
|
||||
(remembered! tagged-structure (z values))
|
||||
(remembered! tagged-structure (w values))
|
||||
(remembered! tagged-structure (tg x y))
|
49
phc-adt-test/phc-adt/test/ck.rkt
Normal file
49
phc-adt-test/phc-adt/test/ck.rkt
Normal file
|
@ -0,0 +1,49 @@
|
|||
#lang typed/racket/base
|
||||
|
||||
(require phc-toolkit
|
||||
typed/rackunit
|
||||
type-expander
|
||||
racket/string
|
||||
(for-syntax racket/base
|
||||
phc-toolkit/untyped)
|
||||
(lib "phc-adt/ctx.hl.rkt"))
|
||||
|
||||
(provide ck
|
||||
ck-not
|
||||
check-print-type)
|
||||
|
||||
(define-syntax (ck stx)
|
||||
(syntax-case stx ()
|
||||
[(_ v t)
|
||||
(quasisyntax/top-loc stx
|
||||
(check-tc
|
||||
(require (only-in (lib "phc-adt/ctx.hl.rkt") set-adt-context-macro))
|
||||
(set-adt-context-macro #,(datum->syntax #'t 'there))
|
||||
(ann v t)))]))
|
||||
|
||||
(define-syntax (ck-not stx)
|
||||
(syntax-case stx ()
|
||||
[(_ v t)
|
||||
(quasisyntax/top-loc stx
|
||||
(check-not-tc
|
||||
(require (only-in (lib "phc-adt/ctx.hl.rkt") set-adt-context-macro))
|
||||
(set-adt-context-macro #,(datum->syntax #'t 'there))
|
||||
(ann v t)))]))
|
||||
|
||||
(: clean-type-str (→ String String))
|
||||
(define (clean-type-str type-str)
|
||||
(string-trim
|
||||
(regexp-replace* #px"(?-s:[ \n]+)"
|
||||
(regexp-replace #px"^- :" type-str "")
|
||||
" ")))
|
||||
|
||||
(define-syntax/case (check-print-type e str) ()
|
||||
(eval-tc
|
||||
(λ (f)
|
||||
(quasisyntax/top-loc stx
|
||||
(check-equal?: (clean-type-str (#,f)) str)))
|
||||
(quasisyntax/top-loc stx
|
||||
(begin (current-print (λ _ (void)))
|
||||
(require (only-in (lib "phc-adt/ctx.hl.rkt") set-adt-context-macro))
|
||||
(set-adt-context-macro #,(datum->syntax #'t 'there))
|
||||
e))))
|
|
@ -0,0 +1,5 @@
|
|||
#lang s-exp phc-adt/declarations
|
||||
(remembered! tagged-structure (foo-ct values))
|
||||
(remembered! tagged-structure (foo values))
|
||||
(remembered! tagged-structure (untagged a b))
|
||||
(remembered! tagged-structure (untagged a c))
|
53
phc-adt-test/phc-adt/test/mailing-list-example/example.rkt
Normal file
53
phc-adt-test/phc-adt/test/mailing-list-example/example.rkt
Normal file
|
@ -0,0 +1,53 @@
|
|||
#lang typed/racket
|
||||
;; phc-adt sets registers some information in its support files when it
|
||||
;; encounters new structure types, so this file has to be compiled twice
|
||||
;; in DrRacket (not on the command-line). The first compilation will fail,
|
||||
;; it is normal and expected (but it should soon give a better error message)
|
||||
(require type-expander multi-id phc-adt typed/rackunit)
|
||||
(adt-init)
|
||||
|
||||
;; This internally does
|
||||
;; (define-multi-id s1
|
||||
;; #:type-expander (λ (stx) …)
|
||||
;; #:match-expander (λ (stx) …)
|
||||
;; #:call (λ (stx) …)
|
||||
;; #:id (λ (stx) …))
|
||||
(define-structure s1 [a : Number] [b : String])
|
||||
|
||||
;; The "structure" identifier is also a multi-id, for on-the-fly usage of a
|
||||
;; structure as a type, match pattern, constructor function or instance creation
|
||||
|
||||
(: foo (→ (U
|
||||
;; type-expander: s1
|
||||
s1
|
||||
;; type-expander: (structure [field : type] …)
|
||||
(structure [a : Number] [c : Symbol]))
|
||||
Number))
|
||||
(define (foo s)
|
||||
(match s
|
||||
;; match-expander: (s1 pat ...)
|
||||
[(s1 (? number? the-a) the-b) (+ the-a (string-length the-b))]
|
||||
;; match-expander: (structure field-name …)
|
||||
[(structure a c) (+ a (string-length (symbol->string c)))]))
|
||||
|
||||
(define instances
|
||||
(append
|
||||
;; identifier macro: s1, to pretend it's a function
|
||||
(map s1
|
||||
(range 5)
|
||||
'("x" "xx" "xxx" "xxxx" "xxxxx"))
|
||||
;; macro: (s1 args …), to pretend it's a function call
|
||||
(list (s1 42 "Why does six times nine equal forty two?"))
|
||||
;; macro: (structure [field : type] …), produces a bulder function
|
||||
(map (structure [a : Number] [c : Symbol])
|
||||
(reverse (range 5))
|
||||
'(x xx xxx xxxx xxxxx))
|
||||
;; macro: (structure [field value] …) or (structure [field : type value]),
|
||||
;; produces an instance
|
||||
(list (structure [a pi] [c 'three-fourteen]))))
|
||||
|
||||
(check-equal? (map foo instances)
|
||||
(append '(1 3 5 7 9)
|
||||
'(82)
|
||||
'(5 5 5 5 5)
|
||||
'(17.141592653589793)))
|
31
phc-adt-test/phc-adt/test/node-low-level-quick-test.rkt
Normal file
31
phc-adt-test/phc-adt/test/node-low-level-quick-test.rkt
Normal file
|
@ -0,0 +1,31 @@
|
|||
#lang typed/racket
|
||||
(require (lib "phc-adt/node-low-level.hl.rkt")
|
||||
typed/rackunit
|
||||
phc-toolkit
|
||||
phc-adt)
|
||||
(adt-init "./mailing-list-example/adt-pre-declarations.rkt")
|
||||
|
||||
(define-constructor foo-ct #:tag foo :)
|
||||
|
||||
(check-not-exn
|
||||
(λ ()
|
||||
(equal-hash-code
|
||||
(|(node foo values)| (delay (list 'a 'b 'c))
|
||||
(raw-node (vector (list 'a 'b 'c)) 0)))))
|
||||
|
||||
(check-not-exn
|
||||
(λ ()
|
||||
(equal-secondary-hash-code
|
||||
(|(node foo values)| (delay (list 'a 'b 'c))
|
||||
(raw-node (vector (list 'a 'b 'c)) 0)))))
|
||||
|
||||
(check-true: (equal?
|
||||
(|(node foo values)| (delay (list 'a 'b 'c))
|
||||
(raw-node (vector (list 'a 'b 'c)) 0))
|
||||
(|(node foo values)| (delay (list 'a 'b'c))
|
||||
(raw-node (vector (list 'a 'b 'c)) 1))))
|
||||
(check-false: (equal?
|
||||
(|(node foo values)| (delay (list 'a 'b 'c))
|
||||
(raw-node (vector (list 'a 'b 'c)) 0))
|
||||
(|(node foo values)| (delay (list 'a 'x 'c))
|
||||
(raw-node (vector (list 'a 'x 'c)) 1))))
|
|
@ -0,0 +1,24 @@
|
|||
#lang s-exp phc-adt/declarations
|
||||
(remembered! tagged-structure (foo a b c))
|
||||
(remembered! tagged-structure (tag b))
|
||||
(remembered! tagged-structure (foo b))
|
||||
(remembered! tagged-structure (foo a c))
|
||||
(remembered! tagged-structure (bar a b c))
|
||||
(remembered! tagged-structure (baz a b))
|
||||
(remembered! tagged-structure (qux b d))
|
||||
(remembered! tagged-structure (bar a c))
|
||||
(remembered! tagged-structure (baz a))
|
||||
(remembered! tagged-structure (qux d))
|
||||
(remembered! tagged-structure (bar b))
|
||||
(remembered! tagged-structure (baz b))
|
||||
(remembered! tagged-structure (qux b))
|
||||
(remembered! tagged-structure (foo d))
|
||||
(remembered! tagged-structure (foo a))
|
||||
(remembered! tagged-structure (foo a b))
|
||||
(remembered! tagged-structure (foo a))
|
||||
(remembered! tagged-structure (foo a b))
|
||||
(remembered! tagged-structure (foo b d))
|
||||
(remembered! tagged-structure (foo a b d))
|
||||
(remembered! tagged-structure (foo a b c d))
|
||||
(remembered! tagged-structure (foo a d))
|
||||
(remembered! tagged-structure (foo a c d))
|
40
phc-adt-test/phc-adt/test/row-polymorphism/test-merge.rkt
Normal file
40
phc-adt-test/phc-adt/test/row-polymorphism/test-merge.rkt
Normal file
|
@ -0,0 +1,40 @@
|
|||
#lang type-expander
|
||||
|
||||
(require phc-adt phc-toolkit)
|
||||
(adt-init)
|
||||
|
||||
(check-equal?:
|
||||
(merge (tagged foo [b 'b1]) (tagged foo [a 'a1] [c 'c1])
|
||||
: (U [(foo b) (foo a c)]))
|
||||
: (tagged foo [a 'a1] [b 'b1] [c 'c1])
|
||||
(tagged foo [a 'a1] [b 'b1] [c 'c1]))
|
||||
|
||||
(check-equal?:
|
||||
(merge (tagged bar [b 'b2]) (tagged bar [a 'a2] [c 'c2])
|
||||
: (U [(bar b) (bar a c)] [(foo b) (foo a c)]))
|
||||
: (tagged bar [a 'a2] [b 'b2] [c 'c2])
|
||||
(tagged bar [a 'a2] [b 'b2] [c 'c2]))
|
||||
|
||||
(check-equal?:
|
||||
(merge (tagged baz [b 'b3]) (tagged baz [a 'a3])
|
||||
: (U [(baz b) (baz a)] [(foo b) (foo a c)]))
|
||||
: (tagged baz [a 'a3] [b 'b3])
|
||||
(tagged baz [a 'a3] [b 'b3]))
|
||||
|
||||
(check-equal?:
|
||||
(merge (tagged qux [b 'b4]) (tagged qux [d 'd4])
|
||||
: (U [(qux b) (qux d)] [(foo b) (foo a c)]))
|
||||
: (tagged qux [b 'b4] [d 'd4])
|
||||
(tagged qux [b 'b4] [d 'd4]))
|
||||
|
||||
;; Different tags
|
||||
(check-equal?:
|
||||
(merge (tagged qux [b 'b4]) (tagged foo [d 'd4])
|
||||
: (U [(qux b) (foo d)] [(foo b) (foo a c)]))
|
||||
: (tagged qux [b 'b4] [d 'd4])
|
||||
(tagged qux [b 'b4] [d 'd4]))
|
||||
(check-equal?:
|
||||
(merge (tagged qux [b 'b4]) (tagged foo [d 'd4])
|
||||
: (U [(qux b) (foo d)] [(foo b) (foo a)]))
|
||||
: (tagged qux [b 'b4] [d 'd4])
|
||||
(tagged qux [b 'b4] [d 'd4]))
|
25
phc-adt-test/phc-adt/test/row-polymorphism/test-split.rkt
Normal file
25
phc-adt-test/phc-adt/test/row-polymorphism/test-split.rkt
Normal file
|
@ -0,0 +1,25 @@
|
|||
#lang type-expander
|
||||
|
||||
(require phc-adt phc-toolkit)
|
||||
(adt-init)
|
||||
|
||||
(define-tagged foo [a : 'a1] [b : 'b1] [c : 'c1])
|
||||
(define-tagged bar [a : 'a2] [b : 'b2] [c : 'c2])
|
||||
(define-tagged baz [a : 'a3] [b : 'b3])
|
||||
(define-tagged qux [b : 'b4] [d : 'd4])
|
||||
|
||||
(let-values ([(x y) (split (foo 'a1 'b1 'c1) : (U (foo a b c)) b)])
|
||||
(check-equal?: x : (tagged foo [b 'b1]) (tagged foo [b 'b1]))
|
||||
(check-equal?: y : (tagged foo [a 'a1] [c 'c1]) (tagged foo [a 'a1] [c 'c1])))
|
||||
|
||||
(let-values ([(x y) (split (bar 'a2 'b2 'c2) : (U (foo a b c) (bar a b c)) b)])
|
||||
(check-equal?: x : (tagged bar [b 'b2]) (tagged bar [b 'b2]))
|
||||
(check-equal?: y : (tagged bar [a 'a2] [c 'c2]) (tagged bar [a 'a2] [c 'c2])))
|
||||
|
||||
(let-values ([(x y) (split (baz 'a3 'b3) : (U (foo a b c) (baz a b)) b)])
|
||||
(check-equal?: x : (tagged baz [b 'b3]) (tagged baz [b 'b3]))
|
||||
(check-equal?: y : (tagged baz [a 'a3]) (tagged baz [a 'a3])))
|
||||
|
||||
(let-values ([(x y) (split (qux 'b4 'd4) : (U (foo a b c) (qux b d)) b)])
|
||||
(check-equal?: x : (tagged qux [b 'b4]) (tagged qux [b 'b4]))
|
||||
(check-equal?: y : (tagged qux [d 'd4]) (tagged qux [d 'd4])))
|
|
@ -0,0 +1,62 @@
|
|||
#lang type-expander
|
||||
|
||||
(require phc-adt phc-toolkit)
|
||||
(adt-init)
|
||||
|
||||
(check-equal?: (with+ (tagged foo [b 'b1]) : (U (foo b)))
|
||||
: (tagged foo [b 'b1])
|
||||
(tagged foo [b 'b1]))
|
||||
(check-equal?: (with+ (tagged foo [b 'b1]) : (U (foo b)) [a 'a1])
|
||||
: (tagged foo [a 'a1] [b 'b1])
|
||||
(tagged foo [a 'a1] [b 'b1]))
|
||||
(check-equal?: (with+ (tagged foo [b 'b1]) : (U (foo b)) [a 'a1] [c 'c1])
|
||||
: (tagged foo [a 'a1] [b 'b1] [c 'c1])
|
||||
(tagged foo [a 'a1] [b 'b1] [c 'c1]))
|
||||
;; Correctly gave an error:
|
||||
;(check-equal?: (with+ (tagged foo [b 'b1]) : (U (foo b)) [a 'a1] [b 'b1-update])
|
||||
; : (tagged foo [a 'a1] [b 'b1-update])
|
||||
; (tagged foo [a 'a1] [b 'b1-update]))
|
||||
;(check-equal?: (with+ (tagged foo [b 'b1]) : (U (foo b)) [b 'b1-update])
|
||||
; : (tagged foo [b 'b1-update])
|
||||
; (tagged foo [b 'b1-update]))
|
||||
;(check-equal?: (with+ (tagged foo [b 'b1]) : (U (foo b)) [a 'a1] [b 'b1-update] [c 'c1])
|
||||
; : (tagged foo [a 'a1] [b 'b1-update] [c 'c1])
|
||||
; (tagged foo [a 'a1] [b 'b1-update] [c 'c1]))
|
||||
|
||||
(check-equal?: (with+ (tagged foo [b 'b1] [d 'd1]) : (U (foo b d)))
|
||||
: (tagged foo [b 'b1] [d 'd1])
|
||||
(tagged foo [b 'b1] [d 'd1]))
|
||||
(check-equal?: (with+ (tagged foo [b 'b1] [d 'd1]) : (U (foo b d)) [a 'a1])
|
||||
: (tagged foo [a 'a1] [b 'b1] [d 'd1])
|
||||
(tagged foo [a 'a1] [b 'b1] [d 'd1]))
|
||||
(check-equal?: (with+ (tagged foo [b 'b1] [d 'd1]) : (U (foo b d)) [a 'a1] [c 'c1])
|
||||
: (tagged foo [a 'a1] [b 'b1] [c 'c1] [d 'd1])
|
||||
(tagged foo [a 'a1] [b 'b1] [c 'c1] [d 'd1]))
|
||||
;(check-equal?: (with+ (tagged foo [b 'b1] [d 'd1]) : (U (foo b d)) [a 'a1] [b 'b1-update])
|
||||
; : (tagged foo [a 'a1] [b 'b1-update] [d 'd1])
|
||||
; (tagged foo [a 'a1] [b 'b1-update] [d 'd1]))
|
||||
;(check-equal?: (with+ (tagged foo [b 'b1] [d 'd1]) : (U (foo b d)) [b 'b1-update])
|
||||
; : (tagged foo [b 'b1-update] [d 'd1])
|
||||
; (tagged foo [b 'b1-update] [d 'd1]))
|
||||
;(check-equal?: (with+ (tagged foo [b 'b1] [d 'd1]) : (U (foo b d)) [a 'a1] [b 'b1-update] [c 'c1])
|
||||
; : (tagged foo [a 'a1] [b 'b1-update] [c 'c1] [d 'd1])
|
||||
; (tagged foo [a 'a1] [b 'b1-update] [c 'c1] [d 'd1]))
|
||||
|
||||
(check-equal?: (with+ (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)))
|
||||
: (tagged foo [b 'b1] [d 'd1])
|
||||
(tagged foo [b 'b1] [d 'd1]))
|
||||
;(check-equal?: (with+ (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)) [a 'a1])
|
||||
; : (tagged foo [a 'a1] [b 'b1] [d 'd1])
|
||||
; (tagged foo [a 'a1] [b 'b1] [d 'd1]))
|
||||
;(check-equal?: (with+ (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)) [a 'a1] [c 'c1])
|
||||
; : (tagged foo [a 'a1] [b 'b1] [c 'c1] [d 'd1])
|
||||
; (tagged foo [a 'a1] [b 'b1] [c 'c1] [d 'd1]))
|
||||
;(check-equal?: (with+ (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)) [a 'a1] [b 'b1-update])
|
||||
; : (tagged foo [a 'a1] [b 'b1-update] [d 'd1])
|
||||
; (tagged foo [a 'a1] [b 'b1-update] [d 'd1]))
|
||||
;(check-equal?: (with+ (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)) [b 'b1-update])
|
||||
; : (tagged foo [b 'b1-update] [d 'd1])
|
||||
; (tagged foo [b 'b1-update] [d 'd1]))
|
||||
;(check-equal?: (with+ (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)) [a 'a1] [b 'b1-update] [c 'c1])
|
||||
; : (tagged foo [a 'a1] [b 'b1-update] [c 'c1] [d 'd1])
|
||||
; (tagged foo [a 'a1] [b 'b1-update] [c 'c1] [d 'd1]))
|
|
@ -0,0 +1,65 @@
|
|||
#lang type-expander
|
||||
|
||||
(require phc-adt phc-toolkit)
|
||||
(adt-init)
|
||||
|
||||
;; Correctly gave an error:
|
||||
(check-equal?: (with!! (tagged foo [b 'b1]) : (U (foo b)))
|
||||
: (tagged foo [b 'b1])
|
||||
(tagged foo [b 'b1]))
|
||||
;(check-equal?: (with!! (tagged foo [b 'b1]) : (U (foo b)) [a 'a1])
|
||||
; : (tagged foo [a 'a1] [b 'b1])
|
||||
; (tagged foo [a 'a1] [b 'b1]))
|
||||
;(check-equal?: (with!! (tagged foo [b 'b1]) : (U (foo b)) [a 'a1] [c 'c1])
|
||||
; : (tagged foo [a 'a1] [b 'b1] [c 'c1])
|
||||
; (tagged foo [a 'a1] [b 'b1] [c 'c1]))
|
||||
;(check-equal?: (with!! (tagged foo [b 'b1]) : (U (foo b)) [a 'a1] [b 'b1-update])
|
||||
; : (tagged foo [a 'a1] [b 'b1-update])
|
||||
; (tagged foo [a 'a1] [b 'b1-update]))
|
||||
(check-equal?: (with!! (tagged foo [b 'b1]) : (U (foo b)) [b 'b1-update])
|
||||
: (tagged foo [b 'b1-update])
|
||||
(tagged foo [b 'b1-update]))
|
||||
;(check-equal?: (with!! (tagged foo [b 'b1]) : (U (foo b)) [a 'a1] [b 'b1-update] [c 'c1])
|
||||
; : (tagged foo [a 'a1] [b 'b1-update] [c 'c1])
|
||||
; (tagged foo [a 'a1] [b 'b1-update] [c 'c1]))
|
||||
|
||||
(check-equal?: (with!! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d)))
|
||||
: (tagged foo [b 'b1] [d 'd1])
|
||||
(tagged foo [b 'b1] [d 'd1]))
|
||||
;(check-equal?: (with!! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d)) [a 'a1])
|
||||
; : (tagged foo [a 'a1] [b 'b1] [d 'd1])
|
||||
; (tagged foo [a 'a1] [b 'b1] [d 'd1]))
|
||||
;(check-equal?: (with!! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d)) [a 'a1] [c 'c1])
|
||||
; : (tagged foo [a 'a1] [b 'b1] [c 'c1] [d 'd1])
|
||||
; (tagged foo [a 'a1] [b 'b1] [c 'c1] [d 'd1]))
|
||||
;(check-equal?: (with!! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d)) [a 'a1] [b 'b1-update])
|
||||
; : (tagged foo [a 'a1] [b 'b1-update] [d 'd1])
|
||||
; (tagged foo [a 'a1] [b 'b1-update] [d 'd1]))
|
||||
(check-equal?: (with!! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d)) [b 'b1-update])
|
||||
: (tagged foo [b 'b1-update] [d 'd1])
|
||||
(tagged foo [b 'b1-update] [d 'd1]))
|
||||
;(check-equal?: (with!! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d)) [a 'a1] [b 'b1-update] [c 'c1])
|
||||
; : (tagged foo [a 'a1] [b 'b1-update] [c 'c1] [d 'd1])
|
||||
; (tagged foo [a 'a1] [b 'b1-update] [c 'c1] [d 'd1]))
|
||||
|
||||
(check-equal?: (with!! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)))
|
||||
: (tagged foo [b 'b1] [d 'd1])
|
||||
(tagged foo [b 'b1] [d 'd1]))
|
||||
;(check-equal?: (with!! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)) [a 'a1])
|
||||
; : (tagged foo [a 'a1] [b 'b1] [d 'd1])
|
||||
; (tagged foo [a 'a1] [b 'b1] [d 'd1]))
|
||||
;(check-equal?: (with!! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)) [a 'a1] [c 'c1])
|
||||
; : (tagged foo [a 'a1] [b 'b1] [c 'c1] [d 'd1])
|
||||
; (tagged foo [a 'a1] [b 'b1] [c 'c1] [d 'd1]))
|
||||
;(check-equal?: (with!! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)) [a 'a1] [b 'b1-update])
|
||||
; : (tagged foo [a 'a1] [b 'b1-update] [d 'd1])
|
||||
; (tagged foo [a 'a1] [b 'b1-update] [d 'd1]))
|
||||
;(check-equal?: (with!! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)) [b 'b1-update])
|
||||
; : (tagged foo [b 'b1-update] [d 'd1])
|
||||
; (tagged foo [b 'b1-update] [d 'd1]))
|
||||
;(check-equal?: (with!! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)) [a 'a1] [b 'b1-update] [c 'c1])
|
||||
; : (tagged foo [a 'a1] [b 'b1-update] [c 'c1] [d 'd1])
|
||||
; (tagged foo [a 'a1] [b 'b1-update] [c 'c1] [d 'd1]))
|
||||
(check-equal?: (with!! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)) [d 'd1-update])
|
||||
: (tagged foo [b 'b1] [d 'd1-update])
|
||||
(tagged foo [b 'b1] [d 'd1-update]))
|
61
phc-adt-test/phc-adt/test/row-polymorphism/test-with.rkt
Normal file
61
phc-adt-test/phc-adt/test/row-polymorphism/test-with.rkt
Normal file
|
@ -0,0 +1,61 @@
|
|||
#lang type-expander
|
||||
|
||||
(require phc-adt phc-toolkit)
|
||||
(adt-init)
|
||||
|
||||
(check-equal?: (with! (tagged foo [b 'b1]) : (U (foo b)))
|
||||
: (tagged foo [b 'b1])
|
||||
(tagged foo [b 'b1]))
|
||||
(check-equal?: (with! (tagged foo [b 'b1]) : (U (foo b)) [a 'a1])
|
||||
: (tagged foo [a 'a1] [b 'b1])
|
||||
(tagged foo [a 'a1] [b 'b1]))
|
||||
(check-equal?: (with! (tagged foo [b 'b1]) : (U (foo b)) [a 'a1] [c 'c1])
|
||||
: (tagged foo [a 'a1] [b 'b1] [c 'c1])
|
||||
(tagged foo [a 'a1] [b 'b1] [c 'c1]))
|
||||
(check-equal?: (with! (tagged foo [b 'b1]) : (U (foo b)) [a 'a1] [b 'b1-update])
|
||||
: (tagged foo [a 'a1] [b 'b1-update])
|
||||
(tagged foo [a 'a1] [b 'b1-update]))
|
||||
(check-equal?: (with! (tagged foo [b 'b1]) : (U (foo b)) [b 'b1-update])
|
||||
: (tagged foo [b 'b1-update])
|
||||
(tagged foo [b 'b1-update]))
|
||||
(check-equal?: (with! (tagged foo [b 'b1]) : (U (foo b)) [a 'a1] [b 'b1-update] [c 'c1])
|
||||
: (tagged foo [a 'a1] [b 'b1-update] [c 'c1])
|
||||
(tagged foo [a 'a1] [b 'b1-update] [c 'c1]))
|
||||
|
||||
(check-equal?: (with! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d)))
|
||||
: (tagged foo [b 'b1] [d 'd1])
|
||||
(tagged foo [b 'b1] [d 'd1]))
|
||||
(check-equal?: (with! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d)) [a 'a1])
|
||||
: (tagged foo [a 'a1] [b 'b1] [d 'd1])
|
||||
(tagged foo [a 'a1] [b 'b1] [d 'd1]))
|
||||
(check-equal?: (with! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d)) [a 'a1] [c 'c1])
|
||||
: (tagged foo [a 'a1] [b 'b1] [c 'c1] [d 'd1])
|
||||
(tagged foo [a 'a1] [b 'b1] [c 'c1] [d 'd1]))
|
||||
(check-equal?: (with! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d)) [a 'a1] [b 'b1-update])
|
||||
: (tagged foo [a 'a1] [b 'b1-update] [d 'd1])
|
||||
(tagged foo [a 'a1] [b 'b1-update] [d 'd1]))
|
||||
(check-equal?: (with! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d)) [b 'b1-update])
|
||||
: (tagged foo [b 'b1-update] [d 'd1])
|
||||
(tagged foo [b 'b1-update] [d 'd1]))
|
||||
(check-equal?: (with! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d)) [a 'a1] [b 'b1-update] [c 'c1])
|
||||
: (tagged foo [a 'a1] [b 'b1-update] [c 'c1] [d 'd1])
|
||||
(tagged foo [a 'a1] [b 'b1-update] [c 'c1] [d 'd1]))
|
||||
|
||||
(check-equal?: (with! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)))
|
||||
: (tagged foo [b 'b1] [d 'd1])
|
||||
(tagged foo [b 'b1] [d 'd1]))
|
||||
(check-equal?: (with! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)) [a 'a1])
|
||||
: (tagged foo [a 'a1] [b 'b1] [d 'd1])
|
||||
(tagged foo [a 'a1] [b 'b1] [d 'd1]))
|
||||
(check-equal?: (with! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)) [a 'a1] [c 'c1])
|
||||
: (tagged foo [a 'a1] [b 'b1] [c 'c1] [d 'd1])
|
||||
(tagged foo [a 'a1] [b 'b1] [c 'c1] [d 'd1]))
|
||||
(check-equal?: (with! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)) [a 'a1] [b 'b1-update])
|
||||
: (tagged foo [a 'a1] [b 'b1-update] [d 'd1])
|
||||
(tagged foo [a 'a1] [b 'b1-update] [d 'd1]))
|
||||
(check-equal?: (with! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)) [b 'b1-update])
|
||||
: (tagged foo [b 'b1-update] [d 'd1])
|
||||
(tagged foo [b 'b1-update] [d 'd1]))
|
||||
(check-equal?: (with! (tagged foo [b 'b1] [d 'd1]) : (U (foo b d) (foo a d)) [a 'a1] [b 'b1-update] [c 'c1])
|
||||
: (tagged foo [a 'a1] [b 'b1-update] [c 'c1] [d 'd1])
|
||||
(tagged foo [a 'a1] [b 'b1-update] [c 'c1] [d 'd1]))
|
18
phc-adt-test/phc-adt/test/test-adt-init-error.rkt
Normal file
18
phc-adt-test/phc-adt/test/test-adt-init-error.rkt
Normal file
|
@ -0,0 +1,18 @@
|
|||
#lang typed/racket
|
||||
|
||||
(require phc-adt
|
||||
syntax/macro-testing
|
||||
typed/rackunit
|
||||
type-expander)
|
||||
|
||||
(define compile-time-exception-regexp
|
||||
#px"phc-adt: adt-init must be called before using the features in phc-adt")
|
||||
(check-exn compile-time-exception-regexp
|
||||
(λ () (convert-compile-time-error
|
||||
(tagged tg [x "x"] [y 2]))))
|
||||
(check-exn compile-time-exception-regexp
|
||||
(λ () (convert-compile-time-error
|
||||
(match '() [(tagged tg x y) 'a] [_ 'b]))))
|
||||
(check-exn compile-time-exception-regexp
|
||||
(λ () (convert-compile-time-error
|
||||
(let () (define-type foo (tagged tg x y)) 1))))
|
11
phc-adt-test/phc-adt/test/test-adt-init-no-error.rkt
Normal file
11
phc-adt-test/phc-adt/test/test-adt-init-no-error.rkt
Normal file
|
@ -0,0 +1,11 @@
|
|||
#lang typed/racket
|
||||
|
||||
(require phc-adt
|
||||
syntax/macro-testing
|
||||
typed/rackunit
|
||||
type-expander)
|
||||
|
||||
(adt-init)
|
||||
(check-not-exn (λ () (tagged t1 [x "x"] [y "2"])))
|
||||
(check-not-exn (λ () (match '() [(tagged tg x y) 'a] [_ 'b])))
|
||||
(check-not-exn (λ () (let () (define-type foo (tagged tg x y)) 1)))
|
18
phc-adt-test/phc-adt/test/test-adt-structure-wrapped.rkt
Normal file
18
phc-adt-test/phc-adt/test/test-adt-structure-wrapped.rkt
Normal file
|
@ -0,0 +1,18 @@
|
|||
#lang typed/racket
|
||||
|
||||
(require phc-adt
|
||||
phc-toolkit
|
||||
type-expander
|
||||
typed/rackunit
|
||||
(only-in (lib "phc-adt/tagged-structure-low-level.hl.rkt")
|
||||
λ-tagged-get-field))
|
||||
(adt-init)
|
||||
|
||||
(define i (structure [a 1 : Number] [b "b" : String]))
|
||||
(define c (structure [a : Number] [b : String]))
|
||||
(define i2 (c 1 "b"))
|
||||
|
||||
(check-equal?: (uniform-get i a) : Number 1)
|
||||
(check-equal?: (uniform-get i b) : String "b")
|
||||
(check-equal?: (uniform-get i2 a) : Number 1)
|
||||
(check-equal?: (uniform-get i2 b) : String "b")
|
22
phc-adt-test/phc-adt/test/test-adt.rkt
Normal file
22
phc-adt-test/phc-adt/test/test-adt.rkt
Normal file
|
@ -0,0 +1,22 @@
|
|||
#lang typed/racket
|
||||
|
||||
(require phc-adt
|
||||
phc-toolkit
|
||||
type-expander
|
||||
typed/rackunit
|
||||
(only-in (lib "phc-adt/tagged-structure-low-level.hl.rkt")
|
||||
λ-tagged-get-field))
|
||||
(adt-init)
|
||||
|
||||
(define-tagged st2 [b String] [a Number])
|
||||
|
||||
((tagged t a b c) 1 'b "c")
|
||||
((tagged t a [b] c) 1 'b "c")
|
||||
((tagged t [a] [b] [c]) 1 'b "c")
|
||||
((tagged t [a : Number] [b : Symbol] [c : String]) 1 'b "c")
|
||||
(tagged t [a : Number 1] [b : Symbol 'b] [c : String "c"])
|
||||
(tagged t [a 1] [b 'b] [c "c"])
|
||||
|
||||
(tagged t [a 1] [b 'b] [c "c"])
|
||||
|
||||
(define-tagged tabc [a 1] [b 'b] [c "c"])
|
126
phc-adt-test/phc-adt/test/test-constructor.rkt
Normal file
126
phc-adt-test/phc-adt/test/test-constructor.rkt
Normal file
|
@ -0,0 +1,126 @@
|
|||
#lang typed/racket
|
||||
|
||||
(require phc-adt phc-toolkit type-expander typed/rackunit)
|
||||
(adt-init)
|
||||
|
||||
; (_ name:id *)
|
||||
(check-not-exn
|
||||
(λ ()
|
||||
(ann (constructor tag1 *)
|
||||
(∀ (A ...) (→ A ... A (tagged tag1 [values (List A ... A)]))))))
|
||||
|
||||
(check-equal?: (constructor-values
|
||||
(ann ((constructor tag1 *))
|
||||
(constructor tag1)))
|
||||
'())
|
||||
(check-equal?: (constructor-values
|
||||
(ann ((constructor tag1 *) 1)
|
||||
(constructor tag1 One)))
|
||||
'(1))
|
||||
(check-equal?: (constructor-values
|
||||
(ann ((constructor tag1 *) 1 "b")
|
||||
(constructor tag1 Number String)))
|
||||
'(1 "b"))
|
||||
(check-equal?: (constructor-values
|
||||
(ann ((constructor tag1 *) 1 "b" 'c)
|
||||
(constructor tag1 Number String 'c)))
|
||||
'(1 "b" c))
|
||||
|
||||
; (_ name:id :colon)
|
||||
(check-not-exn (λ () (ann (constructor tag1 :)
|
||||
(→ (constructor tag1)))))
|
||||
|
||||
; (_ name:id :colon T₀:expr)
|
||||
(check-not-exn (λ () (ann (constructor tag1 : Number)
|
||||
(→ Number (constructor tag1 Number)))))
|
||||
|
||||
; (_ name:id :colon Tᵢ:expr …+)
|
||||
(check-not-exn
|
||||
(λ () (ann (constructor tag1 : Number String)
|
||||
(→ Number String (constructor tag1 Number String)))))
|
||||
(check-not-exn
|
||||
(λ () (ann (constructor tag1 : Number String 'c)
|
||||
(→ Number String 'c (constructor tag1 Number String 'c)))))
|
||||
|
||||
; Call (_ name:id :colon)
|
||||
(check-equal?: (constructor-values (ann ((constructor tag1 :))
|
||||
(constructor tag1)))
|
||||
'())
|
||||
|
||||
; Call (_ name:id :colon T₀:expr)
|
||||
(check-equal?: (constructor-values (ann ((constructor tag1 : Number) 1)
|
||||
(constructor tag1 Number)))
|
||||
'(1))
|
||||
|
||||
; Call (_ name:id :colon Tᵢ:expr …+)
|
||||
(check-equal?: (constructor-values
|
||||
(ann ((constructor tag1 : Number String) 1 "b")
|
||||
(constructor tag1 Number String)))
|
||||
'(1 "b"))
|
||||
(check-equal?: (constructor-values
|
||||
(ann ((constructor tag1 : Number String 'c) 1 "b" 'c)
|
||||
(constructor tag1 Number String 'c)))
|
||||
'(1 "b" c))
|
||||
|
||||
; (_ name:id [val₀:expr :colon T₀:expr])
|
||||
(check-equal?: (constructor-values
|
||||
(ann (constructor tag1 [1 : One])
|
||||
(constructor tag1 One)))
|
||||
'(1))
|
||||
|
||||
; (_ name:id [valᵢ:expr :colon Tᵢ:expr] …)
|
||||
(check-equal?: (constructor-values
|
||||
(ann (constructor tag1 [1 : One] ["b" : (U "b" "B")])
|
||||
(constructor tag1 One (U "b" "B"))))
|
||||
'(1 "b"))
|
||||
|
||||
; (_ name:id val₀:expr)
|
||||
(check-equal?: (constructor-values
|
||||
(ann (constructor tag1 "a")
|
||||
(constructor tag1 String)))
|
||||
'("a"))
|
||||
|
||||
; (_ name:id valᵢ:expr …)
|
||||
(check-equal?: (constructor-values
|
||||
(ann (constructor tag1 "a" "b")
|
||||
(constructor tag1 String String)))
|
||||
'("a" "b"))
|
||||
(check-equal?: (constructor-values
|
||||
(ann (constructor tag1 "a" "b" 'c)
|
||||
(constructor tag1 String String Symbol)))
|
||||
'("a" "b" c))
|
||||
|
||||
(check-equal?: (constructor-values
|
||||
(constructor tag1 "a" #(#\b 3) #t . "a"))
|
||||
: (List* "a" (Vector Char 3) #t "a")
|
||||
'("a" #(#\b 3) #t . "a"))
|
||||
|
||||
(check-equal?: (constructor-values
|
||||
(constructor tag1 "a" #(#\b 3) #t . #\b))
|
||||
: (List* "a" (Vector Char 3) #t Char)
|
||||
'("a" #(#\b 3) #t . #\b))
|
||||
|
||||
(check-equal?: (constructor-values
|
||||
(constructor tag1 "a" #(#\b 3) #t . 3))
|
||||
: (List* "a" (Vector Char 3) #t 3)
|
||||
'("a" #(#\b 3) #t . 3))
|
||||
|
||||
(check-equal?: (constructor-values
|
||||
(constructor tag1 "a" #(#\b 3) #t . #t))
|
||||
: (List* "a" (Vector Char 3) #t #t)
|
||||
'("a" #(#\b 3) #t . #t))
|
||||
|
||||
(check-equal?: (constructor-values
|
||||
(constructor tag1 "a" #(#\b 3) #t . #("x" #\y 26 #f)))
|
||||
: (List* "a" (Vector Char 3) #t (Vector "x" Char 26 #f))
|
||||
'("a" #(#\b 3) #t . #("x" #\y 26 #f)))
|
||||
|
||||
(check-equal?: (constructor-values
|
||||
(constructor tag1 "a" #(#\b 3) #t #("x" #\y 26 #f) . 123))
|
||||
: (List* "a" (Vector Char 3) #t (Vector "x" Char 26 #f) 123)
|
||||
'("a" #(#\b 3) #t #("x" #\y 26 #f) . 123))
|
||||
|
||||
(check-equal?: (constructor-values
|
||||
(constructor tag1 "a" #(#\b 3) #t #("x" #\y 26 #f)))
|
||||
: (List "a" (Vector Char 3) #t (Vector "x" Char 26 #f))
|
||||
'("a" #(#\b 3) #t #("x" #\y 26 #f)))
|
33
phc-adt-test/phc-adt/test/test-constructor2.rkt
Normal file
33
phc-adt-test/phc-adt/test/test-constructor2.rkt
Normal file
|
@ -0,0 +1,33 @@
|
|||
#lang typed/racket
|
||||
|
||||
(require phc-adt phc-toolkit type-expander typed/rackunit)
|
||||
(adt-init)
|
||||
|
||||
(check-equal?: (constructor-values
|
||||
(ann (constructor a 1 "x")
|
||||
;; TODO: Make a (Constructor-AnyTag …) type expander.
|
||||
(tagged a [values (List Number String)])))
|
||||
(list 1 "x"))
|
||||
(check-equal?: (constructor-values
|
||||
(ann (constructor a 1 "x")
|
||||
(tagged a [values Any])))
|
||||
(list 1 "x"))
|
||||
(check-equal?: (constructor-values
|
||||
(ann (constructor a 1 "x")
|
||||
(constructor a Number String)))
|
||||
(list 1 "x"))
|
||||
(check-equal?: (constructor-values
|
||||
(ann (constructor b)
|
||||
(constructor b)))
|
||||
(list))
|
||||
(check-equal?: (constructor-values
|
||||
(ann (constructor c 'd)
|
||||
(constructor c Symbol)))
|
||||
'(d))
|
||||
(check-equal?: (ann (constructor c 2 "y")
|
||||
(constructor c Number String))
|
||||
(constructor c 2 "y"))
|
||||
(check-not-equal?: (constructor d 2 "y")
|
||||
(constructor d 2 "y" 'z))
|
||||
(check-not-equal?: (constructor e 2 "y")
|
||||
(constructor F 2 "y"))
|
69
phc-adt-test/phc-adt/test/test-constructor3.rkt
Normal file
69
phc-adt-test/phc-adt/test/test-constructor3.rkt
Normal file
|
@ -0,0 +1,69 @@
|
|||
#lang typed/racket
|
||||
|
||||
(require phc-adt phc-toolkit type-expander typed/rackunit)
|
||||
(adt-init)
|
||||
|
||||
;; define-constructor
|
||||
(define-constructor tag0 :)
|
||||
(define-constructor tag1 : Number)
|
||||
(define-constructor tag2 : Number String)
|
||||
(define-constructor tag3 : Number String 'c)
|
||||
|
||||
;; Type expander
|
||||
(check-equal?: (constructor-values (ann (constructor tag0) tag0))
|
||||
'())
|
||||
(check-equal?: (constructor-values (ann (constructor tag1 1) tag1))
|
||||
'(1))
|
||||
(check-equal?: (constructor-values (ann (constructor tag2 1 "b") tag2))
|
||||
'(1 "b"))
|
||||
(check-equal?: (constructor-values
|
||||
(ann (constructor tag3 1 "b" (ann 'c 'c)) tag3))
|
||||
'(1 "b" c))
|
||||
|
||||
;; Call
|
||||
(check-equal?: (constructor-values (ann (tag0) (constructor tag0)))
|
||||
'())
|
||||
(check-equal?: (constructor-values (ann (tag1 1) (constructor tag1 Number)))
|
||||
'(1))
|
||||
(check-equal?: (constructor-values
|
||||
(ann (tag2 1 "b") (constructor tag2 Number String)))
|
||||
'(1 "b"))
|
||||
(check-equal?: (constructor-values
|
||||
(ann (tag3 1 "b" 'c) (constructor tag3 Number String 'c)))
|
||||
'(1 "b" c))
|
||||
|
||||
;; Id
|
||||
(check-not-exn (λ () (ann tag0 (→ (constructor tag0)))))
|
||||
(check-not-exn (λ () (ann tag1 (→ Number (constructor tag1 Number)))))
|
||||
(check-not-exn
|
||||
(λ () (ann tag2 (→ Number String (constructor tag2 Number String)))))
|
||||
(check-not-exn
|
||||
(λ () (ann tag3 (→ Number String 'c (constructor tag3 Number String 'c)))))
|
||||
|
||||
;; Match expander
|
||||
(check-equal?: (ann (match (constructor tag0) [(tag0) #t]) #t)
|
||||
#t)
|
||||
(check-equal?: (ann (match (constructor tag1 1) [(tag1 x) (list x)])
|
||||
(List Number))
|
||||
'(1))
|
||||
(check-equal?: (ann (match (constructor tag2 1 "b") [(tag2 x y) (list y x)])
|
||||
(List String Number))
|
||||
'("b" 1))
|
||||
(check-equal?: (ann (match (constructor tag3 1 "b" (ann 'c 'c))
|
||||
[(tag3 x y z) (list z y x)])
|
||||
(List 'c String Number))
|
||||
'(c "b" 1))
|
||||
|
||||
;; Match expander which single pattern
|
||||
(check-equal?: (ann (match (constructor tag0) [(tag0 #:rest whole) whole]) Null)
|
||||
'())
|
||||
(check-equal?: (ann (match (constructor tag1 1) [(tag1 x) x])
|
||||
Number)
|
||||
'1)
|
||||
(check-equal?: (ann (match (constructor tag2 1 "b") [(tag2 x y) (list x y)])
|
||||
(List Number String))
|
||||
'(1 "b"))
|
||||
(check-equal?: (ann (match (constructor tag3 1 "b" (ann 'c 'c))
|
||||
[(tag3 x y z) (list x y z)])
|
||||
(List Number String 'c))
|
||||
'(1 "b" c))
|
195
phc-adt-test/phc-adt/test/test-define-adt.rkt
Normal file
195
phc-adt-test/phc-adt/test/test-define-adt.rkt
Normal file
|
@ -0,0 +1,195 @@
|
|||
#lang typed/racket
|
||||
|
||||
(require phc-adt
|
||||
phc-toolkit
|
||||
type-expander
|
||||
typed/rackunit
|
||||
(only-in (lib "phc-adt/tagged-structure-low-level.hl.rkt")
|
||||
λ-tagged-get-field))
|
||||
(adt-init)
|
||||
|
||||
;(define-constructor c2 : Fixnum String)
|
||||
|
||||
;; define-tagged
|
||||
(begin
|
||||
(define-tagged tagged-s1)
|
||||
(define-tagged tagged-s2 [f Fixnum] [g String])
|
||||
(define-tagged tagged-s3 [g String] [f Fixnum])
|
||||
(define-tagged tagged-s4 [f Fixnum] [g String])
|
||||
|
||||
(check-equal?: (match (ann (tagged-s1) (tagged tagged-s1))
|
||||
[(tagged-s1) #t])
|
||||
#t)
|
||||
|
||||
(check-equal?: (match (ann (tagged-s2 99 "z") tagged-s2)
|
||||
[(tagged-s2 f g) (cons g f)])
|
||||
'("z" . 99))
|
||||
|
||||
(let ()
|
||||
(check-equal?: (match (ann (tagged-s2 99 "in-let") tagged-s2)
|
||||
[(tagged-s2 f g) (cons g f)])
|
||||
'("in-let" . 99)))
|
||||
|
||||
(define (test-match val)
|
||||
(match val
|
||||
[(tagged-s2 x y) (list 'found-s2 y x)]
|
||||
[(tagged-s3 x y) (list 'found-s3 y x)]
|
||||
[(tagged-s4 x y) (list 'found-s4 y x)]))
|
||||
|
||||
(check-equal?:
|
||||
(test-match (ann (tagged-s2 2 "flob")
|
||||
(tagged tagged-s2 [f Fixnum] [g String])))
|
||||
'(found-s2 "flob" 2))
|
||||
|
||||
(check-equal?:
|
||||
(test-match (ann (tagged-s3 "flob" 2)
|
||||
(tagged tagged-s3 [g String] [f Fixnum])))
|
||||
'(found-s3 2 "flob"))
|
||||
|
||||
;; g and f are inverted in the “ann”
|
||||
(check-equal?:
|
||||
(test-match (ann (tagged-s4 2 "flob")
|
||||
(tagged tagged-s4 [g String] [f Fixnum])))
|
||||
'(found-s4 "flob" 2))
|
||||
|
||||
(define (test-match-verbose val)
|
||||
(match val
|
||||
[(tagged tagged-s2 g [f y]) (list 'found-s2 g y)]
|
||||
[(tagged tagged-s3 [g y] f) (list 'found-s2 f y)]
|
||||
[(tagged tagged-s4 [f y] g) (list 'found-s2 g y)]))
|
||||
|
||||
(check-equal?:
|
||||
(test-match (ann (tagged-s2 3 "flob")
|
||||
(tagged tagged-s2 [f Fixnum] [g String])))
|
||||
'(found-s2 "flob" 3))
|
||||
|
||||
;; g and f are inverted in the “ann”
|
||||
(check-equal?:
|
||||
(test-match (ann (tagged-s3 "flob" 3)
|
||||
(tagged tagged-s3 [f Fixnum] [g String])))
|
||||
'(found-s3 3 "flob"))
|
||||
|
||||
(check-equal?:
|
||||
(test-match (ann (tagged-s4 3 "flob")
|
||||
(tagged tagged-s4 [f Fixnum] [g String])))
|
||||
'(found-s4 "flob" 3))
|
||||
|
||||
(check-not-equal?: (tagged-s2 4 "flob")
|
||||
(tagged-s3 "flob" 4))
|
||||
(check-not-equal?: (tagged-s2 4 "flob")
|
||||
(tagged-s4 4 "flob")))
|
||||
|
||||
;; define-constructor
|
||||
(begin
|
||||
(define-constructor c1 :)
|
||||
(define-constructor c2 : Fixnum String)
|
||||
(define-constructor c3 : Fixnum String)
|
||||
|
||||
(check-equal?: (match (ann (c1) (constructor c1))
|
||||
[(c1) #t])
|
||||
#t)
|
||||
|
||||
(check-equal?: (match (ann (c2 99 "z") c2)
|
||||
[(c2 f g) (cons g f)])
|
||||
'("z" . 99))
|
||||
|
||||
(let ()
|
||||
(check-equal?: (match (ann (c2 99 "in-let") c2)
|
||||
[(c2 f g) (cons g f)])
|
||||
'("in-let" . 99)))
|
||||
|
||||
(define (test-c-match val)
|
||||
(match val
|
||||
[(c1) (list 'found-c1)]
|
||||
[(constructor c2 x y z) (list 'found-c2-xyz z y x)]
|
||||
[(c2 x y) (list 'found-c2 y x)]
|
||||
[(c3 x y) (list 'found-c3 y x)]))
|
||||
|
||||
(check-equal?:
|
||||
(test-c-match (ann (c2 2 "flob")
|
||||
(constructor c2 Fixnum String)))
|
||||
'(found-c2 "flob" 2))
|
||||
|
||||
(check-equal?:
|
||||
(test-c-match (ann (c3 2 "flob")
|
||||
(constructor c3 Fixnum String)))
|
||||
'(found-c3 "flob" 2)))
|
||||
|
||||
;; define-tagged (used to use #:private, updated the tests now that the option
|
||||
;; was removed).
|
||||
(begin
|
||||
(define-syntax-rule (defp make mt)
|
||||
(begin
|
||||
(define-tagged txyz #:? txyz?
|
||||
[a Number]
|
||||
[b String])
|
||||
|
||||
(define (make) (txyz 1 "b"))
|
||||
|
||||
(define (mt v)
|
||||
(match v
|
||||
((txyz x y) (list 'macro y x))
|
||||
(_ #f)))))
|
||||
|
||||
(defp make mt)
|
||||
|
||||
(define-tagged txyz #:? txyz?
|
||||
[a Number]
|
||||
[b String])
|
||||
|
||||
(check-equal?: (match (make)
|
||||
((tagged txyz x y) (list 'out y x))
|
||||
(_ #f))
|
||||
#f)
|
||||
|
||||
(check-equal?: (mt (tagged txyz [x 1] [y "b"]))
|
||||
#f)
|
||||
|
||||
(check-equal?: (mt (make))
|
||||
'(macro "b" 1))
|
||||
|
||||
(check-equal?: (make) (txyz 1 "b"))
|
||||
(check-equal?: (match (make)
|
||||
((txyz x y) (list 'out y x))
|
||||
(_ #f))
|
||||
'(out "b" 1))
|
||||
|
||||
(check-equal?: (mt (txyz 1 "b"))
|
||||
'(macro "b" 1)))
|
||||
|
||||
;; define-constructor #:private
|
||||
(begin
|
||||
(define-syntax-rule (defpc makec mtc)
|
||||
(begin
|
||||
(define-constructor cxyz #:? cxyz? : Number String)
|
||||
|
||||
(define (makec) (cxyz 1 "b"))
|
||||
|
||||
(define (mtc v)
|
||||
(match v
|
||||
((cxyz x y) (list 'macro y x))
|
||||
(_ #f)))))
|
||||
|
||||
(defpc makec mtc)
|
||||
|
||||
(define-constructor cxyz #:? cxyz? : Number String)
|
||||
|
||||
(check-equal?: (match (makec)
|
||||
((constructor cxyz e f) (list 'out f e))
|
||||
(_ #f))
|
||||
'(out "b" 1))
|
||||
|
||||
(check-equal?: (mtc (constructor cxyz 1 "b"))
|
||||
'(macro "b" 1))
|
||||
|
||||
(check-equal?: (mtc (makec))
|
||||
'(macro "b" 1))
|
||||
|
||||
(check-equal?: (makec) (cxyz 1 "b"))
|
||||
(check-equal?: (match (makec)
|
||||
((cxyz e f) (list 'out f e))
|
||||
(_ #f))
|
||||
'(out "b" 1))
|
||||
|
||||
(check-equal?: (mtc (cxyz 1 "b"))
|
||||
'(macro "b" 1)))
|
104
phc-adt-test/phc-adt/test/test-empty.rkt
Normal file
104
phc-adt-test/phc-adt/test/test-empty.rkt
Normal file
|
@ -0,0 +1,104 @@
|
|||
#lang typed/racket
|
||||
|
||||
(require phc-adt
|
||||
phc-toolkit
|
||||
type-expander
|
||||
typed/rackunit
|
||||
(only-in (lib "phc-adt/tagged-structure-low-level.hl.rkt")
|
||||
λ-tagged-get-field))
|
||||
(adt-init)
|
||||
|
||||
(define-structure empty-st)
|
||||
(define-tagged empty-tg)
|
||||
(define-constructor empty-ct :)
|
||||
(define-constructor empty-ct-t2 #:tag empty-ct ::)
|
||||
(define-constructor empty-ct-t3 #:tag empty-ct !)
|
||||
(define-constructor empty-ct-2 ::)
|
||||
(define-constructor empty-ct-3 !)
|
||||
|
||||
;(ann empty-ct (→ (constructor empty-ct)))
|
||||
;(ann (empty-ct) (constructor empty-ct))
|
||||
(check-equal?: (if (match (constructor empty-ct)
|
||||
[(empty-ct) #t]
|
||||
[_ #f])
|
||||
#t
|
||||
#f)
|
||||
: #t
|
||||
#t)
|
||||
;(ann (constructor empty-ct) empty-ct)
|
||||
(check-ann empty-ct?
|
||||
(→ Any Boolean : (constructor empty-ct)))
|
||||
(check-equal?: (if (empty-ct? (constructor empty-ct)) #t #f)
|
||||
: #t
|
||||
#t)
|
||||
|
||||
(check-ann (structure #:builder)
|
||||
(→ (structure)))
|
||||
(check-ann (tagged empty-tg #:builder)
|
||||
(→ (tagged empty-tg)))
|
||||
(check-ann (constructor empty-ct :)
|
||||
(→ (constructor empty-ct)))
|
||||
(check-ann (constructor empty-ct ::)
|
||||
(→ Null (constructor empty-ct)))
|
||||
(check-ann (constructor empty-ct !)
|
||||
(→ Any * (constructor empty-ct)))
|
||||
|
||||
(check-true: (match ((constructor empty-ct :))
|
||||
[(tagged empty-ct [values '()]) #t]
|
||||
[_ #f]))
|
||||
|
||||
(check-true: (match ((constructor empty-ct ::) null)
|
||||
[(tagged empty-ct [values '()]) #t]
|
||||
[_ #f]))
|
||||
|
||||
(check-true: (match ((constructor empty-ct !))
|
||||
[(tagged empty-ct [values '()]) #t]
|
||||
[_ #f]))
|
||||
|
||||
(check-true: (match (constructor empty-ct)
|
||||
[(tagged empty-ct [values '()]) #t]
|
||||
[_ #f]))
|
||||
|
||||
(check-ann (let ([v ((constructor empty-ct ::) null)])
|
||||
(if ((|(tagged-cast-predicate empty-ct values)|
|
||||
(make-predicate Null))
|
||||
v)
|
||||
v
|
||||
#f))
|
||||
(constructor empty-ct))
|
||||
|
||||
;; result type should be (constructor empty-ct . Number)
|
||||
(check-ann (constructor empty-ct . #{1 : Number})
|
||||
(constructor empty-ct . Number))
|
||||
(check-ann (constructor empty-ct #:rest 1 : Number)
|
||||
(constructor empty-ct . Number))
|
||||
|
||||
(check-ann (constructor empty-ct . 1)
|
||||
(constructor empty-ct . 1))
|
||||
(check-ann (constructor empty-ct #:rest 1)
|
||||
(constructor empty-ct . 1))
|
||||
(check-ann (constructor empty-ct #:rest (list 1 2))
|
||||
(constructor empty-ct Number Number))
|
||||
|
||||
(check-ann (constructor empty-ct 1)
|
||||
(constructor empty-ct 1))
|
||||
(check-ann (constructor empty-ct [1 : Number])
|
||||
(constructor empty-ct Number))
|
||||
(check-ann (constructor empty-ct)
|
||||
(constructor empty-ct))
|
||||
(check-ann (constructor empty-ct)
|
||||
(tagged empty-ct [values Null]))
|
||||
|
||||
(check-equal?-classes:
|
||||
[(constructor empty-ct . #{1 : Number})
|
||||
(constructor empty-ct #:rest 1 : Number)
|
||||
(constructor empty-ct . 1)
|
||||
(constructor empty-ct #:rest 1)]
|
||||
[(constructor empty-ct #:rest (list 1 2))]
|
||||
[(constructor empty-ct 1)
|
||||
(constructor empty-ct [1 : Number])]
|
||||
[(constructor empty-ct)
|
||||
(constructor empty-ct . #{null : Null})
|
||||
(constructor empty-ct #:rest null : Null)
|
||||
(constructor empty-ct . null)
|
||||
(constructor empty-ct #:rest null)])
|
104
phc-adt-test/phc-adt/test/test-structure-low-level.rkt
Normal file
104
phc-adt-test/phc-adt/test/test-structure-low-level.rkt
Normal file
|
@ -0,0 +1,104 @@
|
|||
#lang typed/racket
|
||||
|
||||
(require phc-adt
|
||||
phc-toolkit
|
||||
type-expander
|
||||
typed/rackunit
|
||||
(lib "phc-adt/tagged-structure-low-level.hl.rkt")
|
||||
(for-syntax phc-toolkit/untyped))
|
||||
(adt-init)
|
||||
|
||||
;; TODO: test all these with unsorted fields too.
|
||||
|
||||
;; Inferred type
|
||||
(define-syntax (test-structure-infer-type stx)
|
||||
(syntax-case stx ()
|
||||
[(_ name . fields)
|
||||
(quasisyntax/top-loc stx
|
||||
(define-type name #,(tagged-infer-type! #'(untagged . fields))))]))
|
||||
|
||||
(test-structure-infer-type test0 test-fa test-fb)
|
||||
(test-structure-infer-type test1)
|
||||
|
||||
;; Explicit type
|
||||
(define-syntax (test-structure-type stx)
|
||||
(syntax-case stx ()
|
||||
[(_ name [field _ type] …)
|
||||
(quasisyntax/top-loc stx
|
||||
(define-type name #,(tagged-type! #'(untagged [field type] …))))]))
|
||||
|
||||
(test-structure-type test2 [test-fa : Number] [test-fb : String])
|
||||
|
||||
;; Builders. Assigning them to a variable can fail at compile-time if TR does
|
||||
;; not see the type properly because it has the wrong scopes).
|
||||
(define c3 (structure #:builder))
|
||||
(define c4 (structure #:builder [test-fa : Number] [test-fb : String]))
|
||||
(define c6 (structure #:builder [test-fa : Number] [test-fc : Number]))
|
||||
(define c5 (structure #:builder test-fa test-fb))
|
||||
|
||||
;; Call constructors, and check the return type
|
||||
(check-not-exn (λ () (ann (c3) test1)))
|
||||
(let ([i4 (c4 7 "ee")]
|
||||
[i5 (c5 8 "ff")])
|
||||
(check-not-exn (λ () (ann i4 test2)))
|
||||
(check-not-exn (λ () (ann i5 test2)))
|
||||
(check-not-exn (λ () (ann i4 (test0 Number String))))
|
||||
(check-not-exn (λ () (ann i5 (test0 Number String)))))
|
||||
|
||||
;; TODO: bug report because using directly (ann v #t) does not work, but
|
||||
;; wrapping it with a no-op if does work.
|
||||
(define-syntax-rule (check-true-type v)
|
||||
(check-equal?: (if (ann v Boolean) #t #f)
|
||||
: #t
|
||||
#t))
|
||||
|
||||
(define-syntax-rule (check-false-type v)
|
||||
(check-false (ann (if (ann v Boolean) #t #f)
|
||||
#f)))
|
||||
|
||||
(let ([i4 (c4 7 "ee")]
|
||||
[i5 (c5 8 "ff")])
|
||||
(check-true-type ((structure? test-fa test-fb) i4))
|
||||
(check-true-type ((structure? test-fa test-fb) i5))
|
||||
(check-false-type ((structure?) i4))
|
||||
(check-true-type ((structure?) (c3)))
|
||||
(check-false-type ((structure? test-fa test-fb) (c3))))
|
||||
|
||||
;; Predicate
|
||||
|
||||
(check-equal?: (tagged-get-field (c4 7 "ee") test-fa 'else)
|
||||
: Number
|
||||
7)
|
||||
(check-equal?: (tagged-get-field (c5 7 "ee") test-fb 'else)
|
||||
: String
|
||||
"ee")
|
||||
(check-equal?: ((λ-tagged-get-field test-fa) (c4 7 "ee"))
|
||||
: Number
|
||||
7)
|
||||
(check-equal?: ((λ-tagged-get-field test-fb) (c5 7 "ee"))
|
||||
: String
|
||||
"ee")
|
||||
|
||||
;; Match-expander
|
||||
(define-match-expander test-structure-match
|
||||
(λ/syntax-case (_ [field pat …] …) ()
|
||||
(quasisyntax/loc stx
|
||||
#,(tagged-match! #'(untagged [field (and pat …)] …)))))
|
||||
|
||||
(check-equal?: (match (c5 7 "ee")
|
||||
[(test-structure-match [test-fa x] [test-fb y])
|
||||
(list y x)])
|
||||
: (List String Number)
|
||||
'("ee" 7))
|
||||
|
||||
;; Supertypes
|
||||
(define-syntax (test-supertypes stx)
|
||||
(syntax-case stx ()
|
||||
[(_ . fields)
|
||||
#`'#,(map cdr (has-fields #'fields))]))
|
||||
|
||||
(check-true (set=?
|
||||
(list->set (test-supertypes test-fa))
|
||||
(set '(untagged test-fa test-fb)
|
||||
'(untagged test-fa test-fc)
|
||||
'(untagged test-fa test-fd))))
|
10
phc-adt-test/phc-adt/test/test-structure-other.rkt
Normal file
10
phc-adt-test/phc-adt/test/test-structure-other.rkt
Normal file
|
@ -0,0 +1,10 @@
|
|||
#lang typed/racket
|
||||
|
||||
(require phc-adt phc-toolkit type-expander typed/rackunit)
|
||||
(adt-init)
|
||||
|
||||
(provide i-other t-other c-other)
|
||||
|
||||
(define i-other (structure [test-fa "a"] [test-fb (ann 'b 'b)]))
|
||||
(define-type t-other (structure [test-fa Number] [test-fc 'c]))
|
||||
(define c-other (structure [test-fa : Number] [test-fd : 'd]))
|
22
phc-adt-test/phc-adt/test/test-structure-parametric.rkt
Normal file
22
phc-adt-test/phc-adt/test/test-structure-parametric.rkt
Normal file
|
@ -0,0 +1,22 @@
|
|||
#lang typed/racket
|
||||
|
||||
(require phc-adt phc-toolkit type-expander typed/rackunit "ck.rkt")
|
||||
(adt-init)
|
||||
|
||||
(define-structure #:∀ (T) a [a : T] [b : T])
|
||||
(check-print-type (a 3 "b") "((tagged untagged a b) (U Positive-Byte String) (U Positive-Byte String))")
|
||||
|
||||
(define-structure b #:∀ (T) [a : T] [b : T])
|
||||
(define-structure c [a : T] [b : T] #:∀ (T))
|
||||
|
||||
(define printed-type
|
||||
"((tagged untagged a b) (U Positive-Byte String) (U Positive-Byte String))")
|
||||
|
||||
(check-print-type (a 3 "b") printed-type)
|
||||
(check-ann (a 3 4) (structure [a Positive-Byte] [b Positive-Byte]))
|
||||
|
||||
(check-print-type (b 3 "b") printed-type)
|
||||
(check-ann (b 3 4) (structure [a Positive-Byte] [b Positive-Byte]))
|
||||
|
||||
(check-print-type (c 3 "b") printed-type)
|
||||
(check-ann (c 3 4) (structure [a Positive-Byte] [b Positive-Byte]))
|
23
phc-adt-test/phc-adt/test/test-structure-speed.rkt
Normal file
23
phc-adt-test/phc-adt/test/test-structure-speed.rkt
Normal file
|
@ -0,0 +1,23 @@
|
|||
#lang typed/racket/base
|
||||
|
||||
(require phc-adt
|
||||
phc-toolkit
|
||||
type-expander
|
||||
typed/rackunit
|
||||
(for-syntax racket/base
|
||||
syntax/stx
|
||||
racket/list))
|
||||
(adt-init)
|
||||
|
||||
(define-syntax (repeat-stx stx)
|
||||
(syntax-case stx ()
|
||||
[(_ n expr)
|
||||
(number? (syntax-e #'n))
|
||||
#`(begin . #,(stx-map (λ _ #'expr)
|
||||
(range (syntax-e #'n))))]))
|
||||
|
||||
(repeat-stx 1 (check-not-exn (λ () (structure [test-fa 2 : Number]
|
||||
[test-fb "b" : String]))))
|
||||
|
||||
;(repeat-stx 1024 (check-not-exn (λ () (structure [test-fa 2 : Number]
|
||||
; [test-fb "b" : String]))))
|
153
phc-adt-test/phc-adt/test/test-structure.rkt
Normal file
153
phc-adt-test/phc-adt/test/test-structure.rkt
Normal file
|
@ -0,0 +1,153 @@
|
|||
#lang typed/racket
|
||||
|
||||
(require phc-adt
|
||||
phc-toolkit
|
||||
type-expander
|
||||
typed/rackunit
|
||||
(only-in (lib "phc-adt/tagged-structure-low-level.hl.rkt")
|
||||
λ-tagged-get-field))
|
||||
(adt-init)
|
||||
|
||||
;; define-structure
|
||||
(define-structure named1 test-fa test-fb)
|
||||
(define-structure named2 test-fb test-fa)
|
||||
(define-structure named3 test-fa test-fc)
|
||||
|
||||
(check-equal?-classes:
|
||||
[#:name "named1 ∪ named2"
|
||||
: (named1 'a 'b)
|
||||
(ann (named1 (ann 'a 'a) (ann 'b 'b)) (named1 'a 'b))
|
||||
(ann (named1 (ann 'a 'a) (ann 'b 'b)) (structure [test-fa 'a] [test-fb 'b]))
|
||||
(ann (named1 (ann 'a 'a) (ann 'b 'b)) ((structure [test-fa] test-fb) 'a 'b))
|
||||
(ann (named2 (ann 'b 'b) (ann 'a 'a)) (named2 'b 'a))
|
||||
(ann (named2 (ann 'b 'b) (ann 'a 'a)) (structure [test-fa 'a] [test-fb 'b]))
|
||||
(ann (named2 (ann 'b 'b) (ann 'a 'a)) ((structure [test-fa] test-fb) 'a 'b))]
|
||||
[#:name "named3"
|
||||
: (named3 'a 'b)
|
||||
(ann (named3 (ann 'a 'a) (ann 'b 'b)) (named3 'a 'b))
|
||||
(ann (named3 (ann 'a 'a) (ann 'b 'b)) (structure [test-fa 'a] [test-fc 'b]))
|
||||
(ann (named3 (ann 'a 'a) (ann 'b 'b)) ((structure [test-fa] test-fc) 'a 'b))])
|
||||
|
||||
(check-equal?: (match (named1 (ann 'a 'a) (ann 'b 'b))
|
||||
[(named1 fa fb) (list fb fa)])
|
||||
: (List 'b 'a)
|
||||
'(b a))
|
||||
|
||||
;; Types
|
||||
(define-type t0 (structure [test-fa Number] [test-fb String]))
|
||||
(define-type t1 (structure))
|
||||
|
||||
(define i2 (structure [test-fa 1] [test-fb "a"]))
|
||||
(define i3 (structure #:instance))
|
||||
|
||||
(check-not-exn (λ () (ann i2 t0)))
|
||||
(check-not-exn (λ () (ann i3 t1)))
|
||||
|
||||
(check-not-exn (λ () (ann i2 StructureTop)))
|
||||
(check-not-exn (λ () (ann i3 StructureTop)))
|
||||
|
||||
(check-true (StructureTop? i2))
|
||||
(check-true (StructureTop? i3))
|
||||
|
||||
;; Instance and make-instance
|
||||
(define c4 (structure test-fa test-fb))
|
||||
(define c5 (structure [test-fa : Number] [test-fb : String]))
|
||||
(define-type test-fa+test-fb (structure [test-fa Number] [test-fb String]))
|
||||
(check-equal?-classes:
|
||||
[#:name "test-fa+test-fb"
|
||||
: test-fa+test-fb
|
||||
((structure test-fa test-fb) 2 "b")
|
||||
((structure [test-fa] test-fb) 2 "b")
|
||||
((structure test-fa [test-fb]) 2 "b")
|
||||
((structure [test-fa] [test-fb]) 2 "b")
|
||||
|
||||
(structure [test-fa 2] [test-fb "b"])
|
||||
|
||||
((structure [test-fa : Number] [test-fb : String]) 2 "b")
|
||||
|
||||
(structure [test-fa 2 : Number] [test-fb "b" : String])
|
||||
|
||||
((structure test-fb test-fa) "b" 2)
|
||||
((structure test-fb [test-fa]) "b" 2)
|
||||
((structure [test-fb] test-fa) "b" 2)
|
||||
((structure [test-fb] [test-fa]) "b" 2)
|
||||
|
||||
(structure [test-fb "b"] [test-fa 2])
|
||||
|
||||
((structure [test-fb : String] [test-fa : Number]) "b" 2)
|
||||
|
||||
(structure [test-fb "b" : String] [test-fa 2 : Number])])
|
||||
|
||||
;; Accessor
|
||||
(check-equal?: ((λ-tagged-get-field test-fb) (c4 7 "ee"))
|
||||
: String
|
||||
"ee")
|
||||
(check-equal?: ((λ-tagged-get-field test-fb) (c5 7 "ee"))
|
||||
: String
|
||||
"ee")
|
||||
|
||||
;; Match
|
||||
((inst check-equal?-classes (List String Number))
|
||||
(cons
|
||||
"match"
|
||||
(list
|
||||
;; Simple
|
||||
(match (c4 7 "ee") [(structure [test-fa fa] [test-fb fb]) (list fb fa)])
|
||||
;; Change order in the struct definition
|
||||
(match (c4 7 "ee") [(structure [test-fb fb] [test-fa fa]) (list fb fa)])
|
||||
;; No patterns
|
||||
(match (c4 7 "ee") [(structure [test-fb] [test-fa]) (list test-fb test-fa)])
|
||||
(match (c4 7 "ee") [(structure test-fb test-fa) (list test-fb test-fa)]))))
|
||||
|
||||
;; supertypes:
|
||||
|
||||
(define fn1 (ann (λ (x) x)
|
||||
(→ (structure-supertype [test-fa Number])
|
||||
(structure-supertype [test-fa Number]))))
|
||||
(check-not-exn
|
||||
(λ ()
|
||||
(ann fn1
|
||||
(→ (U (structure-supertype [test-fa Number] [test-fb Any])
|
||||
(structure-supertype [test-fa Number] [test-fc Any])
|
||||
(structure-supertype [test-fa Number] [test-fd Any]))
|
||||
(U (structure-supertype [test-fa Number] [test-fb Any])
|
||||
(structure-supertype [test-fa Number] [test-fc Any])
|
||||
(structure-supertype [test-fa Number] [test-fd Any]))))))
|
||||
|
||||
(define fn2 (ann (λ (x) x)
|
||||
(→ ((structure-supertype test-fa) Number)
|
||||
((structure-supertype test-fa) Number))))
|
||||
(check-not-exn
|
||||
(λ ()
|
||||
(ann fn2
|
||||
(→ (U (structure-supertype [test-fa Number] [test-fb Any])
|
||||
(structure-supertype [test-fa Number] [test-fc Any])
|
||||
(structure-supertype [test-fa Number] [test-fd Any]))
|
||||
(U (structure-supertype [test-fa Number] [test-fb Any])
|
||||
(structure-supertype [test-fa Number] [test-fc Any])
|
||||
(structure-supertype [test-fa Number] [test-fd Any]))))))
|
||||
|
||||
(check-not-exn (λ () (ann (structure [test-fa 7] [test-fb 'x])
|
||||
(structure-supertype [test-fa Number]))))
|
||||
|
||||
(check-not-exn (λ () (ann (structure [test-fa 8] [test-fc 42])
|
||||
((structure-supertype [test-fa]) Number))))
|
||||
|
||||
(check-not-exn (λ () (ann (structure [test-fa 8] [test-fd "blob"])
|
||||
((structure-supertype test-fa) Number))))
|
||||
|
||||
(check-equal?: (match (structure [test-fa 8] [test-fc 'y])
|
||||
[(structure-supertype [test-fa x]) (+ x 1)])
|
||||
: Number
|
||||
9)
|
||||
|
||||
;; Exchange structures across files (values, types …)
|
||||
(require "test-structure-other.rkt")
|
||||
|
||||
(check-equal? (ann i-other (structure [test-fa String] [test-fb 'b]))
|
||||
(structure [test-fa "a"] [test-fb (ann 'b 'b)]))
|
||||
|
||||
(check-not-exn (λ () (ann (structure [test-fa 1] [test-fc 'c]) t-other)))
|
||||
|
||||
(check-equal? (c-other 7 'd)
|
||||
(structure [test-fa 7 : Number] [test-fd 'd : 'd]))
|
150
phc-adt-test/phc-adt/test/test-structure2.rkt
Normal file
150
phc-adt-test/phc-adt/test/test-structure2.rkt
Normal file
|
@ -0,0 +1,150 @@
|
|||
#lang typed/racket
|
||||
|
||||
(require phc-adt
|
||||
phc-toolkit
|
||||
type-expander
|
||||
typed/rackunit
|
||||
(only-in (lib "phc-adt/tagged-structure-low-level.hl.rkt")
|
||||
λ-tagged-get-field))
|
||||
(adt-init)
|
||||
|
||||
(define-tagged empty-tg)
|
||||
|
||||
;; structure-get field
|
||||
(begin
|
||||
(check-equal?:
|
||||
(uniform-get ((structure #:builder a b c d) 1 "b" 'val-c 4) c)
|
||||
: 'val-c
|
||||
'val-c))
|
||||
|
||||
;; match-expander
|
||||
(begin
|
||||
(let ([test-match
|
||||
(λ ([val : Any])
|
||||
(match val
|
||||
[(structure a b c y) (list a b c y)]
|
||||
[(structure d
|
||||
[a (? number?)]
|
||||
[c (? symbol?) 'value-c]
|
||||
[b bb (? string?)])
|
||||
(list a bb c d)]
|
||||
[else 'other]))])
|
||||
(check-equal?: (test-match
|
||||
((structure #:builder a b c d) 1
|
||||
"b"
|
||||
'value-c
|
||||
4))
|
||||
'(1 "b" value-c 4))
|
||||
(check-equal?: (test-match
|
||||
((structure #:builder a b c y) 1 2 3 4))
|
||||
'(1 2 3 4))
|
||||
(check-equal?: (test-match 'bad) 'other)))
|
||||
|
||||
;; type-expander
|
||||
(begin
|
||||
(check-equal?:
|
||||
(uniform-get (ann ((structure #:builder a b c) 1 "b" #t)
|
||||
(structure [a Number] [c Boolean] [b String]))
|
||||
b)
|
||||
"b"))
|
||||
|
||||
;; structure
|
||||
(begin
|
||||
(let ()
|
||||
(define-structure empty-st)
|
||||
(define-structure stA [a Number])
|
||||
;; BUG 137 (check-equal?: (empty-st) ((structure #:builder)))
|
||||
(check-not-equal?: (empty-st) (structure [a 1]))
|
||||
(check-not-equal?: (structure #:builder) (structure [a 1]))
|
||||
(check-not-equal?: (empty-st) (stA 1))
|
||||
(check-not-equal?: (structure #:builder) (stA 1))
|
||||
(void))
|
||||
|
||||
;; TODO: uncomment these tests:
|
||||
(let ()
|
||||
(define-structure st [a Number] [b String])
|
||||
(define-structure stA [a Number])
|
||||
(define-structure stABC [a Number] [b String] [c Number])
|
||||
(define st1 (st 1 "b"))
|
||||
(define st2 (st 2 "b"))
|
||||
(define sta (stA 1))
|
||||
(define st3 (stABC 1 "b" 3))
|
||||
|
||||
(check-equal?-classes:
|
||||
[#:name st1
|
||||
st1
|
||||
(structure [a 1] [b "b"])
|
||||
(structure [a : Number 1] [b : String "b"])
|
||||
((structure [a : Number] [b : String]) 1 "b")
|
||||
(structure [a : Any 1] [b : Any "b"])
|
||||
((structure [a : Any] [b : Any]) 1 "b")
|
||||
((structure [a] [b]) 1 "b")
|
||||
((structure a b) 1 "b")
|
||||
((structure [a] b) 1 "b")]
|
||||
[(structure [a "1"] [b 'b])
|
||||
(structure [a : String "1"] [b : Symbol 'b])
|
||||
(structure [a : Any "1"] [b : Any 'b])]
|
||||
[st2]
|
||||
[sta]
|
||||
[st3])))
|
||||
|
||||
;; define-structure
|
||||
(begin
|
||||
(define-structure empty-st)
|
||||
(define-structure st [a Number] [b String])
|
||||
(define-structure st2 [b String] [a Number] #:? custom-is-st2?)
|
||||
(define-structure st3 [c String] [a Number] #:? custom-is-st3?))
|
||||
|
||||
;; Constructor:
|
||||
(check-equal?: (empty-st) : empty-st (empty-st))
|
||||
(begin
|
||||
(check-equal?: (uniform-get (st 1 "b") b) : String "b")
|
||||
(check-equal?: (uniform-get (st2 "a" 2) b) : String "a"))
|
||||
|
||||
;; Constructor, as id:
|
||||
(begin
|
||||
(check-equal?: (uniform-get (cadr (map st '(1 2 3) '("x" "y" "z"))) b)
|
||||
: String
|
||||
"y")
|
||||
(check-equal?: (uniform-get (cadr (map st2 '("d" "e" "f") '(1 2 3))) b)
|
||||
: String
|
||||
"e"))
|
||||
|
||||
;; type-expander
|
||||
(begin
|
||||
(check-equal?: (uniform-get (ann (st2 "g" 123) st2) b) "g"))
|
||||
|
||||
;; match-expander
|
||||
(begin
|
||||
(check-equal?: (match (st2 "h" 7) [(st x y) (cons x y)])
|
||||
: (Pairof Number String)
|
||||
'(7 . "h")))
|
||||
|
||||
;; Equality
|
||||
(begin
|
||||
(check-equal?: (ann (st 1 "i") st) (st 1 "i"))
|
||||
(check-equal?: (ann (st2 "j" 2) st2) (st2 "j" 2))
|
||||
(check-equal?: (ann (st 1 "k") st) (st2 "k" 1)))
|
||||
|
||||
;; Predicate
|
||||
(begin
|
||||
(check-equal?: (st? (ann (st 1 "i") (U st st2))) #t)
|
||||
(check-equal?: (custom-is-st2? (ann (st 1 "i") (U st st2))) #t)
|
||||
(check-equal?: (custom-is-st3? (ann (st 1 "i") (U st st2))) #f)
|
||||
(check-equal?: (st? (ann (st 1 "i") (U Number st st2))) #t)
|
||||
(check-equal?: (st? (ann 1 (U Number st st2))) #f)
|
||||
;; Occurrence typing won't work well, if only because fields could be of
|
||||
;; a type for which TR doesn't know how to make-predicate.
|
||||
(define (check-occurrence-typing [x : (U Number st st3)])
|
||||
(if (st? x)
|
||||
(match (ann x st) [(st the-a the-b) (cons the-b the-a)])
|
||||
'other))
|
||||
(check-equal?:
|
||||
(check-occurrence-typing (ann (st 1 "i") (U Number st st3)))
|
||||
'("i" . 1))
|
||||
(check-equal?:
|
||||
(check-occurrence-typing (ann (st2 "j" 2) (U Number st st3)))
|
||||
'("j" . 2))
|
||||
(check-equal?:
|
||||
(check-occurrence-typing (ann 9 (U Number st st3)))
|
||||
'other))
|
110
phc-adt-test/phc-adt/test/test-tagged-call-syntax.rkt
Normal file
110
phc-adt-test/phc-adt/test/test-tagged-call-syntax.rkt
Normal file
|
@ -0,0 +1,110 @@
|
|||
#lang racket/base
|
||||
|
||||
(require (for-template (lib "phc-adt/tagged.hl.rkt"))
|
||||
rackunit
|
||||
syntax/parse)
|
||||
|
||||
(check-equal?
|
||||
(syntax-parse #'(nam #:instance)
|
||||
[(:tagged-call-args-syntax-class)
|
||||
(list* (attribute instance?) (syntax->datum #'(name [fieldᵢ ...])))]
|
||||
[_ 'wrong])
|
||||
'(#t nam []))
|
||||
|
||||
(check-equal?
|
||||
(syntax-parse #'(nam [field0 value0])
|
||||
[(:tagged-call-args-syntax-class)
|
||||
(list* (attribute instance?) (attribute no-types?)
|
||||
(syntax->datum #'(name [fieldᵢ ...] [valueᵢ ...])))]
|
||||
[_ 'wrong])
|
||||
'(#t #t nam [field0] [value0]))
|
||||
|
||||
(check-equal?
|
||||
(syntax-parse #'(nam [field0 value0] [field1 value1])
|
||||
[(:tagged-call-args-syntax-class)
|
||||
(list* (attribute instance?) (attribute no-types?)
|
||||
(syntax->datum #'(name [fieldᵢ ...] [valueᵢ ...])))]
|
||||
[_ 'wrong])
|
||||
'(#t #t nam [field0 field1] [value0 value1]))
|
||||
|
||||
(check-equal?
|
||||
(syntax-parse #'(nam [field0 : type0 value0])
|
||||
[(:tagged-call-args-syntax-class)
|
||||
(list* (attribute instance?) (attribute types?)
|
||||
(syntax->datum #'(name [fieldᵢ ...] [τᵢ ...] [valueᵢ ...])))]
|
||||
[_ 'wrong])
|
||||
'(#t #t nam [field0] [type0] [value0]))
|
||||
|
||||
(check-equal?
|
||||
(syntax-parse #'(nam [field0 : type0 value0] [field1 : type1 value1])
|
||||
[(:tagged-call-args-syntax-class)
|
||||
(list* (attribute instance?) (attribute types?)
|
||||
(syntax->datum #'(name [fieldᵢ ...] [τᵢ ...] [valueᵢ ...])))]
|
||||
[_ 'wrong])
|
||||
'(#t #t nam [field0 field1] [type0 type1] [value0 value1]))
|
||||
|
||||
(check-equal?
|
||||
(syntax-parse #'(nam #:instance)
|
||||
[(:tagged-call-args-syntax-class)
|
||||
(list* (attribute instance?)
|
||||
(syntax->datum #'(name)))]
|
||||
[_ 'wrong])
|
||||
'(#t nam))
|
||||
|
||||
(check-equal?
|
||||
(syntax-parse #'(nam [field0 value0])
|
||||
[(:tagged-call-args-syntax-class)
|
||||
(list* (attribute instance?) (attribute no-types?)
|
||||
(syntax->datum #'(name [fieldᵢ ...] [valueᵢ ...])))]
|
||||
[_ 'wrong])
|
||||
'(#t #t nam [field0] [value0]))
|
||||
|
||||
(check-equal?
|
||||
(syntax-parse #'(nam [field0 : type0 value0])
|
||||
[(:tagged-call-args-syntax-class)
|
||||
(list* (attribute instance?) (attribute types?)
|
||||
(syntax->datum #'(name [fieldᵢ ...] [τᵢ ...] [valueᵢ ...])))]
|
||||
[_ 'wrong])
|
||||
'(#t #t nam [field0] [type0] [value0]))
|
||||
|
||||
(check-equal?
|
||||
(syntax-parse #'(nam [field0] field1)
|
||||
[(:tagged-call-args-syntax-class)
|
||||
(list* (attribute builder?)
|
||||
(syntax->datum #'(name [fieldᵢ ...])))]
|
||||
[_ 'wrong])
|
||||
'(#t nam [field0 field1]))
|
||||
|
||||
(check-equal?
|
||||
(syntax-parse #'(nam [field0] field1)
|
||||
[(:tagged-call-args-syntax-class)
|
||||
(list* (attribute builder?)
|
||||
(syntax->datum #'(name [fieldᵢ ...])))]
|
||||
[_ 'wrong])
|
||||
'(#t nam [field0 field1]))
|
||||
|
||||
(check-equal?
|
||||
(syntax-parse #'(nam [field0] field1)
|
||||
[(:tagged-call-args-syntax-class)
|
||||
(list* (attribute builder?)
|
||||
(syntax->datum #'(name [fieldᵢ ...])))]
|
||||
[_ 'wrong])
|
||||
'(#t nam [field0 field1]))
|
||||
|
||||
(check-equal?
|
||||
(syntax-parse #'(nam)
|
||||
[(:tagged-call-args-syntax-class) 'wrong]
|
||||
[_ 'parse-failed])
|
||||
'parse-failed)
|
||||
|
||||
(check-equal?
|
||||
(syntax-parse #'(#:instance)
|
||||
[(:tagged-call-args-syntax-class) 'wrong]
|
||||
[_ 'parse-failed])
|
||||
'parse-failed)
|
||||
|
||||
(check-equal?
|
||||
(syntax-parse #'()
|
||||
[(:tagged-call-args-syntax-class) 'wrong]
|
||||
[_ 'parse-failed])
|
||||
'parse-failed)
|
67
phc-adt-test/phc-adt/test/test-tagged-define.rkt
Normal file
67
phc-adt-test/phc-adt/test/test-tagged-define.rkt
Normal file
|
@ -0,0 +1,67 @@
|
|||
#lang typed/racket
|
||||
|
||||
(require phc-adt phc-toolkit type-expander "ck.rkt")
|
||||
(adt-init)
|
||||
|
||||
(module m-abc typed/racket
|
||||
(require phc-adt phc-toolkit type-expander "ck.rkt")
|
||||
(adt-init)
|
||||
|
||||
(define-tagged t0 tagged-fc tagged-fb tagged-fa)
|
||||
(define c0 t0)
|
||||
(provide (rename-out [t0 t0-cba]
|
||||
[c0 c0-cba])))
|
||||
|
||||
(module m-xyz typed/racket
|
||||
(require phc-adt phc-toolkit type-expander "ck.rkt")
|
||||
(adt-init)
|
||||
|
||||
(define-tagged t0 tagged-fx tagged-fy tagged-fz)
|
||||
(define c0 t0)
|
||||
(provide (rename-out [t0 t0-xyz]
|
||||
[c0 c0-xyz])))
|
||||
|
||||
(require 'm-abc
|
||||
'm-xyz)
|
||||
|
||||
(define-tagged t0 tagged-fa tagged-fb tagged-fc)
|
||||
(define c0 t0)
|
||||
|
||||
(ck (t0 'a "b" 3) (t0 Symbol String Number))
|
||||
(ck (c0 'a "b" 3) (t0 Symbol String Number))
|
||||
(ck-not (t0 'a "b" 3) (t0 Number Number Number))
|
||||
(ck-not (c0 'a "b" 3) (t0 Number Number Number))
|
||||
|
||||
(ck (t0 'a "b" 3) (t0-cba Number String Symbol))
|
||||
(ck (c0 'a "b" 3) (t0-cba Number String Symbol))
|
||||
(ck-not (t0 'a "b" 3) (t0-cba Number Number Number))
|
||||
(ck-not (c0 'a "b" 3) (t0-cba Number Number Number))
|
||||
|
||||
(ck (t0-cba 'a "b" 3) (t0 Number String Symbol))
|
||||
(ck (c0-cba 'a "b" 3) (t0 Number String Symbol))
|
||||
(ck-not (t0-cba 'a "b" 3) (t0 Number Number Number))
|
||||
(ck-not (c0-cba 'a "b" 3) (t0 Number Number Number))
|
||||
|
||||
(ck (t0-cba 'a "b" 3) (t0-cba Symbol String Number))
|
||||
(ck (c0-cba 'a "b" 3) (t0-cba Symbol String Number))
|
||||
(ck-not (t0-cba 'a "b" 3) (t0-cba Number Number Number))
|
||||
(ck-not (c0-cba 'a "b" 3) (t0-cba Number Number Number))
|
||||
|
||||
(ck (t0-xyz 'a "b" 3) (t0-xyz Symbol String Number))
|
||||
(ck (c0-xyz 'a "b" 3) (t0-xyz Symbol String Number))
|
||||
(ck-not (t0-xyz 'a "b" 3) (t0 Number Number Number))
|
||||
(ck-not (c0-xyz 'a "b" 3) (t0 Number Number Number))
|
||||
(ck-not (t0 'a "b" 3) (t0-xyz Symbol String Number))
|
||||
(ck-not (c0 'a "b" 3) (t0-xyz Symbol String Number))
|
||||
|
||||
(check-equal?-classes:
|
||||
[#:name "abc"
|
||||
(t0 'a "b" 3)
|
||||
(c0 'a "b" 3)
|
||||
(t0-cba 3 "b" 'a)
|
||||
(c0-cba 3 "b" 'a)]
|
||||
[#:name "cba"
|
||||
(t0 3 "b" 'a)
|
||||
(c0 3 "b" 'a)
|
||||
(t0-cba 'a "b" 3)
|
||||
(c0-cba 'a "b" 3)])
|
21
phc-adt-test/phc-adt/test/test-tagged-parametric.rkt
Normal file
21
phc-adt-test/phc-adt/test/test-tagged-parametric.rkt
Normal file
|
@ -0,0 +1,21 @@
|
|||
#lang typed/racket
|
||||
|
||||
(require phc-adt phc-toolkit type-expander typed/rackunit "ck.rkt")
|
||||
(adt-init)
|
||||
|
||||
(define-tagged #:∀ (T) a [a : T] [b : T])
|
||||
(define-tagged b #:∀ (T) [a : T] [b : T])
|
||||
(define-tagged c [a : T] [b : T] #:∀ (T))
|
||||
|
||||
(define (printed-type [t : String])
|
||||
(string-append "((tagged " t " a b)"
|
||||
" (U Positive-Byte String) (U Positive-Byte String))"))
|
||||
|
||||
(check-print-type (a 3 "b") (printed-type "a"))
|
||||
(check-ann (a 3 4) (tagged a [a Positive-Byte] [b Positive-Byte]))
|
||||
|
||||
(check-print-type (b 3 "b") (printed-type "b"))
|
||||
(check-ann (b 3 4) (tagged b [a Positive-Byte] [b Positive-Byte]))
|
||||
|
||||
(check-print-type (c 3 "b") (printed-type "c"))
|
||||
(check-ann (c 3 4) (tagged c [a Positive-Byte] [b Positive-Byte]))
|
20
phc-adt-test/phc-adt/test/test-tagged.rkt
Normal file
20
phc-adt-test/phc-adt/test/test-tagged.rkt
Normal file
|
@ -0,0 +1,20 @@
|
|||
#lang typed/racket
|
||||
|
||||
(require phc-adt phc-toolkit type-expander)
|
||||
(adt-init)
|
||||
|
||||
(check-equal?: (match (ann (tagged t1 [x 1] [y "b"])
|
||||
(tagged t1 [x : Number] [y : String]))
|
||||
[(tagged t1 [x a] [y b]) (list 'ok b a)]
|
||||
[_ #f])
|
||||
'(ok "b" 1))
|
||||
(check-equal?: (match (ann (tagged foo [x "o"] [y 3] [z 'z])
|
||||
(tagged foo
|
||||
[x String]
|
||||
[z 'z]
|
||||
[y Fixnum]))
|
||||
[(tagged foo z x y) (list z y x)])
|
||||
'(z 3 "o"))
|
||||
|
||||
(define-type ma (tagged ma (fav String) (faa ma) (fab mb)))
|
||||
(define-type mb (tagged mb (fbv String) (fba ma)))
|
21
phc-adt-test/phc-adt/test/test-variant.rkt
Normal file
21
phc-adt-test/phc-adt/test/test-variant.rkt
Normal file
|
@ -0,0 +1,21 @@
|
|||
#lang typed/racket
|
||||
|
||||
(require phc-adt phc-toolkit type-expander typed/rackunit)
|
||||
(adt-init)
|
||||
|
||||
(define-variant v1 [x Number String] [y String Number] [z Number String])
|
||||
(check-equal?: (ann (constructor x 1 "a")
|
||||
(U [constructor w Number String]
|
||||
[constructor x Number String]
|
||||
[constructor y String Number]))
|
||||
(constructor x 1 "a"))
|
||||
(check-equal?: (constructor x 1 "a")
|
||||
(constructor x 1 "a"))
|
||||
(check-equal?: (ann (constructor x 1 "a") v1)
|
||||
(constructor x 1 "a"))
|
||||
(check-equal?: (ann (constructor x 1 "a") v1)
|
||||
(ann (constructor x 1 "a") v1))
|
||||
(check-not-equal?: (ann (constructor x 2 "b") v1)
|
||||
(ann (constructor y "b" 2) v1))
|
||||
(check-not-equal?: (ann (constructor x 3 "c") v1)
|
||||
(ann (constructor z 3 "c") v1))
|
12
phc-adt/info.rkt
Normal file
12
phc-adt/info.rkt
Normal file
|
@ -0,0 +1,12 @@
|
|||
#lang info
|
||||
(define collection 'multi)
|
||||
(define deps '("phc-adt-lib"
|
||||
"phc-adt-doc"
|
||||
"phc-adt-test"))
|
||||
(define implies '("phc-adt-lib"
|
||||
"phc-adt-doc"
|
||||
"phc-adt-test"))
|
||||
(define build-deps '())
|
||||
(define pkg-desc "Algebraic Datatypes tailored for writing compilers")
|
||||
(define version "1.1")
|
||||
(define pkg-authors '("Georges Dupéron"))
|
Loading…
Reference in New Issue
Block a user