Analytic Dynamic Tensor

By forming the dynamic tensor analytically at a given Q, we not only can easily determine how to execute first-principle calculations to obtain the actual dynamic tensor most efficiently, but also are capable of performing analytic analysis with the tensor without the need of any first-principles calculations at all, using solely group theory.

This version of the analytic tensor uses the little group analysis method, it does not accounts for all space group symmetry but only a subset. This method is easy to execute and serves as the foundation of the full-group approach.

The basic structure of the code is sound, albeit the renaming of some attributes is required. There is a working version that uses sparse tensor instead of numpy tensor, this should be considered as an alternative feature to save memory, though some linear algebra operations on the sparse tensor (e.g. SVD) might not be available right now.

Here we will show examples for the analytic tensor for both little group approach and the current implementation of full group approach.

[1]:
from analytic_dynamic_tensor import analytic_dynamic_tensor
from sym_analytic_dynamic_tensor import sym_analytic_dynamic_tensor
from coordinate_tools import coord_to_string as c2s

Our example will use graphene as our sample system.

[2]:
poscar = """\
graphene
   1.00000000000000
     2.1217113285639000    1.2249706066897641    0.0000000000000000
     2.1217113285639000   -1.2249706066897641    0.0000000000000000
     0.0000000000000000    0.0000000000000000  -15.0000000000000000
   2
Direct
  0.00000000  0.00000000  0.0000000000000000 C:p
  0.33333333  0.33333333  0.0000000000000000 C:p
""".strip()

Little group approach

[3]:
adt = analytic_dynamic_tensor(
    pos=poscar, # primitive structure
    pgn='D6h', # point group
    pg_loc='1/3 1/3 0', # The site of the symmetry
    kset='0 0 0; 0 0 0', # The Q of interest (the number of q-vectors equals to the order of the derivtive)
)
adt
((0, 0, 0), (0, 0, 0))
[3]:
analytic_dynamic_tensor(graphene
   1.00000000000000
     2.1217113285639000    1.2249706066897641    0.0000000000000000
     2.1217113285639000   -1.2249706066897641    0.0000000000000000
     0.0000000000000000    0.0000000000000000  -15.0000000000000000
   2
Direct
  0.00000000  0.00000000  0.0000000000000000 C:p
  0.33333333  0.33333333  0.0000000000000000 C:p, D6h, pg_loc=1/3 1/3 0, kset=((0, 0, 0), (0, 0, 0)))

The vectorized analytic tensor in the symmetrized basis is stored in the vtensor attribute, and the list of irreducible derivatives the tensor represents is store in globalvar attribute. One can change the globalvar list, and _vtensor attribute needs to be deleted to create the updated vtensor that uses the update list.

The tensor in naive basis is stored in rotated_tensor.

[4]:
print map(c2s, adt.globalvar)
print adt.vtensor.shape
print adt.rotated_tensor.shape
['((((0, 0, 0), B2g), ((0, 0, 0), B2g)), (A1g, 0))', '((((0, 0, 0), E2g), ((0, 0, 0), E2g)), (A1g, 0))']
(6, 6, 2)
(6, 6, 2)
[5]:
for i in range(len(adt.globalvar)):
    print "The dynamic tensor in symmetrized basis with the contribution of the irreducible derivative:"
    print " ", c2s(adt.globalvar[i])
    print adt.vtensor[:, :, i]
The dynamic tensor in symmetrized basis with the contribution of the irreducible derivative:
  ((((0, 0, 0), B2g), ((0, 0, 0), B2g)), (A1g, 0))
[[1.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j]
 [0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j]
 [0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j]
 [0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j]
 [0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j]
 [0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j]]
The dynamic tensor in symmetrized basis with the contribution of the irreducible derivative:
  ((((0, 0, 0), E2g), ((0, 0, 0), E2g)), (A1g, 0))
[[0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j]
 [0.+0.j 1.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j]
 [0.+0.j 0.+0.j 1.+0.j 0.+0.j 0.+0.j 0.+0.j]
 [0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j]
 [0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j]
 [0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j]]
[6]:
for i in range(len(adt.globalvar)):
    print "The dynamic tensor in naive basis with the contribution of the irreducible derivative:"
    print " ", c2s(adt.globalvar[i])
    print adt.rotated_tensor[:, :, i]
The dynamic tensor in naive basis with the contribution of the irreducible derivative:
  ((((0, 0, 0), B2g), ((0, 0, 0), B2g)), (A1g, 0))
[[ 0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j]
 [ 0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j]
 [ 0. +0.j  0. +0.j  0.5+0.j  0. +0.j  0. +0.j -0.5+0.j]
 [ 0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j]
 [ 0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j]
 [ 0. +0.j  0. +0.j -0.5+0.j  0. +0.j  0. +0.j  0.5+0.j]]
The dynamic tensor in naive basis with the contribution of the irreducible derivative:
  ((((0, 0, 0), E2g), ((0, 0, 0), E2g)), (A1g, 0))
