"""
ショットキーダイオードの電流-電圧特性シミュレーションスクリプト。
さまざまなショットキーモデル(単純拡散、拡散、ショットキー障壁低下、トンネル、熱電子電界放出)を用いて、
ショットキーダイオードのIV特性を計算し、プロットします。
物理定数や材料パラメータを入力として受け取り、順方向および逆方向バイアスでの電流を数値的に解きます。
関連リンク: :doc:`schottky_usage` (例として示すリンクであり、実際のドキュメントパスは異なる場合があります)
"""
import numpy as np
import math
import argparse
import matplotlib.pyplot as plt
# 物理定数
KB = 1.380649e-23 # ボルツマン定数 [J/K]
QE = 1.60217663e-19 # 素電荷 [C]
EPS0 = 8.854187e-12 # 真空の誘電率 [F/m]
ME = 9.109383e-31 # 電子の静止質量 [kg]
[ドキュメント]
def calculate_params(args):
"""
ショットキーダイオードの主要な物理パラメータを計算します。
リチャードソン定数、半導体側の有効状態密度とフェルミ準位、
ショットキー障壁高さ、内蔵電位、直列抵抗(単位面積あたり)、
飽和電流密度などを計算します。
:param args: コマンドライン引数から解析されたパラメータを含むオブジェクト。
:type args: argparse.Namespace
:param args.temp: 温度 [K]。
:type args.temp: float
:param args.men: 電子の有効質量比。
:type args.men: float
:param args.ecn: 半導体の電子親和力 [eV]。
:type args.ecn: float
:param args.efm: 金属のフェルミ準位 [eV]。
:type args.efm: float
:param args.ndn: 半導体のドナー濃度 [cm^-3]。
:type args.ndn: float
:param args.mun: 電子の移動度 [cm^2/Vs]。
:type args.mun: float
:param args.dn: 半導体の厚さ [nm]。
:type args.dn: float
:returns: (飽和電流密度 [A/cm^2], 内蔵電位 [V], ショットキー障壁高さ [eV], 半導体側の有効状態密度 [cm^-3], 直列抵抗(単位面積あたり) [Ω cm^2])
:rtype: tuple[float, float, float, float, float]
"""
T = args.temp
# リチャードソン定数の計算
reff = 1.20173e6 * args.men # A/m2/K2
am_cm2 = reff * 1e-4 # A/cm2/K2
# 半導体側の統計
nc = 2.51e19 * (args.men**1.5) * ((T/300)**1.5) # cm-3
efn = args.ecn - (KB * T / QE) * math.log(args.ndn / nc)
# 障壁と内蔵電位
phi_b = -(args.ecn - args.efm)
v_bi = -(efn - args.efm)
# 抵抗と飽和電流
sigma_n = QE * args.ndn * args.mun
rn = 1.0 / sigma_n * args.dn * 1e-7 # ohm cm2
rs = rn
js = am_cm2 * T**2 * math.exp(-QE * phi_b / (KB * T))
return js, v_bi, phi_b, nc, rs
[ドキュメント]
def calculate_schottky_current(model_type, v_target, temp, mus, ncs, nds, epss, js, phi_b, v_bi, n_diode, rs, area, v_initial, men, n_tunnel):
"""
指定されたモデルに基づき、ショットキーダイオードの電流を計算します。
ニュートン法を用いて、直列抵抗効果を考慮したダイオード電圧 `v_diode` を数値的に解き、
それに対応する電流を求めます。
内部関数 `get_current(vd)` は、与えられたダイオード電圧 `vd` に対して、
モデルに応じた電流を計算します。モデルには 'Simple', 'Diffusion', 'Schottky' (障壁低下),
'Tunneling', 'TFE' (熱電子放出と電界放出) があります。
:param model_type: 使用するショットキーモデルの種類 ('Simple', 'Diffusion', 'Schottky', 'Tunneling', 'TFE')。
:type model_type: str
:param v_target: 印加電圧 [V]。
:type v_target: float
:param temp: 温度 [K]。
:type temp: float
:param mus: 電子の移動度 [cm^2/Vs]。
:type mus: float
:param ncs: 半導体の有効状態密度 [cm^-3]。
:type ncs: float
:param nds: 半導体のドナー濃度 [cm^-3]。
:type nds: float
:param epss: 半導体の誘電率 [F/m]。
:type epss: float
:param js: 飽和電流密度 [A/cm^2]。
:type js: float
:param phi_b: ショットキー障壁高さ [eV]。
:type phi_b: float
:param v_bi: 内蔵電位 [V]。
:type v_bi: float
:param n_diode: ダイオードの理想係数。
:type n_diode: float
:param rs: 直列抵抗 [Ω]。
:type rs: float
:param area: ダイオードの面積 [cm^2]。
:type area: float
:param v_initial: ニュートン法の初期ダイオード電圧 [V]。
:type v_initial: float
:param men: 電子の有効質量比。
:type men: float
:param n_tunnel: トンネルモデルの理想係数。
:type n_tunnel: float
:returns: (計算されたダイオード電圧 [V], 計算された電流 [A])
:rtype: tuple[float, float]
"""
if v_target == 0: return 0.0, 0.0
v_diode = v_initial
dv, eps, max_iter = 1e-4, 1e-7, 100
vt = (n_diode * KB * temp) / QE
# 単位換算 (cm -> m)
mus_m = mus * 1e-4
ncs_m = ncs * 1e6
nds_m = nds * 1e6
# get_current をループの外で定義できるよう vd を引数に取る形式に
def get_current(vd):
if v_target >= 0.0 or model_type == 'Simple' or (v_bi - vd) < 0:
return js * (math.exp(vd / vt) - 1.0) * area
elif model_type == 'Diffusion':
e_max = math.sqrt(2.0 * QE * nds_m * (v_bi - vd) / epss)
k_exp = QE * mus_m * ncs_m * e_max * math.exp(-QE * phi_b / (KB * temp))
return k_exp * (math.exp(vd / vt) - 1.0) * 1e-4 * area
elif model_type == 'Schottky':
k1 = (QE / epss)**1.5 * math.sqrt(2) / (4.0 * math.pi) * math.sqrt(nds_m) * math.sqrt(v_bi - vd)
k_exp = math.exp(QE / (KB * temp) * math.sqrt(k1))
return js * k_exp * (math.exp(vd / vt) - 1.0) * area
elif model_type == 'Tunneling':
# 簡易的な指数関数モデル
k_tunnel = math.exp(vd / (n_tunnel * 0.026))
return js * k_tunnel * (math.exp(vd / vt) - 1.0) * area
elif model_type == 'TFE':
hbar = 1.0545718e-34
m_eff = men * ME
# E00の計算
e00_j = (QE * hbar / 2.0) * math.sqrt(nds_m / (epss * m_eff))
e00 = e00_j / QE # [eV]
kt_q = (KB * temp) / QE
e0 = e00 * (1.0 / math.tanh(e00 / kt_q))
if vd >= 0:
# 順方向:傾きが 1/e0 になる
return js * (math.exp(vd / e0) - 1.0) * area
else:
# 逆方向:電界による障壁の薄層化を考慮(簡易WKB)
# 電界が強くなる(vdがマイナスに大きくなる)ほど指数関数的に増大させる
e_max = math.sqrt(2.0 * QE * nds_m * (v_bi - vd) / epss)
# 補正項:E00が大きくなる(ドーピングが高い)ほど透過率が爆発的に増える
tunnel_factor = math.exp(abs(vd) * (e00 / kt_q)**2)
return js * tunnel_factor * (math.exp(vd / e0) - 1.0) * area
return 0
for _ in range(max_iter):
im = get_current(v_diode - dv)
ip = get_current(v_diode + dv)
v_total = 0.5 * ((v_diode - dv + im * rs) + (v_diode + dv + ip * rs))
if abs(v_total - v_target) < eps:
return v_diode, 0.5 * (im + ip)
dv_dv = ((v_diode + dv + ip * rs) - (v_diode - dv + im * rs)) / (2.0 * dv)
v_diode += (v_target - v_total) / dv_dv
return v_diode, get_current(v_diode)
[ドキュメント]
def main():
"""
コマンドライン引数に基づいてショットキーダイオードのIV特性シミュレーションを実行し、結果をプロットします。
`argparse` を使用して入力パラメータを解析し、`calculate_params` で基本パラメータを計算した後、
`calculate_schottky_current` を繰り返し呼び出してIV曲線を生成します。
最後にMatplotlibで結果を可視化します。
:returns: なし
:rtype: None
"""
parser = argparse.ArgumentParser(description='ショットキーダイオードの電流-電圧特性をシミュレートします。')
parser.add_argument('--model', choices=['Simple', 'Diffusion', 'Schottky', 'Tunneling', 'TFE'], default='Simple',
help='使用するショットキーモデル。')
parser.add_argument('--temp', type=float, default=300.0,
help='温度 [K]。')
parser.add_argument('--area', type=float, default=0.01,
help='ダイオードの面積 [cm^2]。')
parser.add_argument('--efm', type=float, default=4.4,
help='金属のフェルミ準位 [eV]。')
parser.add_argument('--ecn', type=float, default=4.05,
help='半導体の電子親和力 [eV]。')
parser.add_argument('--ndn', type=float, default=1.0e16,
help='半導体のドナー濃度 [cm^-3]。')
parser.add_argument('--mun', type=float, default=1500.0,
help='電子の移動度 [cm^2/Vs]。')
parser.add_argument('--eps_r', type=float, default=11.9,
help='比誘電率。')
parser.add_argument('--dn', type=float, default=1000.0,
help='半導体の厚さ [nm]。')
parser.add_argument('--men', type=float, default=0.19,
help='電子の有効質量比。')
# トンネルモデル用のパラメータを追加
parser.add_argument('--n_tunnel', type=float, default=2.0,
help='トンネルモデルの理想係数。')
parser.add_argument('--v0', type=float, default=-2.0,
help='印加電圧の開始値 [V]。')
parser.add_argument('--v1', type=float, default=1.0,
help='印加電圧の終了値 [V]。')
parser.add_argument('--step', type=float, default=0.02,
help='印加電圧のステップサイズ [V]。')
args = parser.parse_args()
# パラメータ計算
js, v_bi, phi_b, ncs, rs_unit = calculate_params(args)
epss = args.eps_r * EPS0
rs = rs_unit / args.area
v_apps = np.arange(args.v0, args.v1 + args.step, args.step)
currents = []
v_prev = args.v0 * 0.2
for v_app in v_apps:
v_d, i = calculate_schottky_current(
args.model, v_app, args.temp, args.mun, ncs, args.ndn, epss,
js, phi_b, v_bi, 1.0, rs, args.area, v_prev,
args.men, args.n_tunnel # 不足していた引数を追加
)
currents.append(i)
v_prev = v_d
# 結果表示
print(f"Model : {args.model}")
print(f"Barrier phiB: {phi_b:.4f} eV")
print(f"V_bi : {v_bi:.4f} V")
print(f"Js : {js:.6e} A/cm2")
# グラフ表示
plt.figure(figsize=(8, 6))
plt.semilogy(v_apps, np.abs(currents), label=f'{args.model} model')
plt.xlabel("Applied Voltage [V]")
plt.ylabel("|Current| [A]")
plt.grid(True, which='both', alpha=0.3)
plt.legend()
plt.title(f"Schottky Diode IV Characteristics ({args.model})")
plt.show()
if __name__ == "__main__":
main()