Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Wrong Derivative of Gate Charge in PSP Model #120

Open
metroid120 opened this issue Apr 16, 2024 · 4 comments
Open

Wrong Derivative of Gate Charge in PSP Model #120

metroid120 opened this issue Apr 16, 2024 · 4 comments

Comments

@metroid120
Copy link
Collaborator

The IHP PDK showed that there is an erroneous calculation of the derivative of the gate charge against gate voltage for the PSP model.
IHP-GmbH/IHP-Open-PDK#64

An exemplary plot:

image

Exemplary ngspice netlist using IHP Open PDK

* Qucs 24.2.1 cov_lv_nmos.sch
.model sg13g2_lv_nmos_psp pspnqs103va type  =  +1

V1 _net0 _net1 DC 0 SIN(0 1 1K 0 0 0) AC 1 ACPHASE 0
Vg _net1 0 DC 1
Vg_dsb _net2 0 DC 0

N1 _net2  _net0  _net2  _net2 sg13g2_lv_nmos_psp

.control
pre_osdi './va_code_psp103p6/psp103_nqs.osdi'
let points = 300
set vgsp = "$&points"
let Cgdsb = vector($vgsp)
let Cgb = vector($vgsp)
let Cgds = vector($vgsp)
let Vg = vector($vgsp)
let ind = 0
while ind < $vgsp
  let v1_act = -1.5+ind*0.01
  alter Vg = $&v1_act
  ac lin 1 1meg 1meg 
  let Cgdsb[ind] = imag(Vg_dsb#branch)/(2*pi*1e6)
  let Vg[ind] = $&v1_act
  let ind = ind + 1
  destroy ac1
end
plot Cgdsb vs Vg
destroy all
reset
.endc
.END

We must locate the erroneous derivative that causes the bug.

@metroid120
Copy link
Collaborator Author

I did some investigations today.
The line of code

SP_S_delta1 = 1.0 / SP_S_delta0; \
in the PSP model seems to be critical somehow for the observed behavior.

@metroid120
Copy link
Collaborator Author

Ok I now know where this comes from. There is a code structure that is repeated three times in the PSP code:

            SP_S_delta0 = exp(SP_S_x0); \
            SP_S_delta1 = 1.0 / SP_S_delta0; \

If this is replace at all occurences with the mathematically equivalent

            SP_S_delta0 = exp(SP_S_x0); \
            SP_S_delta1 = exp(-SP_S_x0); \

the problem is gone. I will go into further detail tomorrow.

@metroid120
Copy link
Collaborator Author

Ok I leave it here, since above mentioned solution is a feasible workaround. Fundamentally the issue must be related
to the derivative generation algorithm in OpenVAF.

My guess is that OpenVAF generates the derivative of $y2$ in the following structure

$y1=exp(a)$

$y2=1/exp(a)$

as

$\frac{dy2}{dx}=-(exp(a))^{-2}exp(a)\frac{da}{dx}$

which causes numerical problems in contrast to the mathematically equivalent

$\frac{dy2}{dx}=-exp(-a)\frac{da}{dx}$

Just a guess.

@dwarning
Copy link

@gjcoram :
The attached patch will solve the problem.
We have to change openvaf/mir_autodiff/src/builder.rs, line 502

  •        Opcode::Exp => res,
    

into

  •        Opcode::Exp => self.ins().exp(arg0),
    

psp_glitch.patch.txt

dwarning pushed a commit to dwarning/OpenVAF that referenced this issue Jul 21, 2024
arpadbuermen pushed a commit to arpadbuermen/OpenVAF that referenced this issue Jul 29, 2024
gjcoram added a commit to gjcoram/OpenVAF that referenced this issue Oct 28, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants