atomics/src/types/i64.rs (77 lines of code) (raw):
// Copyright 2019-2020 Twitter, Inc.
// Licensed under the Apache License, Version 2.0
// http://www.apache.org/licenses/LICENSE-2.0
use crate::*;
#[cfg(feature = "serde")]
use serde::{de::Visitor, Deserialize, Deserializer, Serialize, Serializer};
native!(
/// A signed 64 bit integer which can be shared between threads
pub struct AtomicI64: i64 = core::sync::atomic::AtomicI64;
);
// additional traits
arithmetic!(AtomicI64, i64);
bitwise!(AtomicI64, i64);
fetch_compare_store!(AtomicI64, i64);
saturating_arithmetic!(AtomicI64, i64);
impl Signed for AtomicI64 {}
#[cfg(feature = "serde")]
struct AtomicI64Visitor;
#[cfg(feature = "serde")]
impl<'de> Visitor<'de> for AtomicI64Visitor {
type Value = AtomicI64;
fn expecting(&self, formatter: &mut std::fmt::Formatter) -> std::fmt::Result {
formatter.write_str("a signed 64bit integer")
}
fn visit_i8<E>(self, value: i8) -> Result<Self::Value, E>
where
E: serde::de::Error,
{
Ok(Self::Value::new(i64::from(value)))
}
fn visit_i16<E>(self, value: i16) -> Result<Self::Value, E>
where
E: serde::de::Error,
{
Ok(Self::Value::new(i64::from(value)))
}
fn visit_i32<E>(self, value: i32) -> Result<Self::Value, E>
where
E: serde::de::Error,
{
Ok(Self::Value::new(i64::from(value)))
}
fn visit_i64<E>(self, value: i64) -> Result<Self::Value, E>
where
E: serde::de::Error,
{
Ok(Self::Value::new(value))
}
fn visit_u8<E>(self, value: u8) -> Result<Self::Value, E>
where
E: serde::de::Error,
{
Ok(Self::Value::new(i64::from(value)))
}
fn visit_u16<E>(self, value: u16) -> Result<Self::Value, E>
where
E: serde::de::Error,
{
Ok(Self::Value::new(i64::from(value)))
}
fn visit_u32<E>(self, value: u32) -> Result<Self::Value, E>
where
E: serde::de::Error,
{
Ok(Self::Value::new(i64::from(value)))
}
fn visit_u64<E>(self, value: u64) -> Result<Self::Value, E>
where
E: serde::de::Error,
{
if let Ok(value) = i64::try_from(value) {
Ok(Self::Value::new(value))
} else {
Err(E::custom(format!("i64 is out of range: {}", value)))
}
}
}
#[cfg(feature = "serde")]
impl<'de> Deserialize<'de> for AtomicI64 {
fn deserialize<D>(deserializer: D) -> Result<AtomicI64, D::Error>
where
D: Deserializer<'de>,
{
deserializer.deserialize_i64(AtomicI64Visitor)
}
}
#[cfg(feature = "serde")]
impl Serialize for AtomicI64 {
#[inline]
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
serializer.serialize_some(&self.load(Ordering::SeqCst))
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn load() {
let atomic = AtomicI64::new(0);
assert_eq!(atomic.load(Ordering::SeqCst), 0);
}
#[test]
fn store() {
let atomic = AtomicI64::new(0);
atomic.store(1, Ordering::SeqCst);
assert_eq!(atomic.load(Ordering::SeqCst), 1);
}
#[test]
fn swap() {
let atomic = AtomicI64::new(0);
assert_eq!(atomic.swap(1, Ordering::SeqCst), 0);
}
#[test]
fn compare_exchange() {
let atomic = AtomicI64::new(0);
assert_eq!(
atomic.compare_exchange(0, 1, Ordering::SeqCst, Ordering::SeqCst),
Ok(0)
);
assert_eq!(
atomic.compare_exchange(0, 2, Ordering::SeqCst, Ordering::SeqCst),
Err(1)
);
}
#[test]
fn compare_exchange_weak() {
let atomic = AtomicI64::new(0);
loop {
if atomic
.compare_exchange(0, 1, Ordering::SeqCst, Ordering::SeqCst)
.is_ok()
{
break;
}
}
assert_eq!(atomic.load(Ordering::SeqCst), 1);
}
}