PSCF v1.2
UnitCellBase.h
1#ifndef PRDC_UNIT_CELL_BASE_H
2#define PRDC_UNIT_CELL_BASE_H
3
4/*
5* PSCF - Polymer Self-Consistent Field Theory
6*
7* Copyright 2016 - 2022, The Regents of the University of Minnesota
8* Distributed under the terms of the GNU General Public License.
9*/
10
11#include <util/signal/Signal.h>
12#include <util/containers/FArray.h>
13#include <util/containers/FSArray.h>
14#include <util/containers/FMatrix.h>
15#include <pscf/math/IntVec.h>
16#include <pscf/math/RealVec.h>
17#include <util/math/Constants.h>
18
19namespace Pscf {
20namespace Prdc {
21
22 using namespace Util;
23
29 template <int D>
31 {
32 public:
33
38
43
46
55
59 int nParameter() const;
60
65
71 double parameter(int i) const;
72
79 bool isInitialized() const;
80
84
90 const RealVec<D>& rBasis(int i) const;
91
97 const RealVec<D>& kBasis(int i) const;
98
106 double drBasis(int k, int i, int j) const;
107
115 double dkBasis(int k, int i, int j) const;
116
124 double drrBasis(int k, int i, int j) const;
125
133 double dkkBasis(int k, int i, int j) const;
134
138
144 virtual double ksq(IntVec<D> const & k) const;
145
156 virtual double dksq(IntVec<D> const & vec, int n) const;
157
161
200 template <class Observer>
201 void addObserver(Observer& observer, void (Observer::*methodPtr)());
202
207
215 int nObserver() const;
216
218
219 protected:
220
225
230
238
246
255
264
269
274
279
292
293 private:
294
298 Signal<void>* signalPtr_;
299
303 bool hasSignal() const;
304
311 void initializeToZero();
312
320 virtual void setBasis() = 0;
321
325 void computeDerivatives();
326
330 UnitCellBase(UnitCellBase<D> const & other);
331
335 UnitCellBase<D>& operator = (UnitCellBase<D> const & other);
336
337 };
338
339 /*
340 * Get the number of Parameters in the unit cell.
341 */
342 template <int D>
343 inline
345 { return nParameter_; }
346
347 /*
348 * Get the unit cell parameters.
349 */
350 template <int D>
351 inline
353 {
354 FSArray<double, 6> parameters;
355 for (int i = 0; i < nParameter_; ++i) {
356 parameters.append(parameters_[i]);
357 }
358 return parameters;
359 }
360
361 /*
362 * Get a single parameter of the unit cell.
363 */
364 template <int D>
365 inline
366 double UnitCellBase<D>::parameter(int i) const
367 { return parameters_[i]; }
368
369 /*
370 * Get Bravais basis vector i.
371 */
372 template <int D>
373 inline
375 { return rBasis_[i]; }
376
377 /*
378 * Get reciprocal basis vector i.
379 */
380 template <int D>
381 inline
383 { return kBasis_[i]; }
384
385 /*
386 * Get component j of derivative of r basis vector i w/respect to param k.
387 */
388 template <int D>
389 inline
390 double UnitCellBase<D>::drBasis(int k, int i, int j) const
391 { return drBasis_[k](i,j); }
392
393 /*
394 * Get component j of derivative of r basis vector i w/respect to param k.
395 */
396 template <int D>
397 inline
398 double UnitCellBase<D>::dkBasis(int k, int i, int j) const
399 { return dkBasis_[k](i, j); }
400
401 /*
402 * Get the derivative of ki*kj with respect to parameter k.
403 */
404 template <int D>
405 inline
406 double UnitCellBase<D>::dkkBasis(int k, int i, int j) const
407 { return dkkBasis_[k](i, j); }
408
409 /*
410 * Get the derivative of ri*rj with respect to parameter k.
411 */
412 template <int D>
413 inline
414 double UnitCellBase<D>::drrBasis(int k, int i, int j) const
415 { return drrBasis_[k](i, j); }
416
417 /*
418 * Is this unit cell initialized?
419 */
420 template <int D>
421 inline
423 { return isInitialized_; }
424
425 // Non-inline member functions
426
427 /*
428 * Constructor.
429 */
430 template <int D>
432 : nParameter_(0),
433 isInitialized_(false),
434 signalPtr_(nullptr)
435 { initializeToZero(); }
436
437 /*
438 * Destructor.
439 */
440 template <int D>
442 {
443 if (hasSignal()) {
444 delete signalPtr_;
445 }
446 }
447
448 /*
449 * Set all the parameters in the unit cell.
450 */
451 template <int D>
453 {
454 UTIL_CHECK(parameters.size() == nParameter_);
455 isInitialized_ = false;
456 for (int i = 0; i < nParameter_; ++i) {
457 parameters_[i] = parameters[i];
458 }
459 setLattice();
460 }
461
462 /*
463 * Get square magnitude of reciprocal basis vector.
464 */
465 template <int D>
466 double UnitCellBase<D>::ksq(IntVec<D> const & k) const
467 {
468 RealVec<D> g(0.0);
469 RealVec<D> p;
470 for (int i = 0; i < D; ++i) {
471 p.multiply(kBasis_[i], k[i]);
472 g += p;
473 }
474 double value = 0.0;
475 for (int i = 0; i < D; ++i) {
476 value += g[i]*g[i];
477 }
478 return value;
479 }
480
481 /*
482 * Get magnitude of derivative of square of reciprocal basis vector.
483 */
484 template <int D>
485 double UnitCellBase<D>::dksq(IntVec<D> const & vec, int n) const
486 {
487 double element = 0.0;
488 double value = 0.0;
489
490 for (int p = 0; p < D; ++p){
491 for (int q = 0; q < D; ++q){
492 element = dkkBasis(n, p, q);
493 value += vec[p]*vec[q]*element;
494 }
495 }
496
497 return value;
498 }
499
500 // Functions to manage observers
501
502 template <int D>
503 template <class Observer>
505 void (Observer::*methodPtr)())
506 {
507 if (!hasSignal()) {
508 signalPtr_ = new Signal<void>;
509 }
510 signalPtr_->addObserver(observer, methodPtr);
511 }
512
513 template <int D>
515 {
516 if (hasSignal()) {
517 signalPtr_->clear();
518 delete signalPtr_;
519 signalPtr_ = nullptr;
520 }
521 }
522
523 template <int D>
525 {
526 if (hasSignal()) {
527 return signalPtr_->nObserver();
528 } else {
529 return 0;
530 }
531 }
532
533 // Protected member functions
534
535 /*
536 * Initialize internal arrays to zero.
537 */
538 template <int D>
540 {
541 // Initialize all elements to zero
542 int i, j, k;
543 for (i = 0; i < D; ++i) {
544 for (j = 0; j < D; ++j) {
545 rBasis_[i][j] = 0.0;
546 kBasis_[i][j] = 0.0;
547 }
548 }
549 for (k = 0; k < 6; ++k){
550 for (i = 0; i < D; ++i) {
551 for (j = 0; j < D; ++j) {
552 drBasis_[k](i,j) = 0.0;
553 dkBasis_[k](i,j) = 0.0;
554 drrBasis_[k](i,j) = 0.0;
555 dkkBasis_[k](i,j) = 0.0;
556 }
557 }
558 }
559 }
560
561 /*
562 * Compute quantities involving derivatives.
563 */
564 template <int D>
565 void UnitCellBase<D>::computeDerivatives()
566 {
567 // Compute dkBasis
568 int p, q, r, s, t;
569 for (p = 0; p < nParameter_; ++p) {
570 for (q = 0; q < D; ++q) {
571 for (r = 0; r < D; ++r) {
572
573 // Loop over free indices s, t
574 for (s = 0; s < D; ++s) {
575 for (t = 0; t < D; ++t) {
576 dkBasis_[p](q,r)
577 -= kBasis_[q][s]*drBasis_[p](t,s)*kBasis_[t][r];
578 }
579 }
580 dkBasis_[p](q,r) /= 2.0*Constants::Pi;
581
582 }
583 }
584 }
585
586 // Compute drrBasis and dkkBasis
587 for (p = 0; p < nParameter_; ++p) {
588 for (q = 0; q < D; ++q) {
589 for (r = 0; r < D; ++r) {
590 for (s = 0; s < D; ++s) {
591 drrBasis_[p](q,r) += rBasis_[q][s]*drBasis_[p](r,s);
592 drrBasis_[p](q,r) += rBasis_[r][s]*drBasis_[p](q,s);
593 dkkBasis_[p](q,r) += kBasis_[q][s]*dkBasis_[p](r,s);
594 dkkBasis_[p](q,r) += kBasis_[r][s]*dkBasis_[p](q,s);
595
596 }
597 }
598 }
599 }
600
601 }
602
603 /*
604 * Set all lattice parameters.
605 */
606 template <int D>
608 {
609 initializeToZero();
610 setBasis();
611 computeDerivatives();
612 isInitialized_ = true;
613 if (hasSignal()) {
614 signalPtr_->notify();
615 }
616 }
617
618 template <int D>
619 bool UnitCellBase<D>::hasSignal() const
620 { return bool(signalPtr_); }
621
622}
623}
624#endif
An IntVec<D, T> is a D-component vector of elements of integer type T.
Definition IntVec.h:27
Base class template for a crystallographic unit cell.
bool isInitialized() const
Has this unit cell been initialized?
bool isInitialized_
Has this unit cell been fully initialized?
const RealVec< D > & rBasis(int i) const
Get Bravais basis vector i, denoted by a_i.
void setParameters(FSArray< double, 6 > const &parameters)
Set all the parameters of unit cell.
void addObserver(Observer &observer, void(Observer::*methodPtr)())
Add an Observer to this unit cell.
const RealVec< D > & kBasis(int i) const
Get reciprocal basis vector i, denoted by b_i.
double drrBasis(int k, int i, int j) const
Get derivative of dot product ai.aj with respect to parameter k.
FArray< RealVec< D >, D > rBasis_
Array of Bravais lattice basis vectors.
FArray< FMatrix< double, D, D >, 6 > drBasis_
Array of derivatives of rBasis.
double dkkBasis(int k, int i, int j) const
Get derivative of dot product bi.bj with respect to parameter k.
FArray< RealVec< D >, D > kBasis_
Array of reciprocal lattice basis vectors.
FSArray< double, 6 > parameters() const
Get the parameters of this unit cell.
double drBasis(int k, int i, int j) const
Get component j of derivative of rBasis vector a_i w/respect to k.
FArray< double, 6 > parameters_
Parameters used to describe the unit cell.
void clearObservers()
Clear all observers and delete the Signal<> object.
virtual double ksq(IntVec< D > const &k) const
Compute square magnitude of reciprocal lattice vector.
FArray< FMatrix< double, D, D >, 6 > dkBasis_
Array of derivatives of kBasis.
FArray< FMatrix< double, D, D >, 6 > drrBasis_
Array of derivatives of a_i.a_j.
int nObserver() const
Return the current number of observers.
double dkBasis(int k, int i, int j) const
Get component j of derivative of kBasis vector b_i w/respect to k.
void setLattice()
Compute all protected data, given latticeSystem and parameters.
virtual double dksq(IntVec< D > const &vec, int n) const
Compute derivative of square wavevector w/respect to cell parameter.
int nParameter_
Number of parameters required to specify unit cell.
double parameter(int i) const
Get a single parameter of this unit cell.
FArray< FMatrix< double, D, D >, 6 > dkkBasis_
Array of derivatives of b_i.b_j.
int nParameter() const
Get the number of parameters in the unit cell.
A RealVec<D, T> is D-component vector with elements of floating type T.
Definition RealVec.h:28
Vec< D, T > & multiply(const Vec< D, T > &v, T s)
Multiply a vector v by a scalar s.
Definition Vec.h:505
static const double Pi
Trigonometric constant Pi.
Definition Constants.h:35
A fixed size (static) contiguous array template.
Definition FArray.h:47
A fixed capacity (static) contiguous array with a variable logical size.
Definition rpg/System.h:28
int size() const
Return logical size of this array (i.e., number of elements).
Definition FSArray.h:207
void append(Data const &data)
Append data to the end of the array.
Definition FSArray.h:258
Abstract class template for observer in the observer design pattern.
Definition Observer.h:25
Notifier (or subject) in the Observer design pattern.
Definition Signal.h:39
#define UTIL_CHECK(condition)
Assertion macro suitable for serial or parallel production code.
Definition global.h:68
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.