use std::{ops::*, hash::Hash, fmt::{Display, Debug}};
use crate::homomorphism::*;
use crate::ring::*;
use crate::field::*;
pub struct RingElementWrapper<R>
where R: RingStore
{
ring: R,
element: El<R>
}
impl<R: RingStore> RingElementWrapper<R> {
pub const fn new(ring: R, element: El<R>) -> Self {
Self { ring, element }
}
pub fn pow(self, power: usize) -> Self {
Self {
element: self.ring.pow(self.element, power),
ring: self.ring
}
}
pub fn pow_ref(&self, power: usize) -> Self
where R: Clone
{
Self {
element: self.ring.pow(self.ring.clone_el(&self.element), power),
ring: self.ring.clone()
}
}
pub fn unwrap(self) -> El<R> {
self.element
}
pub fn unwrap_ref(&self) -> &El<R> {
&self.element
}
pub fn parent(&self) -> &R {
&self.ring
}
}
macro_rules! impl_xassign_trait {
($trait_name:ident, $fn_name:ident, $fn_ref_name:ident) => {
impl<R: RingStore> $trait_name for RingElementWrapper<R> {
fn $fn_name(&mut self, rhs: Self) {
debug_assert!(self.ring.get_ring() == rhs.ring.get_ring());
self.ring.$fn_name(&mut self.element, rhs.element);
}
}
impl<'a, R: RingStore> $trait_name<&'a Self> for RingElementWrapper<R> {
fn $fn_name(&mut self, rhs: &'a Self) {
debug_assert!(self.ring.get_ring() == rhs.ring.get_ring());
self.ring.$fn_ref_name(&mut self.element, &rhs.element);
}
}
};
}
macro_rules! impl_trait {
($trait_name:ident, $fn_name:ident, $fn_name_ref_fst:ident, $fn_name_ref_snd:ident, $fn_name_ref:ident) => {
impl<R: RingStore> $trait_name for RingElementWrapper<R> {
type Output = Self;
fn $fn_name(self, rhs: Self) -> Self::Output {
debug_assert!(self.ring.get_ring() == rhs.ring.get_ring());
Self { ring: self.ring, element: rhs.ring.$fn_name(self.element, rhs.element) }
}
}
impl<'a, R: RingStore> $trait_name<RingElementWrapper<R>> for &'a RingElementWrapper<R> {
type Output = RingElementWrapper<R>;
fn $fn_name(self, rhs: RingElementWrapper<R>) -> Self::Output {
debug_assert!(self.ring.get_ring() == rhs.ring.get_ring());
RingElementWrapper { ring: rhs.ring, element: self.ring.$fn_name_ref_fst(&self.element, rhs.element) }
}
}
impl<'a, R: RingStore> $trait_name<&'a RingElementWrapper<R>> for RingElementWrapper<R> {
type Output = RingElementWrapper<R>;
fn $fn_name(self, rhs: &'a RingElementWrapper<R>) -> Self::Output {
debug_assert!(self.ring.get_ring() == rhs.ring.get_ring());
RingElementWrapper { ring: self.ring, element: rhs.ring.$fn_name_ref_snd(self.element, &rhs.element) }
}
}
impl<'a, 'b, R: RingStore + Clone> $trait_name<&'a RingElementWrapper<R>> for &'b RingElementWrapper<R> {
type Output = RingElementWrapper<R>;
fn $fn_name(self, rhs: &'a RingElementWrapper<R>) -> Self::Output {
debug_assert!(self.ring.get_ring() == rhs.ring.get_ring());
RingElementWrapper { ring: self.ring.clone(), element: self.ring.$fn_name_ref(&self.element, &rhs.element) }
}
}
};
}
impl_xassign_trait!{ AddAssign, add_assign, add_assign_ref }
impl_xassign_trait!{ MulAssign, mul_assign, mul_assign_ref }
impl_xassign_trait!{ SubAssign, sub_assign, sub_assign_ref }
impl_trait!{ Add, add, add_ref_fst, add_ref_snd, add_ref }
impl_trait!{ Mul, mul, mul_ref_fst, mul_ref_snd, mul_ref }
impl_trait!{ Sub, sub, sub_ref_fst, sub_ref_snd, sub_ref }
impl<R: RingStore> Div<RingElementWrapper<R>> for RingElementWrapper<R>
where R::Type: Field
{
type Output = Self;
fn div(self, rhs: RingElementWrapper<R>) -> Self::Output {
RingElementWrapper { element: self.ring.div(&self.element, &rhs.element), ring: self.ring }
}
}
impl<'a, 'b, R: RingStore + Clone> Div<&'a RingElementWrapper<R>> for &'b RingElementWrapper<R>
where R::Type: Field
{
type Output = RingElementWrapper<R>;
fn div(self, rhs: &'a RingElementWrapper<R>) -> Self::Output {
RingElementWrapper { element: self.ring.div(&self.element, &rhs.element), ring: self.ring.clone() }
}
}
impl<'a, R: RingStore + Clone> Div<RingElementWrapper<R>> for &'a RingElementWrapper<R>
where R::Type: Field
{
type Output = RingElementWrapper<R>;
fn div(self, rhs: RingElementWrapper<R>) -> Self::Output {
RingElementWrapper { element: self.ring.div(&self.element, &rhs.element), ring: rhs.ring }
}
}
impl<'a, R: RingStore + Clone> Div<&'a RingElementWrapper<R>> for RingElementWrapper<R>
where R::Type: Field
{
type Output = RingElementWrapper<R>;
fn div(self, rhs: &'a RingElementWrapper<R>) -> Self::Output {
RingElementWrapper { element: rhs.ring.div(&self.element, &rhs.element), ring: self.ring }
}
}
macro_rules! impl_xassign_trait_int {
($trait_name:ident, $fn_name:ident) => {
impl<R: RingStore> $trait_name<i32> for RingElementWrapper<R> {
fn $fn_name(&mut self, rhs: i32) {
self.ring.$fn_name(&mut self.element, self.ring.int_hom().map(rhs));
}
}
};
}
macro_rules! impl_trait_int {
($trait_name:ident, $fn_name:ident) => {
impl<R: RingStore> $trait_name<i32> for RingElementWrapper<R> {
type Output = Self;
fn $fn_name(self, rhs: i32) -> Self::Output {
RingElementWrapper { element: self.ring.$fn_name(self.element, self.ring.int_hom().map(rhs)), ring: self.ring }
}
}
impl<R: RingStore> $trait_name<RingElementWrapper<R>> for i32 {
type Output = RingElementWrapper<R>;
fn $fn_name(self, rhs: RingElementWrapper<R>) -> Self::Output {
RingElementWrapper { element: rhs.ring.$fn_name(rhs.ring.int_hom().map(self), rhs.element), ring: rhs.ring }
}
}
impl<'a, R: RingStore + Clone> $trait_name<i32> for &'a RingElementWrapper<R> {
type Output = RingElementWrapper<R>;
fn $fn_name(self, rhs: i32) -> Self::Output {
RingElementWrapper { element: self.ring.$fn_name(self.ring.clone_el(&self.element), self.ring.int_hom().map(rhs)), ring: self.ring.clone() }
}
}
impl<'a, R: RingStore + Clone> $trait_name<&'a RingElementWrapper<R>> for i32 {
type Output = RingElementWrapper<R>;
fn $fn_name(self, rhs: &'a RingElementWrapper<R>) -> Self::Output {
RingElementWrapper { element: rhs.ring.$fn_name(rhs.ring.int_hom().map(self), rhs.ring.clone_el(&rhs.element)), ring: rhs.ring.clone() }
}
}
};
}
impl_xassign_trait_int!{ AddAssign, add_assign }
impl_xassign_trait_int!{ MulAssign, mul_assign }
impl_xassign_trait_int!{ SubAssign, sub_assign }
impl_trait_int!{ Add, add }
impl_trait_int!{ Mul, mul }
impl_trait_int!{ Sub, sub }
impl<R: RingStore> Div<i32> for RingElementWrapper<R>
where R::Type: Field
{
type Output = Self;
fn div(self, rhs: i32) -> Self::Output {
RingElementWrapper { element: self.ring.div(&self.element, &self.ring.int_hom().map(rhs)), ring: self.ring }
}
}
impl<R: RingStore> Div<RingElementWrapper<R>> for i32
where R::Type: Field
{
type Output = RingElementWrapper<R>;
fn div(self, rhs: RingElementWrapper<R>) -> Self::Output {
RingElementWrapper { element: rhs.ring.div(&rhs.ring.int_hom().map(self), &rhs.element), ring: rhs.ring }
}
}
impl<'a, R: RingStore + Clone> Div<i32> for &'a RingElementWrapper<R>
where R::Type: Field
{
type Output = RingElementWrapper<R>;
fn div(self, rhs: i32) -> Self::Output {
RingElementWrapper { element: self.ring.div(&self.element, &self.ring.int_hom().map(rhs)), ring: self.ring.clone() }
}
}
impl<'a, R: RingStore + Clone> Div<&'a RingElementWrapper<R>> for i32
where R::Type: Field
{
type Output = RingElementWrapper<R>;
fn div(self, rhs: &'a RingElementWrapper<R>) -> Self::Output {
RingElementWrapper { element: rhs.ring.div(&rhs.ring.int_hom().map(self), &rhs.element), ring: rhs.ring.clone() }
}
}
impl<R: RingStore + Copy> Copy for RingElementWrapper<R>
where El<R>: Copy
{}
impl<R: RingStore + Clone> Clone for RingElementWrapper<R> {
fn clone(&self) -> Self {
Self { ring: self.ring.clone(), element: self.ring.clone_el(&self.element) }
}
}
impl<R: RingStore> PartialEq for RingElementWrapper<R> {
fn eq(&self, other: &Self) -> bool {
debug_assert!(self.ring.get_ring() == other.ring.get_ring());
self.ring.eq_el(&self.element, &other.element)
}
}
impl<R: RingStore> Eq for RingElementWrapper<R> {}
impl<R: RingStore> Hash for RingElementWrapper<R>
where R::Type: HashableElRing
{
fn hash<H: std::hash::Hasher>(&self, state: &mut H) {
self.ring.hash(&self.element, state)
}
}
impl<R: RingStore> Display for RingElementWrapper<R> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
self.ring.get_ring().dbg(&self.element, f)
}
}
impl<R: RingStore> Debug for RingElementWrapper<R> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
self.ring.get_ring().dbg(&self.element, f)
}
}
impl<R: RingStore> Deref for RingElementWrapper<R> {
type Target = El<R>;
fn deref(&self) -> &Self::Target {
&self.element
}
}
#[cfg(test)]
use crate::rings::finite::FiniteRingStore;
#[cfg(test)]
use crate::rings::zn::zn_64;
#[test]
fn test_arithmetic_expression() {
let ring = zn_64::Zn::new(17);
for x in ring.elements() {
for y in ring.elements() {
for z in ring.elements() {
let expected = ring.add(ring.mul(x, y), ring.mul(ring.add(x, z), ring.sub(y, z)));
let x = RingElementWrapper::new(&ring, x);
let y = RingElementWrapper::new(&ring, y);
let z = RingElementWrapper::new(&ring, z);
assert_el_eq!(ring, expected, (x * y + (x + z) * (y - z)).unwrap());
}
}
}
}
#[test]
fn test_arithmetic_expression_int() {
let ring = zn_64::Zn::new(17);
for x in ring.elements() {
for y in ring.elements() {
for z in ring.elements() {
let expected = ring.add(ring.add(ring.int_hom().mul_map(ring.mul(x, y), 8), ring.mul(ring.add(ring.add(ring.one(), x), ring.int_hom().mul_map(z, 2)), ring.sub(y, ring.int_hom().mul_map(z, 2)))), ring.int_hom().map(5));
let x = RingElementWrapper::new(&ring, x);
let y = RingElementWrapper::new(&ring, y);
let z = RingElementWrapper::new(&ring, z);
assert_el_eq!(ring, expected, (x * 8 * y + (1 + x + 2 * z) * (y - z * 2) + 5).unwrap());
}
}
}
}
#[test]
fn test_arithmetic_expression_ref() {
let ring = zn_64::Zn::new(17);
for x in ring.elements() {
for y in ring.elements() {
for z in ring.elements() {
let expected = ring.add(ring.mul(x, y), ring.mul(ring.add(x, z), ring.sub(y, z)));
let x = RingElementWrapper::new(&ring, x);
let y = RingElementWrapper::new(&ring, y);
let z = RingElementWrapper::new(&ring, z);
assert_el_eq!(ring, expected, (x * &y + (&x + &z) * (&y - z)).unwrap());
}
}
}
}
#[test]
fn test_arithmetic_expression_int_ref() {
let ring = zn_64::Zn::new(17);
for x in ring.elements() {
for y in ring.elements() {
for z in ring.elements() {
let expected = ring.add(ring.add(ring.int_hom().mul_map(ring.mul(x, y), 8), ring.mul(ring.add(ring.add(ring.one(), x), ring.int_hom().mul_map(z, 2)), ring.sub(y, ring.int_hom().mul_map(z, 2)))), ring.int_hom().map(5));
let x = RingElementWrapper::new(&ring, x);
let y = RingElementWrapper::new(&ring, y);
let z = RingElementWrapper::new(&ring, z);
assert_el_eq!(ring, expected, (x * 8 * &y + (1 + &x + 2 * &z) * (&y - z * 2) + 5).unwrap());
}
}
}
}