====== User-defined types ======
User-defined type can be used to create complex circuit structures. A
new user-defined type name is introduced by using ''defproc'',
''defcell'', ''defchan'', or ''deftype''
statements. All user-defined types have the same basic structure:
- a type signature, that provides information about the interface to the type and the ports that are externally visible; and
- a //body//, contained in braces, that specifies the detailed definition of the user-defined type.
The type chosen for each port must be the most specific type used by
that port in the body (see the implementation relation section).
User-defined types can also be parameterized, and this is covered in
detail later.
* [[language:types2:proc|Processes and cells]]
* [[language:types2:data|Data types]]
* [[language:types2:chan|Channel types]]
Process, channel, and data types can include //methods// that provide mechanisms to
manipulate the type or access parts of the type. There are a number of special built-in
method names that can be specified for data types and channel types.
===== Instantiating user-defined types =====
User-defined type variables can be instantiated in much the same manner
as ordinary type variables.
defproc test(bool N, n) { ... }
test x;
// x.N and x.n refer to the ports of ''x''
The ACT description above creates an instance of type ''test'' named
''x''. Creating an instance of a type creates instances of all the
ports listed as well as creating whatever is specified by the body of
the type definition. The list of ports of a user-defined type can be
accessed from the scope outside the type definition by using
dot-notation. These externally visible ports are analogous to the
//fields// of structures or record types in standard programming
languages.
This analogy to records can be used to build complex data types,
albeit with slightly different syntax compared to traditional
programming languages.
Data types can be implemented using Booleans, where Boolean variables
correspond to signals that can be accessed at the circuit level. In this case,
conversions between the higher level description (e.g. an integer) and signals
can be also described using set/get methods as illustrated below.
deftype mystruct <: int<16> (bool b[16])
{
methods {
set {
(;i:16: [self{i} = 1 -> b[i]+ [] self{i} = 0 -> b[i]- ])
}
get {
self := 0;
(;i:16: [b[i] -> self := self | (1 << i) [] ~b[i] -> skip ])
}
}
}
===== Parameterized user-defined types =====
Processes, channels, and datatypes created using ''defproc'',
''defchan'', and ''deftype'' all support
parameterization. Parameters are specified using the ''template''
keyword.
Since the syntax for all three is the same, we use a process
definition to illustrate this. To create a parameterized type, the
definition of the type is preceeded by a template specifier as shown
below.
// A generic adder block
template
defproc adder (e1of2 a[N], b[N]; e1of2 s[N])
{
...
}
This example defines an ''adder'' that takes ''N'' as a
parameter. Note that the value of ''N'' determines the size of the
arrays in the port list for the process. Instances of this
''adder'' can be created in the following way:
adder<4> a1; // a1 is a 4-bit adder
adder<16> a2; // a2 is a 16-bit adder
The value of ''a1.N'' is 4, while the value of ''a2.N'' is
16. To illustrate how one might define this adder block, assume we
have processes ''fulladder'', ''zerosource'', and
''bitbucket'' already defined that implement a full-adder, a
constant source of zeros, and a constant sink respectively. One
possible definition of the adder would be:
template
defproc adder (e1of2 a[N], b[N]; e1of2 s[N])
{
fulladder fa[N];
( i : N-1 : fa[i].a = a[i]; fa[i].b = b[i]; fa[i].s = s[i];
fa[i].co = fa[i+1].ci; )
zerosource z;
bitbucket w;
fa[0].ci=z.x;
fa[N-1].co = w.x;
}
This creates a parameterized ripple-carry adder. Notice the use of
loops and arrays to connect the carry chain for the adder, and the
inputs and outputs of the process to the ''fulladder'' ports.
As shown in the example above, the arguments in the template parameter
list are specified by listing them next to the type name. Trailing
arguments can be omitted from the parameter list attached to the type
as shown in the example below.
template
defproc test (bool n[N]) { ... }
test<5> x;
Channels and data types can also be parameterized in the same way. For
example, the following might be an N-bit dual rail definition.
template
deftype d1of2 <: int (bool d0[N], d1[N]) { ... }
Since the body of the type can use loops and selection statements in
arbitrary ways, changing the parameters for the type can completely
change the structure of the circuit. It can also change the ports for
the type. Hence, when checking for type compatibility, the values of
parameters are also taken into account. Hence, the full type for
instance ''a2'' above is in fact ''adder<5>'', not just
''adder''. Types such as ''fulladder'' that do not have
parameters are more completely specified as ''fulladder<>'',
although the angle brackets can be omitted. Arrays can only correspond
to instances of the same type---so an array cannot contain a three-bit
adder and five-bit adder.
==== Default parameters ====
When defining complex user-defined types with many parameters, it can be useful to have
default parameter values. ACT has syntax to support default parameter values for trailing
parameters in a template definition.
template
defproc driver(bool? inp; bool! outp)
{
bool sig;
prs {
inp => sig-
}
[active_high -> prs { sig => outp- }
[] else -> sig = outp;
]
}
(Note: this is not a real signal driver, but the idea here is the you have a parameterized
driver that can drive a fanout of ''N'' gates.) This definition has a default value for the
''active_high'' parameter as ''true''. So an instance
driver<4> x;
will have four production rules:
x.inp -> sig-
~x.inp -> sig+
sig -> x.outp-
~sig -> x.outp+
However, this behavior can be changed by using:
driver<4,false> x;
In this case, ''sig'' will be connected to ''x.outp''.
Note that ACT is very strict about type-checking; so, for example, ''driver<4>'' and ''driver<4,true>'' are //not// treated as the same type even though the default parameter value for the second template parameter is ''true''.
===== Direction flags =====
Direction specifications can be used for built-in data and channel
types, as well as user-defined types. Consider the ''e1of2''
user-defined channel type that we saw earlier:
defchan e1of2 <: chan(bool) (bool d0,d1,e)
{
spec {
exclhi(d0,d1)
}
...
}
When we use ''e1of2?'' or ''e1of2!'', we need some mechanism to
specify the access permissions for the //port parameters// of the
user-defined type. The convention used is that there are five possible
ways to specify any constraints on access to port parameters. For our
example, we can use one of the following variations in the port parameter list:
bool d0; // No constraints; this port could be read or written
bool! d0; // Both e1of2? and e1of2! have bool! permissions
bool? d0; // Both e1of2? and e1of2! have bool? permissions
bool?! d0; // e1of2? has bool? permissions, and e1of2! has bool! permissions
bool!? d0; // e1of2? has bool! permissions, and e1of2! has bool? permissions
Hence, a better definition of an ''e1of2'' channel would more
completely specify the access permissions for the port parameters in
the following way.
defchan e1of2 <: chan(bool) (bool?! d0,d1; bool!? e) { ... }
A careful examination of the type signature reveals that the sender
and receiver have the appropriate permissions. There is a subtle
interaction between connections and directional types in ACT, and this
is detailed in the section on connections.
===== Macros and Functions within User-defined types =====
User-defined types support additional methods (beyond the special ones for channels and data types).
These methods are of two types:
* //macros//, which correspond to CHP fragments that are used for in-place substitution; and
* //functions// which are supported by [[language:types2:data#pure_structures|pure structures]], which are similar to traditional [[language:expressions#functions|functions]].
==== Macros ====
==== Functions ====
=== Operator overloading ===