use rayon::{
iter::{IntoParallelIterator, ParallelIterator},
slice::{ParallelSlice, ParallelSliceMut},
};
use super::{
QuadratureKind, TRIANGLE_NEAR_SUBDIVISION_DISTANCE_FACTOR, calc_tri_area, map_tri_uv,
triangle_basis_current_density, triangle_quadrature_points,
};
use crate::chunksize;
use crate::macros::{check_length_3tup, mut_par_chunks_3tup, par_chunks_3tup};
use crate::mesh::TriangleMeshView;
use crate::mesh::elements::tri::tri3::{
closest_point as triangle_closest_point,
max_edge_length_squared as triangle_max_edge_length_squared,
subdivide_about_point as triangle_subdivide_about_point,
};
use crate::physics::point_source::current_element::vector_potential_current_element_scalar;
#[inline]
fn triangle_vector_potential_inner(
n0: [f64; 3],
n1: [f64; 3],
n2: [f64; 3],
current_density: [f64; 3],
obs: [f64; 3],
quad_kind: QuadratureKind,
) -> [f64; 3] {
let tri_area = calc_tri_area(n0, n1, n2); let quad_points = triangle_quadrature_points(quad_kind);
let mut a = [0.0; 3];
for qp in quad_points {
let (c, u, v) = (qp[0], qp[1], qp[2]);
let src = map_tri_uv(n0, n1, n2, [u, v]); let moment = [
current_density[0] * c * tri_area, current_density[1] * c * tri_area, current_density[2] * c * tri_area, ];
let contrib = vector_potential_current_element_scalar(src, moment, obs); a[0] += contrib[0]; a[1] += contrib[1]; a[2] += contrib[2]; }
a
}
#[inline]
pub fn triangle_vector_potential_basis(
n0: [f64; 3],
n1: [f64; 3],
n2: [f64; 3],
obs: [f64; 3],
quad_kind: QuadratureKind,
) -> [f64; 3] {
let (_, jref) = triangle_basis_current_density(n0, n1, n2); let max_edge_sq = triangle_max_edge_length_squared(n0, n1, n2); let closest = triangle_closest_point(obs, n0, n1, n2); let dx = obs[0] - closest[0]; let dy = obs[1] - closest[1]; let dz = obs[2] - closest[2]; let dist_sq = dx.mul_add(dx, dy.mul_add(dy, dz * dz)); let subdiv_threshold_sq = TRIANGLE_NEAR_SUBDIVISION_DISTANCE_FACTOR.powi(2) * max_edge_sq;
if dist_sq > subdiv_threshold_sq {
return triangle_vector_potential_inner(n0, n1, n2, jref, obs, quad_kind);
}
let mut a = [0.0; 3]; let min_sub_area = max_edge_sq * 1e-14; for tri in triangle_subdivide_about_point(closest, n0, n1, n2) {
let [a0, b0, c0] = tri;
if calc_tri_area(a0, b0, c0) <= min_sub_area {
continue;
}
let contrib = triangle_vector_potential_inner(a0, b0, c0, jref, obs, quad_kind);
a[0] += contrib[0]; a[1] += contrib[1]; a[2] += contrib[2]; }
a
}
#[inline]
pub fn vector_potential_triangle(
n0: [f64; 3],
n1: [f64; 3],
n2: [f64; 3],
s: [f64; 3],
obs: [f64; 3],
quad_kind: QuadratureKind,
) -> [f64; 3] {
let a_n0 = triangle_vector_potential_basis(n0, n1, n2, obs, quad_kind);
let a_n1 = triangle_vector_potential_basis(n1, n2, n0, obs, quad_kind);
let a_n2 = triangle_vector_potential_basis(n2, n0, n1, obs, quad_kind);
[
s[0] * a_n0[0] + s[1] * a_n1[0] + s[2] * a_n2[0], s[0] * a_n0[1] + s[1] * a_n1[1] + s[2] * a_n2[1], s[0] * a_n0[2] + s[1] * a_n1[2] + s[2] * a_n2[2], ]
}
#[inline]
fn validate_vector_potential_mapping_inputs(
outx: &[f64],
outy: &[f64],
outz: &[f64],
nobs: usize,
nnode: usize,
) -> Result<(), &'static str> {
let expected = nobs
.checked_mul(nnode)
.ok_or("Vector-potential mapping size overflow")?;
if outx.len() != expected || outy.len() != expected || outz.len() != expected {
return Err("Output dimension mismatch");
}
Ok(())
}
#[inline]
fn vector_potential_triangle_mesh_mapping_chunk(
obs: (&[f64], &[f64], &[f64]),
mesh: &TriangleMeshView<'_>,
quad_kind: QuadratureKind,
out: (&mut [f64], &mut [f64], &mut [f64]),
) -> Result<(), &'static str> {
let nobs = obs.0.len(); check_length_3tup!(nobs, obs);
validate_vector_potential_mapping_inputs(out.0, out.1, out.2, nobs, mesh.nnode())?;
out.0.fill(0.0); out.1.fill(0.0); out.2.fill(0.0);
for iobs in 0..nobs {
let row_offset = iobs * mesh.nnode(); let obs_i = [obs.0[iobs], obs.1[iobs], obs.2[iobs]];
for itri in 0..mesh.len() {
let (tri_nodes, idx) = mesh.triangle_nodes_and_indices(itri);
let a0 = triangle_vector_potential_basis(
tri_nodes[0],
tri_nodes[1],
tri_nodes[2],
obs_i,
quad_kind,
); let a1 = triangle_vector_potential_basis(
tri_nodes[1],
tri_nodes[2],
tri_nodes[0],
obs_i,
quad_kind,
); let a2 = triangle_vector_potential_basis(
tri_nodes[2],
tri_nodes[0],
tri_nodes[1],
obs_i,
quad_kind,
);
out.0[row_offset + idx[0]] += a0[0]; out.1[row_offset + idx[0]] += a0[1]; out.2[row_offset + idx[0]] += a0[2];
out.0[row_offset + idx[1]] += a1[0]; out.1[row_offset + idx[1]] += a1[1]; out.2[row_offset + idx[1]] += a1[2];
out.0[row_offset + idx[2]] += a2[0]; out.1[row_offset + idx[2]] += a2[1]; out.2[row_offset + idx[2]] += a2[2]; }
}
Ok(())
}
#[inline]
fn vector_potential_triangle_mesh_inner(
obs: (&[f64], &[f64], &[f64]),
mesh: &TriangleMeshView<'_>,
s: &[f64],
quad_kind: QuadratureKind,
out: (&mut [f64], &mut [f64], &mut [f64]),
) -> Result<(), &'static str> {
let nobs = obs.0.len();
check_length_3tup!(nobs, obs);
check_length_3tup!(nobs, out);
mesh.validate_nodal_values(s)?;
out.0.fill(0.0); out.1.fill(0.0); out.2.fill(0.0);
for i in 0..nobs {
let obs_i = [obs.0[i], obs.1[i], obs.2[i]];
for j in 0..mesh.len() {
let tri_nodes = mesh.triangle_nodes(j);
let tri_s = mesh.triangle_scalars(j, s);
let contrib = vector_potential_triangle(
tri_nodes[0],
tri_nodes[1],
tri_nodes[2],
tri_s,
obs_i,
quad_kind,
);
out.0[i] += contrib[0]; out.1[i] += contrib[1]; out.2[i] += contrib[2]; }
}
Ok(())
}
#[inline]
pub fn vector_potential_triangle_mesh_mapping(
obs: (&[f64], &[f64], &[f64]),
mesh: &TriangleMeshView<'_>,
quad_kind: QuadratureKind,
out: (&mut [f64], &mut [f64], &mut [f64]),
) -> Result<(), &'static str> {
vector_potential_triangle_mesh_mapping_chunk(obs, mesh, quad_kind, out)
}
#[inline]
pub fn vector_potential_triangle_mesh_mapping_par(
obs: (&[f64], &[f64], &[f64]),
mesh: &TriangleMeshView<'_>,
quad_kind: QuadratureKind,
out: (&mut [f64], &mut [f64], &mut [f64]),
) -> Result<(), &'static str> {
let nobs = obs.0.len(); check_length_3tup!(nobs, obs);
validate_vector_potential_mapping_inputs(out.0, out.1, out.2, nobs, mesh.nnode())?;
if nobs == 0 || mesh.nnode() == 0 {
return vector_potential_triangle_mesh_mapping_chunk(obs, mesh, quad_kind, out);
}
let nrow = chunksize(nobs); let nflat = nrow
.checked_mul(mesh.nnode())
.ok_or("Vector-potential mapping size overflow")?;
let (xpc, ypc, zpc) = par_chunks_3tup!(obs, nrow);
let axc = out.0.par_chunks_mut(nflat);
let ayc = out.1.par_chunks_mut(nflat);
let azc = out.2.par_chunks_mut(nflat);
(axc, ayc, azc, xpc, ypc, zpc)
.into_par_iter()
.try_for_each(|(ax, ay, az, xp, yp, zp)| {
vector_potential_triangle_mesh_mapping_chunk(
(xp, yp, zp),
mesh,
quad_kind,
(ax, ay, az),
)
})?;
Ok(())
}
#[inline]
pub fn triangle_mesh_vector_potential_from_potential_vectors(
ax_map: &[f64],
ay_map: &[f64],
az_map: &[f64],
s: &[f64],
out: (&mut [f64], &mut [f64], &mut [f64]),
) -> Result<(), &'static str> {
if ay_map.len() != ax_map.len() || az_map.len() != ax_map.len() {
return Err("Vector-potential mapping dimension mismatch");
}
if s.is_empty() {
if ax_map.is_empty() && out.0.is_empty() && out.1.is_empty() && out.2.is_empty() {
return Ok(());
}
return Err("Vector-potential mapping dimension mismatch");
}
if !ax_map.len().is_multiple_of(s.len()) {
return Err("Vector-potential mapping dimension mismatch");
}
let nobs = ax_map.len() / s.len(); check_length_3tup!(nobs, out);
for iobs in 0..nobs {
let row_offset = iobs * s.len(); let mut ax = 0.0; let mut ay = 0.0; let mut az = 0.0; for inode in 0..s.len() {
ax += ax_map[row_offset + inode] * s[inode]; ay += ay_map[row_offset + inode] * s[inode]; az += az_map[row_offset + inode] * s[inode]; }
out.0[iobs] = ax; out.1[iobs] = ay; out.2[iobs] = az; }
Ok(())
}
#[inline]
pub fn vector_potential_triangle_mesh(
obs: (&[f64], &[f64], &[f64]),
mesh: &TriangleMeshView<'_>,
s: &[f64],
quad_kind: QuadratureKind,
out: (&mut [f64], &mut [f64], &mut [f64]),
) -> Result<(), &'static str> {
vector_potential_triangle_mesh_inner(obs, mesh, s, quad_kind, out)
}
#[inline]
pub fn vector_potential_triangle_mesh_par(
obs: (&[f64], &[f64], &[f64]),
mesh: &TriangleMeshView<'_>,
s: &[f64],
quad_kind: QuadratureKind,
out: (&mut [f64], &mut [f64], &mut [f64]),
) -> Result<(), &'static str> {
mesh.validate_nodal_values(s)?;
let n = chunksize(obs.0.len());
let (xpc, ypc, zpc) = par_chunks_3tup!(obs, n);
let (axc, ayc, azc) = mut_par_chunks_3tup!(out, n);
(axc, ayc, azc, xpc, ypc, zpc)
.into_par_iter()
.try_for_each(|(ax, ay, az, xp, yp, zp)| {
vector_potential_triangle_mesh_inner((xp, yp, zp), mesh, s, quad_kind, (ax, ay, az))
})?;
Ok(())
}