The GNAT compiler supports dimensionality checking. You can specify physical units for objects and the compiler verifies that uses of these objects are compatible with their dimension, in a fashion that is familiar to engineering practice. The dimensions of algebraic expressions (including powers with static exponents) are computed from their constituents.
This feature depends on Ada 2012 aspect specifications and is available for
versions 7.0.1 and later of GNAT.
The GNAT-specific aspect Dimension_System
allows you to define a system of units; the aspect Dimension
allows you to declare dimensioned quantities within a given system.
(These aspects are described in the ‘Implementation Defined Aspects’
chapter of the GNAT Reference Manual).
The major advantage of this model is that it does not require the declaration of multiple operators for all possible combinations of types: you is only need to use the proper subtypes in object declarations.
The simplest way to impose dimensionality checking on a computation is to make
use of one of the instantiations of the package System.Dim.Generic_Mks
, which
is part of the GNAT library. This generic package defines a floating-point
type MKS_Type
, for which a sequence of dimension names are specified,
together with their conventional abbreviations. You should read the following
together with the full specification of the package, in file
s-digemk.ads
.
type Mks_Type is new Float_Type with Dimension_System => ( (Unit_Name => Meter, Unit_Symbol => 'm', Dim_Symbol => 'L'), (Unit_Name => Kilogram, Unit_Symbol => "kg", Dim_Symbol => 'M'), (Unit_Name => Second, Unit_Symbol => 's', Dim_Symbol => 'T'), (Unit_Name => Ampere, Unit_Symbol => 'A', Dim_Symbol => 'I'), (Unit_Name => Kelvin, Unit_Symbol => 'K', Dim_Symbol => "Theta"), (Unit_Name => Mole, Unit_Symbol => "mol", Dim_Symbol => 'N'), (Unit_Name => Candela, Unit_Symbol => "cd", Dim_Symbol => 'J'));
The package then defines a series of subtypes that correspond to these conventional units. For example:
subtype Length is Mks_Type with Dimension => (Symbol => 'm', Meter => 1, others => 0);
and similarly for Mass
, Time
, Electric_Current
,
Thermodynamic_Temperature
, Amount_Of_Substance
, and
Luminous_Intensity
(the standard set of units of the SI system).
The package also defines conventional names for values of each unit, for example:
m : constant Length := 1.0; kg : constant Mass := 1.0; s : constant Time := 1.0; A : constant Electric_Current := 1.0;
as well as useful multiples of these units:
cm : constant Length := 1.0E-02; g : constant Mass := 1.0E-03; min : constant Time := 60.0; day : constant Time := 60.0 * 24.0 * min; ...
There are three instantiations of System.Dim.Generic_Mks
defined in the
GNAT library:
System.Dim.Float_Mks
based on Float
defined in s-diflmk.ads
.
System.Dim.Long_Mks
based on Long_Float
defined in s-dilomk.ads
.
System.Dim.Mks
based on Long_Long_Float
defined in s-dimmks.ads
.
Using one of these packages, you can then define a derived unit by providing the aspect that specifies its dimensions within the MKS system as well as the string to be used for output of a value of that unit:
subtype Acceleration is Mks_Type with Dimension => ("m/sec^2", Meter => 1, Second => -2, others => 0);
Here’s a complete example:
with System.Dim.MKS; use System.Dim.Mks; with System.Dim.Mks_IO; use System.Dim.Mks_IO; with Text_IO; use Text_IO; procedure Free_Fall is subtype Acceleration is Mks_Type with Dimension => ("m/sec^2", 1, 0, -2, others => 0); G : constant acceleration := 9.81 * m / (s ** 2); T : Time := 10.0*s; Distance : Length; begin Put ("Gravitational constant: "); Put (G, Aft => 2, Exp => 0); Put_Line (""); Distance := 0.5 * G * T ** 2; Put ("distance travelled in 10 seconds of free fall "); Put (Distance, Aft => 2, Exp => 0); Put_Line (""); end Free_Fall;
Execution of this program yields:
Gravitational constant: 9.81 m/sec^2 distance travelled in 10 seconds of free fall 490.50 m
However, incorrect assignments such as:
Distance := 5.0; Distance := 5.0 * kg;
are rejected with the following diagnoses:
Distance := 5.0; >>> dimensions mismatch in assignment >>> left-hand side has dimension [L] >>> right-hand side is dimensionless Distance := 5.0 * kg: >>> dimensions mismatch in assignment >>> left-hand side has dimension [L] >>> right-hand side has dimension [M]
The dimensions of an expression are properly displayed even if there is no explicit subtype for it. If we add to the program:
Put ("Final velocity: "); Put (G * T, Aft =>2, Exp =>0); Put_Line ("");
the output includes:
Final velocity: 98.10 m.s**(-1)
The type Mks_Type
is said to be a ‘dimensionable type’ since it has a
Dimension_System
aspect, and the subtypes Length
, Mass
, etc.,
are said to be ‘dimensioned subtypes’ since each one has a Dimension
aspect.
The Dimension
aspect of a dimensioned subtype S
defines a mapping
from the base type’s Unit_Names to integer (or, more generally, rational)
values. This mapping is the ‘dimension vector’ (also referred to as the
‘dimensionality’) for that subtype, denoted by DV(S)
, and thus for each
object of that subtype. Intuitively, the value specified for each
Unit_Name
is the exponent associated with that unit; a zero value
means that the unit is not used. For example:
declare Acc : Acceleration; ... begin ... end;
Here DV(Acc)
= DV(Acceleration)
=
(Meter=>1, Kilogram=>0, Second=>-2, Ampere=>0, Kelvin=>0, Mole=>0, Candela=>0)
.
Symbolically, we can express this as Meter / Second**2
.
The dimension vector of an arithmetic expression is synthesized from the
dimension vectors of its components, with compile-time dimensionality checks
that help prevent mismatches such as using an Acceleration
where a
Length
is required.
The dimension vector of the result of an arithmetic expression ‘expr’, or
DV(expr)
, is defined as follows, assuming conventional
mathematical definitions for the vector operations that are used:
DV(expr)
is the empty vector.
DV(op expr)
, where ‘op’ is a unary operator, is DV(expr)
DV(expr1 op expr2)
, where ‘op’ is “+” or “-”, is DV(expr1)
provided that DV(expr1)
= DV(expr2)
.
If this condition is not met then the construct is illegal.
DV(expr1 * expr2)
is DV(expr1)
+ DV(expr2)
,
and DV(expr1 / expr2)
= DV(expr1)
- DV(expr2)
.
In this context if one of the ‘expr’s is dimensionless then its empty
dimension vector is treated as (others => 0)
.
DV(expr ** power)
is ‘power’ * DV(expr)
,
provided that ‘power’ is a static rational value. If this condition is not
met then the construct is illegal.
Note that, by the above rules, it is illegal to use binary “+” or “-” to
combine a dimensioned and dimensionless value. Thus an expression such as
acc-10.0
is illegal, where acc
is an object of subtype
Acceleration
.
The dimensionality checks for relationals use the same rules as for “+” and “-” except when comparing to a literal; thus
acc > len
is equivalent to
acc-len > 0.0
and is thus illegal, but
acc > 10.0
is accepted with a warning. Analogously, a conditional expression requires the same dimension vector for each branch (with no exception for literals).
The dimension vector of a type conversion T(expr)
is defined
as follows, based on the nature of T
:
T
is a dimensioned subtype, then DV(T(expr))
is DV(T)
provided that either ‘expr’ is dimensionless or
DV(T)
= DV(expr)
. The conversion is illegal
if ‘expr’ is dimensioned and DV(expr)
/= DV(T)
.
Note that vector equality does not require that the corresponding
Unit_Names be the same.
As a consequence of the above rule, you can convert between different dimension systems that follow the same international system of units, with the seven physical components given in the standard order (length, mass, time, etc.). Thus, you can convert a length in meters to a length in inches (with a suitable conversion factor) but not, for example, to a mass in pounds.
T
is the base type for ‘expr’ (and the dimensionless root type of
the dimension system), then DV(T(expr))
is DV(expr)
.
Thus, if ‘expr’ is of a dimensioned subtype of T
, the conversion may
be regarded as a “view conversion” that preserves dimensionality.
This rule means you can write generic code that can be instantiated with compatible dimensioned subtypes. You include in the generic unit conversions that will consequently be present in instantiations, but conversions to the base type will preserve dimensionality and make it possible to write generic code that is correct with respect to dimensionality.
T
is neither a dimensioned subtype nor a dimensionable
base type), DV(T(expr))
is the empty vector. Thus, a dimensioned
value can be explicitly converted to a non-dimensioned subtype, which
of course then escapes dimensionality analysis.
The dimension vector for a type qualification T'(expr)
is the same
as for the type conversion T(expr)
.
An assignment statement
Source := Target;
requires DV(Source)
= DV(Target)
and analogously for parameter
passing (the dimension vector for the actual parameter must be equal to the
dimension vector for the formal parameter).
When using dimensioned types with elementary functions, you need not
instantiate the Ada.Numerics.Generic_Elementary_Functions
package using
the Mks_Type
nor for any of the derived subtypes such as Distance
.
For functions such as Sqrt
, the dimensional analysis will fail when using
the subtypes because both the parameter and return are of the same type.
An example instantiation
package Mks_Numerics is new Ada.Numerics.Generic_Elementary_Functions (System.Dim.Mks.Mks_Type);