Skip to content

Commit

Permalink
Create storage_types! macro to generate type-specific code.
Browse files Browse the repository at this point in the history
All macros now organized in separate files. `storage_types!` allows for
code that explicitly references an underlying storage type, `V`. Part of
#29.
  • Loading branch information
iliekturtles committed Dec 21, 2017
1 parent 3baf7d4 commit 3008a27
Show file tree
Hide file tree
Showing 17 changed files with 1,218 additions and 1,075 deletions.
2 changes: 1 addition & 1 deletion .appveyor.yml
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
# https://github.com/starkat99/appveyor-rust

## Operating System (VM environment) ##
os: Visual Studio 2015
os: Visual Studio 2017

## Build Matrix ##
environment:
Expand Down
4 changes: 2 additions & 2 deletions .travis.yml
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@ rust:
- stable
- beta
- nightly
- 1.15.0
- 1.20.0

matrix:
allow_failures:
Expand All @@ -31,7 +31,7 @@ before_script: |
script: |
cargo build --verbose &&
cargo test --verbose
(test "$TRAVIS_RUST_VERSION" == "1.20.0" || cargo test --verbose)
after_success: |
test "$TRAVIS_RUST_VERSION" != "stable" || cargo coveralls
Expand Down
4 changes: 3 additions & 1 deletion CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,9 @@
availability of the type as an underlying storage type: `usize`, `u8`, `u16`, `u32`, `u64`,
`isize`, `i8`, `i16`, `i32`, `i64`, `bigint`, `biguint`, `rational`, `rational32`, `rational64`,
`bigrational`, `f32`, and `f64`. For compile time reasons only `f32` and `f64` are enabled by
default.
default. A new macro, `storage_types!`, is now available to duplicate code on a per-storage type
basis. See macro documentation for full details. The minimum supported rustc version is now
1.20.0.
* [Breaking] Macro usage and definitions have been simplified and consolidated. `quantities!`,
`replace_ty!`, and `unit!` have been consolidated as "private" match arms of their calling macro.
In order to reduce the chance of macro name collisions `$quantities!` is the only remaining
Expand Down
2 changes: 1 addition & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ Units of measurement is a crate that does automatic type-safe zero-cost
[orbiter]: https://en.wikipedia.org/wiki/Mars_Climate_Orbiter

## Usage
`uom` requires `rustc` 1.15.0 or later. Add this to your `Cargo.toml`:
`uom` requires `rustc` 1.20.0 or later. Add this to your `Cargo.toml`:

```toml
[dependencies]
Expand Down
10 changes: 8 additions & 2 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@
//! [orbiter]: https://en.wikipedia.org/wiki/Mars_Climate_Orbiter
//!
//! ## Usage
//! `uom` requires `rustc` 1.15.0 or later. Add this to your `Cargo.toml`:
//! `uom` requires `rustc` 1.20.0 or later. Add this to your `Cargo.toml`:
//!
//! ```toml
//! [dependencies]
Expand Down Expand Up @@ -188,12 +188,18 @@ pub mod stdlib {
}
}

#[macro_use]
mod storage_types;

#[macro_use]
mod system;

#[macro_use]
mod quantity;

#[cfg(feature = "si")]
#[macro_use]
pub mod si;

