| 
									
										
										
										
											2023-06-23 15:19:08 +02:00
										 |  |  | # Interface Introduction
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ## New style of definitions
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The **mp-units** library decided to use a rather unusual pattern to define entities. | 
					
						
							| 
									
										
										
										
											2023-08-03 21:23:34 +02:00
										 |  |  | Here is how we define `metre` and `second` [SI](../../appendix/glossary.md#si) base units: | 
					
						
							| 
									
										
										
										
											2023-06-23 15:19:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | ```cpp | 
					
						
							|  |  |  | inline constexpr struct metre : named_unit<"m", kind_of<isq::length>> {} metre; | 
					
						
							|  |  |  | inline constexpr struct second : named_unit<"s", kind_of<isq::time>> {} second; | 
					
						
							|  |  |  | ``` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Please note that the above reuses the same identifier for a type and its object. The rationale | 
					
						
							|  |  |  | behind this is that: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - Users always work with objects and never have to spell such a type name. | 
					
						
							|  |  |  | - The types appear in the compilation errors and during debugging. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 11:33:30 +02:00
										 |  |  | !!! important | 
					
						
							| 
									
										
										
										
											2023-06-23 15:19:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     To improve compiler errors' readability and make it easier to correlate them with | 
					
						
							|  |  |  |     a user's written code, a new idiom in the library is to use the same identifier for | 
					
						
							|  |  |  |     a type and its instance. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 20:42:22 +02:00
										 |  |  | ## Strong types instead of aliases
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Let's look again at the above units definitions. Another important point to notice is that | 
					
						
							|  |  |  | all the types describing entities in the library are short, nicely named identifiers | 
					
						
							|  |  |  | that derive from longer, more verbose class template instantiations. This is really important | 
					
						
							|  |  |  | to improve the user experience while debugging the program or analyzing the compilation error. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | !!! note | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Such a practice is rare in the industry. Some popular C++ physical units libraries | 
					
						
							|  |  |  |     generate enormously long error messages where even only the first line failed o fit | 
					
						
							|  |  |  |     on a slide with a tiny font. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-23 15:19:08 +02:00
										 |  |  | ## Entities composability
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Many physical units libraries (in C++ or any other programming language) assign strong types | 
					
						
							|  |  |  | to library entities (i.e. derived units). While `metre_per_second` as a type may not look too | 
					
						
							|  |  |  | scary, consider, for example, units of angular momentum. If we followed this path, its | 
					
						
							| 
									
										
										
										
											2023-08-03 21:23:34 +02:00
										 |  |  | [coherent unit](../../appendix/glossary.md#coherent-derived-unit) would look like | 
					
						
							| 
									
										
										
										
											2023-06-23 15:19:08 +02:00
										 |  |  | `kilogram_metre_sq_per_second`. Now, consider how many scaled versions of this unit would you | 
					
						
							|  |  |  | predefine in the library to ensure that all users are happy with your choice? | 
					
						
							|  |  |  | How expensive would it be from the implementation point of view? | 
					
						
							|  |  |  | What about potential future standardization efforts? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | This is why in **mp-units**, we put a strong requirement to make everything as composable as | 
					
						
							|  |  |  | possible. For example, to create a quantity with a unit of speed, one may write: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ```cpp | 
					
						
							|  |  |  | quantity<si::metre / si::second> q; | 
					
						
							|  |  |  | ``` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | In case you use such an unit often and would prefer to have a handy helper for it, you can | 
					
						
							|  |  |  | always do something like this: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ```cpp | 
					
						
							|  |  |  | constexpr auto metre_per_second = si::metre / si::second; | 
					
						
							|  |  |  | quantity<metre_per_second> q; | 
					
						
							|  |  |  | ``` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | or choose any shorter identifier of your choice. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Coming back to the angular momentum case, thanks to the composability of units, a user can | 
					
						
							|  |  |  | create such a quantity in the following way: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ```cpp | 
					
						
							|  |  |  | using namespace mp_units::si::unit_symbols; | 
					
						
							|  |  |  | auto q = la_vector{1, 2, 3} * isq::angular_momentum[kg * m2 / s]; | 
					
						
							|  |  |  | ``` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | It is a much better solution. It is terse and easy to understand. Please also notice how | 
					
						
							|  |  |  | easy it is to obtain any scaled version of such a unit (i.e. `mg * square(mm) / min`) | 
					
						
							|  |  |  | without having to introduce hundreds of types to predefine them. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 20:42:22 +02:00
										 |  |  | ## Value-based equations
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The **mp-units** library is based on C++20, which greatly improves a user's experience. One of | 
					
						
							|  |  |  | such improvements are value-based equations. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | As we have learned above, the entities are being used as values in the code, and they compose. | 
					
						
							|  |  |  | Moreover, derived entities can be defined in the library using such value-based equations. | 
					
						
							|  |  |  | This is a huge improvement compared to what we can find in other physical units libraries or | 
					
						
							|  |  |  | what we have to deal with when we want to write some equations for `std::ratio`. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | For example, below are a few definitions of the SI derived units showing the power of C++20 | 
					
						
							|  |  |  | extensions to Non-Type Template Parameters, which allows us to directly pass a result of | 
					
						
							| 
									
										
										
										
											2023-08-03 21:23:34 +02:00
										 |  |  | the value-based [unit equation](../../appendix/glossary.md#unit-equation) to a class template | 
					
						
							| 
									
										
										
										
											2023-07-07 20:42:22 +02:00
										 |  |  | definition: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ```cpp | 
					
						
							|  |  |  | inline constexpr struct newton : named_unit<"N", kilogram * metre / square(second)> {} newton; | 
					
						
							|  |  |  | inline constexpr struct pascal : named_unit<"Pa", newton / square(metre)> {} pascal; | 
					
						
							|  |  |  | inline constexpr struct joule : named_unit<"J", newton * metre> {} joule; | 
					
						
							|  |  |  | ``` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-23 15:19:08 +02:00
										 |  |  | ## Expression templates
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The previous chapter provided a rationale for not having predefined types for derived entities. | 
					
						
							|  |  |  | In many libraries, such an approach results in long and unreadable compilation errors, as | 
					
						
							|  |  |  | framework-generated types are typically far from being easy to read and understand. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The **mp-units** library greatly improves the user experience by extensively using | 
					
						
							|  |  |  | expression templates. Such expressions are used consistently throughout the entire library | 
					
						
							|  |  |  | to describe the results of: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-03 21:23:34 +02:00
										 |  |  | - [dimension equation](../../appendix/glossary.md#dimension-equation) - the result is put into | 
					
						
							| 
									
										
										
										
											2023-06-23 15:19:08 +02:00
										 |  |  |   the `derived_dimension<>` class template | 
					
						
							| 
									
										
										
										
											2023-08-03 21:23:34 +02:00
										 |  |  | - [quantity equation](../../appendix/glossary.md#quantity-equation) - the result is put into | 
					
						
							| 
									
										
										
										
											2023-06-23 15:19:08 +02:00
										 |  |  |   the `derived_quantity_spec<>` class template | 
					
						
							| 
									
										
										
										
											2023-08-03 21:23:34 +02:00
										 |  |  | - [unit equation](../../appendix/glossary.md#unit-equation) - the result is put into the | 
					
						
							|  |  |  |   `derived_unit<>` class template | 
					
						
							| 
									
										
										
										
											2023-06-23 15:19:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | For example, if we take the above-defined base units and put the results of their division into | 
					
						
							|  |  |  | the quantity class template like this: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ```cpp | 
					
						
							|  |  |  | quantity<metre / second> q; | 
					
						
							|  |  |  | ``` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | we will observe the following type in the debugger | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ``` | 
					
						
							|  |  |  | (gdb) ptype q | 
					
						
							|  |  |  | type = class mp_units::quantity<mp_units::derived_unit<metre, mp_units::per<second>>(), double> [with Rep = double] { | 
					
						
							|  |  |  | ``` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The same type identifier will be visible in the compilation error (in case it happens). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 11:33:30 +02:00
										 |  |  | !!! important | 
					
						
							| 
									
										
										
										
											2023-06-23 15:19:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Expressions templates are extensively used throughout the library to improve the readability | 
					
						
							|  |  |  |     of the resulting types. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ### Identities
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | As mentioned above, equations can be done on dimensions, quantities, and units. Each such domain must | 
					
						
							|  |  |  | introduce an identity object that can be used in the resulting expressions. Here is the list of | 
					
						
							|  |  |  | identities used in the library: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | | Domain Concept |    Identity     | | 
					
						
							|  |  |  | |----------------|:---------------:| | 
					
						
							|  |  |  | | `Dimension`    | `dimension_one` | | 
					
						
							|  |  |  | | `QuantitySpec` | `dimensionless` | | 
					
						
							|  |  |  | | `Unit`         |      `one`      | | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | In the equations, a user can refer to an identity object either explicitly: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ```cpp | 
					
						
							|  |  |  | constexpr auto my_unit = one / second; | 
					
						
							|  |  |  | ``` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | or implicitly: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ```cpp | 
					
						
							|  |  |  | constexpr auto my_unit = 1 / second; | 
					
						
							|  |  |  | ``` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Both cases with result in the same expression template being generated and put into the wrapper | 
					
						
							|  |  |  | class template. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ### Supported operations and their results
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | There are only a few operations that one can do on such entities and the result of each of them has | 
					
						
							|  |  |  | its unique representation in the library: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | |                   Operation                   | Resulting template expression arguments | | 
					
						
							|  |  |  | |:---------------------------------------------:|:---------------------------------------:| | 
					
						
							|  |  |  | |                    `A * B`                    |                 `A, B`                  | | 
					
						
							|  |  |  | |                    `B * A`                    |                 `A, B`                  | | 
					
						
							|  |  |  | |                    `A * A`                    |              `power<A, 2>`              | | 
					
						
							|  |  |  | |               `{identity} * A`                |                   `A`                   | | 
					
						
							|  |  |  | |               `A * {identity}`                |                   `A`                   | | 
					
						
							|  |  |  | |                    `A / B`                    |               `A, per<B>`               | | 
					
						
							|  |  |  | |                    `A / A`                    |              `{identity}`               | | 
					
						
							|  |  |  | |               `A / {identity}`                |                   `A`                   | | 
					
						
							|  |  |  | |               `{identity} / A`                |          `{identity}, per<A>`           | | 
					
						
							|  |  |  | |                  `pow<2>(A)`                  |              `power<A, 2>`              | | 
					
						
							|  |  |  | |             `pow<2>({identity})`              |              `{identity}`               | | 
					
						
							|  |  |  | |          `sqrt(A)` or `pow<1, 2>(A)`          |            `power<A, 1, 2>`             | | 
					
						
							|  |  |  | | `sqrt({identity})` or `pow<1, 2>({identity})` |              `{identity}`               | | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ### Simplifying the resulting expression templates
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | To limit the length and improve the readability of generated types, there are many rules to simplify | 
					
						
							|  |  |  | the resulting expression template. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 1. **Ordering** | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     The resulting comma-separated arguments of multiplication are always sorted according to | 
					
						
							|  |  |  |     a specific predicate. This is why: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ```cpp | 
					
						
							|  |  |  |     static_assert(A * B == B * A); | 
					
						
							|  |  |  |     static_assert(std::is_same_v<decltype(A * B), decltype(B * A)>); | 
					
						
							|  |  |  |     ``` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     This is probably the most important of all steps, as it allows comparing types and enables the rest of | 
					
						
							|  |  |  |     simplification rules. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 2. **Aggregation** | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     In case two of the same identifiers are found next to each other on the argument list they | 
					
						
							|  |  |  |     will be aggregated in one entry: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     |              Before              |      After       | | 
					
						
							|  |  |  |     |:--------------------------------:|:----------------:| | 
					
						
							|  |  |  |     |              `A, A`              |  `power<A, 2>`   | | 
					
						
							|  |  |  |     |         `A, power<A, 2>`         |  `power<A, 3>`   | | 
					
						
							|  |  |  |     |  `power<A, 1, 2>, power<A, 2>`   | `power<A, 5, 2>` | | 
					
						
							|  |  |  |     | `power<A, 1, 2>, power<A, 1, 2>` |       `A`        | | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 3. **Simplification** | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     In case two of the same identifiers are found in the numerator and denominator argument lists; | 
					
						
							|  |  |  |     they are being simplified into one entry: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     |        Before         |        After         | | 
					
						
							|  |  |  |     |:---------------------:|:--------------------:| | 
					
						
							|  |  |  |     |      `A, per<A>`      |     `{identity}`     | | 
					
						
							|  |  |  |     | `power<A, 2>, per<A>` |         `A`          | | 
					
						
							|  |  |  |     | `power<A, 3>, per<A>` |    `power<A, 2>`     | | 
					
						
							|  |  |  |     | `A, per<power<A, 2>>` | `{identity}, per<A>` | | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 4. **Repacking** | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     In case an expression uses two results of other operations, the components of its arguments are repacked | 
					
						
							|  |  |  |     into one resulting type and simplified there. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     For example, assuming: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ```cpp | 
					
						
							|  |  |  |     constexpr auto X = A / B; | 
					
						
							|  |  |  |     ``` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     then: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     | Operation | Resulting template expression arguments | | 
					
						
							|  |  |  |     |:---------:|:---------------------------------------:| | 
					
						
							|  |  |  |     |  `X * B`  |                   `A`                   | | 
					
						
							|  |  |  |     |  `X * A`  |          `power<A, 2>, per<B>`          | | 
					
						
							|  |  |  |     |  `X * X`  |     `power<A, 2>, per<power<B, 2>>`     | | 
					
						
							|  |  |  |     |  `X / X`  |              `{identity}`               | | 
					
						
							|  |  |  |     |  `X / A`  |          `{identity}, per<B>`           | | 
					
						
							|  |  |  |     |  `X / B`  |          `A, per<power<B, 2>>`          | | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ## Example
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Thanks to all of the features described above, a user may write the code like this one: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ```cpp | 
					
						
							|  |  |  | using namespace mp_units::si::unit_symbols; | 
					
						
							|  |  |  | auto speed = 60. * isq::speed[km / h]; | 
					
						
							|  |  |  | auto duration = 8 * s; | 
					
						
							|  |  |  | auto acceleration = speed / duration; | 
					
						
							| 
									
										
										
										
											2023-08-23 16:46:15 +02:00
										 |  |  | std::cout << "acceleration: " << acceleration << " (" << acceleration.in(m / s2) << ")\n"; | 
					
						
							| 
									
										
										
										
											2023-06-23 15:19:08 +02:00
										 |  |  | ``` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The `acceleration`, being the result of the above code, has the following type | 
					
						
							|  |  |  | (after stripping the `mp_units` namespace for brevity): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ```text | 
					
						
							|  |  |  | quantity<reference<derived_quantity_spec<isq::speed, per<isq::time>>{}, derived_unit<si::kilo_<si::metre{}>, per<non_si::hour, si::second>>{}>{}, int> | 
					
						
							|  |  |  | ``` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | and the text output presents: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ```text | 
					
						
							|  |  |  | acceleration: 7.5 km h⁻¹ s⁻¹ (2.08333 m/s²) | 
					
						
							|  |  |  | ``` |