Finding the steady states of a Duffing oscillator 
Here we show the workflow of HarmonicBalance.jl on a simple example - the driven Duffing oscillator. The equation of motion for the displacement 
In general, there is no analytical solution to the differential equation. Fortunately, some harmonics are more important than others. By truncating the infinite-dimensional Fourier space to a set of judiciously chosen harmonics, we may obtain a soluble system. For the Duffing resonator, we can well try to only consider the drive frequency 
which constraints the spectrum of 
First we need to declare the symbolic variables (the excellent Symbolics.jl is used here).
using HarmonicBalance
@variables α ω ω0 F γ t x(t) # declare constant variables and a function x(t)Next, we have to input the equations of motion. This will be stored as a DifferentialEquation. The input needs to specify that only x is a mathematical variable, the other symbols are parameters:
diff_eq = DifferentialEquation(d(x,t,2) + ω0^2*x + α*x^3 + γ*d(x,t) ~ F*cos(ω*t), x)System of 1 differential equations
Variables:       x(t)
Harmonic ansatz: x(t) => ;   
Differential(t)(Differential(t)(x(t))) + Differential(t)(x(t))*γ + x(t)*(ω0^2) + (x(t)^3)*α ~ F*cos(t*ω)One harmonic 
The harmonic ansatz needs to be specified now – we expand x in a single frequency 
add_harmonic!(diff_eq, x, ω) # specify the ansatz x = u(T) cos(ωt) + v(T) sin(ωt)The object diff_eq now contains all the necessary information to convert the differential equation to the algebraic harmonic equations (coupled polynomials in 
harmonic_eq = get_harmonic_equations(diff_eq)A set of 2 harmonic equations
Variables: u1(T), v1(T)
Parameters: ω, α, γ, ω0, F
Harmonic ansatz: 
x(t) = u1(T)*cos(ωt) + v1(T)*sin(ωt)
Harmonic equations:
(2//1)*Differential(T)(v1(T))*ω + Differential(T)(u1(T))*γ - u1(T)*(ω^2) + u1(T)*(ω0^2) + v1(T)*γ*ω + (3//4)*(u1(T)^3)*α + (3//4)*u1(T)*(v1(T)^2)*α ~ F
Differential(T)(v1(T))*γ - (2//1)*Differential(T)(u1(T))*ω - u1(T)*γ*ω - v1(T)*(ω^2) + v1(T)*(ω0^2) + (3//4)*(u1(T)^2)*v1(T)*α + (3//4)*(v1(T)^3)*α ~ 0//1The variables u1 and v1 were declared automatically to construct the harmonic ansatz. The slow time variable T describes variation of the quadratures on timescales much slower than ω. For a steady state, all derivatives w.r.t T vanish, leaving only algebraic equations to be solved.
We are ready to start plugging in numbers! Let us find steady states by solving harmonic_eq for numerical parameters. Homotopy continuation is especially suited to solving over a range of parameter values. Here we will solve over a range of driving frequencies ω – these are stored as Pairs{Sym, Vector{Float}}:
varied = ω => range(0.9, 1.2, 100); # range of parameter valuesω => 0.9:0.0030303030303030303:1.2The other parameters we be fixed – these are declared as Pairs{Sym, Float} pairs:
fixed = (α => 1., ω0 => 1.0, F => 0.01, γ => 0.01); # fixed parameters(α => 1.0, ω0 => 1.0, F => 0.01, γ => 0.01)Now everything is ready to crank the handle. get_steady_states solves our harmonic_eq using the varied and fixed parameters:
result = get_steady_states(harmonic_eq, varied, fixed)A steady state result for 100 parameter points
Solution branches:   3
   of which real:    3
   of which stable:  2
Classes: stable, physical, HopfThe algorithm has found 3 solution branches in total (out of the hypothetically admissible 
To visualize the results, we can use the Plots.jl extension of HarmonicBalance. In short, the PlotsExt.jl code module gets loaded up conditions that Plots.jl is loaded. To know more about package extensions, you can visit the julia documentation.
Here we plot the solution amplitude, 
using Plots
plot(result, "sqrt(u1^2 + v1^2)")
This is the expected response curve for the Duffing equation.
If you want to use another plotting package then Plots.jl, you can extract the desired the data from the result object and use it in your preferred plotting package.
Y = get_solutions(result, "sqrt(u1^2 + v1^2)")100-element Vector{Vector{ComplexF64}}:
 [0.052018252688548865 + 1.4763556108352277e-39im, NaN + NaN*im, NaN + NaN*im]
 [0.05350409873601037 + 1.7682816237800798e-38im, NaN + NaN*im, NaN + NaN*im]
 [0.05507950952558313 - 1.5042017114615136e-39im, NaN + NaN*im, NaN + NaN*im]
 [0.056752395754592934 + 1.7608932232281176e-38im, NaN + NaN*im, NaN + NaN*im]
 [0.058531557933164474 + 1.1895072065808566e-38im, NaN + NaN*im, NaN + NaN*im]
 [0.06042680006976532 + 1.4084347341650409e-37im, NaN + NaN*im, NaN + NaN*im]
 [0.062449057224280326 + 2.022070885121794e-39im, NaN + NaN*im, NaN + NaN*im]
 [0.06461053756748154 + 3.782899235883258e-37im, NaN + NaN*im, NaN + NaN*im]
 [0.06692487893821529 - 7.159750257578077e-37im, NaN + NaN*im, NaN + NaN*im]
 [0.06940731879175716 - 9.05749869876619e-36im, NaN + NaN*im, NaN + NaN*im]
 ⋮
 [0.7208945671058409 + 9.805071127351202e-71im, 0.02617300479879441 + 0.0im, NaN + NaN*im]
 [0.7272065484880825 + 6.455560019902363e-41im, 0.02569212388375472 - 1.4059605599584656e-45im, NaN + NaN*im]
 [0.7334774314184077 + 1.4513647193008286e-48im, 0.02522748187697229 + 1.9583778876510703e-48im, NaN + NaN*im]
 [0.7397074914670921 + 3.256748601526058e-44im, 0.024778270139119492 - 2.191065954103261e-47im, NaN + NaN*im]
 [0.7458968612891713 + 6.394090250338288e-47im, 0.02434373315416625 + 0.0im, NaN + NaN*im]
 [0.7520454974269711 + 2.0325083638737126e-34im, 0.02392316420636251 + 6.839513540641689e-49im, NaN + NaN*im]
 [0.7581531325641623 + 1.260507399367225e-34im, 0.023515901475943724 - 8.564483740088528e-50im, NaN + NaN*im]
 [0.7642192050744053 - 1.0553764276822929e-46im, 0.023121324506598654 - 1.7448925014840846e-46im, NaN + NaN*im]
 [0.770242751665831 - 1.5760481466634026e-45im, 0.02273885100371398 - 2.2441104597360666e-44im, NaN + NaN*im]Using multiple harmonics 
In the above section, we truncated the Fourier space to a single harmonic 
for small 
which gives a response of the form 
We argued that frequency conversion takes place, to first order from 
Note that this is not a perturbative treatment! The harmonics 
add_harmonic!(diff_eq, x, [ω, 3ω]) # specify the two-harmonics ansatz
harmonic_eq = get_harmonic_equations(diff_eq)A set of 4 harmonic equations
Variables: u1(T), v1(T), u2(T), v2(T)
Parameters: ω, ω0, γ, α, F
Harmonic ansatz: 
x(t) = u1(T)*cos(ωt) + v1(T)*sin(ωt) + u2(T)*cos(3ωt) + v2(T)*sin(3ωt)
Harmonic equations:
(2//1)*Differential(T)(v1(T))*ω + Differential(T)(u1(T))*γ - u1(T)*(ω^2) + u1(T)*(ω0^2) + v1(T)*γ*ω + (3//4)*(u1(T)^3)*α + (3//4)*(u1(T)^2)*u2(T)*α + (3//2)*u1(T)*(v2(T)^2)*α + (3//2)*u1(T)*v2(T)*v1(T)*α + (3//4)*u1(T)*(v1(T)^2)*α + (3//2)*u1(T)*(u2(T)^2)*α - (3//4)*(v1(T)^2)*u2(T)*α ~ F
Differential(T)(v1(T))*γ - (2//1)*Differential(T)(u1(T))*ω - u1(T)*γ*ω - v1(T)*(ω^2) + v1(T)*(ω0^2) + (3//4)*(u1(T)^2)*v2(T)*α + (3//4)*(u1(T)^2)*v1(T)*α - (3//2)*u1(T)*v1(T)*u2(T)*α + (3//2)*(v2(T)^2)*v1(T)*α - (3//4)*v2(T)*(v1(T)^2)*α + (3//4)*(v1(T)^3)*α + (3//2)*v1(T)*(u2(T)^2)*α ~ 0//1
Differential(T)(u2(T))*γ + (6//1)*Differential(T)(v2(T))*ω + (3//1)*v2(T)*γ*ω - (9//1)*u2(T)*(ω^2) + u2(T)*(ω0^2) + (1//4)*(u1(T)^3)*α + (3//2)*(u1(T)^2)*u2(T)*α - (3//4)*u1(T)*(v1(T)^2)*α + (3//4)*(v2(T)^2)*u2(T)*α + (3//2)*(v1(T)^2)*u2(T)*α + (3//4)*(u2(T)^3)*α ~ 0//1
-(6//1)*Differential(T)(u2(T))*ω + Differential(T)(v2(T))*γ - (9//1)*v2(T)*(ω^2) + v2(T)*(ω0^2) - (3//1)*u2(T)*γ*ω + (3//2)*(u1(T)^2)*v2(T)*α + (3//4)*(u1(T)^2)*v1(T)*α + (3//4)*(v2(T)^3)*α + (3//2)*v2(T)*(v1(T)^2)*α + (3//4)*v2(T)*(u2(T)^2)*α - (1//4)*(v1(T)^3)*α ~ 0//1The variables u1, v1 now encode ω and u2, v2 encode 3ω. We see this system is much harder to solve as we now have 4 harmonic variables, resulting in 4 coupled cubic equations. A maximum of 
result = get_steady_states(harmonic_eq, varied, fixed)
plot(result, "sqrt(u1^2 + v1^2)")
For the above parameters (where a perturbative treatment would have been reasonable), the principal response at 
p1=plot(result, "sqrt(u1^2 + v1^2)", legend=false)
p2=plot(result, "sqrt(u2^2 + v2^2)")
plot(p1, p2)
The non-perturbative nature of the ansatz allows us to capture some behaviour which is not a mere extension of the usual single-harmonic Duffing response. Suppose we drive a strongly nonlinear resonator at frequency 
fixed = (α => 10., ω0 => 3, F => 5, γ=>0.01)   # fixed parameters
varied = ω => range(0.9, 1.4, 100)           # range of parameter values
result = get_steady_states(harmonic_eq, varied, fixed)A steady state result for 100 parameter points
Solution branches:   7
   of which real:    2
   of which stable:  2
Classes: stable, physical, HopfAlthough 9 branches were found in total, only 3 remain physical (real-valued). Let us visualise the amplitudes corresponding to the two harmonics, 
p1 = plot(result, "sqrt(u1^2 + v1^2)", legend=false)
p2 = plot(result, "sqrt(u2^2 + v2^2)")
plot(p1, p2)
The contributions of 