#[cfg(test)]
pub mod tests;
mod tests;
250 changes: 250 additions & 0 deletions src/quantity.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,250 @@
/// Macro to implement a [quantity][quantity] and associated [measurement units][measurement]. Note
/// that this macro must be executed in direct submodules of the module where the
/// [`system!`](macro.system.html) macro was executed. `@...` match arms are considered private.
///
/// * `$quantity_attr`: Quantity attributes. Generally used to set documentation comments for the
/// quantity.
/// * `$quantity`: Quantity name (e.g. `Length`).
/// * `$description`: Quantity description (e.g. `"length"`).
/// * `$dim_attr`: Dimension attributes. Generally used to set documentation comments for the
/// quantity's dimension type alias.
/// * `$system`: System of quantities type (e.g. `ISQ`).
/// * `$dimension`: Power of a factor for each base quantity in the system. Power should be
/// represented as a `typenum` type-level integer (e.g. `N1`, `Z0`, `P1`, `P2`, ...).
/// * `$unit`: Unit name (e.g. `meter`, `foot`).
/// * `$conversion`: Conversion from the unit to the base unit of the quantity (e.g. `3.048E-1` to
/// convert `foot` to `meter`).
/// * `$abbreviation`: Unit abbreviation (e.g. `"m"`).
/// * `$singular`: Singular unit description (e.g. `"meter"`).
/// * `$plural`: Plural unit description (e.g. `"meters"`).
///
/// An example invocation is given below for the quantity of length in a meter-kilogram-second
/// system. The `#[macro_use]` attribute must be used when including the `uom` crate to make the
/// `quantity!` macro available.
///
/// ```ignore
/// #[macro_use]
/// extern crate uom;
///
/// # fn main() { }
/// # mod mks {
/// #[macro_use]
/// mod length {
/// quantity! {
/// /// Length (base unit meter, m<sup>1</sup>).
/// quantity: Length; "length";
/// /// Length dimension, m<sup>1</sup>.
/// dimension: Q<P1 /*length*/, Z0 /*mass*/, Z0 /*time*/>;
/// units {
/// @meter: 1.0E0; "m", "meter", "meters";
/// @foot: 3.048E-1; "ft", "foot", "feet";
/// }
/// impl {
/// pub fn custom_description() -> &'static str { "LENGTH" }
/// }
/// }
/// }
/// # #[macro_use]
/// # mod mass {
/// # quantity! {
/// # /// Mass (base unit kilogram, kg<sup>1</sup>).
/// # quantity: Mass; "mass";
/// # /// Mass dimension, kg<sup>1</sup>.
/// # dimension: Q<Z0 /*length*/, P1 /*mass*/, Z0 /*time*/>;
/// # units {
/// # @kilogram: 1.0; "kg", "kilogram", "kilograms";
/// # }
/// # }
/// # }
/// # #[macro_use]
/// # mod time {
/// # quantity! {
/// # /// Time (base unit second, s<sup>1</sup>).
/// # quantity: Time; "time";
/// # /// Time dimension, s<sup>1</sup>.
/// # dimension: Q<Z0 /*length*/, Z0 /*mass*/, P1 /*time*/>;
/// # units {
/// # @second: 1.0; "s", "second", "seconds";
/// # }
/// # }
/// # }
/// # system! {
/// # /// System of quantities, Q.
/// # quantities: Q {
/// # length: meter, L;
/// # mass: kilogram, M;
/// # time: second, T;
/// # }
/// # /// System of units, U.
/// # units: U {
/// # mod length::Length,
/// # mod mass::Mass,
/// # mod time::Time,
/// # }
/// # }
/// # mod f32 {
/// # Q!(mks, f32/*, (centimeter, gram, second)*/);
/// # }
/// # }
/// ```
///
/// [quantity]: http://jcgm.bipm.org/vim/en/1.1.html
/// [measurement]: http://jcgm.bipm.org/vim/en/1.9.html
#[macro_export]
macro_rules! quantity {
(
$(#[$quantity_attr:meta])* quantity: $quantity:ident; $description:expr;
$(#[$dim_attr:meta])* dimension: $system:ident<$($dimension:ident),+>;
units {
$($(#[$unit_attr:meta])* @$unit:ident: $conversion:expr;
$abbreviation:expr, $singular:expr, $plural:expr;)+
}
) => {
$(#[$dim_attr])*
pub type Dimension = super::$system<$($crate::typenum::$dimension),+>;

$(#[$quantity_attr])*
pub type $quantity<U, V> = super::Quantity<Dimension, U, V>;

/// Marker trait to identify measurement units for the quantity. See
/// [`Unit`](../trait.Unit.html).
pub trait Unit<V>: super::Conversion<V> {}

$(quantity!(@unit $(#[$unit_attr])* @$unit);

impl super::Unit for $unit {
#[inline(always)]
fn abbreviation() -> &'static str {
$abbreviation
}

#[inline(always)]
fn singular() -> &'static str {
$singular
}

#[inline(always)]
fn plural() -> &'static str {
$plural
}
})+

/// Quantity description.
#[allow(dead_code)]
#[inline(always)]
pub fn description() -> &'static str {
$description
}

impl<U, V> $quantity<U, V>
where
U: super::Units<Dimension, V>,
V: $crate::num::Num,
{
/// Create a new quantity from the given value and measurement unit.
#[inline(always)]
pub fn new<N>(v: V) -> Self
where
N: Unit<V>,
{
$quantity {
dimension: $crate::stdlib::marker::PhantomData,
units: $crate::stdlib::marker::PhantomData,
value: v * <N as super::Conversion<V>>::conversion()
/ <U as super::Units<Dimension, V>>::conversion(),
}
}

/// Retrieve the value of the quantity in the given measurement unit.
#[inline(always)]
pub fn get<N>(self, _unit: N) -> V
where
N: Unit<V>,
{
self.value * <U as super::Units<Dimension, V>>::conversion()
/ <N as super::Conversion<V>>::conversion()
}

/// Returns the largest integer less than or equal to a number in the given
/// measurement unit.
#[inline(always)]
pub fn floor<N>(self, _unit: N) -> Self
where
V: $crate::num::Float,
N: Unit<V>,
{
Self::new::<N>(self.get(_unit).floor())
}

/// Returns the smallest integer less than or equal to a number in the given
/// measurement unit.
#[inline(always)]
pub fn ceil<N>(self, _unit: N) -> Self
where
V: $crate::num::Float,
N: Unit<V>,
{
Self::new::<N>(self.get(_unit).ceil())
}

/// Returns the nearest integer to a number in the in given measurement unit.
/// Round half-way cases away from 0.0.
#[inline(always)]
pub fn round<N>(self, _unit: N) -> Self
where
V: $crate::num::Float,
N: Unit<V>,
{
Self::new::<N>(self.get(_unit).round())
}

/// Returns the integer part of a number in the given measurement unit.
#[inline(always)]
pub fn trunc<N>(self, _unit: N) -> Self
where
V: $crate::num::Float,
N: Unit<V>,
{
Self::new::<N>(self.get(_unit).trunc())
}

/// Returns the fractional part of a number in the given measurement unit.
#[inline(always)]
pub fn fract<N>(self, _unit: N) -> Self
where
V: $crate::num::Float,
N: Unit<V>,
{
Self::new::<N>(self.get(_unit).fract())
}
}

$(impl<V> Unit<V> for $unit
where
V: $crate::num::Num + $crate::num::FromPrimitive,
{
}

impl<V> super::Conversion<V> for $unit
where
V: $crate::num::Num + $crate::num::FromPrimitive,
{
#[inline(always)]
fn conversion() -> V {
V::from_f64($conversion).unwrap()
}
})+
};
(@unit $(#[$unit_attr:meta])+ @$unit:ident) => {
$(#[$unit_attr])*
#[allow(non_camel_case_types)]
#[derive(Clone, Copy, Debug, Hash)]
pub struct $unit;
};
(@unit @$unit:ident) => {
/// Measurement unit.
#[allow(non_camel_case_types)]
#[derive(Clone, Copy, Debug, Hash)]
pub struct $unit;
};
}
42 changes: 16 additions & 26 deletions src/si/acceleration.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
//! Acceleration (base unit meter per second<sup>2</sup>, m<sup>1</sup> · s<sup>-2</sup>).
quantity! {
/// Acceleration (base unit meter per second<sup>2</sup>, m<sup>1</sup> · s<sup>-2</sup>).
quantity: Acceleration; "acceleration";
Expand Down Expand Up @@ -59,17 +59,20 @@ quantity! {
}

#[cfg(test)]
macro_rules! test {
($V:ident) => {
use ::si::$V::*;
use ::si::acceleration as a;
use ::si::length as l;
use ::si::time as t;
mod tests {
storage_types! {
types: Float;

use num::One;
use si::quantities::*;
use si::acceleration as a;
use si::length as l;
use si::time as t;

#[test]
fn check_dimension() {
let _: Acceleration = Length::new::<l::meter>(1.0) /
(Time::new::<t::second>(1.0) * Time::new::<t::second>(1.0));
let _: Acceleration<V> = Length::new::<l::meter>(V::one()) /
(Time::new::<t::second>(V::one()) * Time::new::<t::second>(V::one()));
}

#[test]
Expand Down Expand Up @@ -97,25 +100,12 @@ macro_rules! test {
test(l::yoctometer, a::yoctometer_per_second_squared);

// TODO #17 Convert to == once PartialEq is implemented.
fn test<L: l::Unit<$V>, A: a::Unit<$V>>(_l: L, a: A) {
assert_eq!(1.0,
(Length::new::<L>(1.0) /
(Time::new::<t::second>(1.0) * Time::new::<t::second>(1.0)))
fn test<L: l::Unit<V>, A: a::Unit<V>>(_l: L, a: A) {
assert_eq!(V::one(),
(Length::new::<L>(V::one()) /
(Time::new::<t::second>(V::one()) * Time::new::<t::second>(V::one())))
.get(a));
}
}
};
}

#[cfg(test)]
mod tests {
#[cfg(feature = "f32")]
mod f32 {
test!(f32);
}

#[cfg(feature = "f64")]
mod f64 {
test!(f64);
}
}
Loading

0 comments on commit 3008a27

Please sign in to comment.