Skip to content

Commit

Permalink
port tetragono to support tat v0.4
Browse files Browse the repository at this point in the history
  • Loading branch information
hzhangxyz committed Nov 22, 2023
1 parent 9082d96 commit c4933b7
Show file tree
Hide file tree
Showing 7 changed files with 74 additions and 93 deletions.
29 changes: 14 additions & 15 deletions tetragono/tetragono/abstract_state.py
Original file line number Diff line number Diff line change
Expand Up @@ -217,7 +217,8 @@ def Edge(self):
type
The edge type of this abstract state.
"""
return self.Tensor.model.Edge
# Get the compat edge constructor.
return self.Tensor.__self__.symmetry.Edge

@property
def Symmetry(self):
Expand All @@ -229,7 +230,8 @@ def Symmetry(self):
type
The symmetry type of this abstract state.
"""
return self.Tensor.model.Symmetry
# Group all input into a tuple
return staticmethod(lambda *args: args[0] if args and isinstance(args[0], tuple) else args)

def _v2_to_v3_rename(self, state):
"""
Expand Down Expand Up @@ -328,10 +330,8 @@ def _construct_symmetry(self, value):
Symmetry
The result symmetry object.
"""
if isinstance(value, self.Symmetry):
return value
else:
return self.Symmetry(value)
# Symmetry is nothing but a tuple wrapper
return self.Symmetry(value)

