source: doc/theses/fangren_yu_MMath/intro.tex @ 5a02308

Last change on this file since 5a02308 was 5a02308, checked in by Peter A. Buhr <pabuhr@…>, 4 weeks ago

respond to Andrew's comments about intro chapter

  • Property mode set to 100644
File size: 32.0 KB
Line 
1\chapter{Introduction}
2
3This thesis is exploratory work I did to understand, fix, and extend the \CFA type-system, specifically, the \newterm{type-resolver} used to satisfy call-site assertions among overloaded variable and function names to allow polymorphic routine calls.
4Assertions are the operations a function uses within its body to perform its computation.
5For example, a polymorphic function summing an array needs a size, zero, assignment, and plus for the array element-type, and a subscript operation for the array type.
6\begin{cfa}
7T sum( T a[$\,$], size_t size ) {
8        @T@ total = { @0@ };  // size, 0 for type T
9        for ( size_t i = 0; i < size; i += 1 )
10                total @+=@ a@[@i@]@; // + and subscript for T
11        return total;
12}
13\end{cfa}
14In certain cases, if the resolver fails to find an exact assertion match, it attempts to find a \emph{best} match using reasonable type conversions.
15Hence, \CFA follows the current trend of replacing nominal inheritance with traits composed of assertions for type matching.
16The over-arching goal is to push the boundary on localized assertion matching, with advanced overloading resolution and type conversions that match programmer expectations in the C programming language.
17Together, the resulting \CFA type-system has a number of unique features making it different from other programming languages with expressive, static, type-systems.
18
19
20\section{Types}
21
22All computers have multiple types because computer architects optimize the hardware around a few basic types with well defined (mathematical) operations: boolean, integral, floating-point, and occasionally strings.
23A programming language and its compiler present ways to declare types that ultimately map into the ones provided by the underlying hardware.
24These language types are thrust upon programmers, with their syntactic and semantic rules and restrictions.
25These rules are used to transform a language expression to a hardware expression.
26Modern programming-languages allow user-defined types and generalize across multiple types using polymorphism.
27Type systems can be static, where each variable has a fixed type during execution and an expression's type is determined once at compile time, or dynamic, where each variable can change type during execution and so an expression's type is reconstructed on each evaluation.
28Expressibility, generalization, and safety are all bound up in a language's type system, and hence, directly affect the capability, build time, and correctness of program development.
29
30
31\section{Overloading}
32
33Overloading allows programmers to use the most meaningful names without fear of name clashes within a program or from external sources, like include files.
34\begin{quote}
35There are only two hard things in Computer Science: cache invalidation and \emph{naming things}. --- Phil Karlton
36\end{quote}
37Experience from \CC and \CFA developers is that the type system implicitly and correctly disambiguates the majority of overloaded names, \ie it is rare to get an incorrect selection or ambiguity, even among hundreds of overloaded (variables and) functions.
38In many cases, a programmer has no idea there are name clashes, as they are silently resolved, simplifying the development process.
39Depending on the language, any ambiguous cases are resolved using some form of qualification and/or casting.
40
41
42\subsection{Operator Overloading}
43
44Virtually all programming languages overload the arithmetic operators across the basic computational types using the number and type of parameters and returns.
45Like \CC, \CFA maps operators to named functions and allows these operators to be overloaded with user-defined types.
46The syntax for operator names uses the @'?'@ character to denote a parameter, \eg left and right unary operators: @?++@ and @++?@, and binary operators @?+?@ and @?<=?@.
47Here, a user-defined type is extended with an addition operation with the same syntax as builtin types.
48\begin{cfa}
49struct S { int i, j };
50S @?+?@( S op1, S op2 ) { return (S){ op1.i + op2.i, op1.j + op2.j }; }
51S s1, s2;
52s1 = s1 @+@ s2;                 $\C[1.75in]{// infix call}$
53s1 = @?+?@( s1, s2 );   $\C{// direct call}\CRT$
54\end{cfa}
55The type system examines each call site and selects the best matching overloaded function based on the number and types of arguments.
56If there are mixed-mode operands, @2 + 3.5@, the type system attempts (safe) conversions, like in C/\CC, converting the argument type(s) to the parameter type(s).
57Conversions are necessary because the hardware rarely supports mix-mode operations, so both operands must be the same type.
58Note, without implicit conversions, programmers must write an exponential number of functions covering all possible exact-match cases among all possible types.
59This approach does not match with programmer intuition and expectation, regardless of any \emph{safety} issues resulting from converted values.
60
61
62\subsection{Function Overloading}
63
64Both \CFA and \CC allow function names to be overloaded, as long as their prototypes differ in the number and type of parameters and returns.
65\begin{cfa}
66void f( void );                 $\C[2in]{// (1): no parameter}$
67void f( char );                 $\C{// (2): overloaded on the number and parameter type}$
68void f( int, int );             $\C{// (3): overloaded on the number and parameter type}$
69f( 'A' );                               $\C{// select (2)}\CRT$
70\end{cfa}
71In this case, the name @f@ is overloaded depending on the number and parameter types.
72The type system examines each call size and selects the best match based on the number and types of the arguments.
73Here, there is a perfect match for the call, @f( 'A' )@ with the number and parameter type of function (2).
74
75Ada, Scala, and \CFA type-systems also use the return type in resolving a call, to pinpoint the best overloaded name.
76For example, in many programming languages with overloading, the following functions are ambiguous without using the return type.
77\begin{cfa}
78int f( int );                   $\C[2in]{// (1); overloaded on return type and parameter}$
79double f( int );                $\C{// (2); overloaded on return type and parameter}$
80int i = f( 3 );                 $\C{// select (1)}$
81double d = f( 3 );              $\C{// select (2)}\CRT$
82\end{cfa}
83Alternatively, if the type system looks at the return type, there is an exact match for each call, which again matches with programmer intuition and expectation.
84This capability can be taken to the extreme, where there are no function parameters.
85\begin{cfa}
86int random( void );             $\C[2in]{// (1); overloaded on return type}$
87double random( void );  $\C{// (2); overloaded on return type}$
88int i = random();               $\C{// select (1)}$
89double d = random();    $\C{// select (2)}\CRT$
90\end{cfa}
91Again, there is an exact match for each call.
92If there is no exact match, a set of minimal, safe conversions can be added to find a best match, as for operator overloading.
93
94
95\subsection{Variable Overloading}
96
97Unlike most programming languages, \CFA has variable overloading within a scope, along with shadow overloading in nested scopes.
98(Shadow overloading is also possible for functions, if a language supports nested function declarations, \eg \CC named, nested, lambda functions.)
99\begin{cfa}
100void foo( double d );
101int v;                              $\C[2in]{// (1)}$
102double v;                               $\C{// (2) variable overloading}$
103foo( v );                               $\C{// select (2)}$
104{
105        int v;                          $\C{// (3) shadow overloading}$
106        double v;                       $\C{// (4) and variable overloading}$
107        foo( v );                       $\C{// select (4)}\CRT$
108}
109\end{cfa}
110It is interesting that shadow overloading is considered a normal programming-language feature with only slight software-engineering problems.
111However, variable overloading within a scope is often considered extremely dangerous, without any evidence to corroborate this claim.
112In contrast, function overloading in \CC occurs silently within the global scope from @#include@ files all the time without problems.
113
114In \CFA, the type system simply treats overloaded variables as an overloaded function returning a value with no parameters.
115Hence, no significant effort is required to support this feature by leveraging the return type to disambiguate as variables have no parameters.
116\begin{cfa}
117int MAX = 2147483647;   $\C[2in]{// (1); overloaded on return type}$
118long int MAX = ...;             $\C{// (2); overloaded on return type}$
119double MAX = ...;               $\C{// (3); overloaded on return type}$
120int i = MAX;                    $\C{// select (1)}$
121long int i = MAX;               $\C{// select (2)}$
122double d = MAX;                 $\C{// select (3)}\CRT$
123\end{cfa}
124Hence, the name @MAX@ can replace all the C type-specific names, \eg @INT_MAX@, @LONG_MAX@, @DBL_MAX@, \etc.
125The result is a significant reduction in names to access typed constants.
126
127As an aside, C has a separate namespace for type and variables allowing overloading between the namespaces, using @struct@ (qualification) to disambiguate.
128\begin{cfa}
129void S() {
130        struct @S@ { int S; };
131        @struct S@ S;
132        void S( @struct S@ S ) { S.S = 1; };
133}
134\end{cfa}
135
136
137\subsection{Constant Overloading}
138
139\CFA is unique in providing restricted constant overloading for the values @0@ and @1@, which have special status in C.
140For example, the value @0@ is both an integer and a pointer literal, so its meaning depends on context.
141In addition, several operations are defined in terms of values @0@ and @1@.
142For example, @if@ and iteration statements in C compare the condition with @0@, and the increment and decrement operators are semantically equivalent to adding or subtracting the value @1@.
143\begin{cfa}
144if ( x ) ++x;        =>    if ( x @!= 0@ ) x @+= 1@;
145for ( ; x; --x )   =>    for ( ; x @!= 0@; x @-= 1@ )
146\end{cfa}
147To generalize this feature, both constants are given types @zero_t@ and @one_t@ in \CFA, which allows overloading various operations for new types that seamlessly work with the special @0@ and @1@ contexts.
148The types @zero_t@ and @one_t@ have special builtin implicit conversions to the various integral types, and a conversion to pointer types for @0@, which allows standard C code involving @0@ and @1@ to work.
149\begin{cfa}
150struct S { int i, j; };
151void ?{}( S & s, zero_t ) { s.[i,j] = 0; } $\C{// constant constructors}$
152void ?{}( S & s, one_t ) { s.[i,j] = 1; }
153S ?=?( S & dst, zero_t ) { dst.[i,j] = 0; return dst; } $\C{// constant assignments}$
154S ?=?( S & dst, one_t ) { dst.[i,j] = 1; return dst; }
155S ?+=?( S & s, one_t ) { s.[i,j] += 1; return s; } $\C{// increment/decrement each field}$
156S ?-=?( S & s, one_t ) { s.[i,j] -= 1; return s; }
157int ?!=?( S s, zero_t ) { return s.i != 0 && s.j != 0; } $\C{// constant comparison}$
158S s = @0@;                      $\C{// initialization}$
159s = @0@;                        $\C{// assignments}$
160s = @1@;
161if ( @s@ ) @++s@;       $\C{// unary ++/-\,- come implicitly from +=/-=}$
162\end{cfa}
163Here, type @S@ is first-class with respect to the basic types, working with all existing implicit C mechanisms.
164
165
166\section{Type Inferencing}
167
168Every variable has a type, but association between them can occur in different ways:
169at the point where the variable comes into existence (declaration) and/or on each assignment to the variable.
170\begin{cfa}
171double x;                               $\C{// type only}$
172float y = 3.1D;                 $\C{// type and initialization}$
173auto z = y;                             $\C{// initialization only}$
174z = "abc";                              $\C{// assignment}$
175\end{cfa}
176For type-only, the programmer specifies the initial type, which remains fixed for the variable's lifetime in statically typed languages.
177For type-and-initialization, the specified and initialization types may not agree.
178For initialization-only, the compiler may select the type by melding programmer and context information.
179When the compiler participates in type selection, it is called \newterm{type inferencing}.
180Note, type inferencing is different from type conversion: type inferencing \emph{discovers} a variable's type before setting its value, whereas conversion has two typed values and performs a (possibly lossy) action to convert one value to the type of the other variable.
181Finally, for assignment, the current variable and expression types may not agree.
182Discovering a variable or function type is complex and has limitations.
183The following covers these issues, and why some schemes are not amenable with the \CFA type system.
184
185One of the first and powerful type-inferencing system is Hindley--Milner~\cite{Damas82}.
186Here, the type resolver starts with the types of the program constants used for initialization and these constant types flow throughout the program, setting all variable and expression types.
187\begin{cfa}
188auto f() {
189        x = 1;   y = 3.5;       $\C{// set types from constants}$
190        x = // expression involving x, y and other local initialized variables
191        y = // expression involving x, y and other local initialized variables
192        return x, y;
193}
194auto w = f();                   $\C{// typing flows outwards}$
195
196void f( auto x, auto y ) {
197        x = // expression involving x, y and other local initialized variables
198        y = // expression involving x, y and other local initialized variables
199}
200s = 1;   t = 3.5;               $\C{// set types from constants}$
201f( s, t );                              $\C{// typing flows inwards}$
202\end{cfa}
203In both overloads of @f@, the type system works from the constant initializations inwards and/or outwards to determine the types of all variables and functions.
204Note, like template meta programming, there could be a new function generated for the second @f@ depending on the types of the arguments, assuming these types are meaningful in the body of @f@.
205Inferring type constraints, by analysing the body of @f@ is possible, and these constraints must be satisfied at each call site by the argument types;
206in this case, parametric polymorphism can allow separate compilation.
207In languages with type inferencing, there is often limited overloading to reduce the search space, which introduces the naming problem.
208Note, return-type inferencing goes in the opposite direction to Hindley--Milner: knowing the type of the result and flowing back through an expression to help select the best possible overloads, and possibly converting the constants for a best match.
209
210In simpler type-inferencing systems, such as C/\CC/\CFA, there are more specific usages.
211\begin{cquote}
212\setlength{\tabcolsep}{10pt}
213\begin{tabular}{@{}lll@{}}
214\multicolumn{1}{c}{\textbf{gcc / \CFA}} & \multicolumn{1}{c}{\textbf{\CC}} \\
215\begin{cfa}
216#define expr 3.0 * i
217typeof(expr) x = expr;
218int y;
219typeof(y) z = y;
220\end{cfa}
221&
222\begin{cfa}
223
224auto x = 3.0 * i;
225int y;
226auto z = y;
227\end{cfa}
228&
229\begin{cfa}
230
231// use type of initialization expression
232
233// use type of initialization expression
234\end{cfa}
235\end{tabular}
236\end{cquote}
237The two important capabilities are:
238\begin{itemize}[topsep=0pt]
239\item
240Not determining or writing long generic types, \eg, given deeply nested generic types.
241\begin{cfa}
242typedef T1(int).T2(float).T3(char).T @ST@;  $\C{// \CFA nested type declaration}$
243@ST@ x, y, x;
244\end{cfa}
245This issue is exaggerated with \CC templates, where type names are 100s of characters long, resulting in unreadable error messages.
246\item
247Ensuring the type of secondary variables, match a primary variable(s).
248\begin{cfa}
249int x; $\C{// primary variable}$
250typeof(x) y, z, w; $\C{// secondary variables match x's type}$
251\end{cfa}
252If the type of @x@ changes, the type of the secondary variables correspondingly updates.
253\end{itemize}
254Note, the use of @typeof@ is more restrictive, and possibly safer, than general type-inferencing.
255\begin{cfa}
256int x;
257type(x) y = ... // complex expression
258type(x) z = ... // complex expression
259\end{cfa}
260Here, the types of @y@ and @z@ are fixed (branded), whereas with type inferencing, the types of @y@ and @z@ are potentially unknown.
261
262
263\subsection{Type-Inferencing Issues}
264
265Each kind of type-inferencing system has its own set of issues that flow onto the programmer in the form of convenience, restrictions, or confusions.
266
267A convenience is having the compiler use its overarching program knowledge to select the best type for each variable based on some notion of \emph{best}, which simplifies the programming experience.
268
269A restriction is the conundrum in type inferencing of when to \emph{brand} a type.
270That is, when is the type of the variable/function more important than the type of its initialization expression.
271For example, if a change is made in an initialization expression, it can cause cascading type changes and/or errors.
272At some point, a variable's type needs to remain constant and the initializing expression needs to be modified or in error when it changes.
273Often type-inferencing systems allow restricting (\newterm{branding}) a variable or function type, so the complier can report a mismatch with the constant initialization.
274\begin{cfa}
275void f( @int@ x, @int@ y ) {  // brand function prototype
276        x = // expression involving x, y and other local initialized variables
277        y = // expression involving x, y and other local initialized variables
278}
279s = 1;   t = 3.5;
280f( s, @t@ ); // type mismatch
281\end{cfa}
282In Haskell, it is common for programmers to brand (type) function parameters.
283
284A confusion is large blocks of code where all declarations are @auto@, as is now common in \CC.
285As a result, understanding and changing the code becomes almost impossible.
286Types provide important clues as to the behaviour of the code, and correspondingly to correctly change or add new code.
287In these cases, a programmer is forced to re-engineer types, which is fragile, or rely on a fancy IDE that can re-engineer types for them.
288For example, given:
289\begin{cfa}
290auto x = @...@
291\end{cfa}
292and the need to write a routine to compute using @x@
293\begin{cfa}
294void rtn( @type of x@ parm );
295rtn( x );
296\end{cfa}
297A programmer must re-engineer the type of @x@'s initialization expression, reconstructing the possibly long generic type-name.
298In this situation, having the type name or its short alias is essential.
299
300The \CFA's type system tries to prevent type-resolution mistakes by relying heavily on the type of the left-hand side of assignment to pinpoint the right types within an expression.
301Type inferencing defeats this goal because there is no left-hand type.
302Fundamentally, type inferencing tries to magic away variable types from the programmer.
303However, this results in lazy programming with the potential for poor performance and safety concerns.
304Types are as important as control-flow in writing a good program, and should not be masked, even if it requires the programmer to think!
305A similar issue is garbage collection, where storage management is magicked away, often resulting in poor program design and performance.\footnote{
306There are full-time Java consultants, who are hired to find memory-management problems in large Java programs.}
307The entire area of Computer-Science data-structures is obsessed with time and space, and that obsession should continue into regular programming.
308Understanding space and time issues is an essential part of the programming craft.
309Given @typedef@ and @typeof@ in \CFA, and the strong desire to use the left-hand type in resolution, implicit type-inferencing is unsupported.
310Should a significant need arise, this feature can be revisited.
311
312
313\section{Polymorphism}
314
315\CFA provides polymorphic functions and types, where a polymorphic function can constrain types using assertions based on traits.
316
317
318\subsection{Polymorphic Function}
319
320The signature feature of the \CFA type-system is parametric-polymorphic functions~\cite{forceone:impl,Cormack90,Duggan96}, generalized using a @forall@ clause (giving the language its name).
321\begin{cfa}
322@forall( T )@ T identity( T val ) { return val; }
323int forty_two = identity( 42 );         $\C{// T is bound to int, forty\_two == 42}$
324\end{cfa}
325This @identity@ function can be applied to an \newterm{object type}, \ie a type with a known size and alignment, which is sufficient to stack allocate, default or copy initialize, assign, and delete.
326The \CFA implementation passes the size and alignment for each type parameter, as well as auto-generated default and copy constructors, assignment operator, and destructor.
327For an incomplete \newterm{data type}, \eg pointer/reference types, this information is not needed.
328\begin{cfa}
329forall( T * ) T * identity( T * val ) { return val; }
330int i, * ip = identity( &i );
331\end{cfa}
332Unlike \CC template functions, \CFA polymorphic functions are compatible with C \emph{separate compilation}, preventing compilation and code bloat.
333
334To constrain polymorphic types, \CFA uses \newterm{type assertions}~\cite[pp.~37-44]{Alphard} to provide further type information, where type assertions may be variable or function declarations that depend on a polymorphic type variable.
335For example, the function @twice@ works for any type @T@ with a matching addition operator.
336\begin{cfa}
337forall( T @| { T ?+?(T, T); }@ ) T twice( T x ) { return x @+@ x; }
338int val = twice( twice( 3 ) );  $\C{// val == 12}$
339\end{cfa}
340For example. parametric polymorphism and assertions occurs in existing type-unsafe (@void *@) C @qsort@ to sort an array.
341\begin{cfa}
342void qsort( void * base, size_t nmemb, size_t size, int (*cmp)( const void *, const void * ) );
343\end{cfa}
344Here, the polymorphism is type-erasure, and the parametric assertion is the comparison routine, which is explicitly passed.
345\begin{cfa}
346enum { N = 5 };
347double val[N] = { 5.1, 4.1, 3.1, 2.1, 1.1 };
348int cmp( const void * v1, const void * v2 ) { $\C{// compare two doubles}$
349        return *(double *)v1 < *(double *)v2 ? -1 : *(double *)v2 < *(double *)v1 ? 1 : 0;
350}
351qsort( val, N, sizeof( double ), cmp );
352\end{cfa}
353The equivalent type-safe version in \CFA is a wrapper over the C version.
354\begin{cfa}
355forall( ET | { int @?<?@( ET, ET ); } ) $\C{// type must have < operator}$
356void qsort( ET * vals, size_t dim ) {
357        int cmp( const void * t1, const void * t2 ) { $\C{// nested function}$
358                return *(ET *)t1 @<@ *(ET *)t2 ? -1 : *(ET *)t2 @<@ *(ET *)t1 ? 1 : 0;
359        }
360        qsort( vals, dim, sizeof(ET), cmp ); $\C{// call C version}$
361}
362qsort( val, N );  $\C{// deduct type double, and pass builtin < for double}$
363\end{cfa}
364The nested function @cmp@ is implicitly built and provides the interface from typed \CFA to untyped (@void *@) C.
365Providing a hidden @cmp@ function in \CC is awkward as lambdas do not use C calling conventions and template declarations cannot appear in block scope.
366% In addition, an alternate kind of return is made available: position versus pointer to found element.
367% \CC's type system cannot disambiguate between the two versions of @bsearch@ because it does not use the return type in overload resolution, nor can \CC separately compile a template @bsearch@.
368Call-site inferencing and nested functions provide a localized form of inheritance.
369For example, the \CFA @qsort@ can be made to sort in descending order by locally changing the behaviour of @<@.
370\begin{cfa}
371{
372        int ?<?( double x, double y ) { return x @>@ y; } $\C{// locally override behaviour}$
373        qsort( vals, 10 );                                                      $\C{// descending sort}$
374}
375\end{cfa}
376The local version of @?<?@ overrides the built-in @?<?@ so it is passed to @qsort@.
377The local version performs @?>?@, making @qsort@ sort in descending order.
378Hence, any number of assertion functions can be overridden locally to maximize the reuse of existing functions and types, without the construction of a named inheritance hierarchy.
379A final example is a type-safe wrapper for C @malloc@, where the return type supplies the type/size of the allocation, which is impossible in most type systems.
380\begin{cfa}
381static inline forall( T & | sized(T) )
382T * malloc( void ) {
383        if ( _Alignof(T) <= __BIGGEST_ALIGNMENT__ ) return (T *)malloc( sizeof(T) ); // C allocation
384        else return (T *)memalign( _Alignof(T), sizeof(T) );
385}
386// select type and size from left-hand side
387int * ip = malloc();  double * dp = malloc();  $@$[aligned(64)] struct S {...} * sp = malloc();
388\end{cfa}
389The @sized@ assertion passes size and alignment as a data object has no implicit assertions.
390Both assertions are used in @malloc@ via @sizeof@ and @_Alignof@.
391
392These mechanism are used to construct type-safe wrapper-libraries condensing hundreds of existing C functions into tens of \CFA overloaded functions.
393Hence, existing C legacy code is leveraged as much as possible;
394other programming languages must build supporting libraries from scratch, even in \CC.
395
396
397\subsection{Traits}
398
399\CFA provides \newterm{traits} to name a group of type assertions, where the trait name allows specifying the same set of assertions in multiple locations, preventing repetition mistakes at each function declaration.
400\begin{cquote}
401\begin{tabular}{@{}l|@{\hspace{10pt}}l@{}}
402\begin{cfa}
403trait @sumable@( T ) {
404        void @?{}@( T &, zero_t ); // 0 literal constructor
405        T ?+?( T, T );           // assortment of additions
406        T @?+=?@( T &, T );
407        T ++?( T & );
408        T ?++( T & );
409};
410\end{cfa}
411&
412\begin{cfa}
413forall( T @| sumable( T )@ ) // use trait
414T sum( T a[$\,$], size_t size ) {
415        @T@ total = { @0@ };  // initialize by 0 constructor
416        for ( size_t i = 0; i < size; i += 1 )
417                total @+=@ a[i]; // select appropriate +
418        return total;
419}
420\end{cfa}
421\end{tabular}
422\end{cquote}
423Traits are simply flatten at the use point, as if written in full by the programmer, where traits often contain overlapping assertions, \eg operator @+@.
424Hence, trait names play no part in type equivalence.
425Note, the type @T@ is an object type, and hence, has the implicit internal trait @otype@.
426\begin{cfa}
427trait otype( T & | sized(T) ) {
428        void ?{}( T & );                                                $\C{// default constructor}$
429        void ?{}( T &, T );                                             $\C{// copy constructor}$
430        void ?=?( T &, T );                                             $\C{// assignment operator}$
431        void ^?{}( T & );                                               $\C{// destructor}$
432};
433\end{cfa}
434The implicit routines are used by the @sumable@ operator @?+=?@ for the right side of @?+=?@ and return.
435
436If the array type is not a builtin type, an extra type parameter and assertions are required, like subscripting.
437This case is generalized in polymorphic container-types, such as a list with @insert@ and @remove@ operations, and an element type with copy and assignment.
438
439
440\subsection{Generic Types}
441
442A significant shortcoming of standard C is the lack of reusable type-safe abstractions for generic data structures and algorithms.
443Broadly speaking, there are three approaches to implement abstract data structures in C.
444\begin{enumerate}[leftmargin=*]
445\item
446Write bespoke data structures for each context they are needed.
447While this approach is flexible and supports integration with the C type checker and tooling, it is tedious and error prone, especially for more complex data structures.
448\item
449Use @void *@-based polymorphism, \eg the C standard library functions @bsearch@ and @qsort@, which allow for the reuse of code with common functionality.
450However, this approach eliminates the type checker's ability to ensure argument types are properly matched, often requiring a number of extra function parameters, pointer indirection, and dynamic allocation that is otherwise unnecessary.
451\item
452Use preprocessor macros, similar to \CC @templates@, to generate code that is both generic and type checked, but errors may be difficult to interpret.
453Furthermore, writing and using preprocessor macros is difficult and inflexible.
454\end{enumerate}
455
456\CC, Java, and other languages use \newterm{generic types} to produce type-safe abstract data-types.
457\CFA generic types integrate efficiently and naturally with the existing polymorphic functions, while retaining backward compatibility with C and providing separate compilation.
458However, for known concrete parameters, the generic-type definition can be inlined, like \CC templates.
459
460A generic type can be declared by placing a @forall@ specifier on a @struct@ or @union@ declaration and instantiated using a parenthesized list of types after the type name.
461\begin{cquote}
462\begin{tabular}{@{}l|@{\hspace{10pt}}l@{}}
463\begin{cfa}
464@forall( F, S )@ struct pair {
465        F first;        S second;
466};
467@forall( F, S )@ // object
468S second( pair( F, S ) p ) { return p.second; }
469@forall( F *, S * )@ // sized
470S * second( pair( F *, S * ) p ) { return p.second; }
471\end{cfa}
472&
473\begin{cfa}
474pair( double, int ) dpr = { 3.5, 42 };
475int i = second( dpr );
476pair( void *, int * ) vipr = { 0p, &i };
477int * ip = second( vipr );
478double d = 1.0;
479pair( int *, double * ) idpr = { &i, &d };
480double * dp = second( idpr );
481\end{cfa}
482\end{tabular}
483\end{cquote}
484\CFA generic types are \newterm{fixed} or \newterm{dynamic} sized.
485Fixed-size types have a fixed memory layout regardless of type parameters, whereas dynamic types vary in memory layout depending on their type parameters.
486For example, the type variable @T *@ is fixed size and is represented by @void *@ in code generation;
487whereas, the type variable @T@ is dynamic and set at the point of instantiation.
488The difference between fixed and dynamic is the complexity and cost of field access.
489For fixed, field offsets are computed (known) at compile time and embedded as displacements in instructions.
490For dynamic, field offsets are computed at compile time at the call site, stored in an array of offset values, passed as a polymorphic parameter, and added to the structure address for each field dereference within a polymorphic routine.
491See~\cite[\S~3.2]{Moss19} for complete implementation details.
492
493Currently, \CFA generic types allow assertion.
494For example, the following declaration of a sorted set-type ensures the set key supports equality and relational comparison.
495\begin{cfa}
496forall( Elem, @Key@ | { _Bool ?==?( Key, Key ); _Bool ?<?( Key, Key ); } )
497struct Sorted_Set { Elem elem; @Key@ key; ... };
498\end{cfa}
499However, the operations that insert/remove elements from the set should not appear as part of the generic-types assertions.
500\begin{cfa}
501forall( @Elem@ | /* any assertions on element type */ ) {
502        void insert( Sorted_Set set, @Elem@ elem ) { ... }
503        bool remove( Sorted_Set set, @Elem@ elem ) { ... } // false => element not present
504        ... // more set operations
505} // distribution
506\end{cfa}
507(Note, the @forall@ clause can be distributed across multiple functions.)
508For software-engineering reasons, the set assertions would be refactored into a trait to allow alternative implementations, like a Java \lstinline[language=java]{interface}.
509
510In summation, the \CFA type system inherits \newterm{nominal typing} for concrete types from C, and adds \newterm{structural typing} for polymorphic types.
511Traits are used like interfaces in Java or abstract base-classes in \CC, but without the nominal inheritance relationships.
512Instead, each polymorphic function or generic type defines the structural type needed for its execution, which is fulfilled at each call site from the lexical environment, like Go~\cite{Go} or Rust~\cite{Rust} interfaces.
513Hence, new lexical scopes and nested functions are used extensively to create local subtypes, as in the @qsort@ example, without having to manage a nominal inheritance hierarchy.
514
515
516\section{Contributions}
517
518\begin{enumerate}
519\item
520\item
521\item
522\end{enumerate}
523
524
525\begin{comment}
526From: Andrew James Beach <ajbeach@uwaterloo.ca>
527To: Peter Buhr <pabuhr@uwaterloo.ca>, Michael Leslie Brooks <mlbrooks@uwaterloo.ca>,
528    Fangren Yu <f37yu@uwaterloo.ca>, Jiada Liang <j82liang@uwaterloo.ca>
529Subject: Re: Haskell
530Date: Fri, 30 Aug 2024 16:09:06 +0000
531
532Do you mean:
533
534one = 1
535
536And then write a bunch of code that assumes it is an Int or Integer (which are roughly int and Int in Cforall) and then replace it with:
537
538one = 1.0
539
540And have that crash? That is actually enough, for some reason Haskell is happy to narrow the type of the first literal (Num a => a) down to Integer but will not do the same for (Fractional a => a) and Rational (which is roughly Integer for real numbers). Possibly a compatibility thing since before Haskell had polymorphic literals.
541
542Now, writing even the first version will fire a -Wmissing-signatures warning, because it does appear to be understood that just from a documentation perspective, people want to know what types are being used. Now, if you have the original case and start updating the signatures (adding one :: Fractional a => a), you can eventually get into issues, for example:
543
544import Data.Array (Array, Ix, (!))
545atOne :: (Ix a, Frational a) => Array a b -> b - - In CFA: forall(a | Ix(a) | Frational(a), b) b atOne(Array(a, b) const & array)
546atOne = (! one)
547
548Which compiles and is fine except for the slightly awkward fact that I don't know of any types that are both Ix and Fractional types. So you might never be able to find a way to actually use that function. If that is good enough you can reduce that to three lines and use it.
549
550Something that just occurred to me, after I did the above examples, is: Are there any classic examples in literature I could adapt to Haskell?
551
552Andrew
553
554PS, I think it is too obvious of a significant change to work as a good example but I did mock up the structure of what I am thinking you are thinking about with a function. If this helps here it is.
555
556doubleInt :: Int -> Int
557doubleInt x = x * 2
558
559doubleStr :: String -> String
560doubleStr x = x ++ x
561
562-- Missing Signature
563action = doubleInt - replace with doubleStr
564
565main :: IO ()
566main = print $ action 4
567\end{comment}
Note: See TracBrowser for help on using the repository browser.