25template <
typename Derived>
26void secDerToHessian(
const gsEigen::DenseBase<Derived> & secDers,
28 gsMatrix<typename Derived::Scalar> & hessian)
30 const index_t sz = dim*(dim+1)/2;
31 const gsAsConstMatrix<typename Derived::Scalar>
32 ders(secDers.derived().data(), sz, secDers.size() / sz );
33 hessian.resize(dim*dim, ders.cols() );
38 hessian = secDers.transpose();
41 hessian.row(0)=ders.row(0);
43 hessian.row(2)=ders.row(2);
44 hessian.row(3)=ders.row(1);
47 hessian.row(0)=ders.row(0);
49 hessian.row(1)=ders.row(3);
51 hessian.row(2)=ders.row(4);
52 hessian.row(4)=ders.row(1);
54 hessian.row(5)=ders.row(5);
55 hessian.row(8)=ders.row(2);
66 fs(&_fs), dim(
give(_dim)), id(
give(_id)) { }
77 return static_cast<size_t>(fs->
size()*dim)==mapper.
mapSize();
108template <
class E>
struct is_arithmetic{
enum{value=0};};
109template <>
struct is_arithmetic<real_t>{
enum{value=1};};
110template <typename E, bool = is_arithmetic<E>::value >
111class _expr {
using E::GISMO_ERROR_expr;};
113template<
class T>
class gsFeSpace;
114template<
class T>
class gsFeVariable;
115template<
class T>
class gsFeSolution;
116template<
class E>
class symm_expr;
117template<
class E>
class symmetrize_expr;
118template<
class E>
class normalized_expr;
119template<
class E>
class trace_expr;
120template<
class E>
class integral_expr;
121template<
class E>
class adjugate_expr;
122template<
class E>
class norm_expr;
123template<
class E>
class sqNorm_expr;
124template<
class E>
class det_expr;
125template<
class E>
class value_expr;
126template<
class E>
class asdiag_expr;
127template<
class E>
class max_expr;
128template<
class E>
class rowsum_expr;
129template<
class E>
class colsum_expr;
130template<
class E>
class col_expr;
131template<
class T>
class meas_expr;
132template<
class E>
class inv_expr;
133template<
class E,
bool cw = false>
class tr_expr;
134template<
class E>
class cb_expr;
135template<
class E>
class abs_expr;
136template<
class E>
class pow_expr;
137template<
class E>
class sign_expr;
138template<
class E>
class ppart_expr;
139template<
class E>
class exp_expr;
140template<
class E>
class ppartval_expr;
141template<
class T>
class cdiam_expr;
142template<
class E>
class temp_expr;
143template<
class E1,
class E2,
bool = E1::ColBlocks && !E1::ScalarValued && !E2::ScalarValued>
class mult_expr
144{
using E1::GISMO_ERROR_mult_expr_has_invalid_template_arguments;};
147template<
class E> pow_expr<E>
148pow(_expr<E>
const& u, real_t q) {
return pow_expr<E>(u,q); }
153template <
typename E>
struct expr_traits
157 typedef real_t Scalar;
158 typedef const E Nested_t;
161# define Temporary_t typename util::conditional<ScalarValued,Scalar, \
162 typename gsMatrix<Scalar>::Base >::type
163#if __cplusplus >= 201402L || _MSVC_LANG >= 201402L
164# define MatExprType auto
165# define AutoReturn_t auto
168# define MatExprType typename gsMatrix<real_t>::constRef
169# define AutoReturn_t typename util::conditional<ScalarValued,real_t,MatExprType>::type
180 _expr(
const _expr&) { }
190 typedef typename expr_traits<E>::Nested_t Nested_t;
191 typedef typename expr_traits<E>::Scalar Scalar;
197 static_cast<E const&
>(*this).print(os);
216 std::ostream & printDetail(std::ostream &os)
const
218 os << (isVectorTr() ?
"VectorTr " :
219 (isVector() ?
"Vector " :
220 (isMatrix() ?
"Matrix " :
222 <<
"expression of size "<< rows()
223 <<
" x "<<cols()<<
"\n";
230 {
return static_cast<E const&
>(*this).eval(k); }
233 tr_expr<E>
tr()
const
234 {
return tr_expr<E,false>(
static_cast<E const&
>(*
this)); }
238 {
return tr_expr<E,true>(
static_cast<E const&
>(*
this)); }
241 cb_expr<E>
cb()
const
242 {
return cb_expr<E>(
static_cast<E const&
>(*
this)); }
246 {
return sign_expr<E>(
static_cast<E const&
>(*
this), tolerance); }
250 {
return exp_expr<E>(
static_cast<E const&
>(*
this)); }
259 mult_expr<real_t, ppart_expr<mult_expr<double,E,false>> ,
false>
260 npart()
const {
return -1* ( -(*this) ).ppart() ; }
264 {
return temp_expr<E>(
static_cast<E const&
>(*
this)); }
267 inv_expr<E>
const inv()
const
268 {
return inv_expr<E>(
static_cast<E const&
>(*
this)); }
272 {
return trace_expr<E>(
static_cast<E const&
>(*
this)); }
275 adjugate_expr<E>
adj()
const
276 {
return adjugate_expr<E>(
static_cast<E const&
>(*
this)); }
280 {
return norm_expr<E>(
static_cast<E const&
>(*
this)); }
284 {
return normalized_expr<E>(
static_cast<E const&
>(*
this)); }
288 {
return det_expr<E>(
static_cast<E const&
>(*
this)); }
291 sqNorm_expr<E> sqNorm()
const
292 {
return sqNorm_expr<E>(
static_cast<E const&
>(*
this)); }
295 mult_expr<E,E,0> (sqr)()
const {
return (*
this)*(*this); }
297 symm_expr<E> symm()
const
298 {
return symm_expr<E>(
static_cast<E const&
>(*
this)); }
300 symmetrize_expr<E> symmetrize()
const
301 {
return symmetrize_expr<E>(
static_cast<E const&
>(*
this)); }
306 {
return value_expr<E>(
static_cast<E const&
>(*
this)); }
314 {
return max_expr<E>(
static_cast<E const&
>(*
this)); }
318 {
return rowsum_expr<E>(
static_cast<E const&
>(*
this)); }
322 {
return colsum_expr<E>(
static_cast<E const&
>(*
this)); }
324 col_expr<E> operator[](
const index_t i)
const
325 {
return col_expr<E>(
static_cast<E const&
>(*
this),i); }
329 {
return static_cast<E const&
>(*this).rows(); }
333 {
return static_cast<E const&
>(*this).cols(); }
336 {
return static_cast<E const&
>(*this).cardinality_impl(); }
338 static index_t cardinality_impl() {
return 1; }
344 static constexpr bool isVector () {
return 1==E::Space; }
345 static constexpr bool isVectorTr() {
return 2==E::Space; }
346 static constexpr bool isMatrix () {
return 3==E::Space; }
351 {
static_cast<E const&
>(*this).parse(evList); }
353 template<
class op>
void apply(op & _op)
const
354 {
static_cast<E const&
>(*this).apply(_op); }
361 return static_cast<E const&
>(*this).rowVar();
369 return static_cast<E const&
>(*this).colVar();
374 operator E&() {
return static_cast< E&
>(*this); }
375 operator E
const&()
const {
return static_cast<const E&
>(*this); }
377 E
const & derived()
const {
return static_cast<const E&
>(*this); }
382std::ostream &
operator<<(std::ostream &os,
const _expr<E> & b)
383{b.print(os);
return os; }
399class gsNullExpr :
public _expr<gsNullExpr<T> >
403 operator const gsFeSpace<T> & ()
const
405 static gsFeSpace<T> vv(-1);
413 void parse(gsExprHelper<T> &)
const { }
415 const gsFeSpace<T> & rowVar()
const {
GISMO_ERROR(
"gsNullExpr"); }
416 const gsFeSpace<T> & colVar()
const {
GISMO_ERROR(
"gsNullExpr"); }
418 void print(std::ostream &os)
const { os <<
"NullExpr"; }
420 static const gsNullExpr & get()
431class symbol_expr :
public _expr<E>
434 typedef typename expr_traits<E>::Scalar Scalar;
438 const gsFunctionSet<Scalar> * m_fs;
439 const gsFuncData<Scalar> * m_fd;
446 bool isAcross()
const {
return m_isAcross; }
450 E ac(this->derived());
452 ac.m_isAcross =
true;
458 E ac(this->derived());
460 ac.m_isAcross =
false;
465 const gsFunctionSet<Scalar> & source()
const {
return *m_fs;}
468 const gsFuncData<Scalar> & data()
const
470 GISMO_ASSERT(NULL!=m_fd,
"FuncData member not registered "<<
this<<
"/"<< m_fs);
475 void parse(gsExprHelper<Scalar> & evList)
const
481 index_t cardinality_impl()
const
483 GISMO_ASSERT(this->data().actives.rows()!=0,
"Cardinality depends on the NEED_ACTIVE flag");
484 return m_d * this->data().actives.rows();
488 void setSource(
const gsFunctionSet<Scalar> & fs) { m_fs = &fs;}
491 void setData(
const gsFuncData<Scalar> & val) { m_fd = &val;}
492 void setDim(
index_t _d) { m_d = _d; }
493 void clear() { m_fs = NULL; }
496 explicit symbol_expr(
index_t _d)
497 : m_fs(NULL), m_fd(NULL), m_d(_d), m_isAcross(false) { }
500 bool isValid()
const {
return NULL!=m_fd && NULL!=m_fs; }
508 MatExprType eval(
const index_t k)
const
509 {
return m_fd->values[0].col(k).blockDiag(m_d); }
512 const gsFeSpace<Scalar> & rowVar()
const {
return gsNullExpr<Scalar>::get();}
513 const gsFeSpace<Scalar> & colVar()
const {
return gsNullExpr<Scalar>::get();}
517 GISMO_ASSERT(NULL!=m_fs,
"FeVariable: Function source not registered");
518 return m_fs->targetDim();
521 index_t cols()
const {
return m_d; }
523 void print(std::ostream &os)
const { os <<
"u"; }
528 index_t dim()
const {
return m_d;}
532 index_t targetDim()
const {
return m_fs->targetDim(); }
535 index_t parDim()
const {
return m_fs->domainDim(); }
539 GISMO_ASSERT(0!=m_fd->values[0].size(),
"Probable error.");
540 return m_fd->values[0].rows();
548class col_expr :
public _expr<col_expr<E> >
550 typename E::Nested_t _c;
553 typedef typename E::Scalar Scalar;
554 typedef const col_expr<E> Nested_t;
556 enum { Space = E::Space, ScalarValued = 0, ColBlocks = 0 };
558 col_expr(
const E & c,
const index_t i) : _c(c), _i(i) { }
563 inline MatExprType eval(
const index_t k)
const {
return _c.eval(k).col(_i); }
565 index_t rows()
const {
return _c.rows(); }
566 index_t cols()
const {
return 1; }
567 void parse(gsExprHelper<Scalar> & evList)
const { _c.parse(evList); }
569 const gsFeSpace<Scalar> & rowVar()
const {
return _c.rowVar(); }
570 const gsFeSpace<Scalar> & colVar()
const {
return _c.colVar(); }
572 void print(std::ostream &os)
const { os<<_c<<
"["<<_i<<
"]"; }
579class _expr<T, true> :
public _expr<_expr<T> >
584 typedef const _expr<T> Nested_t;
586 explicit _expr(Scalar c) : _c(
give(c)) { }
589 enum {Space = 0, ScalarValued = 1, ColBlocks= 0};
591 inline Scalar eval(
const index_t )
const {
return _c; }
593 inline _expr val()
const {
return *
this; }
594 index_t rows()
const {
return 0; }
595 index_t cols()
const {
return 0; }
596 void parse(gsExprHelper<Scalar> &)
const { }
598 const gsFeSpace<T> & rowVar()
const {
return gsNullExpr<T>::get(); }
599 const gsFeSpace<T> & colVar()
const {
return gsNullExpr<T>::get(); }
601 void print(std::ostream &os)
const { os<<_c; }
608class gsGeometryMap :
public _expr<gsGeometryMap<T> >
610 const gsFunctionSet<T> * m_fs;
611 const gsMapData<T> * m_fd;
617 enum {Space = 0, ScalarValued= 0, ColBlocks= 0};
619 bool isAcross()
const {
return m_isAcross; }
621 gsGeometryMap right()
const
625 ac.m_isAcross =
true;
629 gsGeometryMap left()
const
633 ac.m_isAcross =
false;
638 const gsFunctionSet<T> & source()
const {
return *m_fs;}
641 const gsMapData<T> & data()
const
643 GISMO_ASSERT(NULL!=m_fd,
"gsGeometryMap: invalid data "<< m_fs <<
","<<m_fd);
647 index_t targetDim()
const {
return m_fs->targetDim();}
648 index_t domainDim()
const {
return m_fs->domainDim();}
651 void copyCoefs(
const gsGeometryMap<T> & other)
const
653 GISMO_ASSERT(
dynamic_cast<const gsMultiPatch<T>*
>( this->m_fs ),
"error");
654 const gsMultiPatch<T> & thisMP =
static_cast<const gsMultiPatch<T>&
>(*this->m_fs );
655 GISMO_ASSERT(
dynamic_cast<const gsMultiPatch<T>*
>( other.m_fs ),
"error");
656 const gsMultiPatch<T> & otherMP =
static_cast<const gsMultiPatch<T>&
>(*other.m_fs );
657 GISMO_ASSERT( (thisMP.domainDim()==otherMP.domainDim())&&
658 (thisMP.geoDim()==otherMP.geoDim())&&
659 (thisMP.coefsSize() == otherMP.coefsSize())&&
660 (thisMP.nPatches()==otherMP.nPatches()),
661 "The geometryMaps are not compatible!");
664 for (
index_t p=0; p < thisMP.nPatches(); p++ )
667 thisMP.patch(p).coefs() = otherMP.patch(p).coefs();
672 void deformBy(
const gsFeSolution<T> & displacement)
const
674 const index_t dim = m_fs->domainDim();
676 const gsMultiBasis<T> & mb =
static_cast<const gsMultiBasis<T>&
>(displacement.space().source());
677 const gsMultiPatch<T> & mp =
static_cast<const gsMultiPatch<T>&
>(*this->m_fs );
678 GISMO_ASSERT(
dynamic_cast<const gsMultiBasis<T>*
>(&displacement.space().source()),
"error");
679 GISMO_ASSERT(
dynamic_cast<const gsMultiPatch<T>*
>( this->m_fs),
"error");
682 for (
size_t p=0; p < mp.nPatches(); p++ )
685 gsMatrix<T> &result = mp.patch(p).coefs();
688 const index_t sz = mb[p].size();
691 for (
index_t c = 0; c!=dim; c++)
694 for (
index_t i = 0; i < sz; ++i)
696 const int ii = displacement.mapper().index(i, p, c);
697 if ( displacement.mapper().is_free_index(ii) )
699 result(i,c) += displacement.coefs().at(ii);
703 result(i,c) += displacement.fixedPart().at( displacement.mapper().global_to_bindex(ii));
715 void print(std::ostream &os)
const { os <<
"G"; }
717 auto eval(
const index_t k)
const ->
decltype(m_fd->values[0].col(k))
718 {
return m_fd->values[0].col(k); }
722 gsGeometryMap() : m_fs(NULL), m_fd(NULL), m_isAcross(false) { }
724 void setSource(
const gsFunctionSet<Scalar> & fs) { m_fs = &fs;}
725 void setData(
const gsMapData<Scalar> & val) { m_fd = &val;}
729 index_t rows()
const {
return m_fs->targetDim(); }
730 index_t cols()
const {
return 1; }
732 const gsFeSpace<T> & rowVar()
const {
return gsNullExpr<T>::get(); }
733 const gsFeSpace<T> & colVar()
const {
return gsNullExpr<T>::get(); }
735 void parse(gsExprHelper<Scalar> & evList)
const
743template <
typename T>
struct expr_traits<gsGeometryMap<T> >
746 typedef const gsGeometryMap<T> Nested_t;
753class meas_expr :
public _expr<meas_expr<T> >
755 typename gsGeometryMap<T>::Nested_t _G;
758 enum {Space = 0, ScalarValued = 1, ColBlocks = 0};
762 meas_expr(
const gsGeometryMap<T> & G) : _G(G) { }
766 return _G.data().measures.at(k);
769 index_t rows()
const {
return 0; }
770 index_t cols()
const {
return 0; }
771 void parse(gsExprHelper<Scalar> & evList)
const
777 const gsFeSpace<T> & rowVar()
const {
return gsNullExpr<T>::get(); }
778 const gsFeSpace<T> & colVar()
const {
return gsNullExpr<T>::get(); }
780 void print(std::ostream &os)
const { os <<
"meas("; _G.print(os); os <<
")"; }
790 friend class cdiam_expr<T>;
791 const gsExprHelper<T> * m_exprdata;
793 gsFeElement(
const gsFeElement &);
797 gsFeElement(
const gsExprHelper<T> & eh) : m_exprdata(&eh) { }
799 bool isValid()
const {
return nullptr!=m_exprdata; }
801 const gsVector<T> & weights()
const {
return m_exprdata->weights();}
803 template<
class E>
inline
804 integral_expr<E> integral(
const _expr<E>& ff)
const
805 {
return integral_expr<E>(*
this,ff); }
807 typedef integral_expr<T> AreaRetType;
808 AreaRetType area()
const
809 {
return integral(_expr<T,true>(1)); }
811 typedef integral_expr<meas_expr<T> > PHAreaRetType;
813 PHAreaRetType area(
const gsGeometryMap<Scalar> & _G)
const
814 {
return integral(meas_expr<T>(_G)); }
816 typedef pow_expr<integral_expr<T> > DiamRetType;
818 DiamRetType diam() const
819 {
return pow(integral(_expr<T,true>(1)),(T)(1)/(T)(2)); }
821 typedef pow_expr<integral_expr<meas_expr<T> > > PHDiamRetType;
823 PHDiamRetType diam(
const gsGeometryMap<Scalar> & _G)
const
824 {
return pow(integral(meas_expr<T>(_G)),(T)(1)/(T)(2)); }
829 void print(std::ostream &os)
const { os <<
"e"; }
831 void parse(gsExprHelper<T> & evList)
const
833 GISMO_ERROR(
"Call desired member of element expression instead.");
845 typedef real_t Scalar;
846 mutable Scalar m_val;
848 const gsFeElement<Scalar> &
_e;
849 typename _expr<E>::Nested_t _ff;
851 enum {Space= 0, ScalarValued= 1, ColBlocks = 0};
853 integral_expr(
const gsFeElement<Scalar> & el,
const _expr<E> & u)
854 : m_val(-1),
_e(el), _ff(u) { }
856 const Scalar & eval(
const index_t k)
const
859 GISMO_ENSURE(
_e.isValid(),
"Element is valid within integrals only.");
862 const Scalar * w =
_e.weights().data();
863 m_val = (*w) * _ff.val().eval(0);
864 for (
index_t j = 1; j !=
_e.weights().rows(); ++j)
865 m_val += (*(++w)) * _ff.val().eval(j);
870 inline const integral_expr<E> & val()
const {
return *
this; }
871 inline index_t rows()
const {
return 0; }
872 inline index_t cols()
const {
return 0; }
873 void parse(gsExprHelper<Scalar> & evList)
const
878 const gsFeSpace<Scalar> & rowVar()
const {
return gsNullExpr<Scalar>::get(); }
879 const gsFeSpace<Scalar> & colVar()
const {
return gsNullExpr<Scalar>::get(); }
881 void print(std::ostream &os)
const
916struct expr_traits<gsFeVariable<T> >
919 typedef const gsFeVariable<T> Nested_t;
930 typedef symbol_expr< gsFeVariable<T> > Base;
934 enum {Space = 0, ScalarValued = 0, ColBlocks = 0};
938class gsComposition :
public symbol_expr< gsComposition<T> >
941 typedef symbol_expr< gsComposition<T> > Base;
942 typename gsGeometryMap<T>::Nested_t _G;
944 explicit gsComposition(
const gsGeometryMap<T> & G,
index_t _d = 1)
945 : Base(_d), _G(G) { }
947 enum {Space = 0, ScalarValued= 0, ColBlocks= 0};
949 typename gsMatrix<T>::constColumn
950 eval(
const index_t k)
const {
return this->m_fd->values[0].col(k); }
952 const gsGeometryMap<T> & inner()
const {
return _G;};
954 void parse(gsExprHelper<T> & evList)
const
974 friend class gsNullExpr<T>;
977 typedef symbol_expr< gsFeSpace<T> > Base;
983 enum{Space = 1, ScalarValued=0, ColBlocks=0};
993 GISMO_ASSERT(NULL!=m_sd,
"Space/mapper not properly initialized.");
998 {
return const_cast<gsFeSpace*
>(
this)->mapper();}
1000 inline const gsMatrix<T> & fixedPart()
const {
return m_sd->fixedDofs;}
1001 gsMatrix<T> & fixedPart() {
return m_sd->fixedDofs;}
1003 index_t id()
const {
return (m_sd ? m_sd->id : -101); }
1006 index_t interfaceCont()
const {
return m_sd->cont;}
1009 GISMO_ASSERT(_r>-2 && _r<1,
"Invalid or not implemented (r="<<_r<<
").");
1010 return m_sd->cont = _r;
1013 gsFeSolution<T> function(
const gsMatrix<T>& solVector)
const
1014 {
return gsFeSolution<T>(*
this); }
1019 const index_t dim = this->dim();
1025 const index_t sz = mb[p].size();
1026 result.resize(sz, dim!=1 ? dim : solVector.cols());
1028 for (
index_t c = 0; c!=dim; c++)
1031 for (
index_t i = 0; i < sz; ++i)
1033 const int ii = m_sd->mapper.
index(i, p, c);
1035 for(
index_t s = 0; s != solVector.cols(); ++s )
1036 result(i,c+s) = solVector(ii,s);
1049 GISMO_ASSERT( dofsMapper.
mapSize()==
static_cast<size_t>(this->source().size()*dofsMapper.
numComponents()),
"The dof-mapper is not consistent: mapSize()="<<dofsMapper.
mapSize()<<
"!="<<
static_cast<size_t>(this->source().size())<<
"=this->source().size()");
1050 m_sd->mapper =
give(dofsMapper);
1053 void setup(
const index_t _icont = -1)
const
1055 this->setInterfaceCont(_icont);
1063 if ( 0==this->interfaceCont() )
1065 for ( gsBoxTopology::const_iiterator it = mb->topology().iBegin();
1066 it != mb->topology().iEnd(); ++it )
1068 mb->matchInterface(*it, m_sd->mapper);
1073 if (
const gsMappedBasis<2,T> * mb =
1074 dynamic_cast<const gsMappedBasis<2,T>*
>(&this->source()) )
1076 m_sd->mapper.
setIdentity(mb->nPatches(), mb->size() , this->dim());
1083 const index_t _icont = -1)
const
1085 this->setInterfaceCont(_icont);
1087 const index_t dim = this->dim();
1093 if (0 == this->interfaceCont())
1095 for (gsBoxTopology::const_iiterator it = mb->topology().iBegin();
1096 it != mb->topology().iEnd(); ++it) {
1097 if ( it->type() != interaction::contact )
1098 mb->matchInterface(*it, m_sd->mapper);
1104 for (
typename gsBoundaryConditions<T>::const_iterator
1105 it = bc.
begin(
"Dirichlet"); it != bc.
end(
"Dirichlet"); ++it)
1107 const index_t cc = it->unkComponent();
1108 GISMO_ASSERT(
static_cast<size_t>(it->ps.patch) < this->mapper().numPatches(),
1109 "Problem: a boundary condition is set on a patch id which does not exist.");
1111 bnd = mb->basis(it->ps.patch).boundary(it->ps.side());
1116 for (
typename gsBoundaryConditions<T>::const_iterator
1117 it = bc.
begin(
"Clamped"); it != bc.
end(
"Clamped"); ++it)
1119 const index_t cc = it->unkComponent();
1121 GISMO_ASSERT(
static_cast<size_t>(it->ps.patch) < this->mapper().numPatches(),
1122 "Problem: a boundary condition is set on a patch id which does not exist.");
1124 bnd = mb->basis(it->ps.patch).boundaryOffset(it->ps.side(), 0);
1125 bnd1 = mb->basis(it->ps.patch).boundaryOffset(it->ps.side(), 1);
1127 if (!it->ps.parameter())
1129 for (
index_t c = 0; c!=dim; c++)
1131 if (c==cc || cc==-1 )
1132 for (
index_t k = 0; k < bnd.size(); ++k)
1133 m_sd->mapper.
matchDof(it->ps.patch, (bnd)(k, 0),
1134 it->ps.patch, (bnd1)(k, 0), c);
1140 for (
typename gsBoundaryConditions<T>::const_iterator
1141 it = bc.
begin(
"Collapsed"); it != bc.
end(
"Collapsed"); ++it)
1143 const index_t cc = it->unkComponent();
1145 GISMO_ASSERT(
static_cast<size_t>(it->ps.patch) < this->mapper().numPatches(),
1146 "Problem: a boundary condition is set on a patch id which does not exist.");
1148 bnd = mb->basis(it->ps.patch).boundary(it->ps.side());
1151 for (
index_t c = 0; c!=dim; c++)
1153 if (c==cc || cc==-1)
1154 for (
index_t k = 0; k < bnd.size() - 1; ++k)
1155 m_sd->mapper.
matchDof(it->ps.patch, (bnd)(0, 0),
1156 it->ps.patch, (bnd)(k + 1, 0), c);
1161 for (
typename gsBoundaryConditions<T>::const_cpliterator
1164 const index_t cc = it->component;
1166 GISMO_ASSERT(
static_cast<size_t>(it->ifc.first().patch) < this->mapper().numPatches(),
1167 "Problem: a boundary condition is set on a patch id which does not exist.");
1168 GISMO_ASSERT(
static_cast<size_t>(it->ifc.second().patch) < this->mapper().numPatches(),
1169 "Problem: a boundary condition is set on a patch id which does not exist.");
1172 bnd = mb->basis(it->ifc.first().patch).boundary(it->ifc.first().side());
1173 bnd1 = mb->basis(it->ifc.second().patch).boundary(it->ifc.second().side());
1176 for (
index_t c = 0; c!=dim; c++)
1178 if (c==cc || cc==-1)
1180 for (
index_t k = 0; k < bnd.size() - 1; ++k)
1181 m_sd->mapper.
matchDof(it->ifc.first() .patch, (bnd)(0, 0),
1182 it->ifc.first() .patch, (bnd)(k + 1, 0), c);
1183 for (
index_t k = 0; k < bnd1.size(); ++k)
1184 m_sd->mapper.
matchDof(it->ifc.first() .patch, (bnd)(0, 0),
1185 it->ifc.second().patch, (bnd1)(k, 0), c);
1191 for (
typename gsBoundaryConditions<T>::const_citerator
1194 for (
index_t r = 0; r!=this->dim(); ++r)
1196 if (it->component!=-1 && r!=it->component)
continue;
1199 GISMO_ASSERT(
static_cast<size_t>(it->patch) < mb->nBases(),
1200 "Problem: a corner boundary condition is set on a patch id which does not exist.");
1201 m_sd->mapper.
eliminateDof(mb->basis(it->patch).functionAtCorner(it->corner),
1202 it->patch, it->component);
1207 dynamic_cast<const gsBasis<T> *
>(&this->source()))
1211 for (
typename gsBoundaryConditions<T>::const_iterator
1212 it = bc.
begin(
"Dirichlet"); it != bc.
end(
"Dirichlet"); ++it) {
1214 "Problem: a boundary condition is set on a patch id which does not exist.");
1216 bnd = b->boundary(it->ps.side());
1220 for (
typename gsBoundaryConditions<T>::const_iterator
1221 it = bc.
begin(
"Clamped"); it != bc.
end(
"Clamped"); ++it) {
1223 "Problem: a boundary condition is set on a patch id which does not exist.");
1225 bnd = b->boundary(it->ps.side());
1231 for (
typename gsBoundaryConditions<T>::const_iterator
1232 it = bc.
begin(
"Collapsed"); it != bc.
end(
"Collapsed"); ++it) {
1234 "Problem: a boundary condition is set on a patch id which does not exist.");
1236 bnd = b->boundary(it->ps.side());
1240 }
else if (
const gsMappedBasis<2, T> *mapb =
1241 dynamic_cast<const gsMappedBasis<2, T> *
>(&this->source()))
1243 m_sd->mapper.
setIdentity(mapb->nPatches(), mapb->size(), this->dim());
1245 if (0 == this->interfaceCont())
1248 for (gsBoxTopology::const_iiterator it = mapb->getTopol().iBegin();
1249 it != mapb->getTopol().iEnd(); ++it) {
1250 int1 = mapb->basis(it->first().patch).boundaryOffset(it->first().side(), 0);
1251 int2 = mapb->basis(it->second().patch).boundaryOffset(it->second().side(), 0);
1253 m_sd->mapper.
matchDofs(it->first().patch, int1, it->second().patch, int2);
1256 if (1 == this->interfaceCont())
1258 GISMO_ERROR(
"Boundary offset function is not implemented for gsMappedBasis in general.");
1262 for (
typename gsBoundaryConditions<T>::const_iterator
1263 it = bc.
begin(
"Dirichlet"); it != bc.
end(
"Dirichlet"); ++it) {
1264 const index_t cc = it->unkComponent();
1265 GISMO_ASSERT(
static_cast<size_t>(it->ps.patch) < this->mapper().numPatches(),
1266 "Problem: a boundary condition is set on a patch id which does not exist.");
1268 bnd = mapb->basis(it->ps.patch).boundary(it->ps.side());
1274 for (
typename gsBoundaryConditions<T>::const_iterator
1275 it = bc.
begin(
"Clamped"); it != bc.
end(
"Clamped"); ++it) {
1276 const index_t cc = it->unkComponent();
1278 GISMO_ASSERT(
static_cast<size_t>(it->ps.patch) < this->mapper().numPatches(),
1279 "Problem: a boundary condition is set on a patch id which does not exist.");
1281 bnd = mapb->basis(it->ps.patch).boundaryOffset(it->ps.side(), 0);
1282 bnd1 = mapb->basis(it->ps.patch).boundaryOffset(it->ps.side(), 1);
1287 if (!it->ps.parameter())
1289 for (
index_t c = 0; c!=dim; c++)
1291 if (c==cc || cc==-1 )
1292 for (
index_t k = 0; k < bnd.size() - 1; ++k)
1293 m_sd->mapper.
matchDof( it->ps.patch, (bnd)(k, 0),
1294 it->ps.patch, (bnd1)(k, 0), c);
1297 gsWarn <<
"Unable to apply clamped condition.\n";
1301 for (
typename gsBoundaryConditions<T>::const_iterator
1302 it = bc.
begin(
"Collapsed"); it != bc.
end(
"Collapsed"); ++it) {
1303 const index_t cc = it->unkComponent();
1305 GISMO_ASSERT(
static_cast<size_t>(it->ps.patch) < this->mapper().numPatches(),
1306 "Problem: a boundary condition is set on a patch id which does not exist.");
1308 bnd = mapb->basis(it->ps.patch).boundary(it->ps.side());
1314 for (
index_t c = 0; c!=dim; c++)
1316 if (c==cc || cc==-1)
1317 for (
index_t k = 0; k < bnd.size() - 1; ++k)
1318 m_sd->mapper.
matchDof(it->ps.patch, (bnd)(0, 0),
1319 it->ps.patch, (bnd)(k + 1, 0), c);
1325 for (
typename gsBoundaryConditions<T>::const_citerator
1330 "Problem: a corner boundary condition is set on a patch id which does not exist.");
1331 m_sd->mapper.
eliminateDof(mapb->basis(it->patch).functionAtCorner(it->corner), it->patch, it->component);
1335 GISMO_ASSERT(0 == bc.size(),
"Problem: BCs are ignored.");
1336 m_sd->mapper.
setIdentity(this->source().nPieces(), this->source().size());
1342 gsDirichletValues(bc, dir_values, *
this);
1345 void print(std::ostream &os)
const { os <<
"u"; }
1353template<
class T>
inline bool
1355{
return a.id()== b.id() && a.isAcross()==b.isAcross(); }
1365class gsFeSolution :
public _expr<gsFeSolution<T> >
1368 const gsFeSpace<T> _u;
1374 enum {Space = 0, ScalarValued= 0, ColBlocks= 0};
1376 bool isAcross()
const {
return m_isAcross; }
1378 gsFeSolution right()
const
1380 gsFeSolution ac(*
this);
1381 ac.m_isAcross =
true;
1385 gsFeSolution left()
const {
return gsFeSolution(*
this); }
1387 explicit gsFeSolution(
const gsFeSpace<T> & u) : _u(u), _Sv(NULL) { }
1389 gsFeSolution(
const gsFeSpace<T> & u, gsMatrix<T> & Sv) : _u(u), _Sv(&Sv) { }
1391 const gsFeSpace<T> & space()
const {
return _u;};
1393 mutable gsMatrix<T> res;
1394 const gsMatrix<T> & eval(
index_t k)
const
1396 bool singleActives = (1 == _u.data().actives.cols());
1398 res.setZero(_u.dim(), 1);
1399 const gsDofMapper & map = _u.mapper();
1400 GISMO_ASSERT(_Sv->size()==map.freeSize(),
"The solution vector has wrong dimensions: "<<_Sv->size()<<
" != "<<map.freeSize());
1402 for (
index_t c = 0; c!=_u.dim(); c++)
1404 for (
index_t i = 0; i!=_u.data().actives.rows(); ++i)
1406 const index_t ii = map.index(_u.data().actives(i, singleActives ? 0 : k), _u.data().patchId, c);
1407 if ( map.is_free_index(ii) )
1408 res.at(c) += _Sv->
at(ii) * _u.data().values[0](i,k);
1410 res.at(c) += _u.data().values[0](i,k) *
1411 _u.fixedPart().at( map.global_to_bindex(ii) );
1421 const gsFeSpace<Scalar> & rowVar()
const {
return gsNullExpr<Scalar>::get();}
1422 const gsFeSpace<Scalar> & colVar()
const {
return gsNullExpr<Scalar>::get();}
1423 index_t rows()
const {
return _u.dim(); }
1425 static index_t cols() {
return 1; }
1427 void parse(gsExprHelper<Scalar> & evList)
const
1433 void print(std::ostream &os)
const { os <<
"s"; }
1436 index_t dim()
const {
return _u.dim();}
1439 {
return _u.source().domainDim(); }
1442 const gsDofMapper & mapper()
const {
return _u.mapper();}
1444 inline const gsMatrix<T> & fixedPart()
const {
return _u.fixedPart();}
1445 gsMatrix<T> & fixedPart() {
return _u.fixedPart();}
1448 const gsFuncData<T> & data()
const {
return _u.data();}
1450 void setSolutionVector(gsMatrix<T>& solVector)
1451 { _Sv = & solVector; }
1458 void setComponent(
index_t component, real_t value,
index_t patch=-1)
1460 gsMatrix<T> & solVector = *_Sv;
1461 const gsDofMapper & mapper = _u.mapper();
1466 patchEnd = _u.mapper().numPatches();
1470 patchEnd = patch + 1;
1473 for (
index_t p=patchStart; p!=patchEnd; ++p)
1475 for (
size_t i = 0; i != mapper.patchSize(p, component); ++i)
1477 const index_t ii = mapper.index(i, p, component);
1478 if ( mapper.is_free_index(ii) )
1479 solVector.at(ii) = value;
1484 const gsMatrix<T> & coefs()
const {
return *_Sv; }
1496 GISMO_ASSERT(j<_Sv->size(),
"Solution vector is not initialized/allocated, sz="<<_Sv->size() );
1504 void extract(gsMatrix<T> & result,
const index_t p = 0)
const
1505 { _u.getCoeffs(*_Sv, result, p); }
1509 void extractFull(gsMatrix<T> & result)
const
1513 const size_t totalSz = _u.mapper().mapSize();
1514 result.resize(totalSz, 1);
1515 for (
size_t p=0; p!=_u.mapper().numPatches(); ++p)
1517 offset = _u.mapper().offset(p);
1520 for (
index_t c = 0; c!=dim; c++)
1522 const index_t sz = _u.mapper().patchSize(p,c);
1525 for (
index_t i = 0; i < sz; ++i)
1528 const int ii = _u.mapper().index(i, p, c);
1530 if ( _u.mapper().is_free_index(ii) )
1532 result(i+offset,0) = _Sv->
at(ii);
1535 result(i+offset,0) = _u.fixedPart().at( _u.mapper().global_to_bindex(ii) );
1543 void extract(gsMultiPatch<T> & result)
const
1547 if(
const gsMultiBasis<T>* basis =
dynamic_cast<const gsMultiBasis<T>*
>(&_u.source()) )
1548 for (
size_t i = 0; i != basis->nBases(); ++i)
1550 memory::unique_ptr<gsGeometry<T> > p(this->extractPiece(i));
1551 result.addPatch(*p);
1556 void extract(gsMappedSpline<2,T> & result)
const
1558 if(
const gsMappedBasis<2,T>* basis =
dynamic_cast<const gsMappedBasis<2,T>*
>(&_u.source()) )
1561 this->extractFull(coefs);
1562 coefs.resize(coefs.rows()/_u.dim(),_u.dim());
1563 result.init(*basis,coefs);
1568 memory::unique_ptr<gsGeometry<T> > extractPiece(
const index_t p)
const
1570 if (
const gsBasis<T> * b =
dynamic_cast<const gsBasis<T>*
>(&_u.source().piece(p)) )
1574 return b->makeGeometry(
give(cf));
1580 void insert(
const gsGeometry<T> & g,
const index_t p = 0)
const
1582 insert(g.coefs(), p);
1586 void insert(
const gsMatrix<T> & cf,
const index_t p = 0)
const
1588 gsMatrix<T> & sol = *_Sv;
1590 const gsDofMapper & mapper = _u.mapper();
1591 for (
index_t c = 0; c!=_u.dim(); c++)
1593 for (
index_t i = 0; i != cf.rows(); ++i)
1595 const index_t ii = mapper.index(i, p, c);
1596 if ( mapper.is_free_index(ii) )
1597 sol.at(ii) = cf(i, c);
1612template<
class E,
bool cw>
1613class tr_expr :
public _expr<tr_expr<E,cw> >
1615 typename E::Nested_t _u;
1619 typedef typename E::Scalar Scalar;
1621 tr_expr(_expr<E>
const& u)
1625 enum {ColBlocks = E::ColBlocks, ScalarValued=E::ScalarValued};
1626 enum {Space = cw?E::Space:(E::Space==1?2:(E::Space==2?1:E::Space))};
1628 mutable Temporary_t res;
1629 const Temporary_t & eval(
const index_t k)
const
1632 res = _u.eval(k).blockTranspose( _u.cardinality() );
1634 res = _u.eval(k).transpose();
1638 index_t rows()
const {
return _u.cols(); }
1640 index_t cols()
const {
return _u.rows(); }
1642 void parse(gsExprHelper<Scalar> & evList)
const
1643 { _u.parse(evList); }
1645 const gsFeSpace<Scalar> & rowVar()
const {
return cw?_u.rowVar():_u.colVar(); }
1646 const gsFeSpace<Scalar> & colVar()
const {
return cw?_u.colVar():_u.rowVar(); }
1648 index_t cardinality_impl()
const {
return _u.cardinality_impl(); }
1650 void print(std::ostream &os)
const { os<<
"("; _u.print(os); os <<
")\u1D40"; }
1667class cb_expr :
public _expr<cb_expr<E> >
1669 typename E::Nested_t _u;
1673 typedef typename E::Scalar Scalar;
1675 cb_expr(_expr<E>
const& u)
1679 enum {ColBlocks = 1, ScalarValued=E::ScalarValued};
1680 enum {Space = E::Space};
1682 mutable gsMatrix<Scalar> ev, res;
1684 const gsMatrix<Scalar> & eval(
const index_t k)
const
1697 GISMO_ASSERT(res.rows() % _u.rows() == 0 && res.cols() % _u.cols() == 0,
"Result is not a multiple of the space dimensions?");
1700 if ( (cardinality = res.rows() / _u.rows()) >= 1 && res.cols() / _u.cols() == 1 )
1702 res.resize(_u.rows(), cardinality * _u.cols());
1703 for (
index_t r = 0; r!=cardinality; r++)
1704 res.block(0 , r * _u.cols(), _u.rows(), _u.cols()) = ev.block( r * _u.rows(), 0, _u.rows(), _u.cols() );
1706 else if ( (cardinality = res.rows() / _u.rows()) == 1 && res.cols() / _u.cols() >= 1 )
1708 res.resize(_u.rows(), cardinality * _u.cols());
1709 for (
index_t r = 0; r!=cardinality; r++)
1710 res.block(0 , r * _u.cols(), _u.rows(), _u.cols()) = ev.block( 0, r * _u.cols(), _u.rows(), _u.cols() );
1716 index_t rows()
const {
return _u.rows(); }
1718 index_t cols()
const {
return _u.cols(); }
1720 void parse(gsExprHelper<Scalar> & evList)
const
1721 { _u.parse(evList); }
1723 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
1724 const gsFeSpace<Scalar> & colVar()
const {
return _u.colVar(); }
1726 index_t cardinality_impl()
const
1730 if ( res.rows() / _u.rows() >= 1 && res.cols() / _u.cols() == 1 )
1731 cardinality = res.rows() / _u.rows();
1732 else if ( res.rows() / _u.rows() == 1 && res.cols() / _u.cols() >= 1 )
1733 cardinality = res.cols() / _u.cols();
1735 GISMO_ERROR(
"Cardinality for cb_expr cannot be determined.");
1740 void print(std::ostream &os)
const { os<<
"{"; _u.print(os); os <<
"}"; }
1748class temp_expr :
public _expr<temp_expr<E> >
1750 typename E::Nested_t _u;
1751 typedef typename E::Scalar Scalar;
1752 mutable gsMatrix<Scalar> tmp;
1755 temp_expr(_expr<E>
const& u)
1759 enum {Space = E::Space, ScalarValued = E::ScalarValued,
1760 ColBlocks = E::ColBlocks};
1764 const gsMatrix<Scalar> & eval(
const index_t k)
const
1770 index_t rows()
const {
return _u.rows(); }
1771 index_t cols()
const {
return _u.cols(); }
1772 void parse(gsExprHelper<Scalar> & evList)
const { _u.parse(evList); }
1773 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
1774 const gsFeSpace<Scalar> & colVar()
const {
return _u.colVar(); }
1776 void print(std::ostream &os)
const { _u.print(os); }
1783class trace_expr :
public _expr<trace_expr<E> >
1786 typedef typename E::Scalar Scalar;
1787 enum {ScalarValued = 0, Space = E::Space, ColBlocks= 0};
1790 typename E::Nested_t _u;
1791 mutable gsMatrix<Scalar> res;
1794 trace_expr(_expr<E>
const& u) : _u(u)
1801 const gsMatrix<Scalar> & eval(
const index_t k)
const
1803 auto tmp = _u.eval(k);
1805 const index_t r = _u.cardinality();
1811 for (
index_t i = 0; i!=r; ++i)
1812 res.at(i) = tmp.middleCols(i*cb,cb).trace();
1819 index_t rows()
const {
return _u.cols() / _u.rows(); }
1820 index_t cols()
const {
return 1; }
1822 index_t cardinality_impl()
const {
return _u.cardinality(); }
1824 void parse(gsExprHelper<Scalar> & evList)
const
1825 { _u.parse(evList); }
1827 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
1828 const gsFeSpace<Scalar> & colVar()
const {
return _u.colVar(); }
1830 void print(std::ostream &os)
const { os <<
"trace("; _u.print(os); os<<
")"; }
1837class adjugate_expr :
public _expr<adjugate_expr<E> >
1840 typedef typename E::Scalar Scalar;
1841 enum {ScalarValued = 0, ColBlocks = E::ColBlocks};
1842 enum {Space = E::Space};
1844 typename E::Nested_t _u;
1845 mutable gsMatrix<Scalar> res;
1848 adjugate_expr(_expr<E>
const& u) : _u(u)
1855 const gsMatrix<Scalar> & eval(
const index_t k)
const
1857 auto tmp = _u.eval(k);
1859 const index_t r = _u.cols() / cb;
1860 res.resize(_u.rows(),_u.cols());
1861 for (
index_t i = 0; i!=r; ++i){
1862 res.middleCols(i*cb,cb) = tmp.middleCols(i*cb,cb).adjugate();
1870 index_t rows()
const {
return _u.rows(); }
1871 index_t cols()
const {
return _u.cols(); }
1873 void parse(gsExprHelper<Scalar> & evList)
const
1874 { _u.parse(evList); }
1876 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
1877 const gsFeSpace<Scalar> & colVar()
const {
return _u.colVar(); }
1879 void print(std::ostream &os)
const { os <<
"adj("; _u.print(os); os<<
")"; }
1883class reshape_expr :
public _expr<reshape_expr<E> >
1886 typedef typename E::Scalar Scalar;
1887 enum {ScalarValued = 0, ColBlocks = E::ColBlocks};
1888 enum {Space = E::Space};
1890 typename E::Nested_t _u;
1892 mutable gsMatrix<Scalar> tmp;
1897 reshape_expr(_expr<E>
const& u,
index_t n,
index_t m) : _u(u), _n(n), _m(m)
1902 const gsAsConstMatrix<Scalar> eval(
const index_t k)
const
1905 GISMO_ASSERT( _u.rows()*_u.cols() == _n*_m,
"Wrong dimension "
1906 << _u.rows() <<
" x "<<_u.cols() <<
"!=" << _n <<
" * "<< _m );
1908 return gsAsConstMatrix<Scalar>(tmp.data(),_n,_m);
1911 index_t rows()
const {
return _n; }
1912 index_t cols()
const {
return _m; }
1914 void parse(gsExprHelper<Scalar> & evList)
const
1915 { _u.parse(evList); }
1917 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
1918 const gsFeSpace<Scalar> & colVar()
const {
return _u.colVar(); }
1920 void print(std::ostream &os)
const { os <<
"reshape("; _u.print(os); os<<
","<<_n<<
","<<_m<<
")"; }
1924template <
typename E> EIGEN_STRONG_INLINE
1926{
return reshape_expr<E>(u, n, m); }
1929class replicate_expr :
public _expr<replicate_expr<E> >
1932 typedef typename E::Scalar Scalar;
1933 enum {ScalarValued = 0, Space = E::Space, ColBlocks= E::ColBlocks};
1935 typename E::Nested_t _u;
1937 mutable gsMatrix<Scalar> tmp;
1942 replicate_expr(_expr<E>
const& u,
index_t n,
index_t m) : _u(u), _n(n), _m(m)
1946 auto eval(
const index_t k)
const ->
decltype(tmp.replicate(_n,_m))
1949 return tmp.replicate(_n,_m);
1952 index_t rows()
const {
return _n*_u.rows(); }
1953 index_t cols()
const {
return _m*_u.cols(); }
1955 void parse(gsExprHelper<Scalar> & evList)
const
1956 { _u.parse(evList); }
1958 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
1959 const gsFeSpace<Scalar> & colVar()
const {
return _u.colVar(); }
1960 index_t cardinality_impl()
const {
return _u.cardinality_impl(); }
1962 void print(std::ostream &os)
const { os <<
"replicate("; _u.print(os); os<<
","<<_n<<
","<<_m<<
")"; }
1966template <
typename E> EIGEN_STRONG_INLINE
1968{
return replicate_expr<E>(u, n, m); }
1981 typedef typename E::Scalar Scalar;
1982 enum {ScalarValued = 0, Space = E::Space, ColBlocks= 0};
1984 typename E::Nested_t _u;
1997 const index_t numActives = _u.cardinality();
1999 for (
index_t i = 0; i<numActives; ++i)
2001 tmp(0,2*i+1) += tmp(1,2*i);
2002 std::swap(tmp(1,2*i), tmp(1,2*i+1));
2005 tmp.resize(4,numActives);
2006 tmp.conservativeResize(3,numActives);
2009 tmp.transposeInPlace();
2011 tmp.transposeInPlace();
2016 index_t rows()
const {
return 1; }
2017 index_t cols()
const {
return 3; }
2020 { _u.parse(evList); }
2024 index_t cardinality_impl()
const {
return _u.cardinality_impl(); }
2026 void print(std::ostream &os)
const { os <<
"flat("; _u.print(os); os<<
")"; }
2030template <
typename E> EIGEN_STRONG_INLINE
2039 class diag_expr :
public _expr<diag_expr<E> >
2042 typedef typename E::Scalar Scalar;
2043 enum {Space=0, ColBlocks=E::ColBlocks, ScalarValued = 0};
2045 typename E::Nested_t _u;
2046 mutable gsMatrix<Scalar> res;
2049 diag_expr(_expr<E>
const& u) : _u(u)
2051 GISMO_ASSERT(0== _u.cols()%_u.rows(),
"Expecting square-block expression, got "
2052 << _u.rows() <<
" x "<< _u.cols() );
2055 const gsMatrix<Scalar> & eval(
const index_t k)
const
2058 MatExprType tmp = _u.eval(k);
2060 const index_t r = _u.cols() / cb;
2062 for (
index_t i = 0; i!=r; ++i)
2063 res.row(i) = tmp.middleCols(i*cb,cb).diagonal();
2067 index_t rows()
const {
return _u.cols() / _u.rows(); }
2068 index_t cols()
const {
return _u.rows(); }
2070 void parse(gsExprHelper<Scalar> & evList)
const
2071 { _u.parse(evList); }
2073 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
2074 const gsFeSpace<Scalar> & colVar()
const {
return _u.colVar(); }
2077 void print(std::ostream &os)
const { os <<
"diag("; _u.print(os); os<<
")"; }
2081template <
typename E> EIGEN_STRONG_INLINE
2083{
return diag_expr<E>(u); }
2085#define GISMO_EXPR_VECTOR_EXPRESSION(name, mname, isSv) \
2086 template<class E> class name##_##expr : public _expr<name##_##expr<E> > { \
2087 typename E::Nested_t _u; \
2089 typedef typename E::Scalar Scalar; \
2090 enum {Space= E::Space, ScalarValued= isSv, ColBlocks= E::ColBlocks}; \
2091 name##_##expr(_expr<E> const& u) : _u(u) { } \
2092 mutable Temporary_t tmp; \
2093 const Temporary_t & eval(const index_t k) const { \
2094 tmp = _u.eval(k).mname(); return tmp; } \
2095 index_t rows() const { return isSv ? 0 : _u.rows(); } \
2096 index_t cols() const { return isSv ? 0 : _u.cols(); } \
2097 void parse(gsExprHelper<Scalar> & evList) const { _u.parse(evList); } \
2098 const gsFeSpace<Scalar> & rowVar() const {return gsNullExpr<Scalar>::get();} \
2099 const gsFeSpace<Scalar> & colVar() const {return gsNullExpr<Scalar>::get();} \
2100 void print(std::ostream &os) const \
2101 { os << #name <<"("; _u.print(os); os <<")"; } \
2105GISMO_EXPR_VECTOR_EXPRESSION(norm,norm,1);
2107GISMO_EXPR_VECTOR_EXPRESSION(sqNorm,squaredNorm,1);
2109GISMO_EXPR_VECTOR_EXPRESSION(normalized,normalized,0);
2118GISMO_EXPR_VECTOR_EXPRESSION(det,determinant,1);
2122#undef GISMO_EXPR_VECTOR_EXPRESSION
2131 typedef typename E::Scalar Scalar;
2133 typename E::Nested_t _u;
2137 enum{Space = E::Space, ScalarValued= 0, ColBlocks= E::ColBlocks};
2145 auto m = _u.eval(k);
2149 for (
index_t i = 0; i!=c; ++i)
2150 res.middleCols(i*r,r) = m.col(i).asDiagonal();
2157 index_t cardinality_impl()
const {
return _u.cardinality_impl(); }
2159 index_t rows()
const {
return _u.rows(); }
2160 index_t cols()
const {
return _u.rows() * _u.cols(); }
2162 { _u.parse(evList); }
2164 void print(std::ostream &os)
const { os <<
"diag("; _u.print(os); os <<
")";}
2169class max_expr :
public _expr<max_expr<E> >
2172 typedef typename E::Scalar Scalar;
2173 enum {ScalarValued = 0, Space = E::Space, ColBlocks = 1};
2175 typename E::Nested_t _u;
2181 max_expr(_expr<E>
const& u) : _u(u)
2186 const gsMatrix<Scalar> & eval(
const index_t k)
const {
return eval_impl(_u,k); }
2188 index_t rows()
const {
return 1; }
2189 index_t cols()
const {
return 1; }
2190 void setFlag()
const { _u.setFlag(); }
2192 void parse(gsExprHelper<Scalar> & evList)
const
2193 { _u.parse(evList); }
2195 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
2196 const gsFeSpace<Scalar> & colVar()
const {
return _u.colVar(); }
2197 index_t cardinality_impl()
const {
return _u.cardinality_impl(); }
2199 void print(std::ostream &os)
const { os <<
"max("; _u.print(os); os<<
")"; }
2201 template<
class U>
inline
2202 typename util::enable_if< util::is_same<U,gsFeSpace<Scalar> >::value,
const gsMatrix<Scalar> & >::type
2203 eval_impl(
const U & u,
const index_t k)
const
2207 res.resize(1,u.cardinality());
2209 for (
index_t c=0; c!=_u.cardinality(); c++)
2210 res(0,c) = tmp.block(0,c*u.cols(),u.rows(),u.cols()).maxCoeff();
2212 for (
index_t c=0; c!=_u.rows(); c++)
2213 res(0,c) = tmp.block(c*u.rows(),0,u.rows(),u.cols()).maxCoeff();
2218 template<
class U>
inline
2219 typename util::enable_if< !util::is_same<U,gsFeSpace<Scalar> >::value,
const gsMatrix<Scalar> & >::type
2220 eval_impl(
const U & u,
const index_t k)
const
2222 res = u.eval(k).colwise().maxCoeff();
2228class rowsum_expr :
public _expr<rowsum_expr<E> >
2231 typedef typename E::Scalar Scalar;
2232 enum {ScalarValued = 0, Space = E::Space, ColBlocks = E::ColBlocks};
2234 typename E::Nested_t _u;
2235 mutable gsMatrix<Scalar> tmp;
2239 rowsum_expr(_expr<E>
const& u) : _u(u)
2244 const gsMatrix<Scalar> & eval(
const index_t k)
const
2246 tmp = _u.eval(k).rowwise().sum();
2250 index_t rows()
const {
return _u.rows(); }
2251 index_t cols()
const {
return 1; }
2252 void setFlag()
const { _u.setFlag(); }
2254 void parse(gsExprHelper<Scalar> & evList)
const
2255 { _u.parse(evList); }
2257 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
2258 const gsFeSpace<Scalar> & colVar()
const {
return _u.colVar(); }
2259 index_t cardinality_impl()
const {
return _u.cardinality_impl(); }
2261 void print(std::ostream &os)
const { os <<
"rowsum("; _u.print(os); os<<
")"; }
2265class colsum_expr :
public _expr<colsum_expr<E> >
2268 typedef typename E::Scalar Scalar;
2269 enum {ScalarValued = 0, Space = E::Space, ColBlocks = E::ColBlocks};
2271 typename E::Nested_t _u;
2272 mutable gsMatrix<Scalar> tmp;
2276 colsum_expr(_expr<E>
const& u) : _u(u)
2281 const gsMatrix<Scalar> & eval(
const index_t k)
const
2283 tmp = _u.eval(k).colwise().sum();
2287 index_t rows()
const {
return _u.rows(); }
2288 index_t cols()
const {
return 1; }
2289 void setFlag()
const { _u.setFlag(); }
2291 void parse(gsExprHelper<Scalar> & evList)
const
2292 { _u.parse(evList); }
2294 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
2295 const gsFeSpace<Scalar> & colVar()
const {
return _u.colVar(); }
2296 index_t cardinality_impl()
const {
return _u.cardinality_impl(); }
2298 void print(std::ostream &os)
const { os <<
"colsum("; _u.print(os); os<<
")"; }
2307 typedef real_t Scalar;
2308 enum {Space = 0, ScalarValued = 0, ColBlocks = 0};
2322 index_t rows()
const {
return _dim; }
2323 index_t cols()
const {
return _dim; }
2329 void print(std::ostream &os)
const { os <<
"id("<<_dim <<
")";}
2338 typedef real_t Scalar;
2339 enum {Space = 0, ScalarValued = 0, ColBlocks = 0};
2353 index_t rows()
const {
return _mat.rows(); }
2354 index_t cols()
const {
return _mat.cols(); }
2360 void print(std::ostream &os)
const { os <<
"constMat";}
2369 typename E::Nested_t _u;
2370 typename E::Scalar _tol;
2372 typedef typename E::Scalar Scalar;
2373 enum {ScalarValued = 1, Space = E::Space, ColBlocks= 0};
2375 sign_expr(_expr<E>
const& u, Scalar tolerance = 0.0) : _u(u),_tol(tolerance){
2376 GISMO_ASSERT( _tol >= 0,
"Tolerance for sign_expr should be a positive number.");
2379 Scalar eval(
const index_t k)
const
2381 const Scalar v = _u.val().eval(k);
2382 return ( v>_tol ? 1 : ( v<-_tol ? -1 : 0 ) );
2385 static index_t rows() {
return 0; }
2386 static index_t cols() {
return 0; }
2391 static bool isScalar() {
return true; }
2396 void print(std::ostream &os)
const { os<<
"sgn("; _u.print(os); os <<
")"; }
2405 typename E::Nested_t _u;
2407 typedef typename E::Scalar Scalar;
2408 enum {ScalarValued = 1, Space = E::Space, ColBlocks= 0};
2410 exp_expr(_expr<E>
const& u) : _u(u) { }
2412 Scalar eval(
const index_t k)
const
2414 const Scalar v = _u.val().eval(k);
2415 return math::exp(v);
2418 static index_t rows() {
return 0; }
2419 static index_t cols() {
return 0; }
2424 static bool isScalar() {
return true; }
2429 void print(std::ostream &os)
const { os<<
"exp("; _u.print(os); os <<
")"; }
2439 typedef typename E::Scalar Scalar;
2440 enum {ScalarValued = E::ScalarValued, Space = E::Space, ColBlocks= E::ColBlocks};
2442 typename E::Nested_t _u;
2450 res = _u.eval(k).cwiseMax(0.0);
2455 index_t rows()
const {
return _u.rows(); }
2456 index_t cols()
const {
return _u.cols(); }
2464 void print(std::ostream &os)
const { os<<
"posPart("; _u.print(os); os <<
")"; }
2473 typename E::Nested_t _u;
2475 typedef typename E::Scalar Scalar;
2476 enum {ScalarValued = 1, Space = 0, ColBlocks= 0};
2482 Scalar & eval(
index_t k)
const
2484 res = std::max(0.0,_u.eval(k));
2488 index_t rows()
const {
return 0; }
2489 index_t cols()
const {
return 0; }
2492 { _u.parse(evList); }
2497 void print(std::ostream &os)
const { os<<
"posPart("; _u.print(os); os <<
")"; }
2506 typename E::Nested_t _u;
2509 typedef typename E::Scalar Scalar;
2510 enum {ScalarValued = 1, Space = E::Space, ColBlocks= E::ColBlocks};
2514 pow_expr(_expr<E>
const& u, Scalar q) : _u(u), _q(q) { }
2516 Scalar eval(
const index_t k)
const
2518 const Scalar v = _u.val().eval(k);
2519 return math::pow(v,_q);
2522 static index_t rows() {
return 0; }
2523 static index_t cols() {
return 0; }
2528 static bool isScalar() {
return true; }
2533 void print(std::ostream &os)
const { os<<
"pow("; _u.print(os); os <<
")"; }
2541template <
typename E1,
typename E2>
2545 typedef typename E1::Scalar Scalar;
2546 enum {ScalarValued = 0, ColBlocks = 1};
2547 enum {Space = E2::Space};
2550 typename E1::Nested_t _u;
2551 typename E2::Nested_t _v;
2562 const index_t N = _v.cols() / (r*r);
2564 const auto uEv = _u.eval(k);
2565 const auto vEv = _v.eval(k);
2567 res.resize(r, N*r*r);
2569 for (
index_t s = 0; s!=r; ++s)
2570 for (
index_t i = 0; i!=N; ++i)
2572 res.middleCols((s*N + i)*r,r).noalias() =
2573 uEv.col(s) * vEv.middleCols((s*N + i)*r,r).row(s);
2580 index_t rows()
const {
return _u.cols(); }
2581 index_t cols()
const {
return _v.cols(); }
2584 { _u.parse(evList); _v.parse(evList); }
2589 void print(std::ostream &os)
const { os <<
"matrix_by_space("; _u.print(os); os<<
")"; }
2597template <
typename E1,
typename E2>
2601 typedef typename E1::Scalar Scalar;
2602 enum {ScalarValued = 0, ColBlocks = 1};
2603 enum {Space = E2::Space};
2606 typename E1::Nested_t _u;
2607 typename E2::Nested_t _v;
2618 const index_t N = _v.cols() / (r*r);
2620 const auto uEv = _u.eval(k);
2621 const auto vEv = _v.eval(k);
2623 res.resize(r, N*r*r);
2624 for (
index_t s = 0; s!=r; ++s)
2625 for (
index_t i = 0; i!=N; ++i)
2627 res.middleCols((s*N + i)*r,r).noalias() =
2628 uEv.transpose()*vEv.middleCols((s*N + i)*r,r).transpose();
2634 index_t rows()
const {
return _u.cols(); }
2635 index_t cols()
const {
return _v.cols(); }
2638 { _u.parse(evList); _v.parse(evList); }
2643 void print(std::ostream &os)
const { os <<
"matrix_by_space_tr("; _u.print(os); os<<
")"; }
2650class value_expr :
public _expr<value_expr<E> >
2652 typename E::Nested_t _u;
2655 typedef typename E::Scalar Scalar;
2656 value_expr(_expr<E>
const& u) : _u(u)
2663 enum {Space= 0, ScalarValued= 1, ColBlocks= 0};
2665 Scalar eval(
const index_t k)
const {
return eval_impl(_u,k); }
2668 inline value_expr<E> val()
const {
return *
this; }
2669 index_t rows()
const {
return 0; }
2670 index_t cols()
const {
return 0; }
2671 void parse(gsExprHelper<Scalar> & evList)
const
2672 { _u.parse(evList); }
2674 static bool isScalar() {
return true; }
2676 const gsFeSpace<Scalar> & rowVar()
const {
return gsNullExpr<Scalar>::get();}
2677 const gsFeSpace<Scalar> & colVar()
const {
return gsNullExpr<Scalar>::get();}
2679 void print(std::ostream &os)
const { _u.print(os); }
2684 template<
class U>
static inline
2685 typename util::enable_if<U::ScalarValued,Scalar>::type
2686 eval_impl(
const U & u,
const index_t k) {
return u.eval(k); }
2688 template<
class U>
static inline
2689 typename util::enable_if<!U::ScalarValued,Scalar>::type
2690 eval_impl(
const U & u,
const index_t k) {
return u.eval(k).value(); }
2694class abs_expr :
public _expr<abs_expr<E> >
2696 typename E::Nested_t _u;
2699 typedef typename E::Scalar Scalar;
2700 explicit abs_expr(_expr<E>
const& u) : _u(u) { }
2703 enum {Space= 0, ScalarValued= 1, ColBlocks= 0};
2705 Scalar eval(
const index_t k)
const {
return abs_expr::eval_impl(_u,k); }
2707 index_t rows()
const {
return _u.rows(); }
2708 index_t cols()
const {
return _u.cols(); }
2709 void parse(gsExprHelper<Scalar> & evList)
const
2710 { _u.parse(evList); }
2712 static bool isScalar() {
return true; }
2714 const gsFeSpace<Scalar> & rowVar()
const {
return gsNullExpr<Scalar>::get();}
2715 const gsFeSpace<Scalar> & colVar()
const {
return gsNullExpr<Scalar>::get();}
2717 void print(std::ostream &os)
const { _u.print(os); }
2722 template<
class U>
static inline
2723 typename util::enable_if<U::ScalarValued,Scalar>::type
2724 eval_impl(
const U & u,
const index_t k) {
return math::abs(u.eval(k)); }
2725 template<
class U>
static inline
2726 typename util::enable_if<!U::ScalarValued,gsMatrix<Scalar> >::type
2727 eval_impl(
const U & u,
const index_t k) {
return u.eval(k).cwiseAbs(); }
2736class grad_expr :
public _expr<grad_expr<E> >
2738 typename E::Nested_t _u;
2740 enum {Space = E::Space, ScalarValued= 0, ColBlocks= 0};
2742 typedef typename E::Scalar Scalar;
2743 mutable gsMatrix<Scalar> tmp;
2745 grad_expr(
const E & u) : _u(u)
2746 {
GISMO_ASSERT(1==u.dim(),
"grad(.) requires 1D variable, use jac(.) instead.");}
2748 const gsMatrix<Scalar> & eval(
const index_t k)
const
2753 tmp = _u.data().values[1].reshapeCol(k, cols(), cardinality_impl()).transpose();
2757 index_t rows()
const {
return 1 ; }
2759 index_t cols()
const {
return _u.source().domainDim(); }
2761 index_t cardinality_impl()
const
2762 {
return _u.data().values[1].rows() / cols(); }
2764 void parse(gsExprHelper<Scalar> & evList)
const
2767 _u.data().flags |= NEED_GRAD;
2770 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
2771 const gsFeSpace<Scalar> & colVar()
const
2772 {
return gsNullExpr<Scalar>::get();}
2774 void print(std::ostream &os)
const { os <<
"\u2207("; _u.print(os); os <<
")"; }
2777 template<
class U>
static inline
2778 typename util::enable_if<util::is_same<U,gsComposition<Scalar> >::value,
2779 const gsMatrix<Scalar> &>::type
2780 eval_impl(
const U & u,
const index_t k)
2794class grad_expr<gsFeSolution<T> > :
public _expr<grad_expr<gsFeSolution<T> > >
2797 const gsFeSolution<T> _u;
2801 enum {Space = 0, ScalarValued= 0, ColBlocks= 0};
2803 explicit grad_expr(
const gsFeSolution<T> & u) : _u(u) { }
2805 mutable gsMatrix<T> res;
2806 const gsMatrix<T> & eval(
index_t k)
const
2808 GISMO_ASSERT(1==_u.data().actives.cols(),
"Single actives expected");
2810 res.setZero(_u.dim(), _u.parDim());
2811 const gsDofMapper & map = _u.mapper();
2812 for (
index_t c = 0; c!= _u.dim(); c++)
2814 for (
index_t i = 0; i!=_u.data().actives.size(); ++i)
2816 const index_t ii = map.index(_u.data().actives.at(i), _u.data().patchId,c);
2817 if ( map.is_free_index(ii) )
2819 res.row(c) += _u.coefs().at(ii) *
2820 _u.data().values[1].col(k).segment(i*_u.parDim(), _u.parDim()).transpose();
2825 _u.fixedPart().at( map.global_to_bindex(ii) ) *
2826 _u.data().values[1].col(k).segment(i*_u.parDim(), _u.parDim()).transpose();
2833 index_t rows()
const {
return _u.dim();}
2834 index_t cols()
const {
return _u.parDim(); }
2836 const gsFeSpace<Scalar> & rowVar()
const
2837 {
return gsNullExpr<Scalar>::get();}
2838 const gsFeSpace<Scalar> & colVar()
const
2839 {
return gsNullExpr<Scalar>::get();}
2841 void parse(gsExprHelper<Scalar> & evList)
const
2844 evList.add(_u.space());
2848 void print(std::ostream &os)
const { os <<
"\u2207(s)"; }
2858class dJacdc_expr :
public _expr<dJacdc_expr<E> >
2860 typename E::Nested_t _u;
2862 enum{ Space = E::Space, ScalarValued = 0, ColBlocks = (1==E::Space?1:0)};
2864 typedef typename E::Scalar Scalar;
2866 mutable gsMatrix<Scalar> res;
2869 dJacdc_expr(
const E & u,
index_t c) : _u(u), _c(c)
2870 {
GISMO_ASSERT(1==u.dim(),
"grad(.) requires 1D variable, use jac(.) instead.");}
2872 const gsMatrix<Scalar> & eval(
const index_t k)
const
2874 index_t dd = _u.source().domainDim();
2876 res.setZero(dd, dd*n);
2878 gsMatrix<Scalar>
grad = _u.data().values[1].reshapeCol(k, dd, n);
2879 for(
index_t i = 0; i < n; i++){
2880 res.row(_c).segment(i*dd,dd) =
grad.col(i);
2885 index_t rows()
const {
return _u.source().domainDim(); }
2887 index_t cols()
const {
return _u.source().domainDim()*_u.rows(); }
2889 void parse(gsExprHelper<Scalar> & evList)
const
2892 _u.data().flags |= NEED_GRAD;
2895 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
2896 const gsFeSpace<Scalar> & colVar()
const
2897 {
return gsNullExpr<Scalar>::get();}
2899 void print(std::ostream &os)
const { os <<
"dJacdc("; _u.print(os); os <<
")"; }
2907class nabla_expr :
public _expr<nabla_expr<T> >
2909 typename gsFeVariable<T>::Nested_t u;
2920 nabla_expr(
const gsFeVariable<T> & _u) : u(_u)
2926 mutable gsMatrix<Scalar> res;
2928 const gsMatrix<Scalar> & eval(
const index_t k)
const
2932 res.setZero(rows(), d);
2934 for (
index_t i = 0; i!=d; ++i)
2935 res.col(i).segment(i*n,n) = u.data().values[1].reshapeCol(k, d, n).row(i);
2939 index_t rows()
const {
return u.rows(); }
2940 index_t cols()
const {
return u.cols(); }
2942 void parse(gsExprHelper<Scalar> & evList)
const
2945 u.data().flags |= NEED_GRAD;
2948 const gsFeSpace<T> & rowVar()
const {
return u.rowVar(); }
2949 const gsFeSpace<T> & colVar()
const
2950 {
return gsNullExpr<Scalar>::get();}
2952 void print(std::ostream &os)
const { os <<
"nabla("; u.print(os); os <<
")"; }
2962class nabla2_expr :
public _expr<nabla2_expr<T> >
2964 typename gsFeVariable<T>::Nested_t u;
2975 nabla2_expr(
const gsFeVariable<T> & _u)
2979 MatExprType eval(
const index_t k)
const
2982 return u.data().values[2]
2983 .reShapeCol(k, u.data().values[2].rows()/u.cSize(), u.cSize() )
2984 .topRows(u.parDim()).transpose();
2987 index_t rows()
const {
return u.rows(); }
2988 index_t cols()
const {
return u.parDim(); }
2990 void parse(gsExprHelper<Scalar> & evList)
const
2996 const gsFeSpace<T> & rowVar()
const {
return u.rowVar(); }
2997 const gsFeSpace<T> & colVar()
const
2998 {
return gsNullExpr<T>::get();}
3013 typename gsGeometryMap<T>::Nested_t _G;
3017 enum {Space = 0, ScalarValued= 0, ColBlocks= 0};
3019 explicit onormal_expr(
const gsGeometryMap<T> & G) : _G(G) { }
3021 auto eval(
const index_t k)
const ->
decltype(_G.data().outNormals.col(k))
3022 {
return _G.data().outNormals.col(k); }
3024 index_t rows()
const {
return _G.source().targetDim(); }
3025 index_t cols()
const {
return 1; }
3027 const gsFeSpace<T> & rowVar()
const {
return gsNullExpr<T>::get();}
3028 const gsFeSpace<T> & colVar()
const {
return gsNullExpr<T>::get();}
3036 void print(std::ostream &os)
const { os <<
"nv("; _G.print(os); os <<
")"; }
3046 typename gsGeometryMap<T>::Nested_t _G;
3050 enum {Space = 0, ScalarValued= 0, ColBlocks= 0};
3054 GISMO_ENSURE( _G.source().domainDim()+1 == _G.source().targetDim(),
"Surface normal requires codimension 1");
3057 auto eval(
const index_t k)
const ->
decltype(_G.data().normals.col(k))
3058 {
return _G.data().normals.col(k); }
3060 index_t rows()
const {
return _G.source().targetDim(); }
3061 index_t cols()
const {
return 1; }
3063 const gsFeSpace<T> & rowVar()
const {
return gsNullExpr<T>::get();}
3064 const gsFeSpace<T> & colVar()
const {
return gsNullExpr<T>::get();}
3072 void print(std::ostream &os)
const { os <<
"sn("; _G.print(os); os <<
")"; }
3082 typename gsGeometryMap<T>::Nested_t _G;
3086 enum {Space = 0, ScalarValued= 0, ColBlocks= 0};
3093 if (_G.targetDim()==2)
3095 res = _G.data().outNormals.col(k);
3096 std::swap( res(0,0), res(1,0) );
3100 else if (_G.targetDim()==3)
3103 res.
col3d(0) = _G.data().normals.col3d(k)
3104 .cross( _G.data().outNormals.col3d(k) );
3108 GISMO_ERROR(
"Function not implemented for dimension"<<_G.targetDim());
3112 index_t rows()
const {
return _G.source().targetDim(); }
3113 index_t cols()
const {
return 1; }
3125 void print(std::ostream &os)
const { os <<
"tv("; _G.print(os); os <<
")"; }
3134 typename E::Nested_t _u;
3137 typedef typename E::Scalar Scalar;
3138 enum {Space = E::Space, ScalarValued= 0, ColBlocks= 0};
3142 auto eval(
const index_t k)
const ->
decltype(_u.data().laplacians.col(k))
3145 return _u.data().laplacians.col(k);
3151 index_t rows()
const {
return _u.data().laplacians.rows(); }
3152 index_t cols()
const {
return 1; }
3154 index_t cardinality_impl()
const {
return _u.cardinality_impl(); }
3165 void print(std::ostream &os)
const { os <<
"\u2206("; _u.print(os); os <<
")"; }
3172class lapl_expr<gsFeSolution<T> > :
public _expr<lapl_expr<gsFeSolution<T> > >
3175 const gsFeSolution<T> _u;
3179 enum {Space = 0, ScalarValued= 0, ColBlocks= 0};
3181 lapl_expr(
const gsFeSolution<T> & u) : _u(u) { }
3183 mutable gsMatrix<T> res;
3184 const gsMatrix<T> & eval(
const index_t k)
const
3186 GISMO_ASSERT(1==_u.data().actives.cols(),
"Single actives expected");
3188 res.setZero(_u.dim(), 1);
3189 const gsDofMapper & map = _u.mapper();
3191 index_t numActs = _u.data().values[0].rows();
3192 index_t numDers = _u.parDim() * (_u.parDim() + 1) / 2;
3195 for (
index_t c = 0; c!= _u.dim(); c++)
3196 for (
index_t i = 0; i!=numActs; ++i)
3198 const index_t ii = map.index(_u.data().actives.at(i), _u.data().patchId,c);
3199 deriv2 = _u.data().values[2].block(i*numDers,k,_u.parDim(),1);
3200 if ( map.is_free_index(ii) )
3201 res.at(c) += _u.coefs().at(ii) * deriv2.sum();
3203 res.at(c) +=_u.fixedPart().at( map.global_to_bindex(ii) ) * deriv2.sum();
3208 index_t rows()
const {
return _u.dim(); }
3209 index_t cols()
const {
return 1; }
3211 void parse(gsExprHelper<Scalar> & evList)
const
3213 evList.add(_u.space());
3217 const gsFeSpace<Scalar> & rowVar()
const {
return gsNullExpr<T>::get();}
3218 const gsFeSpace<Scalar> & colVar()
const {
return gsNullExpr<T>::get();}
3220 void print(std::ostream &os)
const { os <<
"\u2206(s)"; }
3227class fform2nd_expr :
public _expr<fform2nd_expr<T> >
3229 typename gsGeometryMap<T>::Nested_t _G;
3232 enum {Space = 0, ScalarValued = 0, ColBlocks = 0};
3234 fform2nd_expr(
const gsGeometryMap<T> & G) : _G(G) { }
3236 const gsAsConstMatrix<Scalar> eval(
const index_t k)
const
3238 return gsAsConstMatrix<Scalar>(_G.data().fundForms.col(k).data(),rows(),cols());
3241 index_t rows()
const {
return _G.source().domainDim() ; }
3242 index_t cols()
const {
return _G.source().domainDim() ; }
3244 void parse(gsExprHelper<Scalar> & evList)
const
3250 const gsFeSpace<Scalar> & rowVar()
const {
return gsNullExpr<T>::get();}
3251 const gsFeSpace<Scalar> & colVar()
const {
return gsNullExpr<T>::get();}
3253 void print(std::ostream &os)
const { os <<
"fform2nd("; _G.print(os); os <<
")"; }
3261class jacInv_expr :
public _expr<jacInv_expr<T> >
3263 typename gsGeometryMap<T>::Nested_t _G;
3266 enum {Space = 0, ScalarValued = 0, ColBlocks = 0};
3268 jacInv_expr(
const gsGeometryMap<T> & G) : _G(G)
3274 MatExprType eval(
const index_t k)
const {
return _G.data().jacInvTr.reshapeCol(k,cols(),rows()).transpose(); }
3276 index_t rows()
const {
return _G.source().domainDim(); }
3277 index_t cols()
const {
return _G.source().targetDim(); }
3279 void parse(gsExprHelper<Scalar> & evList)
const
3285 const gsFeSpace<Scalar> & rowVar()
const {
return gsNullExpr<T>::get();}
3286 const gsFeSpace<Scalar> & colVar()
const {
return gsNullExpr<T>::get();}
3291 void print(std::ostream &os)
const { os <<
"jacInv("; _G.print(os); os <<
")"; }
3298class jac_expr :
public _expr<jac_expr<E> >
3300 typename E::Nested_t _u;
3302 enum {ColBlocks = (1==E::Space?1:0) };
3303 enum {Space = E::Space, ScalarValued= 0 };
3305 typedef typename E::Scalar Scalar;
3307 mutable gsMatrix<Scalar> res;
3309 jac_expr(
const E & _u_)
3312 MatExprType eval(
const index_t k)
const
3317 res = _u.data().values[1].col(k).transpose().blockDiag(_u.dim());
3321 res = _u.data().values[1]
3322 .reshapeCol(k, _u.parDim(), _u.targetDim()).transpose()
3323 .blockDiag(_u.dim());
3328 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
3330 const gsFeSpace<Scalar> & colVar()
const {
return gsNullExpr<Scalar>::get(); }
3332 index_t rows()
const {
return rows_impl(_u); }
3333 index_t cols()
const {
return cols_impl(_u); }
3338 index_t cardinality_impl()
const
3340 return _u.dim() * _u.data().actives.rows();
3343 void parse(gsExprHelper<Scalar> & evList)
const
3350 void print(std::ostream &os)
const { os <<
"\u2207("; _u.print(os);os <<
")"; }
3358 template<
class U>
inline
3359 typename util::enable_if<!(util::is_same<U,gsFeSpace<Scalar> >::value),
index_t >::type
3360 rows_impl(
const U & u)
const
3362 return u.source().targetDim();
3365 template<
class U>
inline
3366 typename util::enable_if< (util::is_same<U,gsFeSpace<Scalar> >::value),
index_t >::type
3367 rows_impl(
const U & u)
const
3372 template<
class U>
inline
3373 typename util::enable_if<!(util::is_same<U,gsFeSpace<Scalar> >::value),
index_t >::type
3374 cols_impl(
const U & u)
const
3376 return u.source().domainDim();
3379 template<
class U>
inline
3380 typename util::enable_if< (util::is_same<U,gsFeSpace<Scalar> >::value),
index_t >::type
3381 cols_impl(
const U & u)
const
3383 return u.source().domainDim();
3390 template<
class U>
inline
3391 typename util::enable_if<!(util::is_same<U,gsFeSpace<Scalar> >::value),
const gsFeSpace<Scalar> & >::type
3394 return gsNullExpr<Scalar>::get();
3397 template<
class U>
inline
3398 typename util::enable_if<(util::is_same<U,gsFeSpace<Scalar> >::value),
const gsFeSpace<Scalar> & >::type
3409class jac_expr<gsGeometryMap<T> > :
public _expr<jac_expr<gsGeometryMap<T> > >
3411 typename gsGeometryMap<T>::Nested_t _G;
3415 enum {Space = 0, ScalarValued= 0, ColBlocks= 0};
3417 jac_expr(
const gsGeometryMap<T> & G) : _G(G) { }
3418 MatExprType eval(
const index_t k)
const
3421 return _G.data().values[1]
3422 .reshapeCol(k, _G.data().dim.first, _G.data().dim.second).transpose();
3425 index_t rows()
const {
return _G.source().targetDim(); }
3427 index_t cols()
const {
return _G.source().domainDim(); }
3429 static const gsFeSpace<Scalar> & rowVar() {
return gsNullExpr<Scalar>::get(); }
3430 static const gsFeSpace<Scalar> & colVar() {
return gsNullExpr<Scalar>::get(); }
3432 void parse(gsExprHelper<Scalar> & evList)
const
3438 meas_expr<T> absDet()
const
3440 GISMO_ASSERT(rows() == cols(),
"The Jacobian matrix is not square");
3441 return meas_expr<T>(_G);
3444 jacInv_expr<T> inv()
const
3446 GISMO_ASSERT(rows() == cols(),
"The Jacobian matrix is not square");
3447 return jacInv_expr<T>(_G);
3451 jacInv_expr<T> ginv()
const {
return jacInv_expr<T>(_G); }
3453 void print(std::ostream &os)
const { os <<
"\u2207("; _G.print(os); os <<
")"; }
3457class hess_expr :
public _expr<hess_expr<E> >
3460 typedef typename E::Scalar Scalar;
3462 typename E::Nested_t _u;
3463 mutable gsMatrix<Scalar> res;
3465 enum {ScalarValued = 0, ColBlocks = (1==E::Space?1:0) };
3466 enum {Space = E::Space };
3469 hess_expr(
const E & u) : _u(u)
3475 const gsMatrix<Scalar> & eval(
const index_t k)
const
3477 const gsFuncData<Scalar> & dd = _u.data();
3478 const index_t sz = cardinality_impl();
3479 res.resize(dd.dim.first, sz*dd.dim.first);
3480 secDerToHessian(dd.values[2].col(k), dd.dim.first, res);
3481 res.resize(dd.dim.first, res.cols()*dd.dim.first);
3487 index_t rows()
const {
return _u.data().dim.first; }
3493 index_t cardinality_impl()
const
3495 return 2*_u.data().values[2].rows()/
3496 (_u.data().dim.first*(1+_u.data().dim.first));
3500 void parse(gsExprHelper<Scalar> & evList)
const
3503 _u.data().flags |= NEED_2ND_DER;
3506 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
3507 const gsFeSpace<Scalar> & colVar()
const {
return gsNullExpr<Scalar>::get(); }
3509 void print(std::ostream &os)
const
3511 { os <<
"\u210D(U)"; }
3515class hess_expr<gsFeSolution<T> > :
public _expr<hess_expr<gsFeSolution<T> > >
3518 const gsFeSolution<T> _u;
3522 enum{Space = 0, ScalarValued = 0, ColBlocks = 0 };
3524 hess_expr(
const gsFeSolution<T> & u) : _u(u) { }
3526 mutable gsMatrix<T> res;
3527 const gsMatrix<T> & eval(
const index_t k)
const
3529 GISMO_ASSERT(1==_u.data().actives.cols(),
"Single actives expected. Actives: \n"<<_u.data().actives);
3531 const gsDofMapper & map = _u.mapper();
3533 const index_t numActs = _u.data().values[0].rows();
3534 const index_t pdim = _u.parDim();
3535 index_t numDers = pdim*(pdim+1)/2;
3541 res.setZero(numDers,1);
3542 for (
index_t i = 0; i!=numActs; ++i)
3544 const index_t ii = map.index(_u.data().actives.at(i), _u.data().patchId,0);
3545 deriv2 = _u.data().values[2].block(i*numDers,k,numDers,1);
3546 if ( map.is_free_index(ii) )
3547 res += _u.coefs().at(ii) * deriv2;
3549 res +=_u.fixedPart().at( map.global_to_bindex(ii) ) * deriv2;
3551 secDerToHessian(res, pdim, deriv2);
3553 res.resize(pdim,pdim);
3558 res.setZero(rows(), numDers);
3559 for (
index_t c = 0; c != _u.dim(); c++)
3560 for (
index_t i = 0; i != numActs; ++i) {
3561 const index_t ii = map.index(_u.data().actives.at(i), _u.data().patchId, c);
3562 deriv2 = _u.space().data().values[2].block(i * numDers, k, numDers,
3564 if (map.is_free_index(ii))
3565 res.row(c) += _u.coefs().at(ii) * deriv2;
3567 res.row(c) += _u.fixedPart().at(map.global_to_bindex(ii)) * deriv2;
3586 return _u.parDim() * (_u.parDim() + 1) / 2;
3589 const gsFeSpace<Scalar> & rowVar()
const {
return gsNullExpr<Scalar>::get(); }
3590 const gsFeSpace<Scalar> & colVar()
const {
return gsNullExpr<Scalar>::get(); }
3592 void parse(gsExprHelper<Scalar> & evList)
const
3595 evList.add(_u.space());
3599 void print(std::ostream &os)
const { os <<
"\u210D(s)"; }
3608class dJacG_expr :
public _expr<dJacG_expr<T> >
3610 typename gsGeometryMap<T>::Nested_t _G;
3612 mutable gsMatrix<T> res;
3616 dJacG_expr(
const gsGeometryMap<T> & G) : _G(G) { }
3618 MatExprType eval(
const index_t k)
const
3620 const index_t sz = _G.data().values[0].rows();
3621 const index_t s = _G.data().derivSize();
3623 res.resize(_G.data().dim.second, sz*_G.data().dim.first);
3628 index_t rows()
const {
return _G.source().targetDim(); }
3629 index_t cols()
const {
return _G.source().domainDim(); }
3631 void parse(gsExprHelper<Scalar> & evList)
const
3634 _G.data().flags |= NEED_2ND_DER;
3643class curl_expr :
public _expr<curl_expr<T> >
3648 typename gsFeVariable<T>::Nested_t _u;
3649 mutable gsMatrix<Scalar> res;
3651 enum{ Space = 1, ScalarValued= 0, ColBlocks= 0};
3653 curl_expr(
const gsFeVariable<T> & u) : _u(u)
3654 {
GISMO_ASSERT(3==u.dim(),
"curl(.) requires 3D variable."); }
3656 const gsMatrix<T> & eval(
const index_t k)
const
3658 res.setZero( rows(), _u.dim());
3659 const index_t na = _u.data().values[0].rows();
3660 gsAsConstMatrix<T, Dynamic, Dynamic> pd =
3661 _u.data().values[1].reshapeCol(k, cols(), na);
3663 res.col(0).segment(na ,na) = -pd.row(2);
3664 res.col(0).segment(2*na,na) = pd.row(1);
3665 res.col(1).segment(0 ,na) = pd.row(2);
3666 res.col(1).segment(2*na,na) = -pd.row(0);
3667 res.col(2).segment(0 ,na) = -pd.row(1);
3668 res.col(2).segment(na ,na) = pd.row(0);
3672 index_t rows()
const {
return _u.dim() * _u.data().values[0].rows(); }
3673 index_t cols()
const {
return _u.data().dim.first; }
3675 void parse(gsExprHelper<Scalar> & evList)
const
3678 _u.data().flags |= NEED_GRAD;
3681 const gsFeSpace<T> & rowVar()
const {
return _u.rowVar(); }
3682 const gsFeSpace<T> & colVar()
const {
return gsNullExpr<T>::get();}
3684 void print(std::ostream &os)
const { os <<
"curl("; _u.print(os); os <<
")"; }
3697template <
typename E1,
typename E2>
3698class mult_expr<E1,E2,false> :
public _expr<mult_expr<E1, E2, false> >
3700 typename E1::Nested_t _u;
3701 typename E2::Nested_t _v;
3704 enum {ScalarValued = E1::ScalarValued && E2::ScalarValued,
3705 ColBlocks = E2::ColBlocks};
3706 enum {Space = (int)E1::Space + (
int)E2::Space };
3708 typedef typename E1::Scalar Scalar;
3710 mult_expr(_expr<E1>
const& u,
3714 mutable Temporary_t tmp;
3715 const Temporary_t & eval(
const index_t k)
const
3717 GISMO_ASSERT(0==_u.cols()*_v.rows() || _u.cols() == _v.rows(),
3718 "Wrong dimensions "<<_u.cols()<<
"!="<<_v.rows()<<
" in * operation:\n"
3719 << _u <<
" times \n" << _v );
3722 tmp = _u.eval(k) * _v.eval(k);
3726 index_t rows()
const {
return E1::ScalarValued ? _v.rows() : _u.rows(); }
3727 index_t cols()
const {
return E2::ScalarValued ? _u.cols() : _v.cols(); }
3728 void parse(gsExprHelper<Scalar> & evList)
const
3729 { _u.parse(evList); _v.parse(evList); }
3732 index_t cardinality_impl()
const
3733 {
return 0==E1::Space ? _v.cardinality(): _u.cardinality(); }
3735 const gsFeSpace<Scalar> & rowVar()
const
3736 {
return 0==E1::Space ? _v.rowVar() : _u.rowVar(); }
3737 const gsFeSpace<Scalar> & colVar()
const
3738 {
return 0==E2::Space ? _u.colVar() : _v.colVar(); }
3740 void print(std::ostream &os)
const { _u.print(os); os<<
"*"; _v.print(os); }
3757template <
typename E1,
typename E2>
3758class mult_expr<E1, E2, true> :
public _expr<mult_expr<E1, E2, true> >
3761 typedef typename E2::Scalar Scalar;
3763 typename E1::Nested_t _u;
3764 typename E2::Nested_t _v;
3766 mutable gsMatrix<Scalar> res;
3768 enum {ScalarValued = 0, ColBlocks = E1::ColBlocks};
3769 enum {Space = (int)E1::Space + (
int)E2::Space };
3771 mult_expr(_expr<E1>
const& u,
3778 const gsMatrix<Scalar> & eval(
const index_t k)
const
3782 const index_t nb = _u.cardinality();
3783 const auto tmpA = _u.eval(k);
3784 const auto tmpB = _v.eval(k);
3789 if ( 1 == _v.cardinality() )
3791 res.resize(ur, vc*nb);
3792 GISMO_ASSERT(tmpA.cols()==uc*nb,
"Dimension error.. "<< tmpA.cols()<<
"!="<<uc*nb );
3795 for (
index_t i = 0; i!=nb; ++i)
3796 res.middleCols(i*vc,vc).noalias()
3797 = tmpA.middleCols(i*uc,uc) * tmpB;
3804 const index_t nbv = _v.cardinality();
3805 res.resize(ur*nb, vc*nbv);
3806 for (
index_t i = 0; i!=nb; ++i)
3807 for (
index_t j = 0; j!=nbv; ++j)
3809 res.block(i*ur,j*vc,ur,vc).noalias() =
3810 tmpA.middleCols(i*uc,uc) * tmpB.middleCols(j*vc,vc);
3825 void parse(gsExprHelper<Scalar> & evList)
const
3826 { _u.parse(evList); _v.parse(evList); }
3829 index_t cardinality_impl()
const {
return _u.cardinality(); }
3831 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
3832 const gsFeSpace<Scalar> & colVar()
const
3834 if ( 1 == _v.cardinality() )
3840 void print(std::ostream &os)
const
3841 { os <<
"("; _u.print(os);os <<
"*";_v.print(os);os <<
")"; }
3849template <
typename E2>
3850class mult_expr<typename E2::Scalar, E2, false>
3851 :
public _expr<mult_expr<typename E2::Scalar, E2, false> >
3855 typedef typename E2::Scalar Scalar;
3858 typename E2::Nested_t _v;
3862 enum {ScalarValued = E2::ScalarValued, ColBlocks = E2::ColBlocks};
3863 enum {Space = E2::Space};
3865 mult_expr(Scalar
const & c, _expr<E2>
const& v)
3868 EIGEN_STRONG_INLINE AutoReturn_t eval(
const index_t k)
const
3870 return ( _c * _v.eval(k) );
3874 index_t rows()
const {
return _v.rows(); }
3875 index_t cols()
const {
return _v.cols(); }
3877 void parse(gsExprHelper<Scalar> & evList)
const
3878 { _v.parse(evList); }
3880 index_t cardinality_impl()
const
3881 {
return _v.cardinality(); }
3883 const gsFeSpace<Scalar> & rowVar()
const {
return _v.rowVar(); }
3884 const gsFeSpace<Scalar> & colVar()
const {
return _v.colVar(); }
3886 void print(std::ostream &os)
const { os << _c <<
"*";_v.print(os); }
3890template <
typename E1,
typename E2>
3891class collapse_expr :
public _expr<collapse_expr<E1, E2> >
3893 typename E1::Nested_t _u;
3894 typename E2::Nested_t _v;
3897 enum {ScalarValued = 0, ColBlocks = 0};
3898 enum { Space = (int)E1::Space + (
int)E2::Space };
3900 typedef typename E1::Scalar Scalar;
3902 mutable gsMatrix<Scalar> res;
3904 collapse_expr(_expr<E1>
const& u,
3909 const gsMatrix<Scalar> &
3913 const auto tmpA = _u.eval(k);
3914 const auto tmpB = _v.eval(k);
3920 for (
index_t i = 0; i!=nb; ++i)
3922 res.row(i).transpose().noalias() = tmpA.middleCols(i*ur,ur) * tmpB;
3925 else if (E2::ColBlocks)
3929 for (
index_t i = 0; i!=nb; ++i)
3931 res.row(i).noalias() = tmpA * tmpB.middleCols(i*ur,ur);
3938 index_t rows()
const {
return E1::ColBlocks ? _u.cols() / _v.rows() : _v.cols() / _u.cols() ; }
3939 index_t cols()
const {
return E1::ColBlocks ? _v.rows() : _u.cols(); }
3941 void parse(gsExprHelper<Scalar> & evList)
const
3942 { _u.parse(evList); _v.parse(evList); }
3944 const gsFeSpace<Scalar> & rowVar()
const
3945 {
return E1::ColBlocks ? _u.rowVar() : _v.rowVar(); }
3946 const gsFeSpace<Scalar> & colVar()
const
3951 void print(std::ostream &os)
const { _u.print(os); os<<
"~"; _v.print(os); }
3955template <
typename E1,
typename E2>
3957collapse_expr<E1,E2> collapse( _expr<E1>
const& u, _expr<E2>
const& v)
3958{
return collapse_expr<E1, E2>(u, v); }
3971template <
typename E1,
typename E2,
bool = E2::ColBlocks>
3972class frprod_expr :
public _expr<frprod_expr<E1, E2> >
3975 typedef typename E1::Scalar Scalar;
3976 enum {ScalarValued = 0, ColBlocks=E2::ColBlocks};
3977 enum { Space = (int)E1::Space + (
int)E2::Space };
3985 typename E1::Nested_t _u;
3986 typename E2::Nested_t _v;
3988 mutable gsMatrix<Scalar> res;
3992 frprod_expr(_expr<E1>
const& u, _expr<E2>
const& v)
4002 const gsMatrix<Scalar> & eval(
const index_t k)
const
4006 const index_t nb = _u.cardinality();
4007 auto A = _u.eval(k);
4008 auto B = _v.eval(k);
4010 for (
index_t i = 0; i!=nb; ++i)
4011 for (
index_t j = 0; j!=nb; ++j)
4013 (A.middleCols(i*rb,rb).array() * B.middleCols(j*rb,rb).array()).sum();
4017 index_t rows()
const {
return _u.cols() / _u.rows(); }
4018 index_t cols()
const {
return _u.cols() / _u.rows(); }
4020 void parse(gsExprHelper<Scalar> & evList)
const
4021 { _u.parse(evList); _v.parse(evList); }
4023 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
4024 const gsFeSpace<Scalar> & colVar()
const {
return _v.colVar(); }
4026 void print(std::ostream &os)
const
4027 { os <<
"("; _u.print(os); os<<
" % "; _v.print(os); os<<
")";}
4037template <
typename E1,
typename E2>
4038class frprod_expr<E1,E2,false> :
public _expr<frprod_expr<E1, E2,false> >
4041 typedef typename E1::Scalar Scalar;
4042 enum {ScalarValued = 0, Space = E1::Space, ColBlocks= E1::ColBlocks};
4045 typename E1::Nested_t _u;
4046 typename E2::Nested_t _v;
4048 mutable gsMatrix<Scalar> res;
4052 frprod_expr(_expr<E1>
const& u, _expr<E2>
const& v)
4062 const gsMatrix<Scalar> & eval(
const index_t k)
const
4065 auto A = _u.eval(k);
4066 auto B = _v.eval(k);
4068 const index_t nb = _u.cardinality();
4070 for (
index_t i = 0; i!=nb; ++i)
4072 (A.middleCols(i*rb,rb).array() * B.array()).sum();
4076 index_t rows()
const {
return _u.cols() / _u.rows(); }
4077 index_t cols()
const {
return 1; }
4079 void parse(gsExprHelper<Scalar> & evList)
const
4080 { _u.parse(evList); _v.parse(evList); }
4083 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
4084 const gsFeSpace<Scalar> & colVar()
const {
return _v.rowVar(); }
4086 void print(std::ostream &os)
const
4087 { os <<
"("; _u.print(os); os<<
" % "; _v.print(os); os<<
")";}
4093template <
typename E1,
typename E2>
4094class divide_expr :
public _expr<divide_expr<E1,E2> >
4096 typename E1::Nested_t _u;
4097 typename E2::Nested_t _v;
4100 typedef typename E1::Scalar Scalar;
4103 enum {ScalarValued = E1::ScalarValued, ColBlocks= E2::ColBlocks};
4104 enum {Space = E1::Space};
4106 divide_expr(_expr<E1>
const& u, _expr<E2>
const& v)
4109 GISMO_STATIC_ASSERT(E2::ScalarValued,
"The denominator needs to be scalar valued.");
4112 AutoReturn_t eval(
const index_t k)
const
4113 {
return ( _u.eval(k) / _v.eval(k) ); }
4115 index_t rows()
const {
return _u.rows(); }
4116 index_t cols()
const {
return _u.cols(); }
4118 void parse(gsExprHelper<Scalar> & evList)
const
4119 { _u.parse(evList); _v.parse(evList); }
4122 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
4123 const gsFeSpace<Scalar> & colVar()
const {
return _u.colVar(); }
4125 void print(std::ostream &os)
const
4126 { os <<
"("; _u.print(os);os <<
" / ";_v.print(os);os <<
")"; }
4132template <
typename E1>
4133class divide_expr<E1,typename E1::Scalar>
4134 :
public _expr<divide_expr<E1,typename E1::Scalar> >
4137 typedef typename E1::Scalar Scalar;
4140 typename E1::Nested_t _u;
4144 enum {Space= E1::Space, ScalarValued = E1::ScalarValued, ColBlocks= E1::ColBlocks};
4146 divide_expr(_expr<E1>
const& u, Scalar
const c)
4149 AutoReturn_t eval(
const index_t k)
const
4150 {
return ( _u.eval(k) / _c ); }
4152 index_t rows()
const {
return _u.rows(); }
4153 index_t cols()
const {
return _u.cols(); }
4155 void parse(gsExprHelper<Scalar> & evList)
const
4156 { _u.parse(evList); }
4159 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
4160 const gsFeSpace<Scalar> & colVar()
const {
return _u.colVar(); }
4162 void print(std::ostream &os)
const
4163 { os <<
"("; _u.print(os);os <<
"/"<< _c <<
")"; }
4170template <
typename E2>
4171class divide_expr<typename E2::Scalar,E2>
4172 :
public _expr<divide_expr<typename E2::Scalar,E2> >
4175 typedef typename E2::Scalar Scalar;
4179 typename E2::Nested_t _u;
4181 enum {Space= 0, ScalarValued = 1, ColBlocks= 0};
4183 divide_expr(Scalar
const c, _expr<E2>
const& u)
4185 { GISMO_STATIC_ASSERT(E2::ScalarValued,
"The denominator needs to be scalar valued."); }
4187 Scalar eval(
const index_t k)
const
4188 {
return ( _c / _u.val().eval(k) ); }
4190 index_t rows()
const {
return 0; }
4191 index_t cols()
const {
return 0; }
4193 void parse(gsExprHelper<Scalar> & evList)
const
4194 { _u.parse(evList); }
4197 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
4198 const gsFeSpace<Scalar> & colVar()
const {
return _u.colVar(); }
4200 void print(std::ostream &os)
const
4201 { os <<
"("<< _c <<
"/";_u.print(os);os <<
")";}
4207template <
typename E1,
typename E2>
4208class add_expr :
public _expr<add_expr<E1, E2> >
4210 typename E1::Nested_t _u;
4211 typename E2::Nested_t _v;
4214 enum {ScalarValued = E1::ScalarValued && E2::ScalarValued,
4215 ColBlocks = E1::ColBlocks || E2::ColBlocks };
4216 enum {Space = E1::Space};
4218 typedef typename E1::Scalar Scalar;
4220 add_expr(_expr<E1>
const& u, _expr<E2>
const& v)
4224 _u.rowVar()==_v.rowVar() && _u.colVar()==_v.colVar(),
4225 "Error: adding apples and oranges (use comma instead),"
4226 " namely:\n" << _u <<
"\n"<<_v<<
4227 " \nvars:\n" << _u.rowVar().id()<<
"!="<<_v.rowVar().id() <<
", "<< _u.colVar().id()<<
"!="<<_v.colVar().id()<<
4228 " \nspaces:\n" << (
int)E1::Space<<
"!="<< (
int)E2::Space
4232 mutable Temporary_t res;
4233 const Temporary_t & eval(
const index_t k)
const
4236 "Wrong dimensions "<<_u.rows()<<
"!="<<_v.rows()<<
" in + operation:\n"
4237 << _u <<
" plus \n" << _v );
4239 "Wrong dimensions "<<_u.cols()<<
"!="<<_v.cols()<<
" in + operation:\n"
4240 << _u <<
" plus \n" << _v );
4241 res = _u.eval(k) + _v.eval(k);
4245 index_t rows()
const {
return _u.rows(); }
4246 index_t cols()
const {
return _u.cols(); }
4248 void parse(gsExprHelper<Scalar> & evList)
const
4249 { _u.parse(evList); _v.parse(evList); }
4252 index_t cardinality_impl()
const {
return _u.cardinality_impl(); }
4254 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
4255 const gsFeSpace<Scalar> & colVar()
const {
return _v.colVar(); }
4257 void print(std::ostream &os)
const
4258 { os <<
"("; _u.print(os);os <<
" + ";_v.print(os);os <<
")"; }
4274template <
typename E1,
typename E2>
4275class summ_expr :
public _expr<summ_expr<E1,E2> >
4278 typedef typename E1::Scalar Scalar;
4280 enum {Space = E1::Space, ScalarValued= 0, ColBlocks= E2::ColBlocks};
4282 summ_expr(E1
const& u, E2
const& M) : _u(u), _M(M) { }
4284 const gsMatrix<Scalar> & eval(
const index_t k)
const
4286 auto sl = _u.eval(k);
4288 auto ml = _M.eval(k);
4290 const index_t mb = _M.cardinality();
4292 GISMO_ASSERT(_M.cols()==_M.rows(),
"Matrix must be square: "<< _M.rows()<<
" x "<< _M.cols() <<
" expr: "<< _M );
4293 GISMO_ASSERT(mb==_u.cols(),
"cardinality must match vector, but card(M)="<<_M.cardinality()<<
" and cols(u)="<<_u.cols());
4295 res.setZero(mr, sr * mr);
4296 for (
index_t i = 0; i!=sr; ++i)
4297 for (
index_t t = 0; t!=mb; ++t)
4298 res.middleCols(i*mr,mr) += sl(i,t) * ml.middleCols(t*mr,mr);
4302 index_t rows()
const {
return _M.rows(); }
4303 index_t cols()
const {
return _M.rows(); }
4305 void parse(gsExprHelper<Scalar> & evList)
const
4306 { _u.parse(evList); _M.parse(evList); }
4308 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
4309 const gsFeSpace<Scalar> & colVar()
const {
return gsNullExpr<Scalar>::get(); }
4311 index_t cardinality_impl()
const
4314 void print(std::ostream &os)
const
4315 { os <<
"sum("; _M.print(os); os<<
","; _u.print(os); os<<
")"; }
4318 typename E1::Nested_t _u;
4319 typename E2::Nested_t _M;
4321 mutable gsMatrix<Scalar> res;
4328template <
typename E1,
typename E2>
4329class sub_expr :
public _expr<sub_expr<E1, E2> >
4331 typename E1::Nested_t _u;
4332 typename E2::Nested_t _v;
4335 enum {ScalarValued = E1::ScalarValued && E2::ScalarValued,
4336 ColBlocks = E1::ColBlocks || E2::ColBlocks };
4337 enum {Space = E1::Space};
4339 typedef typename E1::Scalar Scalar;
4341 sub_expr(_expr<E1>
const& u, _expr<E2>
const& v)
4345 _u.rowVar()==_v.rowVar() && _u.colVar()==_v.colVar(),
4346 "Error: substracting apples from oranges (use comma instead),"
4347 " namely:\n" << _u <<
"\n"<<_v);
4350 mutable Temporary_t res;
4351 const Temporary_t & eval(
const index_t k)
const
4358 "Wrong dimensions "<<_u.rows()<<
"!="<<_v.rows()<<
" in - operation:\n" << _u <<
" minus \n" << _v );
4360 "Wrong dimensions "<<_u.cols()<<
"!="<<_v.cols()<<
" in - operation:\n" << _u <<
" minus \n" << _v );
4362 "Cardinality "<< _u.cardinality()<<
" != "<< _v.cardinality());
4365 res = _u.eval(k) - _v.eval(k);
4369 index_t rows()
const {
return _u.rows(); }
4370 index_t cols()
const {
return _u.cols(); }
4372 void parse(gsExprHelper<Scalar> & evList)
const
4373 { _u.parse(evList); _v.parse(evList); }
4375 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
4376 const gsFeSpace<Scalar> & colVar()
const {
return _v.colVar(); }
4378 index_t cardinality_impl()
const
4381 "Cardinality "<< _u.cardinality()<<
" != "<< _v.cardinality());
4382 return _u.cardinality();
4385 void print(std::ostream &os)
const
4386 { os <<
"("; _u.print(os); os<<
" - ";_v.print(os); os <<
")";}
4392template <
typename E>
4393class symm_expr :
public _expr<symm_expr<E> >
4395 typename E::Nested_t _u;
4397 mutable gsMatrix<typename E::Scalar> tmp;
4399 typedef typename E::Scalar Scalar;
4401 enum { Space = (0==E::Space ? 0 : E::Space), ScalarValued= E::ScalarValued, ColBlocks= E::ColBlocks };
4403 symm_expr(_expr<E>
const& u)
4406 MatExprType eval(
const index_t k)
const
4411 return tmp * tmp.transpose();
4414 index_t rows()
const {
return _u.rows(); }
4415 index_t cols()
const {
return _u.rows(); }
4417 void parse(gsExprHelper<Scalar> & evList)
const
4418 { _u.parse(evList); }
4420 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
4421 const gsFeSpace<Scalar> & colVar()
const {
return _u.rowVar(); }
4423 void print(std::ostream &os)
const { os <<
"symm("; _u.print(os); os <<
")"; }
4426template <
typename E>
4427class symmetrize_expr :
public _expr<symmetrize_expr<E> >
4429 typename E::Nested_t _u;
4431 mutable gsMatrix<typename E::Scalar> tmp;
4433 enum { Space = (0==E::Space ? 0 : 3), ScalarValued=E::ScalarValued, ColBlocks= E::ColBlocks };
4434 typedef typename E::Scalar Scalar;
4436 symmetrize_expr(_expr<E>
const& u)
4439 MatExprType eval(
const index_t k)
const
4444 return tmp + tmp.transpose();
4447 index_t rows()
const {
return _u.rows(); }
4448 index_t cols()
const {
return _u.rows(); }
4450 void parse(gsExprHelper<Scalar> & evList)
const
4451 { _u.parse(evList); }
4453 const gsFeSpace<Scalar> & rowVar()
const {
return _u.rowVar(); }
4454 const gsFeSpace<Scalar> & colVar()
const {
return _u.rowVar(); }
4455 index_t cardinality_impl()
const {
return _u.cardinality_impl(); }
4457 void print(std::ostream &os)
const { os <<
"symmetrize("; _u.print(os); os <<
")"; }
4472EIGEN_STRONG_INLINE constMat_expr ones(
const index_t dim) {
4475 return constMat_expr(ones);
4478EIGEN_STRONG_INLINE constMat_expr mat(
const gsMatrix<real_t> mat) {
return constMat_expr(mat); }
4485template<
class E> EIGEN_STRONG_INLINE
4486abs_expr<E>
abs(
const E & u) {
return abs_expr<E>(u); }
4489template<
class E> EIGEN_STRONG_INLINE
4490grad_expr<E>
grad(
const E & u) {
return grad_expr<E>(u); }
4493template<
class E> EIGEN_STRONG_INLINE
4497template<
class T> EIGEN_STRONG_INLINE
4501template<
class T> EIGEN_STRONG_INLINE
4505template<
class T> EIGEN_STRONG_INLINE
4508template<
class T> EIGEN_STRONG_INLINE
4509normal_expr<T> sn(
const gsGeometryMap<T> & u) {
return normal_expr<T>(u); }
4512template<
class T> EIGEN_STRONG_INLINE
4515template<
class E> EIGEN_STRONG_INLINE
4516lapl_expr<E> lapl(
const symbol_expr<E> & u) {
return lapl_expr<E>(u); }
4518template<
class T> EIGEN_STRONG_INLINE
4519lapl_expr<gsFeSolution<T> > lapl(
const gsFeSolution<T> & u)
4520{
return lapl_expr<gsFeSolution<T> >(u); }
4523template<
class T> EIGEN_STRONG_INLINE fform2nd_expr<T>
fform2nd(
const gsGeometryMap<T> & G)
4524{
return fform2nd_expr<T>(G); }
4527template<
class E> EIGEN_STRONG_INLINE
4528jac_expr<E>
jac(
const symbol_expr<E> & u) {
return jac_expr<E>(u); }
4531template<
class T> EIGEN_STRONG_INLINE
4532jac_expr<gsGeometryMap<T> >
jac(
const gsGeometryMap<T> & G) {
return jac_expr<gsGeometryMap<T> >(G);}
4535template<
class T> EIGEN_STRONG_INLINE
4536grad_expr<gsFeSolution<T> >
jac(
const gsFeSolution<T> & s) {
return grad_expr<gsFeSolution<T> >(s);}
4538template<
class E> EIGEN_STRONG_INLINE
4539hess_expr<E> hess(
const symbol_expr<E> & u) {
return hess_expr<E>(u); }
4542template<
class T> EIGEN_STRONG_INLINE
4543hess_expr<gsGeometryMap<T> > hess(
const gsGeometryMap<T> & u) {
return hess_expr<gsGeometryMap<T> >(u); }
4546template<
class T> EIGEN_STRONG_INLINE
4547hess_expr<gsFeSolution<T> > hess(
const gsFeSolution<T> & u) {
return hess_expr<gsFeSolution<T> >(u); }
4550template<
class T> EIGEN_STRONG_INLINE
4551dJacG_expr<T>
dJac(
const gsGeometryMap<T> & G) {
return dJacG_expr<T>(G); }
4554template<
class T> EIGEN_STRONG_INLINE
4555meas_expr<T>
meas(
const gsGeometryMap<T> & G) {
return meas_expr<T>(G); }
4558template <
typename E1,
typename E2> EIGEN_STRONG_INLINE
4559mult_expr<E1,E2>
const operator*(_expr<E1>
const& u, _expr<E2>
const& v)
4560{
return mult_expr<E1, E2>(u, v); }
4562template <
typename E2> EIGEN_STRONG_INLINE
4563mult_expr<typename E2::Scalar,E2,false>
const
4564operator*(
typename E2::Scalar
const& u, _expr<E2>
const& v)
4565{
return mult_expr<typename E2::Scalar, E2, false>(u, v); }
4567template <
typename E1> EIGEN_STRONG_INLINE
4568mult_expr<typename E1::Scalar,E1,false>
const
4569operator*(_expr<E1>
const& v,
typename E1::Scalar
const& u)
4570{
return mult_expr<typename E1::Scalar,E1, false>(u, v); }
4572template <
typename E1> EIGEN_STRONG_INLINE
4573mult_expr<typename E1::Scalar,E1,false>
const
4574operator-(_expr<E1>
const& u)
4575{
return mult_expr<typename E1::Scalar,E1, false>(-1, u); }
4577template <
typename E> mult_expr<constMat_expr, E>
const
4578operator*( gsMatrix<typename E::Scalar>
const& u, _expr<E>
const& v)
4579{
return mult_expr<constMat_expr, E>(mat(u), v); }
4581template <
typename E> mult_expr<E, constMat_expr>
const
4582operator*(_expr<E>
const& u, gsMatrix<typename E::Scalar>
const& v)
4583{
return mult_expr<E, constMat_expr>(u, mat(v) ); }
4586template <
typename E1,
typename E2> EIGEN_STRONG_INLINE
4587frprod_expr<E1,E2>
const operator%(_expr<E1>
const& u, _expr<E2>
const& v)
4588{
return frprod_expr<E1, E2>(u, v); }
4591template <
typename E1,
typename E2> EIGEN_STRONG_INLINE
4592divide_expr<E1,E2>
const operator/(_expr<E1>
const& u, _expr<E2>
const& v)
4593{
return divide_expr<E1,E2>(u, v); }
4595template <
typename E> EIGEN_STRONG_INLINE
4596divide_expr<E,typename E::Scalar>
const
4597operator/(_expr<E>
const& u,
const typename E::Scalar v)
4598{
return divide_expr<E,typename E::Scalar>(u, v); }
4600template <
typename E> EIGEN_STRONG_INLINE
4601divide_expr<typename E::Scalar,E>
const
4602operator/(
const typename E::Scalar u, _expr<E>
const& v)
4603{
return divide_expr<typename E::Scalar,E>(u, v); }
4606template <
typename E1,
typename E2> EIGEN_STRONG_INLINE
4607add_expr<E1,E2>
const operator+(_expr<E1>
const& u, _expr<E2>
const& v)
4608{
return add_expr<E1, E2>(u, v); }
4611template <
typename E> EIGEN_STRONG_INLINE
4612add_expr< E, _expr<typename E::Scalar, true> >
4614{
return add_expr<E,_expr<typename E::Scalar>>(u, _expr<typename E::Scalar,true>(v)); }
4617template <
typename E> EIGEN_STRONG_INLINE
4618add_expr< E, _expr<typename E::Scalar, true> >
4620{
return add_expr<E,_expr<typename E::Scalar>>(u, _expr<typename E::Scalar,true>(v)); }
4623template <
typename E1,
typename E2> EIGEN_STRONG_INLINE
4624summ_expr<E1,E2>
const summ(E1
const & u, E2
const& M)
4625{
return summ_expr<E1,E2>(u, M); }
4629template <
typename E1,
typename E2> EIGEN_STRONG_INLINE
4635template <
typename E1,
typename E2> EIGEN_STRONG_INLINE
4640template <
typename E1,
typename E2> EIGEN_STRONG_INLINE
4641sub_expr<E1,E2>
const operator-(_expr<E1>
const& u, _expr<E2>
const& v)
4642{
return sub_expr<E1, E2>(u, v); }
4644template <
typename E2> EIGEN_STRONG_INLINE
4645sub_expr<_expr<typename E2::Scalar>,E2>
const
4646operator-(
typename E2::Scalar
const& s, _expr<E2>
const& v)
4649 return sub_expr<_expr<typename E2::Scalar>, E2>(_expr<typename E2::Scalar>(s), v);
4655#define GISMO_SHORTCUT_VAR_EXPRESSION(name,impl) template<class E> EIGEN_STRONG_INLINE \
4656 auto name(const E & u) -> decltype(impl) { return impl; }
4657#define GISMO_SHORTCUT_MAP_EXPRESSION(name,impl) template<class T> EIGEN_STRONG_INLINE \
4658 auto name(const gsGeometryMap<T> & G) -> decltype(impl) { return impl; }
4659#define GISMO_SHORTCUT_PHY_EXPRESSION(name,impl) template<class E> EIGEN_STRONG_INLINE \
4660 auto name(const E & u, const gsGeometryMap<typename E::Scalar> & G) -> decltype(impl) { return impl; }
4663GISMO_SHORTCUT_VAR_EXPRESSION( div,
jac(u).trace() )
4664GISMO_SHORTCUT_PHY_EXPRESSION( idiv, ijac(u,G).trace() )
4667GISMO_SHORTCUT_MAP_EXPRESSION(unv,
nv(G).normalized() )
4669GISMO_SHORTCUT_MAP_EXPRESSION(usn, sn(G).normalized() )
4671GISMO_SHORTCUT_PHY_EXPRESSION(igrad,
grad(u)*
jac(G).ginv() )
4672GISMO_SHORTCUT_VAR_EXPRESSION(igrad,
grad(u) )
4674GISMO_SHORTCUT_PHY_EXPRESSION( ijac,
jac(u) *
jac(G).ginv())
4677GISMO_SHORTCUT_PHY_EXPRESSION(ihess,
4678 jac(G).ginv().tr()*( hess(u) -
summ(igrad(u,G),hess(G)) ) *
jac(G).ginv() )
4679GISMO_SHORTCUT_VAR_EXPRESSION(ihess, hess(u) )
4681GISMO_SHORTCUT_PHY_EXPRESSION(ilapl, ihess(u,G).trace() )
4682GISMO_SHORTCUT_VAR_EXPRESSION(ilapl, hess(u).trace() )
4684GISMO_SHORTCUT_VAR_EXPRESSION(fform,
jac(u).tr()*
jac(u) )
4685GISMO_SHORTCUT_VAR_EXPRESSION(shapeop, fform(u).inv() *
fform2nd(u) )
4687#undef GISMO_SHORTCUT_PHY_EXPRESSION
4688#undef GISMO_SHORTCUT_VAR_EXPRESSION
4689#undef GISMO_SHORTCUT_MAP_EXPRESSION
index_t rows() const
Returns the row-size of the expression.
Definition gsExpressions.h:328
mult_expr< real_t, ppart_expr< mult_expr< double, E, false > >, false > npart() const
Returns the expression's negative part.
Definition gsExpressions.h:260
static constexpr bool isVector()
rowSpan && !colSpan
Definition gsExpressions.h:344
max_expr< E > max() const
Returns the rowSum of a matrix.
Definition gsExpressions.h:313
normalized_expr< E > normalized() const
Returns the vector normalized to unit length.
Definition gsExpressions.h:283
det_expr< E > det() const
Returns the determinant of the expression.
Definition gsExpressions.h:287
value_expr< E > val() const
Definition gsExpressions.h:305
temp_expr< E > temp() const
Returns an evaluation of the (sub-)expression in temporary memory.
Definition gsExpressions.h:263
tr_expr< E, true > cwisetr() const
Returns the coordinate-wise transpose of the expression.
Definition gsExpressions.h:237
exp_expr< E > exp() const
Returns exp(expression)
Definition gsExpressions.h:249
ppart_expr< E > ppart() const
Returns the expression's positive part.
Definition gsExpressions.h:253
void parse(gsExprHelper< Scalar > &evList) const
Parse the expression and discover the list of evaluation sources, also sets the required evaluation f...
Definition gsExpressions.h:350
tr_expr< E > tr() const
Returns the transpose of the expression.
Definition gsExpressions.h:233
const gsFeSpace< Scalar > & rowVar() const
Definition gsExpressions.h:358
const gsFeSpace< Scalar > & colVar() const
Definition gsExpressions.h:366
adjugate_expr< E > adj() const
Returns the adjugate of the expression (for matrix-valued expressions)
Definition gsExpressions.h:275
norm_expr< E > norm() const
Returns the Euclidean norm of the expression.
Definition gsExpressions.h:279
sign_expr< E > sgn(Scalar tolerance=0) const
Returns the sign of the expression.
Definition gsExpressions.h:245
bool isScalar() const
Returns true iff the expression is scalar-valued.
Definition gsExpressions.h:342
trace_expr< E > trace() const
Returns the trace of the expression (for matrix-valued expressions)
Definition gsExpressions.h:271
colsum_expr< E > colSum() const
Returns the colSum of a matrix.
Definition gsExpressions.h:321
rowsum_expr< E > rowSum() const
Returns the rowSum of a matrix.
Definition gsExpressions.h:317
index_t cols() const
Returns the column-size of the expression.
Definition gsExpressions.h:332
asdiag_expr< E > asDiag() const
Returns a diagonal matrix expression of the vector expression.
Definition gsExpressions.h:309
MatExprType eval(const index_t k) const
Evaluates the expression at evaluation point indexed by k.
Definition gsExpressions.h:229
void print(std::ostream &os) const
Prints the expression as a string to os.
Definition gsExpressions.h:194
cb_expr< E > cb() const
Returns the puts the expression to colBlocks.
Definition gsExpressions.h:241
inv_expr< E > const inv() const
Returns the inverse of the expression (for matrix-valued expressions)
Definition gsExpressions.h:267
Definition gsExpressions.h:2129
Definition gsExpressions.h:2336
Definition gsExpressions.h:2404
Definition gsExpressions.h:1979
Definition gsExpressions.h:973
Definition gsExpressions.h:928
Definition gsExpressions.h:2305
Definition gsExpressions.h:842
const gsFeElement< Scalar > & _e
Reference to the element.
Definition gsExpressions.h:848
Definition gsExpressions.h:3133
Definition gsExpressions.h:2599
Definition gsExpressions.h:2543
Definition gsExpressions.h:3045
Definition gsExpressions.h:3012
Definition gsExpressions.h:2505
Definition gsExpressions.h:2437
Definition gsExpressions.h:2472
Definition gsExpressions.h:2368
Definition gsExpressions.h:3081
A basis represents a family of scalar basis functions defined over a common parameter domain.
Definition gsBasis.h:79
Class containing a set of boundary conditions.
Definition gsBoundaryConditions.h:342
const_cpliterator coupledBegin() const
Definition gsBoundaryConditions.h:571
const_iterator begin(const std::string &label) const
Returns a const-iterator to the beginning of the Bc container of type label.
Definition gsBoundaryConditions.h:471
const_citerator cornerBegin() const
Definition gsBoundaryConditions.h:561
const_iterator end(const std::string &label) const
Returns a const-iterator to the end of the Bc container of type label.
Definition gsBoundaryConditions.h:477
const_citerator cornerEnd() const
Definition gsBoundaryConditions.h:566
const_cpliterator coupledEnd() const
Definition gsBoundaryConditions.h:576
Maintains a mapping from patch-local dofs to global dof indices and allows the elimination of individ...
Definition gsDofMapper.h:69
void markBoundary(index_t k, const gsMatrix< index_t > &boundaryDofs, index_t comp=0)
Mark the local dofs boundaryDofs of patch k as eliminated.
Definition gsDofMapper.cpp:188
void finalize()
Must be called after all boundaries and interfaces have been marked to set up the dof numbering.
Definition gsDofMapper.cpp:240
void matchDof(index_t u, index_t i, index_t v, index_t j, index_t comp=0)
Couples dof i of patch u with dof j of patch v such that they refer to the same global dof at compone...
Definition gsDofMapper.cpp:99
bool is_free_index(index_t gl) const
Returns true if global dof gl is not eliminated.
Definition gsDofMapper.h:376
size_t mapSize() const
Returns the total number of patch-local degrees of freedom that are being mapped.
Definition gsDofMapper.h:473
bool isFinalized() const
Checks whether finalize() has been called.
Definition gsDofMapper.h:236
void matchDofs(index_t u, const gsMatrix< index_t > &b1, index_t v, const gsMatrix< index_t > &b2, index_t comp=0)
Couples dofs b1 of patch u with dofs b2 of patch v one by one such that they refer to the same global...
Definition gsDofMapper.cpp:159
void setIdentity(index_t nPatches, size_t nDofs, size_t nComp=1)
Set this mapping to be the identity.
Definition gsDofMapper.cpp:364
index_t numComponents() const
Returns the number of components present in the mapper.
Definition gsDofMapper.h:417
void eliminateDof(index_t i, index_t k, index_t comp=0)
Mark the local dof i of patch k as eliminated.
Definition gsDofMapper.cpp:214
index_t global_to_bindex(index_t gl) const
Returns the boundary index of global dof gl.
Definition gsDofMapper.h:364
index_t index(index_t i, index_t k=0, index_t c=0) const
Returns the global dof index associated to local dof i of patch k.
Definition gsDofMapper.h:325
Definition gsExprHelper.h:27
Interface for the set of functions defined on a domain (the total number of functions in the set equa...
Definition gsFunctionSet.h:219
virtual index_t size() const
size
Definition gsFunctionSet.h:613
A matrix with arbitrary coefficient type and fixed or dynamic size.
Definition gsMatrix.h:41
Col3DType col3d(index_t c)
Returns column c as a fixed-size 3D vector.
Definition gsMatrix.h:244
T at(index_t i) const
Returns the i-th element of the vectorization of the matrix.
Definition gsMatrix.h:211
Holds a set of patch-wise bases and their topology information.
Definition gsMultiBasis.h:37
A vector with arbitrary coefficient type and fixed or dynamic size.
Definition gsVector.h:37
#define index_t
Definition gsConfig.h:32
#define GISMO_ERROR(message)
Definition gsDebug.h:118
#define gsWarn
Definition gsDebug.h:50
#define GISMO_UNUSED(x)
Definition gsDebug.h:112
#define GISMO_ENSURE(cond, message)
Definition gsDebug.h:102
#define GISMO_ASSERT(cond, message)
Definition gsDebug.h:89
The functions compute Dirichlet degrees of freedom using various methods.
This object is a cache for computed values from an evaluator.
Provides declaration of Basis abstract interface.
Matrix< Scalar, Dynamic, Dynamic > cramerInverse() const
Inversion for small matrices using Cramer's Rule.
Definition gsMatrixAddons.h:55
EIGEN_STRONG_INLINE idMat_expr id(const index_t dim)
The identity matrix of dimension dim.
Definition gsExpressions.h:4470
EIGEN_STRONG_INLINE frprod_expr< E1, E2 > const operator%(_expr< E1 > const &u, _expr< E2 > const &v)
Frobenious product (also known as double dot product) operator for expressions.
Definition gsExpressions.h:4587
EIGEN_STRONG_INLINE add_expr< E1, E2 > const operator+(_expr< E1 > const &u, _expr< E2 > const &v)
Addition operator for expressions.
Definition gsExpressions.h:4607
EIGEN_STRONG_INLINE reshape_expr< E > const reshape(E const &u, index_t n, index_t m)
Reshape an expression.
Definition gsExpressions.h:1925
EIGEN_STRONG_INLINE abs_expr< E > abs(const E &u)
Absolute value.
Definition gsExpressions.h:4486
EIGEN_STRONG_INLINE fform2nd_expr< T > fform2nd(const gsGeometryMap< T > &G)
The second fundamental form of G.
Definition gsExpressions.h:4523
EIGEN_STRONG_INLINE tangent_expr< T > tv(const gsGeometryMap< T > &u)
The tangent boundary vector of a geometry map in 2D.
Definition gsExpressions.h:4513
EIGEN_STRONG_INLINE matrix_by_space_expr_tr< E1, E2 > const matrix_by_space_tr(E1 const &u, E2 const &v)
Definition gsExpressions.h:4636
EIGEN_STRONG_INLINE meas_expr< T > meas(const gsGeometryMap< T > &G)
The measure of a geometry map.
Definition gsExpressions.h:4555
EIGEN_STRONG_INLINE grad_expr< E > grad(const E &u)
The gradient of a variable.
Definition gsExpressions.h:4490
EIGEN_STRONG_INLINE replicate_expr< E > const replicate(E const &u, index_t n, index_t m=1)
Replicate an expression.
Definition gsExpressions.h:1967
EIGEN_STRONG_INLINE onormal_expr< T > nv(const gsGeometryMap< T > &u)
The (outer pointing) boundary normal of a geometry map.
Definition gsExpressions.h:4506
EIGEN_STRONG_INLINE dJacG_expr< T > dJac(const gsGeometryMap< T > &G)
The partial derivatives of the Jacobian matrix of a geometry map.
Definition gsExpressions.h:4551
EIGEN_STRONG_INLINE diag_expr< E > const diagonal(E const &u)
Get diagonal elements of matrix as a vector.
Definition gsExpressions.h:2082
EIGEN_STRONG_INLINE curl_expr< T > curl(const gsFeVariable< T > &u)
The curl of a finite element variable.
Definition gsExpressions.h:4498
EIGEN_STRONG_INLINE summ_expr< E1, E2 > const summ(E1 const &u, E2 const &M)
Matrix-summation operator for expressions.
Definition gsExpressions.h:4624
EIGEN_STRONG_INLINE dJacdc_expr< E > dJacdc(const E &u, index_t c)
The derivative of the jacobian of a geometry map with respect to a coordinate.
Definition gsExpressions.h:4494
EIGEN_STRONG_INLINE matrix_by_space_expr< E1, E2 > const matrix_by_space(E1 const &u, E2 const &v)
Definition gsExpressions.h:4630
nabla2_expr< T > nabla2(const gsFeVariable< T > &u)
The nabla2 ( ) of a finite element variable.
Definition gsExpressions.h:3003
EIGEN_STRONG_INLINE nabla_expr< T > nabla(const gsFeVariable< T > &u)
The nabla ( ) of a finite element variable.
Definition gsExpressions.h:4502
std::ostream & operator<<(std::ostream &os, const _expr< E > &b)
Stream operator for expressions.
Definition gsExpressions.h:382
EIGEN_STRONG_INLINE flat_expr< E > const flat(E const &u)
Make a matrix 2x2 expression "flat".
Definition gsExpressions.h:2031
EIGEN_STRONG_INLINE divide_expr< E1, E2 > const operator/(_expr< E1 > const &u, _expr< E2 > const &v)
Scalar division operator for expressions.
Definition gsExpressions.h:4592
EIGEN_STRONG_INLINE jac_expr< E > jac(const symbol_expr< E > &u)
The Jacobian matrix of a FE variable.
Definition gsExpressions.h:4528
EIGEN_STRONG_INLINE mult_expr< E1, E2 > const operator*(_expr< E1 > const &u, _expr< E2 > const &v)
Multiplication operator for expressions.
Definition gsExpressions.h:4559
The G+Smo namespace, containing all definitions for the library.
@ NEED_LAPLACIAN
Laplacian.
Definition gsForwardDeclarations.h:83
@ NEED_VALUE
Value of the object.
Definition gsForwardDeclarations.h:72
@ NEED_DERIV2
Second derivatives.
Definition gsForwardDeclarations.h:80
@ NEED_DERIV
Gradient of the object.
Definition gsForwardDeclarations.h:73
@ NEED_GRAD_TRANSFORM
Gradient transformation matrix.
Definition gsForwardDeclarations.h:77
@ NEED_NORMAL
Normal vector of the object.
Definition gsForwardDeclarations.h:85
@ NEED_MEASURE
The density of the measure pull back.
Definition gsForwardDeclarations.h:76
@ NEED_OUTER_NORMAL
Outward normal on the boundary.
Definition gsForwardDeclarations.h:86
@ NEED_ACTIVE
Active function ids.
Definition gsForwardDeclarations.h:84
@ NEED_2ND_FFORM
Second fundamental form.
Definition gsForwardDeclarations.h:87
S give(S &x)
Definition gsMemory.h:266
Defines expression precomputed_expr.
Struct containing information for matrix assembly.
Definition gsExpressions.h:64