Topological Flat Band Model
This will be a rather extensive example and crosscheck with the 2020 paper "Superconductivity, pseudogap, and phase separation in topological flat bands:'a quantum Monte Carlo study" (arxiv) by Hofmann, Berg and Chowdhury.
Introduction
The paper investigates an attractive square lattice Hubbard model with complex hoppings up to the 5th order. It generates a flat and a dispersive band, where the flatness of the former can increased with the right ratio of hoppings. At partial filling of the flat band superconductivity is observed. The model reads
where $t_n$ refers to n-th nearest neighbor hopping and $\langle i, j \rangle_n$ refers to the involved site pairs. We will discuss the prefactors more closely when implementing the lattice model. The interactive term is a variation of what we use in our standard attractive Hubbard model and can be dealt with using the same Hirsch transformation.
Implementation
The Lattice
The model is defined for a square lattice, however the paper suggests defining it via two site basis $A = (0, 0)$, $B = (0, 1)$ with lattice vector $a_1 = (1, 1)$ and $a_2 = (1, -1)$. We will follow this suggestion. The model uses first, second and fifth neighbor hoppings.
The nearest neighbors are directed, catching different values for $\phi_{ij}^\sigma$ as a result. We need to create two groups, one with directions as indicated in figure 1a) in the paper, and one with the reverse. For second nearest neighbors the prefactor $s_{\langle i, j \rangle_2}$ depends on the combination of sublattice and direction. In $a_1$ direction the value is positive (negative) on the A (B) sublattice, and in $a_2$ it is negative (positive) on the A (B) sublattice. The fifth nearest neighbors always have the same weight and thus do not require special grouping.
using MonteCarlo: UnitCell, Bond, Lattice
function HBCLattice(Lx, Ly = Lx)
uc = UnitCell(
# name
"HBC Square",
# Bravais lattice vectors
([1.0, +1.0], [1.0, -1.0]),
# Sites
[[0.0, 0.0], [0.0, 1.0]],
# Bonds
[
# NN, directed
# bonds from ref plot, π/4 weight for spin up
Bond(1, 2, ( 0, 1), 1),
Bond(1, 2, (-1, 0), 1),
Bond(2, 1, (+1, -1), 1),
Bond(2, 1, ( 0, 0), 1),
# NN reversal
Bond(2, 1, ( 0, -1), 2),
Bond(2, 1, (+1, 0), 2),
Bond(1, 2, (-1, +1), 2),
Bond(1, 2, ( 0, 0), 2),
# NNN
# positive weight (we need forward and backward facing bonds here too)
Bond(1, 1, (+1, 0), 3),
Bond(1, 1, (-1, 0), 3),
Bond(2, 2, ( 0, +1), 3),
Bond(2, 2, ( 0, -1), 3),
# negative weight
Bond(1, 1, ( 0, +1), 4),
Bond(1, 1, ( 0, -1), 4),
Bond(2, 2, (+1, 0), 4),
Bond(2, 2, (-1, 0), 4),
# Fifth nearest neighbors
Bond(1, 1, (2, 0), 5),
Bond(2, 2, (2, 0), 5),
Bond(1, 1, (0, 2), 5),
Bond(2, 2, (0, 2), 5),
# backwards facing bonds
Bond(1, 1, (-2, 0), 5),
Bond(2, 2, (-2, 0), 5),
Bond(1, 1, ( 0, -2), 5),
Bond(2, 2, ( 0, -2), 5),
]
)
return Lattice(uc, (Lx, Ly))
end
With this implementation we can then generate a lattice of arbitrary size with
L = 8
l = HBCLattice(L)
where L
is the linear system size. Note that due to the two basis sites the total number of sites is $2L^2$. To verify our lattice implementation it is useful to create a comparable plot. In Makie, for example, we may run
using GLMakie
# get small lattice without periodic bonds
l = HBCLattice(3)
# create figure and axis without background grid and stretching
fig = Figure()
ax = Axis(fig[1, 1], aspect=DataAspect(), xgridvisible = false, ygridvisible = false)
# collect list of bonds grouped by label
ps = Point2f.(positions(l))
ls = [Point2f[] for _ in 1:5]
for b in bonds_open(l, true)
push!(ls[b.label], ps[b.from], ps[b.to])
end
# Draw arrows for NN groups
ds = ls[1][2:2:end] .- ls[1][1:2:end]
a = arrows!(ax, ls[1][1:2:end] .+ 0.35 .* ds, 0.55 .* ds, color = :black, arrowsize = 16)
ds = ls[2][2:2:end] .- ls[2][1:2:end]
arrows!(ax, ls[2][1:2:end] .+ 0.65 .* ds, 0.25 .* ds, color = :lightgray, arrowsize = 16)
# NNN
linesegments!(ax, ls[3], color = :black, linewidth=1)
linesegments!(ax, ls[4], color = :black, linewidth=1, linestyle = :dash)
# 5th nearest neighbors
linesegments!(ax, ls[5] .+ Point2f(0, 0.05), color = :red)
# draw A and B sites
As = ps[1, :, :][:]
Bs = ps[2, :, :][:]
scatter!(ax, As, color = :black, markersize = 10)
scatter!(ax, Bs, color = :black, marker='■', markersize = 16)
# Label A and B sites
text!(ax, "A", position = Point2f(2-0.2, 0), align = (:right, :center))
text!(ax, "B", position = Point2f(2-0.2, 1), align = (:right, :center))
Makie.save("HBC_lattice.png", fig)
fig
In the plot we indicate the first group of nearest neighbors with black arrows and the second, i.e. the reversals with light gray ones. Next nearest neighbors are indicated with full (group 3) or dashed lines (group 4) like in the paper. The fifth nearest neighbors (group 5) are drawn in red like in the reference.
Hopping and Interaction Matrix
Now that we have the lattice we can generate a fitting hopping matrix. But before we do this, let us briefly discuss some optimizations.
LoopVectorization.jl is a great tool when pushing for peak single threaded/single core linear algebra performance. The linear algebra needed for DQMC is reimplemented in MonteCarlo.jl using it for both Float64
and ComplexF64
. The latter uses MonteCarlo.CMat64
and MonteCarlo.CVec64
as concrete array types which are based on StructArrays.jl under the hood. They should be used in this model. Furthermore we can make use of MonteCarlo.BlockDiagonal
as we have no terms with differing spin indices. Thus we set
using MonteCarlo: StructArray, BlockDiagonal, CMat64
MonteCarlo.@with_kw_noshow struct HBCModel <: Model
# parameters with defaults based on paper
mu::Float64 = 0.0
U::Float64 = 1.0
@assert U >= 0. "U must be positive."
t1::Float64 = 1.0
t2::Float64 = 1.0 / sqrt(2.0)
t5::Float64 = (1 - sqrt(2)) / 4
# lattice
l::Lattice{2}
@assert l.unitcell.name == "HBC Square"
end
MonteCarlo.hopping_eltype(::Type{DQMC}, ::HBCModel) = ComplexF64
MonteCarlo.hopping_matrix_type(::Type{DQMC}, ::HBCModel) = BlockDiagonal{ComplexF64, 2, CMat64}
MonteCarlo.greens_eltype(::Type{DQMC}, ::HBCModel) = ComplexF64
MonteCarlo.greens_matrix_type( ::Type{DQMC}, ::HBCModel) = BlockDiagonal{ComplexF64, 2, CMat64}
for our model. The definition of the hopping matrix then follows from the various weights in the Hamiltonian as
function MonteCarlo.hopping_matrix(m::HBCModel)
# number of sites
N = length(m.l)
# spin up and spin down blocks of T
tup = diagm(0 => fill(-ComplexF64(m.mu), N))
tdown = diagm(0 => fill(-ComplexF64(m.mu), N))
# positive and negative prefactors for t1, t2
t1p = m.t1 * cis(+pi/4) # ϕ_ij^↑ = + π/4
t1m = m.t1 * cis(-pi/4) # ϕ_ij^↓ = - π/4
t2p = + m.t2
t2m = - m.t2
for b in bonds(m.l, Val(true))
# NN paper direction
if b.label == 1
tup[b.from, b.to] = - t1p
tdown[b.from, b.to] = - t1m
# NN reverse direction
elseif b.label == 2
tup[b.from, b.to] = - t1m
tdown[b.from, b.to] = - t1p
# NNN solid bonds
elseif b.label == 3
tup[b.from, b.to] = - t2p
tdown[b.from, b.to] = - t2p
# NNN dashed bonds
elseif b.label == 4
tup[b.from, b.to] = - t2m
tdown[b.from, b.to] = - t2m
# Fifth nearest neighbors
else
tup[b.from, b.to] = - m.t5
tdown[b.from, b.to] = - m.t5
end
end
return BlockDiagonal(StructArray(tup), StructArray(tdown))
end
We note that the hermitian conjugates of a hopping $c_j^\dagger c_i$ can also be understood as reversing the bond direction. Since we include both directions in our lattice definitions, second and fifth nearest neighbor hermitian conjugates are taken care of. First nearest neighbors get a phase shift from complex conjugation, which is included by swapping t1p
and t1m
between group one and two.
To finish off the mandatory model interface we need to provide three more methods. The first is lattice(model)
which simply return the lattice of the model. The other two are unique_flavors(model)
and total_flavors(model)
. The latter returns the total number of flavors a model has without making use of any symmetries, i.e. 2 for a spin 1/2 model. The former returns the number of flavors that will result in unique entries in the hopping matrix. For this model we have a hopping directly depending on spin, thus also 2 flavors here.
MonteCarlo.lattice(m::HBCModel) = m.l
MonteCarlo.unique_flavors(::HBCModel) = 2
MonteCarlo.total_flavors(::HBCModel) = 2
There are a few more methods we can implement for convenience. The most important of these is choose_field(model)
, which sets a default field for our model. The best choice here should be DensityHirschField
or DensityGHQField
as the model uses an attractive interaction. Beyond this we could implement parameters(model)
, save_model
, _load_model
and printing.
HBCModel(l::MonteCarlo.AbstractLattice; kwargs...) = HBCModel(l = l; kwargs...)
MonteCarlo.choose_field(::HBCModel) = DensityHirschField
The full code including these convenience functions can be found here
Simulation Setup
To keep the runtime of this crosscheck reasonable we used the smallest linear system size the paper considers, L = 8
. We also set U = 1
and the fifth nearest neighbor hopping t5 = 0
. This corresponds to a flatness ratio $F = 0.2$. To be comparable to the paper we will need to tune the chemical potential $\mu$ to hit half filling. This can be done through trial and error on a smaller lattice. The optimal $\mu$, after running the main simulation with a small set of different values, seems to be $\mu \approx -2.206$. Thus the basic setup for our simulation becomes
l = HBCLattice(8)
m = HBCModel(l, t5 = 0.0, mu = -2.206) # other defaults match F = 0.2 setup
mc = DQMC(
m, beta = beta, thermalization = 1000, sweeps = 5000,
measure_rate = 5, print_rate = 100, recorder = Discarder()
)
where beta needs to run over a reasonable set of inverse temperatures. We will use [2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 10.0, 12.0, 14.0, 17.0, 20.0, 25.0, 30.0, 35.0, 40.0]
.
In our actual simulation we used a BufferedConfigRecorder
to record configurations. That way the simulation can be replayed with different measurements. This is very useful when you are still unsure about what you want to measure or how exactly those measurements are supposed to work.
Measurements
We will consider the following measurements for comparison:
- Z-spin susceptibility, solid red line in figure 1d)
- Superfluid stiffness, figure 2b)
- Reciprocal s-wave pairing susceptibility, solid red line in figure 4a)
- Reciprocal charge susceptibility, solid blue line in figure 4a)
Z-Spin Susceptibility
The z-spin susceptibility $\int_0^\beta d\tau \langle m_z(r^\prime, \tau) m_z(r, 0) \rangle$ can be measure with
mc[:SDSz] = spin_density_susceptibility(mc, m, :z)
The integral will be evaluated by MonteCarlo.jl and the result, accessible with mean(mc[:SDCz])
, will return the average result by direction. I.e. mean(mc[:SDCz])[i]
will contain the average z-spin susceptibility in directions(mc)[i]
.
Reciprocal Charge Susceptibility
The paper defines the charge susceptibility as $\int_0^\beta d\tau \langle N(\tau) N(0) \rangle$ where $N(\tau) = \sum_j (n_j(\tau) - \nu)$ and $\nu$ is the filling. The charge density susceptibility that MonteCarlo.jl defines, on the other hand, is $\langle n_j(\tau) n_i(0)\rangle$. To connect these two we expand the papers' definition:
In the last step we associated $\nu = \sum_i \langle n_i \rangle / N$, i.e. the average occupation. We can use this representation to calculate the reciprocal charge susceptibility as $1 / (O - \nu^2)$ where $O$ is MonteCarlo.jl's charge density susceptibility and $\nu$ is the average occupation. We measure
mc[:occ] = occupation(mc, m)
mc[:CDS] = charge_density_susceptibility(mc, m)
Reciprocal S-Wave Pairing Susceptibility
The paper defines the (s-wave) pairing susceptibilities as $O(\tau) = \sum_j c_{j, \uparrow} c_{j, \downarrow} + h.c.$. More generally you would consider a site offset between the pairs of operators and use weighted sums to get pairing susceptibilities of various symmetries like d-wave, p-wave, etc. For s-wave this offset is $\vec{0}$. In MonteCarlo.jl these offsets are set via the lattice iterator. For example you may use EachLocalQuadByDistance([2, 4, 5])
to consider the directions(mc)[[2, 4, 5]]
as offsets.
The pairing_suceptibility
constructors from MonteCarlo.jl is written with these offsets in mind. By default it will include offsets for all nearest neighbors as well as offsets of $\vec{0}$. To reduce computational complexity we may reduce these to just $\vec{0}$ offsets by requesting just first offset:
mc[:PS] = pairing_susceptibility(mc, m, K=1)
Superfluid Stiffness
The superfluid stiffness is given by $0.25 [- K_x - \Lambda_{xx}(q = 0)]$ in the paper. Both the diamagnetic contribution $K_x$ and the Fourier transformed current-current correlation $\Lambda_{xx}(q)$ are things we need to measure individually.
The diamagnetic contribution $K_x$ is the simpler one. For that we refer to equations 15a - 15j in the paper. The sum of all of these is the $K_x$ we seek. Since all terms are quadratic in creation and annihilation operators we do not need to worry about expanding them with Wicks theorem. Instead we can simply measure the Greens matrix during the simulation. If we compare the equations with the Hamiltonian we will also notice that they are (almost) the same as the hopping terms. Thus we can get weights from the hopping matrix and apply them afterwards. We measure
mc[:G] = greens_measurement(mc, m)
For current-current correlations we need to measure $\int_0^\beta d \tau \langle J_x^\alpha(r^\prime, \tau) J_x^\beta(r, 0) \rangle$ where $J_x(r, \tau)$ is given in equation 14a - 14j. These terms are partially implemented in MonteCarlo.jl with rest implemented in MonteCarloAnalysis.jl.
Specifically MonteCarlo.jl implements cc_kernel
as a generic version of the terms 14a - 14j with the directional prefactor $\langle \hat{x}, \Delta r_\alpha$ removed. These prefactors are instead introduced in cached_para_ccc
from MonteCarloAnalysis.jl, which computes $\Lambda_{xx}(q)$. The hopping directions considered in cc_kernel
are set by the lattice iterator. By default, it will consider half the bonds of the lattice, dropping reverse bonds.
The included bonds can be adjusted by passing a lattice_iterator = EachBondPairByBravaisDistance([1,6,9,11,13,15])
to the measurements. The indices apply to lattice(mc).unitcell.bonds
and in this case pick the bonds fitting terms 14a - 14f. The remaing terms are 0 with the choice of $t_5 = 0$.
mc[:CCS] = current_current_susceptibility(
mc, m, lattice_iterator = EachBondPairByBravaisDistance([1,6,9,11,13,15])
)
Running the simulations
To run the simulation we simply use run!(mc)
.
We should point out that these simulations are lot more complex than the other two examples. We are working with 128 sites as opposed to 16 and inverse temperatures as large as 40 instead of $\le 12$. We are also using complex matrices which bring $2 - 4$ times the complexity and we need to consider both a spin up and down sector in the greens matrix.
It is therefore advised that you run this on a cluster, in parallel. To figure out how much time is needed you can check the sweep time for the smallest $\beta$ with measurements. The scaling should be roughly linear (w.r.t. $\beta$). Note that you can pass a safe_before::TimeType
to make sure the simulation saves and exits in time. If your cluster restricts you to full nodes it might be useful to create files for each simulation beforehand and distribute filenames to different cores on the same node. (src/mpi.jl might be helpful.)
Results
In this section we plot the results from our simulations on top of the results from the paper. There are 5 points per temperature coming from different chemical potentials $\mu$. The filling varies from 0.22 to 0.265 between them. Note also that not every simulation did the full number of sweeps. The shortest simulation ran for about 3000 sweeps, which includes 1000 thermalization sweeps.
Z-Spin Susceptibility
To get the results from the paper we need to perform a $q = 0$ Fourier transform which is simply a sum. We calculate real(sum(mean(mc[:SDCz])))
and plot against 1 / mc.parameters.beta
.
Reciprocal Charge Susceptibility
For the reciprocal charge susceptibility we plot
CDS = real(sum(mean(mc[:CDS])))
occ = real(sum(mean(mc[:occ])))
xs = 1 / mc.parameters.beta
ys = 1 / (CDS - mc.parameters.beta * occ^2 / length(lattice(mc)))
where the factor mc.parameters.beta / length(lattice(mc))
comes from the imaginary time integral.
Reciprocal Pairing Susceptibility
The pairing susceptibility comes with three directional indices after taking mean(mc[:PS])
. The first is associated with the distance $r - r^\prime$ between the two pairing operators $\Delta(r)$ in $\langle \Delta^\alpha(r) \Delta^\beta(r^\prime) \rangle$. The second and third are displacements inside them. Since we only care about s-wave pairing the internal displacements are zero or index 1. Thus we plot 1 / real(sum(mean(mc[:PS])[:, 1, 1]))
against 1 / mc.parameters.beta
for the reciprocal pairing susceptibility.
Superfluid Stiffness
To compute the superfluid stiffness we make use of the relevant functions from MonteCarloAnalysis.jl. It is given by:
using MonteCarloAnalysis
# The diamagnetic contribution follows from the greens function. We set the
# direction of the current response to x = [1, 0] like the reference paper.
K_x = dia_K(mc, :G, [1, 0])
# The paramagnetic contribution is calculated in full, i.e. Λxx(q) from the
# CCS measurement.
Λxx = cached_para_ccc(mc, :CCS, [1,0])
# Following the paper the superfluid stiffness becomes
SFS = 0.25 * (-K_x - Λxx[1, 1])