[[ 0.5+0.j  0. +0.j  0. +0.j -0.5+0.j  0. +0.j  0. +0.j]
 [ 0. +0.j  0.5+0.j  0. +0.j  0. +0.j -0.5+0.j  0. +0.j]
 [ 0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j]
 [-0.5+0.j  0. +0.j  0. +0.j  0.5+0.j  0. +0.j  0. +0.j]
 [ 0. +0.j -0.5+0.j  0. +0.j  0. +0.j  0.5+0.j  0. +0.j]
 [ 0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j]]

Full group approach

Almost everything except the symmetry analysis is the same as the little group approach.

One small difference is that in the symmetrized basis, the full group approach does not keep placeholder for the removed acoustic does, contrary to the little group apprach. There is no significant difference either way, but it’s a good experiment to have both implemented. Thus choice which will go to the new version can be discussed.

[7]:
sadt = sym_analytic_dynamic_tensor(
    pos=poscar, # primitive structure
    pgn='D6h', # point group
    pg_loc='1/3 1/3 0', # The site of the symmetry
    kset='0 0 0; 0 0 0', # The Q of interest (the number of q-vectors equals to the order of the derivtive)
)
sadt
((0, 0, 0), (0, 0, 0))
[7]:
sym_analytic_dynamic_tensor(graphene
   1.00000000000000
     2.1217113285639000    1.2249706066897641    0.0000000000000000
     2.1217113285639000   -1.2249706066897641    0.0000000000000000
     0.0000000000000000    0.0000000000000000  -15.0000000000000000
   2
Direct
  0.00000000  0.00000000  0.0000000000000000 C:p
  0.33333333  0.33333333  0.0000000000000000 C:p, D6h, pg_loc=1/3 1/3 0, kset=((0, 0, 0), (0, 0, 0)))
[8]:
print map(c2s, sadt.globalvar)
print

for i in range(len(sadt.globalvar)):
    print "The dynamic tensor in symmetrized basis with the contribution of the irreducible derivative:"
    print " ", c2s(sadt.globalvar[i])
    print sadt.vtensor[:, :, i]
    print

for i in range(len(sadt.globalvar)):
    print "The dynamic tensor in naive basis with the contribution of the irreducible derivative:"
    print " ", c2s(sadt.globalvar[i])
    print sadt.rotated_tensor[:, :, i]
    print
['((((0, 0, 0), (B2g, ((B2g, 0), 0))), ((0, 0, 0), (B2g, ((B2g, 0), 0)))), (A1g, 0))', '((((0, 0, 0), (E2g, ((E2g, 0), 0))), ((0, 0, 0), (E2g, ((E2g, 0), 0)))), (A1g, 0))']

The dynamic tensor in symmetrized basis with the contribution of the irreducible derivative:
  ((((0, 0, 0), (B2g, ((B2g, 0), 0))), ((0, 0, 0), (B2g, ((B2g, 0), 0)))), (A1g, 0))
[[1.+0.j 0.+0.j 0.+0.j]
 [0.+0.j 0.+0.j 0.+0.j]
 [0.+0.j 0.+0.j 0.+0.j]]

The dynamic tensor in symmetrized basis with the contribution of the irreducible derivative:
  ((((0, 0, 0), (E2g, ((E2g, 0), 0))), ((0, 0, 0), (E2g, ((E2g, 0), 0)))), (A1g, 0))
[[0.+0.j 0.+0.j 0.+0.j]
 [0.+0.j 1.+0.j 0.+0.j]
 [0.+0.j 0.+0.j 1.+0.j]]

The dynamic tensor in naive basis with the contribution of the irreducible derivative:
  ((((0, 0, 0), (B2g, ((B2g, 0), 0))), ((0, 0, 0), (B2g, ((B2g, 0), 0)))), (A1g, 0))
[[ 0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j]
 [ 0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j]
 [ 0. +0.j  0. +0.j  0.5+0.j  0. +0.j  0. +0.j -0.5+0.j]
 [ 0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j]
 [ 0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j]
 [ 0. +0.j  0. +0.j -0.5+0.j  0. +0.j  0. +0.j  0.5+0.j]]

The dynamic tensor in naive basis with the contribution of the irreducible derivative:
  ((((0, 0, 0), (E2g, ((E2g, 0), 0))), ((0, 0, 0), (E2g, ((E2g, 0), 0)))), (A1g, 0))
[[ 0.5+0.j  0. +0.j  0. +0.j -0.5+0.j  0. +0.j  0. +0.j]
 [ 0. +0.j  0.5+0.j  0. +0.j  0. +0.j -0.5+0.j  0. +0.j]
 [ 0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j]
 [-0.5+0.j  0. +0.j  0. +0.j  0.5+0.j  0. +0.j  0. +0.j]
 [ 0. +0.j -0.5+0.j  0. +0.j  0. +0.j  0.5+0.j  0. +0.j]
 [ 0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j  0. +0.j]]