@property
def total_symmetry(self):
Expand Down Expand Up @@ -367,7 +367,7 @@ def _total_symmetry_edge(self):
Edge
The result virtual edge.
"""
return self.Edge([(-self._total_symmetry, 1)], False)
return self.Edge([(self._total_symmetry, 1)], False).conjugate()

def _construct_edge(self, value):
"""
Expand All @@ -383,10 +383,8 @@ def _construct_edge(self, value):
Edge
The result edge object.
"""
if isinstance(value, self.Edge):
return value
else:
return self.Edge(value)
# Edge accept argument with type Edge. So previous condition is not needed.
return self.Edge(value)

def _construct_physics_edge(self, edge):
"""
Expand Down Expand Up @@ -444,13 +442,14 @@ def _set_hamiltonian(self, points, tensor):
The hamiltonian tensor.
"""
body = len(points)
if not isinstance(tensor, self.Tensor):
raise TypeError("Wrong hamiltonian type")
# Do not check hamiltonian type temporarily
#if not isinstance(tensor, self.Tensor):
# raise TypeError("Wrong hamiltonian type")
if {f"{i}" for i in tensor.names} != {f"{i}{j}" for i in ["I", "O"] for j in range(body)}:
raise ValueError("Wrong hamiltonian name")
for i in range(body):
edge_out = tensor.edges(f"O{i}")
edge_in = tensor.edges(f"I{i}")
edge_out = tensor.edge_by_name(f"O{i}")
edge_in = tensor.edge_by_name(f"I{i}")
if edge_out != self.physics_edges[points[i]]:
raise ValueError("Wrong hamiltonian edge")
if edge_out.conjugated() != edge_in:
Expand Down
20 changes: 12 additions & 8 deletions tetragono/tetragono/common_tensor/No.py
Original file line number Diff line number Diff line change
Expand Up @@ -21,19 +21,23 @@

Tensor = TAT.No.Z.Tensor

identity = Tensor(["I0", "O0"], [2, 2])
identity.blocks[identity.names] = [[1, 0], [0, 1]]
identity = Tensor(["I0", "O0"], [2, 2]).zero()
identity[{"I0": 0, "O0": 0}] = 1
identity[{"I0": 1, "O0": 1}] = 1

pauli_x = Tensor(["I0", "O0"], [2, 2])
pauli_x.blocks[pauli_x.names] = [[0, 1], [1, 0]]
pauli_x = Tensor(["I0", "O0"], [2, 2]).zero()
pauli_x[{"I0": 0, "O0": 1}] = 1
pauli_x[{"I0": 1, "O0": 0}] = 1
Sx = pauli_x / 2

pauli_y = Tensor(["I0", "O0"], [2, 2])
pauli_y.blocks[pauli_y.names] = [[0, -1j], [1j, 0]]
pauli_y = Tensor(["I0", "O0"], [2, 2]).zero()
pauli_y[{"I0": 0, "O0": 1}] = -1j
pauli_y[{"I0": 1, "O0": 0}] = +1j
Sy = pauli_y / 2

pauli_z = Tensor(["I0", "O0"], [2, 2])
pauli_z.blocks[pauli_z.names] = [[1, 0], [0, -1]]
pauli_z = Tensor(["I0", "O0"], [2, 2]).zero()
pauli_z[{"I0": 0, "O0": 0}] = +1
pauli_z[{"I0": 1, "O0": 1}] = -1
Sz = pauli_z / 2

pauli_x_pauli_x = kronecker_product(
Expand Down
5 changes: 4 additions & 1 deletion tetragono/tetragono/sampling_lattice/lattice.py
Original file line number Diff line number Diff line change
Expand Up @@ -313,7 +313,10 @@ def _get_shrinker(self, l1l2, configuration):
# P side is dimension one edge
# Q side is connected to lattice
shrinker = self.Tensor(["P", "Q"], [[(symmetry, 1)], edge.conjugated()]).zero()
shrinker[{"Q": (-symmetry, index), "P": (symmetry, 0)}] = 1
shrinker[{
"Q": (tuple(sym if isinstace(sym, bool) else -sym for sym in symmetry), index),
"P": (symmetry, 0)
}] = 1
yield orbit, shrinker

def _shrink_configuration(self, l1l2, configuration):
Expand Down
27 changes: 14 additions & 13 deletions tetragono/tetragono/sampling_lattice/observer.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,8 @@

import os
import numpy as np
import PyScalapack
# PyScalapack support should be dropped, I think...
#import PyScalapack
from ..utility import (show, showln, allreduce_lattice_buffer, allreduce_buffer, allreduce_number, bcast_buffer,
lattice_update, lattice_prod_sum, lattice_conjugate, mpi_rank, mpi_size, mpi_comm, pickle)
from ..tensor_element import tensor_element
Expand Down Expand Up @@ -274,8 +275,10 @@ def add_observer(self, name, observers):
if self._start:
raise RuntimeError("Cannot enable hole after sampling start")
for positions, observer in observers.items():
if not isinstance(observer, self.owner.Tensor):
raise TypeError("Wrong observer type")
pass
# Do not check the type of observer temporarily
#if not isinstance(observer, self.owner.Tensor):
# raise TypeError("Wrong observer type")
self._observer[name] = observers

def add_energy(self):
Expand Down Expand Up @@ -398,7 +401,7 @@ def __call__(self, possibility, configuration):
# Es should be complex here when calculating gradient
if name == "energy" and self._enable_gradient:
Es = whole_name_value
if self.owner.Tensor.is_real:
if self.owner[0, 0].is_real:
Es = Es.real
holes = configuration.holes() # <psi|s|partial_x psi> / <psi|s|psi>
for l1, l2 in self.owner.sites():
Expand Down Expand Up @@ -519,7 +522,7 @@ def total_energy(self):

def _total_energy_with_imaginary_part(self):
name = "energy"
if self.owner.Tensor.is_complex:
if self.owner[0, 0].is_complex:
return (self._whole_result_reweight[name] + self._total_imaginary_energy_reweight * 1j) / self._total_weight
else:
return self._whole_result_reweight[name] / self._total_weight
Expand Down Expand Up @@ -592,17 +595,15 @@ def natural_gradient_by_conjugate_gradient(self, step, error):
energy = self._total_energy_with_imaginary_part()
delta = self._delta_to_array(self._Delta) / self._total_weight

dtype = np.dtype(self.owner.Tensor.dtype)
btype = self.owner.Tensor.btype

Delta = []
Energy = []
for reweight_s, energy_s, delta_s in self._weights_and_deltas():
param = (reweight_s / self._total_weight)**(1 / 2)
param = param.item()
Delta.append((self._delta_to_array(delta_s) - delta) * param)
Energy.append((energy_s.conjugate() - energy) * param)
Delta = np.asarray(Delta, dtype=dtype)
Energy = np.asarray(Energy, dtype=dtype)
Delta = np.asarray(Delta)
Energy = np.asarray(Energy)

# A x = b
# DT r D x = DT r E
Expand Down Expand Up @@ -663,8 +664,8 @@ def _delta_to_array(self, delta):
# Both delta and result array is in bra space
result = []
for l1, l2 in self.owner.sites():
result.append(delta[l1][l2].transpose(self._Delta[l1][l2].names).storage)
result = np.concatenate(result, dtype=self.owner.Tensor.dtype)
result.append(delta[l1][l2].transpose(self._Delta[l1][l2].names).copy().storage)
result = np.concatenate(result, dtype=result[0].dtype)
return result

def _array_to_delta(self, array):
Expand All @@ -675,7 +676,7 @@ def _array_to_delta(self, array):
for row in result:
for tensor in row:
size = len(tensor.storage)
tensor.storage = array[index:index + size]
tensor.storage[:] = array[index:index + size]
index += size
return result

Expand Down
7 changes: 1 addition & 6 deletions tetragono/tetragono/sampling_lattice/sampling.py
Original file line number Diff line number Diff line change
Expand Up @@ -320,12 +320,7 @@ def __call__(self):
.transpose(["I", "O"]))
hole_edge = hole.edge_by_name("O")
# Calculate rho for all the segments of the physics edge of this orbit
rho = []
for seg in hole_edge.segments:
symmetry, _ = seg
block_rho = hole.blocks[[("I", -symmetry), ("O", symmetry)]]
diag_rho = np.diagonal(block_rho)
rho = [*rho, *diag_rho]
rho = hole.data.diagonal()
rho = np.array(rho).real
rho = np.maximum(rho, 0) # Sometimes there is some negative value because of numeric error.
if np.sum(rho) == 0:
Expand Down
8 changes: 4 additions & 4 deletions tetragono/tetragono/simple_update_lattice.py
Original file line number Diff line number Diff line change
Expand Up @@ -535,9 +535,9 @@ def _update_virtual_bond(self):
for l1, l2 in self.sites():
# Update half of virtual bond, another part will be updated automatically.
if l1 != self.L1 - 1:
self.virtual_bond[l1, l2, "D"] = self[l1, l2].edges("D")
self.virtual_bond[l1, l2, "D"] = self[l1, l2].edge_by_name("D")
if l2 != self.L2 - 1:
self.virtual_bond[l1, l2, "R"] = self[l1, l2].edges("R")
self.virtual_bond[l1, l2, "R"] = self[l1, l2].edge_by_name("R")

def _single_term_simple_update(self, coordinates, index_and_orbit, evolution_operator, new_dimension):
"""
Expand Down Expand Up @@ -629,7 +629,7 @@ def _single_term_simple_update_double_site_nearest_horizontal(self, coordinates,
left = self[i, j]
right = self[i, j + 1]
right = self._try_multiple(right, i, j + 1, "L", division=True)
original_dimension = left.edges("R").dimension
original_dimension = left.edge_by_name("R").dimension
left_q, left_r = left.qr("r", {*(f"P{orbit}" for body_index, orbit in left_index_and_orbit), "R"}, "R", "L")
right_q, right_r = right.qr("r", {*(f"P{orbit}" for body_index, orbit in right_index_and_orbit), "L"}, "L", "R")
u, s, v = (
Expand Down Expand Up @@ -697,7 +697,7 @@ def _single_term_simple_update_double_site_nearest_vertical(self, coordinates, i
up = self[i, j]
down = self[i + 1, j]
down = self._try_multiple(down, i + 1, j, "U", division=True)
original_dimension = up.edges("D").dimension
original_dimension = up.edge_by_name("D").dimension
up_q, up_r = up.qr("r", {*(f"P{orbit}" for body_index, orbit in up_index_and_orbit), "D"}, "D", "U")
down_q, down_r = down.qr("r", {*(f"P{orbit}" for body_index, orbit in down_index_and_orbit), "U"}, "U", "D")
u, s, v = (
Expand Down
71 changes: 25 additions & 46 deletions tetragono/tetragono/tensor_element.py
Original file line number Diff line number Diff line change
Expand Up @@ -27,66 +27,45 @@ def tensor_element(tensor):
return element_pool[tensor_id]


def loop_nonzero_block(block, symmetries, rank, names, template):
indices = [0 for _ in range(rank)]
for i in range(rank):
if block.shape[i] == 0:
# dims == 0
return
while True:
value = block[tuple(indices)]
if value != 0:
# yield
template[{names[i]: 0 for i in range(rank)}] = value
yield [(symmetries[i], indices[i]) for i in range(rank)], template.copy()

edge_position = rank - 1

indices[edge_position] += 1
while indices[edge_position] == block.shape[edge_position]:
if edge_position == 0:
return
indices[edge_position] = 0
edge_position -= 1
indices[edge_position] += 1


def loop_nonzero_tensor(tensor, names, rank):
# see include/TAT/structure/edge.hpp
edges = [tensor.edges(i).segments for i in range(rank)]
arrow = [tensor.edges(i).arrow for i in range(rank)]
Edge = tensor.model.Edge
if tensor.data.nelement() == 0:
return
if rank == 0:
# rank == 0
yield [], tensor
return
symmetry_indices = [0 for _ in range(rank)]
for i in range(rank):
if len(edges[i]) == 0:
# dims == 0
return
zero_symmetry = tensor.model.Symmetry()
indices = [0 for _ in range(rank)]
while True:
symmetries = [edges[i][symmetry_indices[i]][0] for i in range(rank)]
if sum(symmetries, start=zero_symmetry) == zero_symmetry:
block = tensor.blocks[[(names[i], symmetries[i]) for i in range(rank)]]
template_edges = [Edge([symmetries[i]], arrow[i]) for i in range(rank)]
template = type(tensor)(names, template_edges)
yield from loop_nonzero_block(block, symmetries, rank, names, template)
if tensor.data[tuple(indices)] != 0:
element = tensor.__class__(
names=tensor.names,
edges=tuple(tensor.edges[i].__class__(
fermion=tensor.fermion,
dtypes=tensor.dtypes,
symmetry=tuple(symmetryp[indices[i]].reshape([1]) for symmetry in tensor.edges[i].symmetry),
dimension=1,
arrow=tensor.edges[i].arrow,
parity=tensor.edges[i].parity[indices[i]].reshape([1]),
) for i in range(rank)),
fermion=tensor.fermion,
dtypes=tensor.dtypes,
data=tensor.data[tuple(indices)].reshape([1 for _ in range(rank)]),
mask=tensor.mask[tuple(indices)].reshape([1 for _ in range(rank)]),
)
yield [tensor.edges[i].point_by_index(indices[i]) for i in range(rank)], element

edge_position = rank - 1

symmetry_indices[edge_position] += 1
while symmetry_indices[edge_position] == len(edges[edge_position]):
indices[edge_position] += 1
while indices[edge_position] == tensor.edges[edge_position].dimension:
if edge_position == 0:
return
symmetry_indices[edge_position] = 0
indices[edge_position] = 0
edge_position -= 1
symmetry_indices[edge_position] += 1
indices[edge_position] += 1


def conjugate_symmetry(edge_point):
return (-edge_point[0], edge_point[1])
return (tuple(sym if isinstance(sym) else -sym for sym in edge_point[0]), edge_point[1])


def calculate_element(tensor):
Expand Down

0 comments on commit c4933b7

Please sign in to comment.