Skip to content

Commit

Permalink
Merge pull request #32 from firedrakeproject/ksagiyam/fix_serendipity
Browse files Browse the repository at this point in the history
Ksagiyam/fix serendipity
  • Loading branch information
connorjward authored Jan 13, 2023
2 parents 43bc840 + 92e90d8 commit 8b26ef7
Show file tree
Hide file tree
Showing 4 changed files with 47 additions and 13 deletions.
50 changes: 41 additions & 9 deletions FIAT/serendipity.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,8 @@
#
# Modified by David A. Ham (david.ham@imperial.ac.uk), 2019

import numbers
import sympy
from sympy import symbols, legendre, Array, diff, lambdify
import numpy as np
from FIAT.finite_element import FiniteElement
Expand All @@ -30,6 +32,24 @@ def tr(n):
return int((n-3)*(n-2)/2)


def _replace_numbers_with_symbols(polynomials):
# Replace numbers with symbols to work around issue with numpy>=1.24.1;
# see https://github.com/firedrakeproject/fiat/pull/32.
extra_vars = {} # map from numbers to symbols
polynomials_list = []
for poly in polynomials.tolist():
if isinstance(poly, numbers.Real):
if poly not in extra_vars:
extra_vars[poly] = symbols('num_' + str(len(extra_vars)))
polynomials_list.append(extra_vars[poly])
elif isinstance(poly, sympy.core.Expr):
polynomials_list.append(poly)
else:
raise TypeError(f"Unexpected type: {type(poly)}")
polynomials = Array(polynomials_list)
return polynomials, extra_vars


class Serendipity(FiniteElement):

def __new__(cls, ref_el, degree):
Expand Down Expand Up @@ -104,8 +124,10 @@ def __init__(self, ref_el, degree):
super(Serendipity, self).__init__(ref_el=ref_el, dual=None, order=degree, formdegree=formdegree)

self.basis = {(0,)*dim: Array(s_list)}
self.basis_callable = {(0,)*dim: lambdify(variables[:dim], Array(s_list),
modules="numpy", dummify=True)}
polynomials, extra_vars = _replace_numbers_with_symbols(Array(s_list))
self.basis_callable = {(0,)*dim: [lambdify(variables[:dim], polynomials,
modules="numpy", dummify=True),
extra_vars]}
topology = ref_el.get_topology()
unflattening_map = compute_unflattening_map(topology)
unflattened_entity_ids = {}
Expand Down Expand Up @@ -161,16 +183,26 @@ def tabulate(self, order, points, entity=None):
alphas = mis(dim, o)
for alpha in alphas:
try:
callable = self.basis_callable[alpha]
callable, extra_vars = self.basis_callable[alpha]
except KeyError:
polynomials = diff(self.basis[(0,)*dim], *zip(variables, alpha))
callable = lambdify(variables[:dim], polynomials, modules="numpy", dummify=True)
polynomials, extra_vars = _replace_numbers_with_symbols(polynomials)
callable = lambdify(variables[:dim] + tuple(extra_vars.values()), polynomials, modules="numpy", dummify=True)
self.basis[alpha] = polynomials
self.basis_callable[alpha] = callable
tabulation = callable(*(points[:, i] for i in range(pointdim)))
T = np.asarray([np.broadcast_to(tab, (npoints, ))
for tab in tabulation])
phivals[alpha] = T
self.basis_callable[alpha] = [callable, extra_vars]
# Can no longer make a numpy array from objects of inhomogeneous shape
# (unless we specify `dtype==object`);
# see https://github.com/firedrakeproject/fiat/pull/32.
#
# Casting `key`s to float() is needed, otherwise we somehow get the following error:
#
# E TypeError: unsupported type for persistent hash keying: <class 'complex'>
#
# ../../lib/python3.8/site-packages/pytools/persistent_dict.py:243: TypeError
#
# `key`s have been checked to be numbers.Real.
extra_arrays = [np.ones((npoints, ), dtype=points.dtype) * float(key) for key in extra_vars]
phivals[alpha] = callable(*([points[:, i] for i in range(pointdim)] + extra_arrays))
return phivals

def entity_dofs(self):
Expand Down
6 changes: 4 additions & 2 deletions setup.cfg
Original file line number Diff line number Diff line change
@@ -1,14 +1,16 @@
[flake8]
ignore = E501,E226,E731,W504,
E741 # ambiguous variable name
# ambiguous variable name
E741
exclude = .git,__pycache__,doc/sphinx/source/conf.py,build,dist
min-version = 3.0

[pydocstyle]
# Work on removing these ignores
ignore = D100,D101,D102,D103,D104,D105,D107,
D200,D202,
D203, # this error should be disabled
# this error should be disabled
D203,
D204,D205,D208,D209,D212,D213,
D300,
D400,D401,D404,D412,D415,D416
Expand Down
2 changes: 1 addition & 1 deletion test/unit/test_discontinuous_pc.py
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,7 @@ def test_basis_values(dim, degree):

for test_degree in range(degree + 1):
coefs = [n(lambda x: x[0]**test_degree) for n in fe.dual.nodes]
integral = np.float(np.dot(coefs, np.dot(tab, q.wts)))
integral = float(np.dot(coefs, np.dot(tab, q.wts)))
reference = np.dot([x[0]**test_degree
for x in q.pts], q.wts)
assert np.isclose(integral, reference, rtol=1e-14)
Expand Down
2 changes: 1 addition & 1 deletion test/unit/test_discontinuous_taylor.py
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,7 @@ def test_basis_values(dim, degree):

for test_degree in range(degree + 1):
coefs = [n(lambda x: x[0]**test_degree) for n in fe.dual.nodes]
integral = np.float(np.dot(coefs, np.dot(tab, q.wts)))
integral = float(np.dot(coefs, np.dot(tab, q.wts)))
reference = np.dot([x[0]**test_degree
for x in q.pts], q.wts)
assert np.isclose(integral, reference, rtol=1e-14)
Expand Down

0 comments on commit 8b26ef7

Please sign in to comment.