-
Notifications
You must be signed in to change notification settings - Fork 97
/
lib.rs
629 lines (556 loc) · 21.6 KB
/
lib.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
//! Units of measurement is a crate that does automatic type-safe zero-cost
//! [dimensional analysis][analysis]. You can create your own systems or use the pre-built
//! [International System of Units][si] (SI) which is based on the
//! [International System of Quantities][isq] (ISQ) and includes numerous [quantities][quantity]
//! (length, mass, time, ...) with conversion factors for even more numerous
//! [measurement units][measurement] (meter, kilometer, foot, mile, ...). No more crashing your
//! [climate orbiter][orbiter]!
//!
//! [analysis]: https://en.wikipedia.org/wiki/Dimensional_analysis
//! [si]: http://jcgm.bipm.org/vim/en/1.16.html
//! [isq]: http://jcgm.bipm.org/vim/en/1.6.html
//! [quantity]: http://jcgm.bipm.org/vim/en/1.1.html
//! [measurement]: http://jcgm.bipm.org/vim/en/1.9.html
//! [orbiter]: https://en.wikipedia.org/wiki/Mars_Climate_Orbiter
//!
//! ## Usage
//! `uom` requires `rustc` 1.24.1 or later. Add this to your `Cargo.toml`:
//!
//! ```toml
//! [dependencies]
//! uom = "0.23.1"
//! ```
//!
//! and this to your crate root:
//!
//! ```rust
//! extern crate uom;
//! ```
//!
//! The simple example below shows how to use quantities and units as well as how `uom` stops
//! invalid operations:
//!
#![cfg_attr(all(feature = "si", feature = "f32"), doc = " ```rust")]
#![cfg_attr(not(all(feature = "si", feature = "f32")), doc = " ```rust,ignore")]
//! extern crate uom;
//!
//! use uom::si::f32::*;
//! use uom::si::length::kilometer;
//! use uom::si::time::second;
//!
//! fn main() {
//! let length = Length::new::<kilometer>(5.0);
//! let time = Time::new::<second>(15.0);
//! let velocity/*: Velocity*/ = length / time;
//! let _acceleration = calc_acceleration(velocity, time);
//! //let error = length + time; // error[E0308]: mismatched types
//! }
//!
//! fn calc_acceleration(velocity: Velocity, time: Time) -> Acceleration {
//! velocity / time
//! }
//! ```
//!
//! See examples provided with the source for more advanced usage including how to create `Quantity`
//! type aliases for a different set of base units and how to create an entirely new system of
//! quantities.
//!
//! ## Features
//! `uom` has multiple `Cargo` features for controlling available underlying storage types, the
//! inclusion of the pre-built [International System of Units][si] (SI), support for
//! [Serde][serde], and `no_std` functionality. The features are described below. `f32`, `f64`,
//! `std`, and `si` are enabled by default. Features can be cherry-picked by using the
//! `--no-default-features` and `--features "..."` flags when compiling `uom` or specifying
//! features in Cargo.toml:
//!
//! ```toml
//! [dependencies]
//! uom = {
//! version = "0.23.1",
//! default-features = false,
//! features = [
//! "autoconvert", # automatic base unit conversion.
//! "usize", "u8", "u16", "u32", "u64", # Unsigned integer storage types.
//! "isize", "i8", "i16", "i32", "i64", # Signed interger storage types.
//! "bigint", "biguint", # Arbitrary width integer storage types.
//! "rational", "rational32", "rational64", "bigrational", # Integer ratio storage types.
//! "f32", "f64", # Floating point storage types.
//! "si", "std", # Built-in SI system and std library support.
//! "use_serde", # Serde support.
//! ]
//! }
//! ```
//!
//! * `autoconvert` -- Feature to enable automatic conversion between base units in binary
//! operators. Disabling the feature only allows for quantities with the same base units to
//! directly interact. The feature exists to account for compiler limitations where zero-cost
//! code is not generated for non-floating point underlying storage types.
//! * `usize`, `u8`, `u16`, `u32`, `u64`, `isize`, `i8`, `i16`, `i32`, `i64`, `bigint`, `biguint`,
//! `rational`, `rational32`, `rational64`, `bigrational`, `f32`, `f64` -- Features to enable
//! underlying storage types. At least one of these features must be enabled. `f32` and `f64` are
//! enabled by default. See the [Design](#design) section for implications of choosing different
//! underlying storage types.
//! * `si` -- Feature to include the pre-built [International System of Units][si] (SI). Enabled by
//! default.
//! * `std` -- Feature to compile with standard library support. Disabling this feature compiles
//! `uom` with `no_std`. Enabled by default.
//! * `use_serde` -- Feature to enable support for serialization and deserialization of quantities
//! with the [Serde][serde] crate. Disabled by default.
//!
//! [si]: http://jcgm.bipm.org/vim/en/1.16.html
//! [serde]: https://serde.rs/
//!
//! ## Design
//! Rather than working with [measurement units](http://jcgm.bipm.org/vim/en/1.9.html) (meter,
//! kilometer, foot, mile, ...) `uom` works with [quantities](http://jcgm.bipm.org/vim/en/1.1.html)
//! (length, mass, time, ...). This simplifies usage because units are only involved at interface
//! boundaries: the rest of your code only needs to be concerned about the quantities involved. This
//! also makes operations on quantities (+, -, \*, /, ...) have zero runtime cost<sup>1</sup> over
//! using the raw storage type (e.g. `f32`).
//!
//! `uom` normalizes values to the [base unit](http://jcgm.bipm.org/vim/en/1.10.html) for the
//! quantity. Alternative base units can be used by executing the macro defined for the system of
//! quantities (`ISQ!` for the SI). `uom` supports `usize`, `u8`, `u16`, `u32`, `u64`, `isize`,
//! `i8`, `i16`, `i32`, `i64`, `bigint`, `biguint`, `rational`, `rational32`, `rational64`,
//! `bigrational`, `f32`, and `f64` as the underlying storage type.
//!
//! A consequence of normalizing values to the base unit is that some values may not be able to be
//! represented or can't be precisely represented for floating point and rational underlying
//! storage types. For example if the base unit of `length` is `meter` and the underlying storage
//! type is `i32` then values like `1 centimeter` or `1.1 meter` cannot be represented. `1
//! centimeter` is normalized to `0.01 meter` which can't be stored in an `i32`. `uom` only allows
//! units to be used safely. Users of this library will still need to be aware of implementation
//! details of the underlying storage type including limits and precision.
//!
//! 1. As of `rustc` 1.25.0 where codegen bug
//! [#38269](https://github.com/rust-lang/rust/issues/38269) is resolved.
//!
//! ## Contributing
//! Contributions are welcome from everyone. Submit a pull request, an issue, or just add comments
//! to an existing item. The [International Bureau of Weights and Measures][BIPM] is an
//! international standards organization that publishes the [SI Brochure][brochure]. This document
//! defines the [SI] and can be used as a comprehensive reference for changes to `uom`. Conversion
//! factors for non-SI units can be found in NIST [Special Publication 811][nist811].
//!
//! Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in
//! the work by you, as defined in the Apache-2.0 license, shall be dual licensed as below, without
//! any additional terms or conditions.
//!
//! ### License
//! Licensed under either of
//!
//! * Apache License, Version 2.0, ([LICENSE-APACHE](LICENSE-APACHE) or
//! <http://www.apache.org/licenses/LICENSE-2.0>)
//! * MIT license ([LICENSE-MIT](LICENSE-MIT) or <http://opensource.org/licenses/MIT>)
//!
//! at your option.
//!
//! [BIPM]: http://www.bipm.org/en/about-us/
//! [brochure]: http://www.bipm.org/en/publications/si-brochure/
//! [si]: http://jcgm.bipm.org/vim/en/1.16.html
//! [nist811]: https://www.nist.gov/pml/nist-guide-si-appendix-b9-factors-units-listed-kind-quantity-or-field-science
// Compile with `no_std` when the `std` feature is not specified.
#![cfg_attr(not(feature = "std"), no_std)]
// Rustc lints.
#![forbid(unsafe_code)]
#![warn(
//bare_trait_objects, // Requires rustc 1.27.
missing_copy_implementations,
missing_debug_implementations,
missing_docs,
trivial_casts,
trivial_numeric_casts,
unused_extern_crates,
unused_import_braces,
unused_qualifications,
unused_results
)]
// Clippy lints.
#![cfg_attr(
feature = "cargo-clippy",
allow(clippy::deprecated_cfg_attr, clippy::excessive_precision, clippy::inline_always)
)]
// Lints allowed in tests because they are unavoidable in the generic code when a type may or may
// not need to be dereferenced or cloned.
#![cfg_attr(
all(feature = "cargo-clippy", test),
allow(clippy::op_ref, clippy::clone_on_copy, clippy::float_cmp)
)]
// Fail to compile if no underlying storage type features are specified.
#[cfg_attr(rustfmt, rustfmt_skip)]
#[cfg(not(any(
feature = "usize", feature = "u8", feature = "u16", feature = "u32", feature = "u64",
feature = "isize", feature = "i8", feature = "i16", feature = "i32", feature = "i64",
feature = "bigint", feature = "biguint",
feature = "rational", feature = "rational32", feature = "rational64", feature = "bigrational",
feature = "f32", feature = "f64", )))]
compile_error!("A least one underlying storage type must be enabled. See the features section of \
uom documentation for available underlying storage type options.");
#[doc(hidden)]
pub extern crate num_traits;
#[doc(hidden)]
#[cfg(feature = "bigint-support")]
pub extern crate num_bigint;
#[doc(hidden)]
#[cfg(any(feature = "rational-support", feature = "bigint-support"))]
pub extern crate num_rational;
#[doc(hidden)]
#[cfg(feature = "serde")]
pub extern crate serde;
#[doc(hidden)]
pub extern crate typenum;
#[cfg(all(test, any(feature = "f32", feature = "f64")))]
#[macro_use]
extern crate approx;
#[cfg(test)]
#[macro_use]
extern crate quickcheck;
#[cfg(all(test, feature = "serde"))]
extern crate serde_json;
#[cfg(test)]
#[macro_use]
extern crate static_assertions;
// Conditionally import `core` or `std` based on feature selection.
#[doc(hidden)]
pub mod lib {
#[cfg(not(feature = "std"))]
pub use core::*;
#[cfg(feature = "std")]
pub use std::*;
// Re-export `ops` module along with `typenum::ops` to provide all types in a single mod. This
// allows the `system!` macro to reference all operations by the absolute path. Macro paths and
// idents can't easily be combined without a `use` statement that pollutes the macro execution
// location's namespace.
pub mod ops {
#[cfg(not(feature = "std"))]
pub use core::ops::*;
#[cfg(feature = "std")]
pub use std::ops::*;
pub use typenum::type_operators::*;
}
}
// Conditionally import num sub-crate types based on feature selection.
#[doc(hidden)]
pub mod num {
#[cfg(feature = "std")]
pub use num_traits::float::Float;
#[cfg(not(feature = "std"))]
pub use num_traits::float::FloatCore as Float;
pub use num_traits::{pow, FromPrimitive, Num, One, Saturating, Signed, Zero};
#[cfg(feature = "bigint-support")]
pub use num_bigint::{BigInt, BigUint};
#[cfg(feature = "rational-support")]
pub use num_rational::Rational;
#[cfg(feature = "bigint-support")]
pub use num_rational::BigRational;
#[cfg(any(feature = "rational-support", feature = "bigint-support"))]
pub mod rational {
pub use num_rational::*;
}
}
/// Primitive traits and types representing basic properties of types.
pub mod marker {
/// Trait to denote that a quantity is able to be added with a quantity of the same dimensions.
/// When a specific quantity's kind inherits this trait `ops::Add` is implemented
/// automatically.
pub trait Add {}
/// Trait to denote that a quantity is able to be added with a quantity of the same dimensions.
/// When a specific quantity's kind inherits this trait `ops::AddAssign` is implemented
/// automatically.
pub trait AddAssign {}
/// Trait to denote that a quantity is able to be subtracted with a quantity of the same
/// dimensions. When a specific quantity's kind inherits this trait `ops::Sub` is implemented
/// automatically.
pub trait Sub {}
/// Trait to denote that a quantity is able to be subtracted with a quantity of the same
/// dimensions. When a specific quantity's kind inherits this trait `ops::SubAssign` is
/// implemented automatically.
pub trait SubAssign {}
/// Trait to denote that a quantity is able to be multiplied with a quantity of the same
/// dimensions. When a specific quantity's kind inherits this trait `ops::Mul` is implemented
/// automatically.
pub trait Mul {}
/// Trait to denote that a quantity is able to be multiplied with a quantity of the same
/// dimensions. When a specific quantity's kind inherits this trait `ops::MulAssign` is
/// implemented automatically.
pub trait MulAssign {}
/// Trait to denote that a quantity is able to be divided with a quantity of the same
/// dimensions. When a specific quantity's kind inherits this trait `ops::Div` is implemented
/// automatically.
pub trait Div {}
/// Trait to denote that a quantity is able to be divided with a quantity of the same
/// dimensions. When a specific quantity's kind inherits this trait `ops::DivAssign` is
/// implemented automatically.
pub trait DivAssign {}
/// Trait to denote that a quantity is able to be negated. When a specific quantity's kind
/// inherits this trait `ops::Neg` is implemented automatically.
pub trait Neg {}
/// Trait to denote that a quantity is able to calculate a remainder with a quantity of the
/// same dimensions. When a specific quantity's kind inherits this trait `ops::Rem` is
/// implemented automatically.
pub trait Rem {}
/// Trait to denote that a quantity is able to calculate a remainder with a quantity of the
/// same dimensions. When a specific quantity's kind inherits this trait `ops::RemAssign` is
/// implemented automatically.
pub trait RemAssign {}
/// Trait to denote that a quantity is able to perform saturating additions and subtractions
/// with a quantity of the same dimensions. When a specific quantity's kind inherits this trait
/// `ops::Saturating` is implemented automatically.
pub trait Saturating {}
}
#[macro_use]
mod features;
#[macro_use]
mod storage_types;
#[macro_use]
mod system;
#[macro_use]
mod quantity;
#[cfg(feature = "si")]
#[macro_use]
pub mod si;
#[cfg(test)]
mod tests;
/// Operations performed on the constant portion of the [conversion factor][factor]. Used to help
/// guide optimizations when floating point underlying storage types are used.
///
/// For a value, `v: Float`, adding `-0.0` is a no-op while adding `0.0` will change the sign if
/// `v` is `-0.0`. The opposite is true for subtraction.
///
/// ```ignore
/// v
/// 0.0 + -0.0 = 0.0
/// -0.0 + 0.0 = 0.0 // v + 0.0 != v
/// -0.0 + -0.0 = -0.0
/// 0.0 - -0.0 = 0.0
/// -0.0 - 0.0 = 0.0
/// -0.0 - -0.0 = 0.0 // v - -0.0 != v
/// ```
///
/// [factor]: https://jcgm.bipm.org/vim/en/1.24.html
#[derive(Clone, Copy, Debug)]
pub enum ConstantOp {
/// Hint that the constant is being added to a value.
Add,
/// Hint that the constant is being subtracted from a value.
Sub,
}
/// Trait to identify [units][units] which have a [conversion factor][factor].
///
/// [units]: http://jcgm.bipm.org/vim/en/1.13.html
/// [factor]: https://jcgm.bipm.org/vim/en/1.24.html
pub trait Conversion<V> {
/// Conversion factor type specific to the underlying storage type.
type T: ConversionFactor<V>;
/// Coefficient portion of [conversion factor][factor] for converting the given unit to the
/// base unit for the quantity: `(value * coefficient()) + constant()`.
///
/// Default implementation returns `Self::T::one()`.
///
/// [factor]: https://jcgm.bipm.org/vim/en/1.24.html
#[inline(always)]
fn coefficient() -> Self::T {
<Self::T as num::One>::one()
}
/// Constant portion of [conversion factor][factor] for converting the given unit to the base
/// unit for the quantity: `(value * coefficient()) + constant()`.
///
/// Default implementation returns `Self::T::zero()`.
///
/// See [ConstantOp](enum.ConstantOp.html) documentation for details about parameter use.
///
/// [factor]: https://jcgm.bipm.org/vim/en/1.24.html
#[inline(always)]
#[allow(unused_variables)]
fn constant(op: ConstantOp) -> Self::T {
<Self::T as num::Zero>::zero()
}
/// Instance [conversion factor][factor].
///
/// Default implementation returns the coefficient: `Self::coefficient()`.
///
/// [factor]: https://jcgm.bipm.org/vim/en/1.24.html
#[inline(always)]
fn into_conversion(&self) -> Self::T
where
Self: Sized,
{
Self::coefficient()
}
}
/// Trait representing a [conversion factor][factor].
///
/// [factor]: https://jcgm.bipm.org/vim/en/1.24.html
pub trait ConversionFactor<V>:
lib::ops::Add<Self, Output = Self>
+ lib::ops::Sub<Self, Output = Self>
+ lib::ops::Mul<Self, Output = Self>
+ lib::ops::Div<Self, Output = Self>
+ num::Zero
+ num::One
{
/// Raises a `ConversionFactor<V>` to an integer power.
fn powi(self, e: i32) -> Self;
/// Converts a `ConversionFactor<V>` into its underlying type.
fn value(self) -> V;
}
/// Default [kind][kind] of quantities to allow addition, subtraction, multiplication, division,
/// remainder, negation, and saturating addition/subtraction.
///
/// [kind]: https://jcgm.bipm.org/vim/en/1.2.html
pub trait Kind:
marker::Add
+ marker::AddAssign
+ marker::Sub
+ marker::SubAssign
+ marker::Mul
+ marker::MulAssign
+ marker::Div
+ marker::DivAssign
+ marker::Rem
+ marker::RemAssign
+ marker::Neg
+ marker::Saturating
{
}
storage_types! {
types: Float;
impl ::Conversion<V> for V {
type T = V;
#[inline(always)]
fn into_conversion(&self) -> Self::T {
*self
}
}
impl ::ConversionFactor<V> for V {
#[inline(always)]
fn powi(self, e: i32) -> Self {
<V as ::num::Float>::powi(self, e)
}
#[inline(always)]
fn value(self) -> V {
self
}
}
}
storage_types! {
types: PrimInt;
impl ::Conversion<V> for V {
type T = ::num::rational::Ratio<V>;
#[inline(always)]
fn into_conversion(&self) -> Self::T {
(*self).into()
}
}
impl ::ConversionFactor<V> for ::num::rational::Ratio<V> {
#[inline(always)]
fn powi(self, e: i32) -> Self {
self.pow(e)
}
#[inline(always)]
fn value(self) -> V {
self.to_integer()
}
}
}
storage_types! {
types: BigInt, BigUint;
impl ::Conversion<V> for V {
type T = ::num::rational::Ratio<V>;
#[inline(always)]
fn into_conversion(&self) -> Self::T {
self.clone().into()
}
}
impl ::ConversionFactor<V> for ::num::rational::Ratio<V> {
#[inline(always)]
fn powi(self, e: i32) -> Self {
match e.cmp(&0) {
::lib::cmp::Ordering::Equal => <Self as ::num::One>::one(),
::lib::cmp::Ordering::Less => ::num::pow::pow(self.recip(), (-e) as usize),
::lib::cmp::Ordering::Greater => ::num::pow::pow(self, e as usize),
}
}
#[inline(always)]
fn value(self) -> V {
self.to_integer()
}
}
}
storage_types! {
types: Rational, Rational32, Rational64;
impl ::Conversion<V> for V {
type T = V;
#[inline(always)]
fn into_conversion(&self) -> Self::T {
*self
}
}
impl ::ConversionFactor<V> for V {
#[inline(always)]
fn powi(self, e: i32) -> Self {
self.pow(e)
}
#[inline(always)]
fn value(self) -> V {
self
}
}
}
storage_types! {
types: BigRational;
impl ::Conversion<V> for V {
type T = V;
#[inline(always)]
fn into_conversion(&self) -> Self::T {
self.clone()
}
}
impl ::ConversionFactor<V> for V {
#[inline(always)]
fn powi(self, e: i32) -> Self {
match e.cmp(&0) {
::lib::cmp::Ordering::Equal => <Self as ::num::One>::one(),
::lib::cmp::Ordering::Less => ::num::pow::pow(self.recip(), (-e) as usize),
::lib::cmp::Ordering::Greater => ::num::pow::pow(self, e as usize),
}
}
#[inline(always)]
fn value(self) -> V {
self
}
}
}
/// Utilities for formatting and printing quantities.
pub mod fmt {
/// An enum to specify the display style to use.
#[derive(Clone, Copy, Debug)]
pub enum DisplayStyle {
/// Display the value and a unit abbreviation, e.g. "1.0 m", "327 s".
Abbreviation,
/// Display the value and full unit name (pluralized as appropriate),
/// e.g. "1 kilogram", "756 feet".
Description,
}
}
/// Unicode string slice manipulation for quantities.
pub mod str {
/// Represents an error encountered while parsing a string into a `Quantity`.
#[allow(missing_copy_implementations)]
#[derive(Clone, Debug, Eq, PartialEq)]
pub enum ParseQuantityError {
/// No separators (spaces) were encountered.
NoSeparator,
/// An error occurred while parsing the value (first) portion of the string.
///
/// Due to exhaustiveness and type system limitations, this variant does not encode
/// the underlying parse error.
ValueParseError,
/// The unit used wasn't found for this quantity.
///
/// ### Notes
/// For now, only abbreviations are supported, so this error may be encountered even if the
/// unit name (description) is correct.
UnknownUnit,
}
}