Skip to content

Commit

Permalink
Fix Jiles-Atherton-based inductor/transformer model
Browse files Browse the repository at this point in the history
And add tests to ensure they don't bit-rot again.
  • Loading branch information
martinholters committed Aug 4, 2021
1 parent 8e48f7e commit 3a5031b
Show file tree
Hide file tree
Showing 2 changed files with 49 additions and 7 deletions.
14 changes: 7 additions & 7 deletions src/elements.jl
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
# Copyright 2015, 2016, 2017, 2018, 2019, 2020 Martin Holters
# Copyright 2015, 2016, 2017, 2018, 2019, 2020, 2021 Martin Holters
# See accompanying license file.

export resistor, potentiometer, capacitor, inductor, transformer,
Expand Down Expand Up @@ -91,7 +91,7 @@ of nonlinear inductor circuits,” IEEE Trans. Magn., vol. 30, no. 5, pp.
2795–2801, 1994, where the definition of `c` is taken from the latter. The ACME
implementation is discussed in [M. Holters, U. Zölzer, "Circuit Simulation with
Inductors and Transformers Based on the Jiles-Atherton Model of
Magnetization"](http://ant-s4.unibw-hamburg.de/paper-archive/2016/dafxpapers/08-DAFx-16_paper_10-PN.pdf).
Magnetization"](http://dafx.de/paper-archive/2016/dafxpapers/08-DAFx-16_paper_10-PN.pdf).
Pins: `1` and `2` for primary winding, `3` and `4` for secondary winding, and so
on
Expand Down Expand Up @@ -120,13 +120,13 @@ function transformer(::Type{Val{:JA}}; D=2.4e-2, A=4.54e-5, ns=[],
J_1_2 = (1e-4/Ms) * -(1-c)^2*k * δM*δ/den^2 * q[3]
J_1_3 = (1e-4/Ms) * ((1-c) * δM*(Man-q[2])/den + (c*Ms/a)*Ld_q1)
J_1_4 = (1e-4/Ms) * ((c * Ms/a * α)*Ld_q1 - 1)
return (res, @SMatrix [J_1_1 J_1_2; J_2_1 J_2_2])
return (res, @SMatrix [J_1_1 J_1_2 J_1_3 J_1_4])
end
Element(mv=[speye(length(ns)); spzeros(5, length(ns))],
Element(mv=Matrix{Int}(I, length(ns)+5, length(ns)),
mi=[spzeros(length(ns), length(ns)); ns'; spzeros(4, length(ns))],
mx=[spzeros(length(ns), 2); -π*D 0; -1/a -α/a; 0 -1; 0 0; 0 0],
mxd=[-μ0*A*ns -μ0*ns*A; 0 0; 0 0; 0 0; -1 0; 0 -1],
mq=[zeros(length(ns)+1,4); eye(4)], nonlinear_eq = nonlinear_eq)
mq=[zeros(length(ns)+1,4); Matrix(I, 4, 4)], nonlinear_eq = nonlinear_eq)
end


Expand All @@ -148,14 +148,14 @@ parameters are set using named arguments:
| `k` | amount of hysteresis (in Ampere-per-meter) |
| `Ms` | saturation magnetization (in Ampere-per-meter) |
A detailed discussion of the paramters can be found in D. C. Jiles and D. L.
A detailed discussion of the parameters can be found in D. C. Jiles and D. L.
Atherton, “Theory of ferromagnetic hysteresis,” J. Magn. Magn. Mater., vol.
61, no. 1–2, pp. 48–60, Sep. 1986 and J. H. B. Deane, “Modeling the dynamics
of nonlinear inductor circuits,” IEEE Trans. Magn., vol. 30, no. 5, pp.
2795–2801, 1994, where the definition of `c` is taken from the latter. The ACME
implementation is discussed in [M. Holters, U. Zölzer, "Circuit Simulation with
Inductors and Transformers Based on the Jiles-Atherton Model of
Magnetization"](http://ant-s4.unibw-hamburg.de/paper-archive/2016/dafxpapers/08-DAFx-16_paper_10-PN.pdf).
Magnetization"](http://dafx.de/paper-archive/2016/dafxpapers/08-DAFx-16_paper_10-PN.pdf).
Pins: `1`, `2`
"""
Expand Down
42 changes: 42 additions & 0 deletions test/runtests.jl
Original file line number Diff line number Diff line change
Expand Up @@ -412,6 +412,48 @@ end
@test run!(model, zeros(0,1)) [10/100000]
end

@testset "Jiles-Atherton inductor/transformer" begin
@testset "Jiles-Atherton inductor" begin
circ = @circuit begin
Jin = voltagesource()
Jout = currentprobe(), [+] Jin[+]
L = inductor(Val{:JA}), [1] Jout[-], [2] Jin[-]
end
model = DiscreteModel(circ, 1//44100)
# starting non-magnetized, the inductor first behaves sub-linear
y = run!(model, fill(0.1, 1, 750))
d = 2*y[1]
@test all(y' .< 2d*range(0.5, step=1, length=750))
# until it reaches saturation, where it becomes super-linear
run!(model, fill(0.1, 1, 500))
y = run!(model, fill(0.1, 1, 750))
@test all(y' .> 2d*range(0.5, step=1, length=750))
# due to hysteresis, applying the negative voltage for the same time
# drives current below zero
y = run!(model, fill(-0.1, 1, 2000))
@test y[end] < -2e-3
# with voltage at zero (i.e. shorted), the current stays constant
y = run!(model, zeros(1, 1000))
@test y[1] < -2e-3 && all(y[1] .≈ y)
end
@testset "Jiles-Atherton transformer" begin
circ = @circuit begin
Jin = voltagesource(;rs=10)
L = transformer(Val{:JA}; ns = [10, 100]), [1] Jin[+], [2] Jin[-]
Jout = voltageprobe(;gp=1e-3), [+] L[3], [-] L[4]
end
model = DiscreteModel(circ, 1//44100)
u = [sin(2pi*1000/44100*n) for n in (0:499)]'
# almost linear for small input
y1 = run!(model, 0.001*u)[200:end]
y2 = run!(model, 0.002*u)[200:end]
@test isapprox(2y1, y2; rtol = 1e-3)
y3 = run!(model, 10*u)[200:end]
# not at all linear for large input
@test !isapprox(10000y1, y2; rtol = 0.75)
end
end

@testset "BJT" begin
isc=1e-6
ise=2e-6
Expand Down

0 comments on commit 3a5031b

Please sign in to comment.