ParaSail can support the association of physical units with numeric types by adding interface parameters which represent the name of the physical unit (e.g. "centimeters"), its shorthand (e.g. "cm"), and the relevant power of the various possible base units (SI, the International System of Units, identifies 7 different base units), and then a scale factor relative to some base unit (if meters are the base unit, then centimeters would have a scale of 0.01).
For a numeric type with an associated unit, the "simple" predefined operators would be "+", "-", and "=?". These are the ones that expect the same unit for both operands. The "simple" multiplication and division operators would require one of the operands to be unitless (the divisor for division must be unitless, either multiplicand can be unitless), producing a result of the same unit as the non-unitless operand. All of the simple operations can be declared as part of the original interface.
The more "complex" multiplication and division operators would support operands with different units, producing a result with yet a third unit. So how do these operators come into being? Where are they declared? One possibility is to have an interface which is designed to take two existing types each with some set of units and produce a new type that corresponds to the product or the quotient of the constituent types. However, you could end up producing multiple types that are effectively equivalent, depending on the order of the operands in the product, or on the particular sequence or products and quotients used to produce the final type. What you really want to declare is a generic operator, which produces a particular result type, and which has a precondition that checks that the units of the operands when combined produce the units of the result type.
It seems relatively straightforward to define a nested interface which takes two other types as interface parameters, and declares a "*" operator that has the appropriate precondition. In fact the precondition could really be part of the nested interface, rather than the operator it defines. The slightly odd thing is that the nested interface, once instantiated, is merely used for the operator it defines. There is never an object of the interface type created. Furthermore, we would like it to be instantiated implicitly (a la C++ generic functions), with the operator being directly "visible"/usable without having to write something like:
ProductWe simply want to write "X * Y" and end up with the right thing. The simplest approach seems to simply allow functions and procedures to have some of their parameters be of "generic" types:::"*"(X, Y)
interface Float_With_Units <Base is Float<>; Name : Univ_String; Short_Hand : Univ_String; Unit_Dimensions : Array< Element_Type => Univ_Real, Index_Type => Dimension_Enum> := (others => 0.0); Scale : Univ_Real> is operator "from_univ"(Value : Univ_Real) {Value in Base::First*Scale .. Base::Last*Scale} -> Float_With_Units; operator "to_univ"(Value : Float_With_Units) -> Result : Univ_Real {Result in Base::First*Scale .. Base::Last*Scale}; operator "+"(Left, Right : Float_With_Units) -> Result : Float_With_Units {[[Result]] == [[Left]] + [[Right]]}; operator "-"(Left, Right : Float_With_Units) -> Float_With_Units; operator "=?"(Left, Right : Float_With_Units) -> Ordering; operator "*"(Left : Float_With_Units; Right : Right_Type is Float_With_Units<>) -> Result : Result_Type is Float_With_Units <Unit_Dimensions => Unit_Dimensions + Right_Type.Unit_Dimensions> {[[Result]] == [[Left]] * [[Right]]}; operator "/"(Left : Left_Type is ... end Float_With_Units;This presumes that "+" is defined for arrays. We might need to have a distinct array interface that defines +/-/=? component-wise.
This approach of having a parameter type of a function/operator be "generic" is similar to the way generic functions work in C++. It would seem to work pretty well, so long as at least one of the parameters is not of a generic type, so the name resolution can take place based on the type of that parameter. For example, if we write:
Z : T := X * Y;we could use the type of X to "find" the "*" operator, and then see whether the types of Y and Z satisfy the other requirements. We couldn't easily "find" the operator if both operands and the result type were all "generic."