3558 lines
135 KiB
C
3558 lines
135 KiB
C
/**********
|
|
Copyright 1999 Regents of the University of California. All rights reserved.
|
|
Author: Weidong Liu and Pin Su Feb 1999
|
|
Author: 1998 Samuel Fung, Dennis Sinitsky and Stephen Tang
|
|
Modified by Pin Su, Wei Jin 99/9/27
|
|
Modified by Paolo Nenzi 2002
|
|
File: b3soifdld.c 98/5/01
|
|
**********/
|
|
|
|
/*
|
|
* Revision 2.1 99/9/27 Pin Su
|
|
* BSIMFD2.1 release
|
|
*/
|
|
|
|
#include "ngspice.h"
|
|
#include "cktdefs.h"
|
|
#include "b3soifddef.h"
|
|
#include "trandefs.h"
|
|
#include "const.h"
|
|
#include "sperror.h"
|
|
#include "devdefs.h"
|
|
#include "suffix.h"
|
|
|
|
#ifdef _MSC_VER
|
|
extern int _isnan(double);
|
|
#endif
|
|
|
|
#define MAX_EXP 5.834617425e14
|
|
#define MIN_EXP 1.713908431e-15
|
|
#define EXP_THRESHOLD 34.0
|
|
#define EPSOX 3.453133e-11
|
|
#define EPSSI 1.03594e-10
|
|
#define Charge_q 1.60219e-19
|
|
#define KboQ 8.617087e-5 /* Kb / q */
|
|
#define Eg300 1.115 /* energy gap at 300K */
|
|
#define DELTA_1 0.02
|
|
#define DELTA_2 0.02
|
|
#define DELTA_3 0.02
|
|
#define DELTA_4 0.02
|
|
#define DELT_Vbs0eff 0.02
|
|
#define DELT_Vbsmos 0.005
|
|
#define DELT_Vbseff 0.005
|
|
#define DELT_Xcsat 0.2
|
|
#define DELT_Vbs0dio 1e-7
|
|
#define DELTA_VFB 0.02
|
|
#define DELTA_Vcscv 0.0004
|
|
#define DELT_Vbsdio 0.01
|
|
#define CONST_2OV3 0.6666666666
|
|
#define OFF_Vbsdio 2e-2
|
|
#define OFF_Vbs0_dio 2.02e-2
|
|
#define QEX_FACT 20
|
|
|
|
|
|
/* B3SOIFDSmartVbs(Vbs, Old, here, check)
|
|
* Smart Vbs guess.
|
|
*/
|
|
|
|
double
|
|
B3SOIFDSmartVbs(double New, double Old, B3SOIFDinstance *here,
|
|
CKTcircuit *ckt, int *check)
|
|
{
|
|
/* only do it for floating body and DC */
|
|
if (here->B3SOIFDfloat && (ckt->CKTmode & (MODEDC | MODEDCOP)))
|
|
{
|
|
/* Vbs cannot be negative in DC */
|
|
if (New < 0.0) New = 0.0;
|
|
}
|
|
return(New);
|
|
}
|
|
|
|
|
|
/* B3SOIFDlimit(vnew,vold)
|
|
* limits the per-iteration change of any absolute voltage value
|
|
*/
|
|
|
|
double
|
|
B3SOIFDlimit(double vnew, double vold, double limit, int *check)
|
|
{
|
|
double T0, T1;
|
|
|
|
if (isnan (vnew) || isnan (vold))
|
|
{
|
|
fprintf(stderr, "Alberto says: YOU TURKEY! The limiting function received NaN.\n");
|
|
fprintf(stderr, "New prediction returns to 0.0!\n");
|
|
vnew = 0.0;
|
|
*check = 1;
|
|
}
|
|
T0 = vnew - vold;
|
|
T1 = fabs(T0);
|
|
if (T1 > limit) {
|
|
if (T0 > 0.0)
|
|
vnew = vold + limit;
|
|
else
|
|
vnew = vold - limit;
|
|
*check = 1;
|
|
}
|
|
return vnew;
|
|
}
|
|
|
|
|
|
|
|
int
|
|
B3SOIFDload(GENmodel *inModel, CKTcircuit *ckt)
|
|
{
|
|
B3SOIFDmodel *model = (B3SOIFDmodel*)inModel;
|
|
B3SOIFDinstance *here;
|
|
int selfheat;
|
|
|
|
double ag0, qgd, qgs, von, cbhat, VgstNVt, ExpVgst = 0.0;
|
|
double cdhat, cdreq, ceqbd, ceqbs, ceqqb, ceqqd, ceqqg, ceq, geq;
|
|
double delvbd, delvbs, delvds, delvgd, delvgs;
|
|
double Vfbeff, dVfbeff_dVd, dVfbeff_dVb, V3, V4;
|
|
double gcgdb, gcggb, gcgsb, gcgeb, gcgT;
|
|
double gcsdb, gcsgb, gcssb, gcseb, gcsT;
|
|
double gcddb, gcdgb, gcdsb, gcdeb, gcdT;
|
|
double gcbdb, gcbgb, gcbsb, gcbeb, gcbT;
|
|
double gcedb, gcegb, gcesb, gceeb, gceT;
|
|
double gcTt, gTtg, gTtb, gTte, gTtdp, gTtt, gTtsp;
|
|
double vbd, vbs, vds, vgb, vgd, vgs, vgdo, xfact;
|
|
double vg, vd, vs, vp, ve, vb;
|
|
double Vds, Vgs, Vbs, Gmbs, FwdSum, RevSum;
|
|
double Vgs_eff, Vfb, dVfb_dVb, dVfb_dVd, dVfb_dT;
|
|
double Phis, dPhis_dVb, sqrtPhis, dsqrtPhis_dVb, Vth = 0.0, dVth_dVb;
|
|
double dVth_dVd, dVth_dT;
|
|
double Vgst, dVgs_eff_dVg;
|
|
double n, dn_dVb, Vtm;
|
|
double ExpArg, V0;
|
|
double ueff = 0.0, dueff_dVg, dueff_dVd, dueff_dVb, dueff_dT;
|
|
double Esat, Vdsat = 0.0;
|
|
double EsatL, dEsatL_dVg, dEsatL_dVd, dEsatL_dVb, dEsatL_dT;
|
|
double dVdsat_dVg, dVdsat_dVb, dVdsat_dVd, dVdsat_dT, Vasat;
|
|
double dVasat_dVg, dVasat_dVb, dVasat_dVd, dVasat_dT;
|
|
double Va, dVa_dVd, dVa_dVg, dVa_dVb, dVa_dT;
|
|
double Vbseff, dVbseff_dVb;
|
|
double One_Third_CoxWL, Two_Third_CoxWL, CoxWL;
|
|
double T0, dT0_dVg, dT0_dVd, dT0_dVb, dT0_dVc, dT0_dVe, dT0_dT;
|
|
double T1, dT1_dVg, dT1_dVd, dT1_dVb, dT1_dVc, dT1_dVe, dT1_dT;
|
|
double T2, dT2_dVg, dT2_dVd, dT2_dVb, dT2_dVc, dT2_dVe, dT2_dT;
|
|
double T3, dT3_dVg, dT3_dVd, dT3_dVb, dT3_dVc, dT3_dVe, dT3_dT;
|
|
double T4, dT4_dVg, dT4_dVd, dT4_dVb, dT4_dVe, dT4_dT;
|
|
double T5, dT5_dVe;
|
|
double T6, dT6_dVe, dT6_dT;
|
|
double T7;
|
|
double T8;
|
|
double T9;
|
|
double T10;
|
|
double T11;
|
|
double Abulk, dAbulk_dVb, Abulk0, dAbulk0_dVb;
|
|
double VACLM, dVACLM_dVg, dVACLM_dVd, dVACLM_dVb, dVACLM_dT;
|
|
double VADIBL, dVADIBL_dVg, dVADIBL_dVd, dVADIBL_dVb, dVADIBL_dT;
|
|
double Xdep, dXdep_dVb, lt1, dlt1_dVb, ltw, dltw_dVb;
|
|
double Delt_vth, dDelt_vth_dVb, dDelt_vth_dT;
|
|
double Theta0, dTheta0_dVb;
|
|
double TempRatio, tmp1, tmp2, tmp3, tmp4;
|
|
double DIBL_Sft, dDIBL_Sft_dVd, Lambda, dLambda_dVg;
|
|
double a1;
|
|
|
|
double Vgsteff = 0.0, dVgsteff_dVg, dVgsteff_dVd, dVgsteff_dVb;
|
|
double dVgsteff_dVe, dVgsteff_dT;
|
|
double Vdseff = 0.0, dVdseff_dVg, dVdseff_dVd, dVdseff_dVb, dVdseff_dT;
|
|
double VdseffCV, dVdseffCV_dVg, dVdseffCV_dVd, dVdseffCV_dVb;
|
|
double diffVds;
|
|
double dAbulk_dVg, dn_dVd ;
|
|
double beta, dbeta_dVg, dbeta_dVd, dbeta_dVb, dbeta_dT;
|
|
double gche, dgche_dVg, dgche_dVd, dgche_dVb, dgche_dT;
|
|
double fgche1, dfgche1_dVg, dfgche1_dVd, dfgche1_dVb, dfgche1_dT;
|
|
double fgche2, dfgche2_dVg, dfgche2_dVd, dfgche2_dVb, dfgche2_dT;
|
|
double Idl, dIdl_dVg, dIdl_dVd, dIdl_dVb, dIdl_dT;
|
|
double Ids = 0.0, Gm, Gds = 0.0, Gmb;
|
|
double CoxWovL;
|
|
double Rds, dRds_dVg, dRds_dVb, dRds_dT, WVCox, WVCoxRds;
|
|
double Vgst2Vtm, dVgst2Vtm_dT, VdsatCV, dVdsatCV_dVg, dVdsatCV_dVb;
|
|
double Leff, Weff, dWeff_dVg, dWeff_dVb;
|
|
double AbulkCV, dAbulkCV_dVb;
|
|
double qgdo, qgso, cgdo, cgso;
|
|
|
|
double dxpart, sxpart;
|
|
|
|
struct b3soifdSizeDependParam *pParam;
|
|
int ByPass, Check, ChargeComputationNeeded = 0, error;
|
|
|
|
double gbbsp, gbbdp, gbbg, gbbb, gbbe, gbbp, gbbT;
|
|
double gddpsp, gddpdp, gddpg, gddpb, gddpe, gddpT;
|
|
double gsspsp, gsspdp, gsspg, gsspb, gsspe, gsspT;
|
|
double Gbpbs, Gbpgs, Gbpds, Gbpes, Gbpps, GbpT;
|
|
double ves, ved, veb, vge = 0.0, delves, vedo, delved;
|
|
double vps, vpd, Vps, delvps;
|
|
double Vbd, Ves, Vesfb, sqrtXdep, DeltVthtemp, dDeltVthtemp_dT;
|
|
double Vbp, dVbp_dVp, dVbp_dVb, dVbp_dVg, dVbp_dVd, dVbp_dVe, dVbp_dT;
|
|
double Vpsdio, dVpsdio_dVg, dVpsdio_dVd, dVpsdio_dVe, dVpsdio_dVp, dVpsdio_dT;
|
|
double DeltVthw, dDeltVthw_dVb, dDeltVthw_dT;
|
|
double dVbseff_dVd, dVbseff_dVe, dVbseff_dT;
|
|
double dVdsat_dVc, dVasat_dVc, dVACLM_dVc, dVADIBL_dVc, dVa_dVc;
|
|
double dfgche1_dVc, dfgche2_dVc, dgche_dVc, dVdseff_dVc, dIdl_dVc;
|
|
double Gm0, Gds0, Gmb0, GmT0, Gmc, Gme, GmT, dVbseff_dVg;
|
|
double dDIBL_Sft_dVb;
|
|
double diffVdsii ;
|
|
double Idgidl = 0.0, Gdgidld, Gdgidlg, Isgidl = 0.0, Gsgidlg;
|
|
double Gjsd, Gjsb, GjsT, Gjdd, Gjdb, GjdT;
|
|
double Ibp = 0.0, Iii = 0.0, Giid, Giig, Giib, Giie, GiiT, Gcd, Gcb, GcT;
|
|
double ceqbody, ceqbodcon = 0.0;
|
|
double gppg = 0.0, gppdp = 0.0, gppb = 0.0, gppe = 0.0;
|
|
double gppp = 0.0, gppsp = 0.0, gppT;
|
|
double delTemp, deldelTemp, Temp;
|
|
double ceqth, ceqqth;
|
|
double K1;
|
|
double qjs = 0.0, gcjsbs, gcjsT;
|
|
double qjd = 0.0, gcjdbs, gcjdds, gcjdT;
|
|
double qge;
|
|
double ceqqe;
|
|
double ni, Eg, Cbox, Nfb, CboxWL;
|
|
double dVfbeff_dVrg, Cbe = 0.0;
|
|
double qinv = 0.0, qgate = 0.0, qbody = 0.0, qdrn = 0.0, qsrc, qsub = 0.0;
|
|
double cqgate, cqbody = 0.0, cqdrn = 0.0, cqsub, cqtemp;
|
|
double Cgg, Cgd, Cgb, Cge;
|
|
double Csg, Csd, Csb, Cse, Cbg = 0.0, Cbd = 0.0, Cbb = 0.0;
|
|
double Cgg1, Cgb1, Cgd1, Csg1, Csd1, Csb1;
|
|
double Vbs0t = 0.0, dVbs0t_dT ;
|
|
double Vbs0 = 0.0,dVbs0_dVe, dVbs0_dT;
|
|
double Vbs0eff = 0.0 ,dVbs0eff_dVg ,dVbs0eff_dVd ,dVbs0eff_dVe, dVbs0eff_dT;
|
|
double Vbs0teff = 0.0, dVbs0teff_dVg, dVbs0teff_dVd;
|
|
double dVbs0teff_dVe, dVbs0teff_dT;
|
|
double dVbsdio_dVg, dVbsdio_dVd, dVbsdio_dVe;
|
|
double dVbsdio_dVb, dVbsdio_dT;
|
|
double Vthfd = 0.0,dVthfd_dVd ,dVthfd_dVe, dVthfd_dT;
|
|
double Vbs0mos = 0.0 ,dVbs0mos_dVe, dVbs0mos_dT;
|
|
double Vbsmos ,dVbsmos_dVg ,dVbsmos_dVb ,dVbsmos_dVd, dVbsmos_dVe, dVbsmos_dT;
|
|
double Abeff ,dAbeff_dVg ,dAbeff_dVb, dAbeff_dVc;
|
|
double Vcs ,dVcs_dVg ,dVcs_dVb ,dVcs_dVd ,dVcs_dVe, dVcs_dT;
|
|
double Xcsat = 0.0 ,dXcsat_dVg , dXcsat_dVc;
|
|
double Vdsatii ,dVdsatii_dVg ,dVdsatii_dVd, dVdsatii_dVb, dVdsatii_dT;
|
|
double Vdseffii ,dVdseffii_dVg ,dVdseffii_dVd, dVdseffii_dVb, dVdseffii_dT;
|
|
double VcsCV = 0.0;
|
|
double VdsCV = 0.0, dVdsCV_dVg = 0.0, dVdsCV_dVb = 0.0;
|
|
double dVdsCV_dVd = 0.0, dVdsCV_dVc = 0.0;
|
|
double Phisd ,dPhisd_dVg ,dPhisd_dVb ,dPhisd_dVd, dPhisd_dVc;
|
|
double sqrtPhisd;
|
|
double Xc = 0.0;
|
|
double Ic = 0.0;
|
|
double Ibs = 0.0;
|
|
double Ibd = 0.0;
|
|
double Denomi ,dDenomi_dVg ,dDenomi_dVd ,dDenomi_dVb , dDenomi_dT;
|
|
double Qbf = 0.0;
|
|
double Qsubs1 = 0.0;
|
|
double Qsubs2 = 0.0;
|
|
double Qsub0 = 0.0 ,dQsub0_dVg ,dQsub0_dVb ,dQsub0_dVd ;
|
|
double Qac0 = 0.0 ,dQac0_dVb ,dQac0_dVd;
|
|
double Qdep0 ,dQdep0_dVb;
|
|
double Qe1 = 0.0;
|
|
double Ce1g ,Ce1b ,Ce1d ,Ce1e, Ce1T;
|
|
double Ce2g ,Ce2b ,Ce2d ,Ce2e, Ce2T;
|
|
double Qe2 = 0.0;
|
|
double dQac0_dVrg, Vbsdio = 0.0, dQsub0_dVrg;
|
|
|
|
/* for self-heating */
|
|
double vbi, vfbb, phi, sqrtPhi, Xdep0, jbjt, jdif, jrec, jtun, u0temp, vsattemp;
|
|
double rds0, ua, ub, uc;
|
|
double dvbi_dT, dvfbb_dT, djbjt_dT, djdif_dT, djrec_dT, djtun_dT, du0temp_dT;
|
|
double dvsattemp_dT, drds0_dT, dua_dT, dub_dT, duc_dT, dni_dT, dVtm_dT;
|
|
double dVfbeff_dT, dQac0_dT, dQsub0_dT, dVdsCV_dT = 0.0, dPhisd_dT;
|
|
double CbT, CsT, CgT;
|
|
|
|
double Qex, dQex_dVg, dQex_dVb, dQex_dVd, dQex_dVe, dQex_dT;
|
|
|
|
/* clean up last */
|
|
FILE *fpdebug = NULL;
|
|
/* end clean up */
|
|
int nandetect;
|
|
static int nanfound = 0;
|
|
char nanmessage [12];
|
|
|
|
double m;
|
|
|
|
for (; model != NULL; model = model->B3SOIFDnextModel)
|
|
{ for (here = model->B3SOIFDinstances; here != NULL;
|
|
here = here->B3SOIFDnextInstance)
|
|
{
|
|
|
|
if (here->B3SOIFDowner != ARCHme)
|
|
continue;
|
|
|
|
Check = 0;
|
|
ByPass = 0;
|
|
selfheat = (model->B3SOIFDshMod == 1) && (here->B3SOIFDrth0 != 0.0);
|
|
pParam = here->pParam;
|
|
|
|
if (here->B3SOIFDdebugMod > 3)
|
|
{
|
|
if (model->B3SOIFDtype > 0)
|
|
fpdebug = fopen("b3soifdn.log", "a");
|
|
else
|
|
fpdebug = fopen("b3soifdp.log", "a");
|
|
|
|
fprintf(fpdebug, "******* Time : %.5e ******* Device: %s Iteration: %d\n",
|
|
ckt->CKTtime, here->B3SOIFDname, here->B3SOIFDiterations);
|
|
}
|
|
|
|
if ((ckt->CKTmode & MODEINITSMSIG))
|
|
{ vbs = *(ckt->CKTstate0 + here->B3SOIFDvbs);
|
|
vgs = *(ckt->CKTstate0 + here->B3SOIFDvgs);
|
|
ves = *(ckt->CKTstate0 + here->B3SOIFDves);
|
|
vps = *(ckt->CKTstate0 + here->B3SOIFDvps);
|
|
vds = *(ckt->CKTstate0 + here->B3SOIFDvds);
|
|
delTemp = *(ckt->CKTstate0 + here->B3SOIFDdeltemp);
|
|
|
|
vg = *(ckt->CKTrhsOld + here->B3SOIFDgNode);
|
|
vd = *(ckt->CKTrhsOld + here->B3SOIFDdNodePrime);
|
|
vs = *(ckt->CKTrhsOld + here->B3SOIFDsNodePrime);
|
|
vp = *(ckt->CKTrhsOld + here->B3SOIFDpNode);
|
|
ve = *(ckt->CKTrhsOld + here->B3SOIFDeNode);
|
|
vb = *(ckt->CKTrhsOld + here->B3SOIFDbNode);
|
|
|
|
if (here->B3SOIFDdebugMod > 2)
|
|
{
|
|
fprintf(fpdebug, "... INIT SMSIG ...\n");
|
|
}
|
|
if (here->B3SOIFDdebugMod > 0)
|
|
{
|
|
fprintf(stderr,
|
|
"DC op. point converge with %d iterations\n",
|
|
here->B3SOIFDiterations);
|
|
}
|
|
}
|
|
else if ((ckt->CKTmode & MODEINITTRAN))
|
|
{ vbs = *(ckt->CKTstate1 + here->B3SOIFDvbs);
|
|
vgs = *(ckt->CKTstate1 + here->B3SOIFDvgs);
|
|
ves = *(ckt->CKTstate1 + here->B3SOIFDves);
|
|
vps = *(ckt->CKTstate1 + here->B3SOIFDvps);
|
|
vds = *(ckt->CKTstate1 + here->B3SOIFDvds);
|
|
delTemp = *(ckt->CKTstate1 + here->B3SOIFDdeltemp);
|
|
|
|
vg = *(ckt->CKTrhsOld + here->B3SOIFDgNode);
|
|
vd = *(ckt->CKTrhsOld + here->B3SOIFDdNodePrime);
|
|
vs = *(ckt->CKTrhsOld + here->B3SOIFDsNodePrime);
|
|
vp = *(ckt->CKTrhsOld + here->B3SOIFDpNode);
|
|
ve = *(ckt->CKTrhsOld + here->B3SOIFDeNode);
|
|
vb = *(ckt->CKTrhsOld + here->B3SOIFDbNode);
|
|
|
|
if (here->B3SOIFDdebugMod > 2)
|
|
{
|
|
fprintf(fpdebug, "... Init Transient ....\n");
|
|
}
|
|
if (here->B3SOIFDdebugMod > 0)
|
|
{
|
|
fprintf(stderr, "Transient operation point converge with %d iterations\n",
|
|
here->B3SOIFDiterations);
|
|
}
|
|
here->B3SOIFDiterations = 0;
|
|
}
|
|
else if ((ckt->CKTmode & MODEINITJCT) && !here->B3SOIFDoff)
|
|
{ vds = model->B3SOIFDtype * here->B3SOIFDicVDS;
|
|
vgs = model->B3SOIFDtype * here->B3SOIFDicVGS;
|
|
ves = model->B3SOIFDtype * here->B3SOIFDicVES;
|
|
vbs = model->B3SOIFDtype * here->B3SOIFDicVBS;
|
|
vps = model->B3SOIFDtype * here->B3SOIFDicVPS;
|
|
|
|
vg = vd = vs = vp = ve = 0.0;
|
|
|
|
here->B3SOIFDiterations = 0; /* initialize iteration number */
|
|
|
|
delTemp = 0.0;
|
|
here->B3SOIFDphi = pParam->B3SOIFDphi;
|
|
|
|
|
|
if (here->B3SOIFDdebugMod > 2)
|
|
fprintf(fpdebug, "... INIT JCT ...\n");
|
|
|
|
if ((vds == 0.0) && (vgs == 0.0) && (vbs == 0.0) &&
|
|
((ckt->CKTmode & (MODETRAN | MODEAC|MODEDCOP |
|
|
MODEDCTRANCURVE)) || (!(ckt->CKTmode & MODEUIC))))
|
|
{ vbs = 0.0;
|
|
vgs = model->B3SOIFDtype*0.1 + pParam->B3SOIFDvth0;
|
|
vds = 0.0;
|
|
ves = 0.0;
|
|
vps = 0.0;
|
|
}
|
|
}
|
|
else if ((ckt->CKTmode & (MODEINITJCT | MODEINITFIX)) &&
|
|
(here->B3SOIFDoff))
|
|
{ delTemp = vps = vbs = vgs = vds = ves = 0.0;
|
|
vg = vd = vs = vp = ve = 0.0;
|
|
here->B3SOIFDiterations = 0; /* initialize iteration number */
|
|
}
|
|
else
|
|
{
|
|
#ifndef PREDICTOR
|
|
if ((ckt->CKTmode & MODEINITPRED))
|
|
{ xfact = ckt->CKTdelta / ckt->CKTdeltaOld[1];
|
|
*(ckt->CKTstate0 + here->B3SOIFDvbs) =
|
|
*(ckt->CKTstate1 + here->B3SOIFDvbs);
|
|
vbs = (1.0 + xfact)* (*(ckt->CKTstate1 + here->B3SOIFDvbs))
|
|
- (xfact * (*(ckt->CKTstate2 + here->B3SOIFDvbs)));
|
|
*(ckt->CKTstate0 + here->B3SOIFDvgs) =
|
|
*(ckt->CKTstate1 + here->B3SOIFDvgs);
|
|
vgs = (1.0 + xfact)* (*(ckt->CKTstate1 + here->B3SOIFDvgs))
|
|
- (xfact * (*(ckt->CKTstate2 + here->B3SOIFDvgs)));
|
|
*(ckt->CKTstate0 + here->B3SOIFDves) =
|
|
*(ckt->CKTstate1 + here->B3SOIFDves);
|
|
ves = (1.0 + xfact)* (*(ckt->CKTstate1 + here->B3SOIFDves))
|
|
- (xfact * (*(ckt->CKTstate2 + here->B3SOIFDves)));
|
|
*(ckt->CKTstate0 + here->B3SOIFDvps) =
|
|
*(ckt->CKTstate1 + here->B3SOIFDvps);
|
|
vps = (1.0 + xfact)* (*(ckt->CKTstate1 + here->B3SOIFDvps))
|
|
- (xfact * (*(ckt->CKTstate2 + here->B3SOIFDvps)));
|
|
*(ckt->CKTstate0 + here->B3SOIFDvds) =
|
|
*(ckt->CKTstate1 + here->B3SOIFDvds);
|
|
vds = (1.0 + xfact)* (*(ckt->CKTstate1 + here->B3SOIFDvds))
|
|
- (xfact * (*(ckt->CKTstate2 + here->B3SOIFDvds)));
|
|
*(ckt->CKTstate0 + here->B3SOIFDvbd) =
|
|
*(ckt->CKTstate0 + here->B3SOIFDvbs)
|
|
- *(ckt->CKTstate0 + here->B3SOIFDvds);
|
|
|
|
*(ckt->CKTstate0 + here->B3SOIFDvg) = *(ckt->CKTstate1 + here->B3SOIFDvg);
|
|
*(ckt->CKTstate0 + here->B3SOIFDvd) = *(ckt->CKTstate1 + here->B3SOIFDvd);
|
|
*(ckt->CKTstate0 + here->B3SOIFDvs) = *(ckt->CKTstate1 + here->B3SOIFDvs);
|
|
*(ckt->CKTstate0 + here->B3SOIFDvp) = *(ckt->CKTstate1 + here->B3SOIFDvp);
|
|
*(ckt->CKTstate0 + here->B3SOIFDve) = *(ckt->CKTstate1 + here->B3SOIFDve);
|
|
|
|
/* Only predict ve */
|
|
ve = (1.0 + xfact)* (*(ckt->CKTstate1 + here->B3SOIFDve))
|
|
|
|
- (xfact * (*(ckt->CKTstate2 + here->B3SOIFDve)));
|
|
/* Then update vg, vs, vb, vd, vp base on ve */
|
|
vs = ve - model->B3SOIFDtype * ves;
|
|
vg = model->B3SOIFDtype * vgs + vs;
|
|
vd = model->B3SOIFDtype * vds + vs;
|
|
vb = model->B3SOIFDtype * vbs + vs;
|
|
vp = model->B3SOIFDtype * vps + vs;
|
|
|
|
delTemp = (1.0 + xfact)* (*(ckt->CKTstate1 +
|
|
here->B3SOIFDdeltemp))-(xfact * (*(ckt->CKTstate2 +
|
|
here->B3SOIFDdeltemp)));
|
|
|
|
if (selfheat)
|
|
{
|
|
here->B3SOIFDphi = 2.0 * here->B3SOIFDvtm
|
|
* log (pParam->B3SOIFDnpeak /
|
|
here->B3SOIFDni);
|
|
}
|
|
|
|
if (here->B3SOIFDdebugMod > 0)
|
|
{
|
|
fprintf(stderr, "Time = %.6e converge with %d iterations\n", ckt->CKTtime, here->B3SOIFDiterations);
|
|
}
|
|
if (here->B3SOIFDdebugMod > 2)
|
|
{
|
|
fprintf(fpdebug, "... PREDICTOR calculation ....\n");
|
|
}
|
|
here->B3SOIFDiterations = 0;
|
|
}
|
|
else
|
|
{
|
|
#endif /* PREDICTOR */
|
|
|
|
vg = B3SOIFDlimit(*(ckt->CKTrhsOld + here->B3SOIFDgNode),
|
|
*(ckt->CKTstate0 + here->B3SOIFDvg), 3.0, &Check);
|
|
vd = B3SOIFDlimit(*(ckt->CKTrhsOld + here->B3SOIFDdNodePrime),
|
|
*(ckt->CKTstate0 + here->B3SOIFDvd), 3.0, &Check);
|
|
vs = B3SOIFDlimit(*(ckt->CKTrhsOld + here->B3SOIFDsNodePrime),
|
|
*(ckt->CKTstate0 + here->B3SOIFDvs), 3.0, &Check);
|
|
vp = B3SOIFDlimit(*(ckt->CKTrhsOld + here->B3SOIFDpNode),
|
|
*(ckt->CKTstate0 + here->B3SOIFDvp), 3.0, &Check);
|
|
ve = B3SOIFDlimit(*(ckt->CKTrhsOld + here->B3SOIFDeNode),
|
|
*(ckt->CKTstate0 + here->B3SOIFDve), 3.0, &Check);
|
|
delTemp = *(ckt->CKTrhsOld + here->B3SOIFDtempNode);
|
|
|
|
vbs = model->B3SOIFDtype * (*(ckt->CKTrhsOld+here->B3SOIFDbNode)
|
|
- *(ckt->CKTrhsOld+here->B3SOIFDsNodePrime));
|
|
|
|
vps = model->B3SOIFDtype * (vp - vs);
|
|
vgs = model->B3SOIFDtype * (vg - vs);
|
|
ves = model->B3SOIFDtype * (ve - vs);
|
|
vds = model->B3SOIFDtype * (vd - vs);
|
|
|
|
if (here->B3SOIFDdebugMod > 2)
|
|
{
|
|
fprintf(fpdebug, "... DC calculation ....\n");
|
|
fprintf(fpdebug, "Vg = %.10f; Vb = %.10f; Vs = %.10f\n",
|
|
*(ckt->CKTrhsOld + here->B3SOIFDgNode),
|
|
*(ckt->CKTrhsOld + here->B3SOIFDbNode),
|
|
*(ckt->CKTrhsOld + here->B3SOIFDsNode));
|
|
fprintf(fpdebug, "Vd = %.10f; Vsp = %.10f; Vdp = %.10f\n",
|
|
*(ckt->CKTrhsOld + here->B3SOIFDdNode),
|
|
*(ckt->CKTrhsOld + here->B3SOIFDsNodePrime),
|
|
*(ckt->CKTrhsOld + here->B3SOIFDdNodePrime));
|
|
fprintf(fpdebug, "Ve = %.10f; Vp = %.10f; delTemp = %.10f\n",
|
|
*(ckt->CKTrhsOld + here->B3SOIFDeNode),
|
|
*(ckt->CKTrhsOld + here->B3SOIFDpNode),
|
|
*(ckt->CKTrhsOld + here->B3SOIFDtempNode));
|
|
|
|
}
|
|
|
|
#ifndef PREDICTOR
|
|
}
|
|
#endif /* PREDICTOR */
|
|
|
|
vbd = vbs - vds;
|
|
vgd = vgs - vds;
|
|
ved = ves - vds;
|
|
vgdo = *(ckt->CKTstate0 + here->B3SOIFDvgs)
|
|
- *(ckt->CKTstate0 + here->B3SOIFDvds);
|
|
vedo = *(ckt->CKTstate0 + here->B3SOIFDves)
|
|
- *(ckt->CKTstate0 + here->B3SOIFDvds);
|
|
delvbs = vbs - *(ckt->CKTstate0 + here->B3SOIFDvbs);
|
|
delvbd = vbd - *(ckt->CKTstate0 + here->B3SOIFDvbd);
|
|
delvgs = vgs - *(ckt->CKTstate0 + here->B3SOIFDvgs);
|
|
delves = ves - *(ckt->CKTstate0 + here->B3SOIFDves);
|
|
delvps = vps - *(ckt->CKTstate0 + here->B3SOIFDvps);
|
|
deldelTemp = delTemp - *(ckt->CKTstate0 + here->B3SOIFDdeltemp);
|
|
delvds = vds - *(ckt->CKTstate0 + here->B3SOIFDvds);
|
|
delvgd = vgd - vgdo;
|
|
delved = ved - vedo;
|
|
|
|
if (here->B3SOIFDmode >= 0)
|
|
{
|
|
cdhat = here->B3SOIFDcd + (here->B3SOIFDgm-here->B3SOIFDgjdg) * delvgs
|
|
+ (here->B3SOIFDgds - here->B3SOIFDgjdd) * delvds
|
|
+ (here->B3SOIFDgmbs - here->B3SOIFDgjdb) * delvbs
|
|
+ (here->B3SOIFDgme - here->B3SOIFDgjde) * delves
|
|
+ (here->B3SOIFDgmT - here->B3SOIFDgjdT) * deldelTemp;
|
|
}
|
|
else
|
|
{
|
|
cdhat = here->B3SOIFDcd + (here->B3SOIFDgm-here->B3SOIFDgjdg) * delvgd
|
|
- (here->B3SOIFDgds - here->B3SOIFDgjdd) * delvds
|
|
+ (here->B3SOIFDgmbs - here->B3SOIFDgjdb) * delvbd
|
|
+ (here->B3SOIFDgme - here->B3SOIFDgjde) * delved
|
|
+ (here->B3SOIFDgmT - here->B3SOIFDgjdT) * deldelTemp;
|
|
|
|
}
|
|
cbhat = here->B3SOIFDcb + here->B3SOIFDgbgs * delvgs
|
|
+ here->B3SOIFDgbbs * delvbs + here->B3SOIFDgbds * delvds
|
|
+ here->B3SOIFDgbes * delves + here->B3SOIFDgbps * delvps
|
|
+ here->B3SOIFDgbT * deldelTemp;
|
|
|
|
#ifndef NOBYPASS
|
|
/* following should be one big if connected by && all over
|
|
* the place, but some C compilers can't handle that, so
|
|
* we split it up here to let them digest it in stages
|
|
*/
|
|
|
|
if (here->B3SOIFDdebugMod > 3)
|
|
{
|
|
fprintf(fpdebug, "Convergent Criteria : vbs %d vds %d vgs %d ves %d vps %d temp %d\n",
|
|
((fabs(delvbs) < (ckt->CKTreltol * MAX(fabs(vbs),
|
|
fabs(*(ckt->CKTstate0+here->B3SOIFDvbs))) + ckt->CKTvoltTol))) ? 1 : 0,
|
|
((fabs(delvds) < (ckt->CKTreltol * MAX(fabs(vds),
|
|
fabs(*(ckt->CKTstate0+here->B3SOIFDvds))) + ckt->CKTvoltTol))) ? 1 : 0,
|
|
((fabs(delvgs) < (ckt->CKTreltol * MAX(fabs(vgs),
|
|
fabs(*(ckt->CKTstate0+here->B3SOIFDvgs))) + ckt->CKTvoltTol))) ? 1 : 0,
|
|
((fabs(delves) < (ckt->CKTreltol * MAX(fabs(ves),
|
|
fabs(*(ckt->CKTstate0+here->B3SOIFDves))) + ckt->CKTvoltTol))) ? 1 : 0,
|
|
((fabs(delvps) < (ckt->CKTreltol * MAX(fabs(vps),
|
|
fabs(*(ckt->CKTstate0+here->B3SOIFDvps))) + ckt->CKTvoltTol))) ? 1 : 0,
|
|
((fabs(deldelTemp) < (ckt->CKTreltol * MAX(fabs(delTemp),
|
|
fabs(*(ckt->CKTstate0+here->B3SOIFDdeltemp))) + ckt->CKTvoltTol*1e4))) ? 1 : 0);
|
|
fprintf(fpdebug, "delCd %.4e, delCb %.4e\n", fabs(cdhat - here->B3SOIFDcd) ,
|
|
fabs(cbhat - here->B3SOIFDcb));
|
|
|
|
}
|
|
if ((!(ckt->CKTmode & MODEINITPRED)) && (ckt->CKTbypass) && Check == 0)
|
|
if ((fabs(delvbs) < (ckt->CKTreltol * MAX(fabs(vbs),
|
|
fabs(*(ckt->CKTstate0+here->B3SOIFDvbs))) + ckt->CKTvoltTol)) )
|
|
if ((fabs(delvbd) < (ckt->CKTreltol * MAX(fabs(vbd),
|
|
fabs(*(ckt->CKTstate0+here->B3SOIFDvbd))) + ckt->CKTvoltTol)) )
|
|
if ((fabs(delvgs) < (ckt->CKTreltol * MAX(fabs(vgs),
|
|
fabs(*(ckt->CKTstate0+here->B3SOIFDvgs))) + ckt->CKTvoltTol)))
|
|
if ((fabs(delves) < (ckt->CKTreltol * MAX(fabs(ves),
|
|
fabs(*(ckt->CKTstate0+here->B3SOIFDves))) + ckt->CKTvoltTol)))
|
|
if ( (here->B3SOIFDbodyMod == 0) || (here->B3SOIFDbodyMod == 2) ||
|
|
(fabs(delvps) < (ckt->CKTreltol * MAX(fabs(vps),
|
|
fabs(*(ckt->CKTstate0+here->B3SOIFDvps))) + ckt->CKTvoltTol)) )
|
|
if ( (here->B3SOIFDtempNode == 0) ||
|
|
(fabs(deldelTemp) < (ckt->CKTreltol * MAX(fabs(delTemp),
|
|
fabs(*(ckt->CKTstate0+here->B3SOIFDdeltemp)))
|
|
+ ckt->CKTvoltTol*1e4)))
|
|
if ((fabs(delvds) < (ckt->CKTreltol * MAX(fabs(vds),
|
|
fabs(*(ckt->CKTstate0+here->B3SOIFDvds))) + ckt->CKTvoltTol)))
|
|
if ((fabs(cdhat - here->B3SOIFDcd) < ckt->CKTreltol
|
|
* MAX(fabs(cdhat),fabs(here->B3SOIFDcd)) + ckt->CKTabstol))
|
|
if ((fabs(cbhat - here->B3SOIFDcb) < ckt->CKTreltol
|
|
* MAX(fabs(cbhat),fabs(here->B3SOIFDcb)) + ckt->CKTabstol) )
|
|
{ /* bypass code */
|
|
vbs = *(ckt->CKTstate0 + here->B3SOIFDvbs);
|
|
vbd = *(ckt->CKTstate0 + here->B3SOIFDvbd);
|
|
vgs = *(ckt->CKTstate0 + here->B3SOIFDvgs);
|
|
ves = *(ckt->CKTstate0 + here->B3SOIFDves);
|
|
vps = *(ckt->CKTstate0 + here->B3SOIFDvps);
|
|
vds = *(ckt->CKTstate0 + here->B3SOIFDvds);
|
|
delTemp = *(ckt->CKTstate0 + here->B3SOIFDdeltemp);
|
|
|
|
/* calculate Vds for temperature conductance calculation
|
|
in bypass (used later when filling Temp node matrix) */
|
|
Vds = here->B3SOIFDmode > 0 ? vds : -vds;
|
|
|
|
vgd = vgs - vds;
|
|
vgb = vgs - vbs;
|
|
veb = ves - vbs;
|
|
|
|
if (here->B3SOIFDdebugMod > 2)
|
|
{
|
|
fprintf(stderr, "Bypass for %s...\n", here->B3SOIFDname);
|
|
fprintf(fpdebug, "... By pass ....\n");
|
|
fprintf(fpdebug, "vgs=%.4f, vds=%.4f, vbs=%.4f, ",
|
|
vgs, vds, vbs);
|
|
fprintf(fpdebug, "ves=%.4f, vps=%.4f\n", ves, vps);
|
|
}
|
|
if ((ckt->CKTmode & (MODETRAN | MODEAC)) ||
|
|
((ckt->CKTmode & MODETRANOP) && (ckt->CKTmode & MODEUIC)))
|
|
{ ByPass = 1;
|
|
goto line755;
|
|
}
|
|
else
|
|
{ goto line850;
|
|
}
|
|
}
|
|
|
|
|
|
#endif /*NOBYPASS*/
|
|
von = here->B3SOIFDvon;
|
|
|
|
if ((here->B3SOIFDdebugMod > 1) || (here->B3SOIFDdebugMod == -1))
|
|
{
|
|
here->B3SOIFDdum1 = here->B3SOIFDdum2 = here->B3SOIFDdum3 = 0.0;
|
|
here->B3SOIFDdum4 = here->B3SOIFDdum5 = 0.0;
|
|
Qac0 = Qsub0 = Qsubs1 = Qsubs2 = Qbf = Qe1 = Qe2 = 0.0;
|
|
qjs = qjd = Cbg = Cbb = Cbd = Cbe = Xc = qdrn = qgate = 0.0;
|
|
qbody = qsub = 0.0;
|
|
}
|
|
|
|
if (here->B3SOIFDdebugMod > 2) {
|
|
fprintf(fpdebug, "Limited : vgs = %.8f\n", vgs);
|
|
fprintf(fpdebug, "Limited : vds = %.8f\n", vds);
|
|
}
|
|
|
|
if (*(ckt->CKTstate0 + here->B3SOIFDvds) >= 0.0)
|
|
T0 = *(ckt->CKTstate0 + here->B3SOIFDvbs);
|
|
else
|
|
T0 = *(ckt->CKTstate0 + here->B3SOIFDvbd);
|
|
|
|
if (here->B3SOIFDdebugMod > 2)
|
|
fprintf(fpdebug, "Before lim : vbs = %.8f, after = ", T0);
|
|
|
|
if (vds >= 0.0)
|
|
{
|
|
vbs = B3SOIFDlimit(vbs, T0, 0.2, &Check);
|
|
vbs = B3SOIFDSmartVbs(vbs, T0, here, ckt, &Check);
|
|
vbd = vbs - vds;
|
|
vb = model->B3SOIFDtype * vbs + vs;
|
|
if (here->B3SOIFDdebugMod > 2)
|
|
fprintf(fpdebug, "%.8f\n", vbs);
|
|
} else
|
|
{
|
|
vbd = B3SOIFDlimit(vbd, T0, 0.2, &Check);
|
|
vbd = B3SOIFDSmartVbs(vbd, T0, here, ckt, &Check);
|
|
vbs = vbd + vds;
|
|
vb = model->B3SOIFDtype * vbs + vd;
|
|
if (here->B3SOIFDdebugMod > 2)
|
|
fprintf(fpdebug, "%.8f\n", vbd);
|
|
}
|
|
|
|
delTemp =B3SOIFDlimit(delTemp, *(ckt->CKTstate0 + here->B3SOIFDdeltemp),5.0,&Check);
|
|
|
|
}
|
|
|
|
/* Calculate temperature dependent values for self-heating effect */
|
|
Temp = delTemp + ckt->CKTtemp;
|
|
/* for debugging
|
|
Temp = ckt->CKTtemp;
|
|
selfheat = 1;
|
|
if (here->B3SOIFDname[1] == '2')
|
|
{
|
|
Temp += 0.01;
|
|
} */
|
|
TempRatio = Temp / model->B3SOIFDtnom;
|
|
|
|
if (selfheat) {
|
|
Vtm = KboQ * Temp;
|
|
|
|
T0 = 1108.0 + Temp;
|
|
T5 = Temp * Temp;
|
|
Eg = 1.16 - 7.02e-4 * T5 / T0;
|
|
T1 = ((7.02e-4 * T5) - T0 * (14.04e-4 * Temp)) / T0 / T0;
|
|
/* T1 = dEg / dT */
|
|
|
|
T2 = 1.9230584e-4; /* T2 = 1 / 300.15^(3/2) */
|
|
T5 = sqrt(Temp);
|
|
T3 = 1.45e10 * Temp * T5 * T2;
|
|
T4 = exp(21.5565981 - Eg / (2.0 * Vtm));
|
|
ni = T3 * T4;
|
|
dni_dT = 2.175e10 * T2 * T5 * T4 + T3 * T4 *
|
|
(-Vtm * T1 + Eg * KboQ) / (2.0 * Vtm * Vtm);
|
|
|
|
T0 = log(1.0e20 * pParam->B3SOIFDnpeak / (ni * ni));
|
|
vbi = Vtm * T0;
|
|
dvbi_dT = KboQ * T0 + Vtm * (-2.0 * dni_dT / ni);
|
|
|
|
if (pParam->B3SOIFDnsub > 0) {
|
|
T0 = log(pParam->B3SOIFDnpeak / pParam->B3SOIFDnsub);
|
|
vfbb = -model->B3SOIFDtype * Vtm*T0;
|
|
dvfbb_dT = -model->B3SOIFDtype * KboQ*T0;
|
|
}
|
|
else {
|
|
T0 = log(-pParam->B3SOIFDnpeak*pParam->B3SOIFDnsub/ni/ni);
|
|
vfbb = -model->B3SOIFDtype * Vtm*T0;
|
|
dvfbb_dT = -model->B3SOIFDtype *
|
|
(KboQ * T0 + Vtm * 2.0 * dni_dT / ni);
|
|
}
|
|
|
|
/* phi = 2.0 * Vtm * log(pParam->B3SOIFDnpeak / ni); */
|
|
phi = here->B3SOIFDphi;
|
|
sqrtPhi = sqrt(phi);
|
|
Xdep0 = sqrt(2.0 * EPSSI / (Charge_q
|
|
* pParam->B3SOIFDnpeak * 1.0e6))
|
|
* sqrtPhi;
|
|
/* Save the values below for phi calculation in B3SOIFDaccept() */
|
|
here->B3SOIFDvtm = Vtm;
|
|
here->B3SOIFDni = ni;
|
|
|
|
/* Use dTx_dVe variables to act as dTx_dT variables */
|
|
|
|
T8 = 1 / model->B3SOIFDtnom;
|
|
T7 = model->B3SOIFDxbjt / pParam->B3SOIFDndiode;
|
|
T0 = pow(TempRatio, T7);
|
|
dT0_dVe = T7 * pow(TempRatio, T7 - 1.0) * T8;
|
|
|
|
T7 = model->B3SOIFDxdif / pParam->B3SOIFDndiode;
|
|
T1 = pow(TempRatio, T7);
|
|
dT1_dVe = T7 * pow(TempRatio, T7 - 1.0) * T8;
|
|
|
|
T7 = model->B3SOIFDxrec / pParam->B3SOIFDndiode / 2.0;
|
|
T2 = pow(TempRatio, T7);
|
|
dT2_dVe = T7 * pow(TempRatio, T7 - 1.0) * T8;
|
|
|
|
T3 = TempRatio - 1.0;
|
|
T4 = Eg300 / pParam->B3SOIFDndiode / Vtm * T3;
|
|
dT4_dVe = Eg300 / pParam->B3SOIFDndiode / Vtm / Vtm *
|
|
(Vtm * T8 - T3 * KboQ);
|
|
T5 = exp(T4);
|
|
dT5_dVe = dT4_dVe * T5;
|
|
T6 = sqrt(T5);
|
|
dT6_dVe = 0.5 / T6 * dT5_dVe;
|
|
|
|
jbjt = pParam->B3SOIFDisbjt * T0 * T5;
|
|
jdif = pParam->B3SOIFDisdif * T1 * T5;
|
|
jrec = pParam->B3SOIFDisrec * T2 * T6;
|
|
djbjt_dT = pParam->B3SOIFDisbjt * (T0 * dT5_dVe + T5 * dT0_dVe);
|
|
djdif_dT = pParam->B3SOIFDisdif * (T1 * dT5_dVe + T5 * dT1_dVe);
|
|
djrec_dT = pParam->B3SOIFDisrec * (T2 * dT6_dVe + T6 * dT2_dVe);
|
|
|
|
T7 = model->B3SOIFDxtun / pParam->B3SOIFDntun;
|
|
T0 = pow(TempRatio, T7);
|
|
jtun = pParam->B3SOIFDistun * T0;
|
|
djtun_dT = pParam->B3SOIFDistun * T7 * pow(TempRatio, T7 - 1.0) * T8;
|
|
|
|
u0temp = pParam->B3SOIFDu0 * pow(TempRatio, pParam->B3SOIFDute);
|
|
du0temp_dT = pParam->B3SOIFDu0 * pParam->B3SOIFDute *
|
|
pow(TempRatio, pParam->B3SOIFDute - 1.0) * T8;
|
|
|
|
vsattemp = pParam->B3SOIFDvsat - pParam->B3SOIFDat * T3;
|
|
dvsattemp_dT = -pParam->B3SOIFDat * T8;
|
|
|
|
rds0 = (pParam->B3SOIFDrdsw + pParam->B3SOIFDprt
|
|
* T3) / pParam->B3SOIFDrds0denom;
|
|
drds0_dT = pParam->B3SOIFDprt / pParam->B3SOIFDrds0denom * T8;
|
|
|
|
ua = pParam->B3SOIFDuatemp + pParam->B3SOIFDua1 * T3;
|
|
ub = pParam->B3SOIFDubtemp + pParam->B3SOIFDub1 * T3;
|
|
uc = pParam->B3SOIFDuctemp + pParam->B3SOIFDuc1 * T3;
|
|
dua_dT = pParam->B3SOIFDua1 * T8;
|
|
dub_dT = pParam->B3SOIFDub1 * T8;
|
|
duc_dT = pParam->B3SOIFDuc1 * T8;
|
|
}
|
|
else {
|
|
vbi = pParam->B3SOIFDvbi;
|
|
vfbb = pParam->B3SOIFDvfbb;
|
|
phi = pParam->B3SOIFDphi;
|
|
sqrtPhi = pParam->B3SOIFDsqrtPhi;
|
|
Xdep0 = pParam->B3SOIFDXdep0;
|
|
jbjt = pParam->B3SOIFDjbjt;
|
|
jdif = pParam->B3SOIFDjdif;
|
|
jrec = pParam->B3SOIFDjrec;
|
|
jtun = pParam->B3SOIFDjtun;
|
|
u0temp = pParam->B3SOIFDu0temp;
|
|
vsattemp = pParam->B3SOIFDvsattemp;
|
|
rds0 = pParam->B3SOIFDrds0;
|
|
ua = pParam->B3SOIFDua;
|
|
ub = pParam->B3SOIFDub;
|
|
uc = pParam->B3SOIFDuc;
|
|
dni_dT = dvbi_dT = dvfbb_dT = djbjt_dT = djdif_dT = 0.0;
|
|
djrec_dT = djtun_dT = du0temp_dT = dvsattemp_dT = 0.0;
|
|
drds0_dT = dua_dT = dub_dT = duc_dT = 0.0;
|
|
}
|
|
|
|
/* TempRatio used for Vth and mobility */
|
|
if (selfheat) {
|
|
TempRatio = Temp / model->B3SOIFDtnom - 1.0;
|
|
}
|
|
else {
|
|
TempRatio = ckt->CKTtemp / model->B3SOIFDtnom - 1.0;
|
|
}
|
|
|
|
/* determine DC current and derivatives */
|
|
vbd = vbs - vds;
|
|
vgd = vgs - vds;
|
|
vgb = vgs - vbs;
|
|
ved = ves - vds;
|
|
veb = ves - vbs;
|
|
vge = vgs - ves;
|
|
vpd = vps - vds;
|
|
|
|
|
|
if (vds >= 0.0)
|
|
{ /* normal mode */
|
|
here->B3SOIFDmode = 1;
|
|
Vds = vds;
|
|
Vgs = vgs;
|
|
Vbs = vbs;
|
|
Vbd = vbd;
|
|
Ves = ves;
|
|
Vps = vps;
|
|
}
|
|
else
|
|
{ /* inverse mode */
|
|
here->B3SOIFDmode = -1;
|
|
Vds = -vds;
|
|
Vgs = vgd;
|
|
Vbs = vbd;
|
|
Vbd = vbs;
|
|
Ves = ved;
|
|
Vps = vpd;
|
|
}
|
|
|
|
|
|
if (here->B3SOIFDdebugMod > 2)
|
|
{
|
|
fprintf(fpdebug, "Vgs=%.4f, Vds=%.4f, Vbs=%.4f, ",
|
|
Vgs, Vds, Vbs);
|
|
fprintf(fpdebug, "Ves=%.4f, Vps=%.4f, Temp=%.1f\n",
|
|
Ves, Vps, Temp);
|
|
}
|
|
|
|
Vesfb = Ves - vfbb;
|
|
Cbox = model->B3SOIFDcbox;
|
|
K1 = pParam->B3SOIFDk1;
|
|
|
|
ChargeComputationNeeded =
|
|
((ckt->CKTmode & (MODEAC | MODETRAN | MODEINITSMSIG)) ||
|
|
((ckt->CKTmode & MODETRANOP) && (ckt->CKTmode & MODEUIC)))
|
|
? 1 : 0;
|
|
|
|
if (here->B3SOIFDdebugMod == -1)
|
|
ChargeComputationNeeded = 1;
|
|
|
|
|
|
|
|
|
|
/* Poly Gate Si Depletion Effect */
|
|
T0 = pParam->B3SOIFDvfb + phi;
|
|
if ((pParam->B3SOIFDngate > 1.e18) && (pParam->B3SOIFDngate < 1.e25)
|
|
&& (Vgs > T0))
|
|
/* added to avoid the problem caused by ngate */
|
|
{ T1 = 1.0e6 * Charge_q * EPSSI * pParam->B3SOIFDngate
|
|
/ (model->B3SOIFDcox * model->B3SOIFDcox);
|
|
T4 = sqrt(1.0 + 2.0 * (Vgs - T0) / T1);
|
|
T2 = T1 * (T4 - 1.0);
|
|
T3 = 0.5 * T2 * T2 / T1; /* T3 = Vpoly */
|
|
T7 = 1.12 - T3 - 0.05;
|
|
T6 = sqrt(T7 * T7 + 0.224);
|
|
T5 = 1.12 - 0.5 * (T7 + T6);
|
|
Vgs_eff = Vgs - T5;
|
|
dVgs_eff_dVg = 1.0 - (0.5 - 0.5 / T4) * (1.0 + T7 / T6);
|
|
}
|
|
else
|
|
{ Vgs_eff = Vgs;
|
|
dVgs_eff_dVg = 1.0;
|
|
}
|
|
|
|
|
|
Leff = pParam->B3SOIFDleff;
|
|
|
|
if (selfheat) {
|
|
Vtm = KboQ * Temp;
|
|
dVtm_dT = KboQ;
|
|
}
|
|
else {
|
|
Vtm = model->B3SOIFDvtm;
|
|
dVtm_dT = 0.0;
|
|
}
|
|
|
|
V0 = vbi - phi;
|
|
|
|
/* Prepare Vbs0t */
|
|
T0 = -pParam->B3SOIFDdvbd1 * pParam->B3SOIFDleff / pParam->B3SOIFDlitl;
|
|
T1 = pParam->B3SOIFDdvbd0 * (exp(0.5*T0) + 2*exp(T0));
|
|
T2 = T1 * (vbi - phi);
|
|
T3 = 0.5 * model->B3SOIFDqsi / model->B3SOIFDcsi;
|
|
Vbs0t = phi - T3 + pParam->B3SOIFDvbsa + T2;
|
|
if (selfheat)
|
|
dVbs0t_dT = T1 * dvbi_dT;
|
|
else
|
|
dVbs0t_dT = 0.0;
|
|
|
|
/* Prepare Vbs0 */
|
|
T0 = 1 + model->B3SOIFDcsieff / Cbox;
|
|
T1 = pParam->B3SOIFDkb1 / T0;
|
|
T2 = T1 * (Vbs0t - Vesfb);
|
|
|
|
/* T6 is Vbs0 before limiting */
|
|
T6 = Vbs0t - T2;
|
|
dT6_dVe = T1;
|
|
if (selfheat)
|
|
dT6_dT = dVbs0t_dT - T1 * (dVbs0t_dT + dvfbb_dT);
|
|
else
|
|
dT6_dT = 0.0;
|
|
|
|
/* limit Vbs0 to below phi */
|
|
T1 = phi - pParam->B3SOIFDdelp;
|
|
T2 = T1 - T6 - DELT_Vbseff;
|
|
T3 = sqrt(T2 * T2 + 4.0 * DELT_Vbseff);
|
|
Vbs0 = T1 - 0.5 * (T2 + T3);
|
|
T4 = 0.5 * (1 + T2/T3);
|
|
dVbs0_dVe = T4 * dT6_dVe;
|
|
if (selfheat) dVbs0_dT = T4 * dT6_dT;
|
|
else dVbs0_dT = 0.0;
|
|
|
|
T1 = Vbs0t - Vbs0 - DELT_Vbsmos;
|
|
T2 = sqrt(T1 * T1 + DELT_Vbsmos * DELT_Vbsmos);
|
|
T3 = 0.5 * (T1 + T2);
|
|
T4 = T3 * model->B3SOIFDcsieff / model->B3SOIFDqsieff;
|
|
Vbs0mos = Vbs0 - 0.5 * T3 * T4;
|
|
T5 = 0.5 * T4 * (1 + T1 / T2);
|
|
dVbs0mos_dVe = dVbs0_dVe * (1 + T5);
|
|
if (selfheat)
|
|
dVbs0mos_dT = dVbs0_dT - (dVbs0t_dT - dVbs0_dT) * T5;
|
|
else
|
|
dVbs0mos_dT = 0.0;
|
|
|
|
/* Prepare Vthfd - treat Vbs0mos as if it were independent variable Vb */
|
|
Phis = phi - Vbs0mos;
|
|
dPhis_dVb = -1;
|
|
sqrtPhis = sqrt(Phis);
|
|
dsqrtPhis_dVb = -0.5 / sqrtPhis;
|
|
Xdep = Xdep0 * sqrtPhis / sqrtPhi;
|
|
dXdep_dVb = (Xdep0 / sqrtPhi)
|
|
* dsqrtPhis_dVb;
|
|
sqrtXdep = sqrt(Xdep);
|
|
|
|
T0 = pParam->B3SOIFDdvt2 * Vbs0mos;
|
|
if (T0 >= - 0.5)
|
|
{ T1 = 1.0 + T0;
|
|
T2 = pParam->B3SOIFDdvt2;
|
|
}
|
|
else /* Added to avoid any discontinuity problems caused by dvt2*/
|
|
{ T4 = 1.0 / (3.0 + 8.0 * T0);
|
|
T1 = (1.0 + 3.0 * T0) * T4;
|
|
T2 = pParam->B3SOIFDdvt2 * T4 * T4;
|
|
}
|
|
lt1 = model->B3SOIFDfactor1 * sqrtXdep * T1;
|
|
dlt1_dVb = model->B3SOIFDfactor1 * (0.5 / sqrtXdep * T1 * dXdep_dVb
|
|
+ sqrtXdep * T2);
|
|
|
|
T0 = pParam->B3SOIFDdvt2w * Vbs0mos;
|
|
if (T0 >= - 0.5)
|
|
{ T1 = 1.0 + T0;
|
|
T2 = pParam->B3SOIFDdvt2w;
|
|
}
|
|
else /* Added to avoid any discontinuity problems caused by
|
|
dvt2w */
|
|
{ T4 = 1.0 / (3.0 + 8.0 * T0);
|
|
T1 = (1.0 + 3.0 * T0) * T4;
|
|
T2 = pParam->B3SOIFDdvt2w * T4 * T4;
|
|
}
|
|
ltw= model->B3SOIFDfactor1 * sqrtXdep * T1;
|
|
dltw_dVb = model->B3SOIFDfactor1 * (0.5 / sqrtXdep * T1 * dXdep_dVb
|
|
+ sqrtXdep * T2);
|
|
|
|
T0 = -0.5 * pParam->B3SOIFDdvt1 * Leff / lt1;
|
|
if (T0 > -EXP_THRESHOLD)
|
|
{ T1 = exp(T0);
|
|
dT1_dVb = -T0 / lt1 * T1 * dlt1_dVb;
|
|
Theta0 = T1 * (1.0 + 2.0 * T1);
|
|
dTheta0_dVb = (1.0 + 4.0 * T1) * dT1_dVb;
|
|
}
|
|
else
|
|
{ T1 = MIN_EXP;
|
|
Theta0 = T1 * (1.0 + 2.0 * T1);
|
|
dTheta0_dVb = 0.0;
|
|
}
|
|
here->B3SOIFDthetavth = pParam->B3SOIFDdvt0 * Theta0;
|
|
Delt_vth = here->B3SOIFDthetavth * V0;
|
|
dDelt_vth_dVb = pParam->B3SOIFDdvt0 * dTheta0_dVb * V0;
|
|
if (selfheat) dDelt_vth_dT = here->B3SOIFDthetavth * dvbi_dT;
|
|
else dDelt_vth_dT = 0.0;
|
|
|
|
T0 = -0.5*pParam->B3SOIFDdvt1w * pParam->B3SOIFDweff*Leff/ltw;
|
|
if (T0 > -EXP_THRESHOLD)
|
|
{ T1 = exp(T0);
|
|
T2 = T1 * (1.0 + 2.0 * T1);
|
|
dT1_dVb = -T0 / ltw * T1 * dltw_dVb;
|
|
dT2_dVb = (1.0 + 4.0 * T1) * dT1_dVb;
|
|
}
|
|
else
|
|
{ T1 = MIN_EXP;
|
|
T2 = T1 * (1.0 + 2.0 * T1);
|
|
dT2_dVb = 0.0;
|
|
}
|
|
T0 = pParam->B3SOIFDdvt0w * T2;
|
|
DeltVthw = T0 * V0;
|
|
dDeltVthw_dVb = pParam->B3SOIFDdvt0w * dT2_dVb * V0;
|
|
if (selfheat) dDeltVthw_dT = T0 * dvbi_dT;
|
|
else dDeltVthw_dT = 0.0;
|
|
|
|
T0 = sqrt(1.0 + pParam->B3SOIFDnlx / Leff);
|
|
T1 = (pParam->B3SOIFDkt1 + pParam->B3SOIFDkt1l / Leff
|
|
+ pParam->B3SOIFDkt2 * Vbs0mos);
|
|
DeltVthtemp = pParam->B3SOIFDk1 * (T0 - 1.0) * sqrtPhi + T1 * TempRatio;
|
|
if (selfheat) dDeltVthtemp_dT = T1 / model->B3SOIFDtnom;
|
|
else dDeltVthtemp_dT = 0.0;
|
|
|
|
tmp2 = model->B3SOIFDtox * phi
|
|
/ (pParam->B3SOIFDweff + pParam->B3SOIFDw0);
|
|
|
|
T3 = pParam->B3SOIFDeta0 + pParam->B3SOIFDetab * Vbs0mos;
|
|
if (T3 < 1.0e-4) /* avoid discontinuity problems caused by etab */
|
|
{ T9 = 1.0 / (3.0 - 2.0e4 * T3);
|
|
T3 = (2.0e-4 - T3) * T9;
|
|
T4 = T9 * T9 * pParam->B3SOIFDetab;
|
|
dT3_dVb = T4;
|
|
}
|
|
else
|
|
{
|
|
dT3_dVb = pParam->B3SOIFDetab;
|
|
}
|
|
DIBL_Sft = T3 * pParam->B3SOIFDtheta0vb0 * Vds;
|
|
dDIBL_Sft_dVd = T3 * pParam->B3SOIFDtheta0vb0;
|
|
dDIBL_Sft_dVb = pParam->B3SOIFDtheta0vb0 * Vds * dT3_dVb;
|
|
|
|
Vthfd = model->B3SOIFDtype * pParam->B3SOIFDvth0 + pParam->B3SOIFDk1
|
|
* (sqrtPhis - sqrtPhi) - pParam->B3SOIFDk2
|
|
* Vbs0mos-Delt_vth-DeltVthw +(pParam->B3SOIFDk3 +pParam->B3SOIFDk3b
|
|
* Vbs0mos) * tmp2 + DeltVthtemp - DIBL_Sft;
|
|
|
|
T6 = pParam->B3SOIFDk3b * tmp2 - pParam->B3SOIFDk2 +
|
|
pParam->B3SOIFDkt2 * TempRatio;
|
|
dVthfd_dVd = -dDIBL_Sft_dVd;
|
|
T7 = pParam->B3SOIFDk1 * dsqrtPhis_dVb
|
|
- dDelt_vth_dVb - dDeltVthw_dVb
|
|
+ T6 - dDIBL_Sft_dVb;
|
|
dVthfd_dVe = T7 * dVbs0mos_dVe;
|
|
if (selfheat)
|
|
dVthfd_dT = dDeltVthtemp_dT - dDelt_vth_dT - dDeltVthw_dT
|
|
+ T7 * dVbs0mos_dT;
|
|
else
|
|
dVthfd_dT = 0.0;
|
|
|
|
/* Effective Vbs0 and Vbs0t for all Vgs */
|
|
T1 = Vthfd - Vgs_eff - DELT_Vbs0eff;
|
|
T2 = sqrt(T1 * T1 + DELT_Vbs0eff * DELT_Vbs0eff );
|
|
|
|
Vbs0teff = Vbs0t - 0.5 * (T1 + T2);
|
|
dVbs0teff_dVg = 0.5 * (1 + T1/T2) * dVgs_eff_dVg;
|
|
dVbs0teff_dVd = - 0.5 * (1 + T1 / T2) * dVthfd_dVd;
|
|
dVbs0teff_dVe = - 0.5 * (1 + T1 / T2) * dVthfd_dVe;
|
|
if (selfheat)
|
|
dVbs0teff_dT = dVbs0t_dT - 0.5 * (1 + T1 / T2) * dVthfd_dT;
|
|
else
|
|
dVbs0teff_dT = 0.0;
|
|
|
|
/* Calculate nfb */
|
|
T3 = 1 / (K1 * K1);
|
|
T4 = pParam->B3SOIFDkb3 * Cbox / model->B3SOIFDcox;
|
|
T8 = sqrt(phi - Vbs0mos);
|
|
T5 = sqrt(1 + 4 * T3 * (phi + K1 * T8 - Vbs0mos));
|
|
T6 = 1 + T4 * T5;
|
|
Nfb = model->B3SOIFDnfb = 1 / T6;
|
|
T7 = 2 * T3 * T4 * Nfb * Nfb / T5 * (0.5 * K1 / T8 + 1);
|
|
Vbs0eff = Vbs0 - Nfb * 0.5 * (T1 + T2);
|
|
dVbs0eff_dVg = Nfb * 0.5 * (1 + T1/T2) * dVgs_eff_dVg;
|
|
dVbs0eff_dVd = - Nfb * 0.5 * (1 + T1 / T2) * dVthfd_dVd;
|
|
dVbs0eff_dVe = dVbs0_dVe - Nfb * 0.5 * (1 + T1 / T2)
|
|
* dVthfd_dVe - T7 * 0.5 * (T1 + T2) * dVbs0mos_dVe;
|
|
if (selfheat)
|
|
dVbs0eff_dT = dVbs0_dT - Nfb * 0.5 * (1 + T1 / T2)
|
|
* dVthfd_dT - T7 * 0.5 * (T1 + T2) * dVbs0mos_dT;
|
|
else
|
|
dVbs0eff_dT = 0.0;
|
|
|
|
/* Simple check of Vbs */
|
|
/* Prepare Vbsdio */
|
|
Vbs = Vbsdio = Vbs0eff;
|
|
dVbsdio_dVg = dVbs0eff_dVg;
|
|
dVbsdio_dVd = dVbs0eff_dVd;
|
|
dVbsdio_dVe = dVbs0eff_dVe;
|
|
dVbsdio_dT = dVbs0eff_dT;
|
|
dVbsdio_dVb = 0.0;
|
|
|
|
/* Prepare Vbseff */
|
|
T1 = Vbs0teff - Vbsdio - DELT_Vbsmos;
|
|
T2 = sqrt(T1 * T1 + DELT_Vbsmos * DELT_Vbsmos);
|
|
T3 = 0.5 * (T1 + T2);
|
|
T5 = 0.5 * (1 + T1/T2);
|
|
dT3_dVg = T5 * (dVbs0teff_dVg - dVbsdio_dVg);
|
|
dT3_dVd = T5 * (dVbs0teff_dVd - dVbsdio_dVd);
|
|
dT3_dVb = - T5 * dVbsdio_dVb;
|
|
dT3_dVe = T5 * (dVbs0teff_dVe - dVbsdio_dVe);
|
|
if (selfheat) dT3_dT = T5 * (dVbs0teff_dT - dVbsdio_dT);
|
|
else dT3_dT = 0.0;
|
|
T4 = T3 * model->B3SOIFDcsieff / model->B3SOIFDqsieff;
|
|
|
|
Vbsmos = Vbsdio - 0.5 * T3 * T4;
|
|
dVbsmos_dVg = dVbsdio_dVg - T4 * dT3_dVg;
|
|
dVbsmos_dVd = dVbsdio_dVd - T4 * dT3_dVd;
|
|
dVbsmos_dVb = dVbsdio_dVb - T4 * dT3_dVb;
|
|
dVbsmos_dVe = dVbsdio_dVe - T4 * dT3_dVe;
|
|
if (selfheat) dVbsmos_dT = dVbsdio_dT - T4 * dT3_dT;
|
|
else dVbsmos_dT = 0.0;
|
|
|
|
/* Prepare Vcs */
|
|
Vcs = Vbsdio - Vbs0eff;
|
|
dVcs_dVb = dVbsdio_dVb;
|
|
dVcs_dVg = dVbsdio_dVg - dVbs0eff_dVg;
|
|
dVcs_dVd = dVbsdio_dVd - dVbs0eff_dVd;
|
|
dVcs_dVe = dVbsdio_dVe - dVbs0eff_dVe;
|
|
dVcs_dT = dVbsdio_dT - dVbs0eff_dT;
|
|
|
|
/* Check Vps */
|
|
/* Note : if Vps is less Vbs0eff => non-physical */
|
|
T1 = Vps - Vbs0eff + DELT_Vbs0dio;
|
|
T2 = sqrt(T1 * T1 + DELT_Vbs0dio * DELT_Vbs0dio);
|
|
T3 = 0.5 * (1 + T1/T2);
|
|
Vpsdio = Vbs0eff + 0.5 * (T1 + T2);
|
|
dVpsdio_dVg = (1 - T3) * dVbs0eff_dVg;
|
|
dVpsdio_dVd = (1 - T3) * dVbs0eff_dVd;
|
|
dVpsdio_dVe = (1 - T3) * dVbs0eff_dVe;
|
|
if (selfheat) dVpsdio_dT = (1 - T3) * dVbs0eff_dT;
|
|
else dVpsdio_dT = 0.0;
|
|
dVpsdio_dVp = T3;
|
|
Vbp = Vbsdio - Vpsdio;
|
|
dVbp_dVb = dVbsdio_dVb;
|
|
dVbp_dVg = dVbsdio_dVg - dVpsdio_dVg;
|
|
dVbp_dVd = dVbsdio_dVd - dVpsdio_dVd;
|
|
dVbp_dVe = dVbsdio_dVe - dVpsdio_dVe;
|
|
dVbp_dT = dVbsdio_dT - dVpsdio_dT;
|
|
dVbp_dVp = - dVpsdio_dVp;
|
|
|
|
here->B3SOIFDvbsdio = Vbsdio;
|
|
here->B3SOIFDvbs0eff = Vbs0eff;
|
|
|
|
T1 = phi - pParam->B3SOIFDdelp;
|
|
T2 = T1 - Vbsmos - DELT_Vbseff;
|
|
T3 = sqrt(T2 * T2 + 4.0 * DELT_Vbseff * T1);
|
|
Vbseff = T1 - 0.5 * (T2 + T3);
|
|
T4 = 0.5 * (1 + T2/T3);
|
|
dVbseff_dVg = T4 * dVbsmos_dVg;
|
|
dVbseff_dVd = T4 * dVbsmos_dVd;
|
|
dVbseff_dVb = T4 * dVbsmos_dVb;
|
|
dVbseff_dVe = T4 * dVbsmos_dVe;
|
|
if (selfheat) dVbseff_dT = T4 * dVbsmos_dT;
|
|
else dVbseff_dT = 0.0;
|
|
|
|
here->B3SOIFDvbseff = Vbseff;
|
|
|
|
Phis = phi - Vbseff;
|
|
dPhis_dVb = -1;
|
|
sqrtPhis = sqrt(Phis);
|
|
dsqrtPhis_dVb = -0.5 / sqrtPhis ;
|
|
|
|
Xdep = Xdep0 * sqrtPhis / sqrtPhi;
|
|
dXdep_dVb = (Xdep0 / sqrtPhi)
|
|
* dsqrtPhis_dVb;
|
|
|
|
/* Vth Calculation */
|
|
T3 = sqrt(Xdep);
|
|
|
|
T0 = pParam->B3SOIFDdvt2 * Vbseff;
|
|
if (T0 >= - 0.5)
|
|
{ T1 = 1.0 + T0;
|
|
T2 = pParam->B3SOIFDdvt2 ;
|
|
}
|
|
else /* Added to avoid any discontinuity problems caused by dvt2 */
|
|
{ T4 = 1.0 / (3.0 + 8.0 * T0);
|
|
T1 = (1.0 + 3.0 * T0) * T4;
|
|
T2 = pParam->B3SOIFDdvt2 * T4 * T4 ;
|
|
}
|
|
lt1 = model->B3SOIFDfactor1 * T3 * T1;
|
|
dlt1_dVb =model->B3SOIFDfactor1 * (0.5 / T3 * T1 * dXdep_dVb + T3 * T2);
|
|
|
|
T0 = pParam->B3SOIFDdvt2w * Vbseff;
|
|
if (T0 >= - 0.5)
|
|
{ T1 = 1.0 + T0;
|
|
T2 = pParam->B3SOIFDdvt2w ;
|
|
}
|
|
else /* Added to avoid any discontinuity problems caused by dvt2w */
|
|
{ T4 = 1.0 / (3.0 + 8.0 * T0);
|
|
T1 = (1.0 + 3.0 * T0) * T4;
|
|
T2 = pParam->B3SOIFDdvt2w * T4 * T4 ;
|
|
}
|
|
ltw= model->B3SOIFDfactor1 * T3 * T1;
|
|
dltw_dVb=model->B3SOIFDfactor1*(0.5 / T3 * T1 * dXdep_dVb + T3 * T2);
|
|
|
|
T0 = -0.5 * pParam->B3SOIFDdvt1 * Leff / lt1;
|
|
if (T0 > -EXP_THRESHOLD)
|
|
{ T1 = exp(T0);
|
|
Theta0 = T1 * (1.0 + 2.0 * T1);
|
|
dT1_dVb = -T0 / lt1 * T1 * dlt1_dVb;
|
|
dTheta0_dVb = (1.0 + 4.0 * T1) * dT1_dVb;
|
|
}
|
|
else
|
|
{ T1 = MIN_EXP;
|
|
Theta0 = T1 * (1.0 + 2.0 * T1);
|
|
dTheta0_dVb = 0.0;
|
|
}
|
|
|
|
here->B3SOIFDthetavth = pParam->B3SOIFDdvt0 * Theta0;
|
|
Delt_vth = here->B3SOIFDthetavth * V0;
|
|
dDelt_vth_dVb = pParam->B3SOIFDdvt0 * dTheta0_dVb * V0;
|
|
if (selfheat) dDelt_vth_dT = here->B3SOIFDthetavth * dvbi_dT;
|
|
else dDelt_vth_dT = 0.0;
|
|
|
|
T0 = -0.5 * pParam->B3SOIFDdvt1w * pParam->B3SOIFDweff * Leff / ltw;
|
|
if (T0 > -EXP_THRESHOLD)
|
|
{ T1 = exp(T0);
|
|
T2 = T1 * (1.0 + 2.0 * T1);
|
|
dT1_dVb = -T0 / ltw * T1 * dltw_dVb;
|
|
dT2_dVb = (1.0 + 4.0 * T1) * dT1_dVb;
|
|
}
|
|
else
|
|
{ T1 = MIN_EXP;
|
|
T2 = T1 * (1.0 + 2.0 * T1);
|
|
dT2_dVb = 0.0;
|
|
}
|
|
|
|
T0 = pParam->B3SOIFDdvt0w * T2;
|
|
DeltVthw = T0 * V0;
|
|
dDeltVthw_dVb = pParam->B3SOIFDdvt0w * dT2_dVb * V0;
|
|
if (selfheat) dDeltVthw_dT = T0 * dvbi_dT;
|
|
else dDeltVthw_dT = 0.0;
|
|
|
|
T0 = sqrt(1.0 + pParam->B3SOIFDnlx / Leff);
|
|
T1 = (pParam->B3SOIFDkt1 + pParam->B3SOIFDkt1l / Leff
|
|
+ pParam->B3SOIFDkt2 * Vbseff);
|
|
DeltVthtemp = pParam->B3SOIFDk1 * (T0 - 1.0) * sqrtPhi + T1 * TempRatio;
|
|
if (selfheat)
|
|
dDeltVthtemp_dT = T1 / model->B3SOIFDtnom;
|
|
else
|
|
dDeltVthtemp_dT = 0.0;
|
|
|
|
tmp2 = model->B3SOIFDtox * phi
|
|
/ (pParam->B3SOIFDweff + pParam->B3SOIFDw0);
|
|
|
|
T3 = pParam->B3SOIFDeta0 + pParam->B3SOIFDetab * Vbseff;
|
|
if (T3 < 1.0e-4) /* avoid discontinuity problems caused by etab */
|
|
{ T9 = 1.0 / (3.0 - 2.0e4 * T3);
|
|
T3 = (2.0e-4 - T3) * T9;
|
|
T4 = T9 * T9 * pParam->B3SOIFDetab;
|
|
dT3_dVb = T4 ;
|
|
}
|
|
else
|
|
{
|
|
dT3_dVb = pParam->B3SOIFDetab ;
|
|
}
|
|
DIBL_Sft = T3 * pParam->B3SOIFDtheta0vb0 * Vds;
|
|
dDIBL_Sft_dVd = pParam->B3SOIFDtheta0vb0 * T3;
|
|
dDIBL_Sft_dVb = pParam->B3SOIFDtheta0vb0 * Vds * dT3_dVb;
|
|
|
|
Vth = model->B3SOIFDtype * pParam->B3SOIFDvth0 + pParam->B3SOIFDk1
|
|
* (sqrtPhis - sqrtPhi) - pParam->B3SOIFDk2
|
|
* Vbseff- Delt_vth - DeltVthw +(pParam->B3SOIFDk3 + pParam->B3SOIFDk3b
|
|
* Vbseff) * tmp2 + DeltVthtemp - DIBL_Sft;
|
|
|
|
here->B3SOIFDvon = Vth;
|
|
|
|
T6 = pParam->B3SOIFDk3b * tmp2 - pParam->B3SOIFDk2
|
|
+ pParam->B3SOIFDkt2 * TempRatio;
|
|
dVth_dVb = pParam->B3SOIFDk1 * dsqrtPhis_dVb
|
|
- dDelt_vth_dVb - dDeltVthw_dVb
|
|
+ T6 - dDIBL_Sft_dVb; /* this is actually dVth_dVbseff */
|
|
dVth_dVd = -dDIBL_Sft_dVd;
|
|
if (selfheat) dVth_dT = dDeltVthtemp_dT - dDelt_vth_dT - dDeltVthw_dT;
|
|
else dVth_dT = 0.0;
|
|
|
|
/* Calculate n */
|
|
T2 = pParam->B3SOIFDnfactor * EPSSI / Xdep;
|
|
dT2_dVb = - T2 / Xdep * dXdep_dVb;
|
|
|
|
T3 = pParam->B3SOIFDcdsc + pParam->B3SOIFDcdscb * Vbseff
|
|
+ pParam->B3SOIFDcdscd * Vds;
|
|
dT3_dVb = pParam->B3SOIFDcdscb;
|
|
dT3_dVd = pParam->B3SOIFDcdscd;
|
|
|
|
T4 = (T2 + T3 * Theta0 + pParam->B3SOIFDcit) / model->B3SOIFDcox;
|
|
dT4_dVb = (dT2_dVb + Theta0 * dT3_dVb + dTheta0_dVb * T3)
|
|
/ model->B3SOIFDcox;
|
|
dT4_dVd = Theta0 * dT3_dVd / model->B3SOIFDcox;
|
|
|
|
if (T4 >= -0.5)
|
|
{ n = 1.0 + T4;
|
|
dn_dVb = dT4_dVb;
|
|
dn_dVd = dT4_dVd;
|
|
}
|
|
else
|
|
/* avoid discontinuity problems caused by T4 */
|
|
{ T0 = 1.0 / (3.0 + 8.0 * T4);
|
|
n = (1.0 + 3.0 * T4) * T0;
|
|
T0 *= T0;
|
|
dn_dVb = T0 * dT4_dVb;
|
|
dn_dVd = T0 * dT4_dVd;
|
|
}
|
|
|
|
/* Effective Vgst (Vgsteff) Calculation */
|
|
|
|
Vgst = Vgs_eff - Vth;
|
|
|
|
T10 = 2.0 * n * Vtm;
|
|
VgstNVt = Vgst / T10;
|
|
ExpArg = (2.0 * pParam->B3SOIFDvoff - Vgst) / T10;
|
|
|
|
/* MCJ: Very small Vgst */
|
|
if (VgstNVt > EXP_THRESHOLD)
|
|
{ Vgsteff = Vgst;
|
|
/* T0 is dVgsteff_dVbseff */
|
|
T0 = -dVth_dVb;
|
|
dVgsteff_dVg = dVgs_eff_dVg + T0 * dVbseff_dVg;
|
|
dVgsteff_dVd = -dVth_dVd + T0 * dVbseff_dVd;
|
|
dVgsteff_dVb = T0 * dVbseff_dVb;
|
|
dVgsteff_dVe = T0 * dVbseff_dVe;
|
|
if (selfheat)
|
|
dVgsteff_dT = -dVth_dT + T0 * dVbseff_dT;
|
|
else
|
|
dVgsteff_dT = 0.0;
|
|
}
|
|
else if (ExpArg > EXP_THRESHOLD)
|
|
{ T0 = (Vgst - pParam->B3SOIFDvoff) / (n * Vtm);
|
|
ExpVgst = exp(T0);
|
|
Vgsteff = Vtm * pParam->B3SOIFDcdep0 / model->B3SOIFDcox * ExpVgst;
|
|
T3 = Vgsteff / (n * Vtm) ;
|
|
/* T1 is dVgsteff_dVbseff */
|
|
T1 = -T3 * (dVth_dVb + T0 * Vtm * dn_dVb);
|
|
dVgsteff_dVg = T3 * dVgs_eff_dVg + T1 * dVbseff_dVg;
|
|
dVgsteff_dVd = -T3 * (dVth_dVd + T0 * Vtm * dn_dVd) + T1 * dVbseff_dVd;
|
|
dVgsteff_dVe = T1 * dVbseff_dVe;
|
|
dVgsteff_dVb = T1 * dVbseff_dVb;
|
|
if (selfheat)
|
|
dVgsteff_dT = -T3 * (dVth_dT + T0 * dVtm_dT * n)
|
|
+ Vgsteff / Temp + T1 * dVbseff_dT;
|
|
else
|
|
dVgsteff_dT = 0.0;
|
|
}
|
|
else
|
|
{ ExpVgst = exp(VgstNVt);
|
|
T1 = T10 * log(1.0 + ExpVgst);
|
|
dT1_dVg = ExpVgst / (1.0 + ExpVgst);
|
|
dT1_dVb = -dT1_dVg * (dVth_dVb + Vgst / n * dn_dVb)
|
|
+ T1 / n * dn_dVb;
|
|
dT1_dVd = -dT1_dVg * (dVth_dVd + Vgst / n * dn_dVd)
|
|
+ T1 / n * dn_dVd;
|
|
T3 = (1.0 / Temp);
|
|
if (selfheat)
|
|
dT1_dT = -dT1_dVg * (dVth_dT + Vgst * T3) + T1 * T3;
|
|
else
|
|
dT1_dT = 0.0;
|
|
|
|
dT2_dVg = -model->B3SOIFDcox / (Vtm * pParam->B3SOIFDcdep0)
|
|
* exp(ExpArg);
|
|
T2 = 1.0 - T10 * dT2_dVg;
|
|
dT2_dVd = -dT2_dVg * (dVth_dVd - 2.0 * Vtm * ExpArg * dn_dVd)
|
|
+ (T2 - 1.0) / n * dn_dVd;
|
|
dT2_dVb = -dT2_dVg * (dVth_dVb - 2.0 * Vtm * ExpArg * dn_dVb)
|
|
+ (T2 - 1.0) / n * dn_dVb;
|
|
if (selfheat)
|
|
dT2_dT = -dT2_dVg * (dVth_dT - ExpArg * T10 * T3);
|
|
else
|
|
dT2_dT = 0.0;
|
|
|
|
Vgsteff = T1 / T2;
|
|
T3 = T2 * T2;
|
|
/* T4 is dVgsteff_dVbseff */
|
|
T4 = (T2 * dT1_dVb - T1 * dT2_dVb) / T3;
|
|
dVgsteff_dVb = T4 * dVbseff_dVb;
|
|
dVgsteff_dVe = T4 * dVbseff_dVe;
|
|
dVgsteff_dVg = (T2 * dT1_dVg - T1 * dT2_dVg) / T3 * dVgs_eff_dVg
|
|
+ T4 * dVbseff_dVg;
|
|
dVgsteff_dVd = (T2 * dT1_dVd - T1 * dT2_dVd) / T3 + T4 * dVbseff_dVd;
|
|
if (selfheat)
|
|
dVgsteff_dT = (T2 * dT1_dT - T1 * dT2_dT) / T3 + T4 * dVbseff_dT;
|
|
else
|
|
dVgsteff_dT = 0.0;
|
|
}
|
|
Vgst2Vtm = Vgsteff + 2.0 * Vtm;
|
|
if (selfheat) dVgst2Vtm_dT = 2.0 * dVtm_dT;
|
|
else dVgst2Vtm_dT = 0.0;
|
|
|
|
/* Calculate Effective Channel Geometry */
|
|
T9 = sqrtPhis - sqrtPhi;
|
|
Weff = pParam->B3SOIFDweff - 2.0 * (pParam->B3SOIFDdwg * Vgsteff
|
|
+ pParam->B3SOIFDdwb * T9);
|
|
dWeff_dVg = -2.0 * pParam->B3SOIFDdwg;
|
|
dWeff_dVb = -2.0 * pParam->B3SOIFDdwb * dsqrtPhis_dVb;
|
|
|
|
if (Weff < 2.0e-8) /* to avoid the discontinuity problem due to Weff*/
|
|
{ T0 = 1.0 / (6.0e-8 - 2.0 * Weff);
|
|
Weff = 2.0e-8 * (4.0e-8 - Weff) * T0;
|
|
T0 *= T0 * 4.0e-16;
|
|
dWeff_dVg *= T0;
|
|
dWeff_dVb *= T0;
|
|
}
|
|
|
|
T0 = pParam->B3SOIFDprwg * Vgsteff + pParam->B3SOIFDprwb * T9;
|
|
if (T0 >= -0.9)
|
|
{ Rds = rds0 * (1.0 + T0);
|
|
dRds_dVg = rds0 * pParam->B3SOIFDprwg;
|
|
dRds_dVb = rds0 * pParam->B3SOIFDprwb * dsqrtPhis_dVb;
|
|
if (selfheat) dRds_dT = (1.0 + T0) * drds0_dT;
|
|
else dRds_dT = 0.0;
|
|
}
|
|
else
|
|
/* to avoid the discontinuity problem due to prwg and prwb*/
|
|
{ T1 = 1.0 / (17.0 + 20.0 * T0);
|
|
Rds = rds0 * (0.8 + T0) * T1;
|
|
T1 *= T1;
|
|
dRds_dVg = rds0 * pParam->B3SOIFDprwg * T1;
|
|
dRds_dVb = rds0 * pParam->B3SOIFDprwb * dsqrtPhis_dVb
|
|
* T1;
|
|
if (selfheat) dRds_dT = (0.8 + T0) * T1 * drds0_dT;
|
|
else dRds_dT = 0.0;
|
|
}
|
|
|
|
/* Calculate Abulk */
|
|
if (pParam->B3SOIFDa0 == 0.0)
|
|
{
|
|
Abulk0 = Abulk = dAbulk0_dVb = dAbulk_dVg = dAbulk_dVb = 0.0;
|
|
}
|
|
else
|
|
{
|
|
T1 = 0.5 * pParam->B3SOIFDk1 / sqrtPhi;
|
|
T9 = sqrt(model->B3SOIFDxj * Xdep);
|
|
tmp1 = Leff + 2.0 * T9;
|
|
T5 = Leff / tmp1;
|
|
tmp2 = pParam->B3SOIFDa0 * T5;
|
|
tmp3 = pParam->B3SOIFDweff + pParam->B3SOIFDb1;
|
|
tmp4 = pParam->B3SOIFDb0 / tmp3;
|
|
T2 = tmp2 + tmp4;
|
|
dT2_dVb = -T9 * tmp2 / tmp1 / Xdep * dXdep_dVb;
|
|
T6 = T5 * T5;
|
|
T7 = T5 * T6;
|
|
|
|
Abulk0 = T1 * T2;
|
|
dAbulk0_dVb = T1 * dT2_dVb;
|
|
|
|
T8 = pParam->B3SOIFDags * pParam->B3SOIFDa0 * T7;
|
|
dAbulk_dVg = -T1 * T8;
|
|
Abulk = Abulk0 + dAbulk_dVg * Vgsteff;
|
|
|
|
dAbulk_dVb = dAbulk0_dVb - T8 * Vgsteff * 3.0 * T1 * dT2_dVb
|
|
/ tmp2;
|
|
}
|
|
|
|
if (Abulk0 < 0.01)
|
|
{
|
|
T9 = 1.0 / (3.0 - 200.0 * Abulk0);
|
|
Abulk0 = (0.02 - Abulk0) * T9;
|
|
dAbulk0_dVb *= T9 * T9;
|
|
}
|
|
|
|
if (Abulk < 0.01)
|
|
{
|
|
T9 = 1.0 / (3.0 - 200.0 * Abulk);
|
|
Abulk = (0.02 - Abulk) * T9;
|
|
dAbulk_dVb *= T9 * T9;
|
|
}
|
|
|
|
T2 = pParam->B3SOIFDketa * Vbseff;
|
|
if (T2 >= -0.9)
|
|
{ T0 = 1.0 / (1.0 + T2);
|
|
dT0_dVb = -pParam->B3SOIFDketa * T0 * T0 ;
|
|
}
|
|
else
|
|
/* added to avoid the problems caused by Keta */
|
|
{ T1 = 1.0 / (0.8 + T2);
|
|
T0 = (17.0 + 20.0 * T2) * T1;
|
|
dT0_dVb = -pParam->B3SOIFDketa * T1 * T1 ;
|
|
}
|
|
dAbulk_dVg *= T0;
|
|
dAbulk_dVb = dAbulk_dVb * T0 + Abulk * dT0_dVb;
|
|
dAbulk0_dVb = dAbulk0_dVb * T0 + Abulk0 * dT0_dVb;
|
|
Abulk *= T0;
|
|
Abulk0 *= T0;
|
|
|
|
Abulk += 1;
|
|
Abulk0 += 1;
|
|
|
|
/* Prepare Abeff */
|
|
T0 = pParam->B3SOIFDabp * Vgst2Vtm;
|
|
T1 = 1 - Vcs / T0 - DELT_Xcsat;
|
|
T2 = sqrt(T1 * T1 + DELT_Xcsat * DELT_Xcsat);
|
|
T3 = 1 - 0.5 * (T1 + T2);
|
|
T5 = -0.5 * (1 + T1 / T2);
|
|
dT1_dVg = Vcs / Vgst2Vtm / T0;
|
|
dT3_dVg = T5 * dT1_dVg;
|
|
dT1_dVc = - 1 / T0;
|
|
dT3_dVc = T5 * dT1_dVc;
|
|
|
|
Xcsat = pParam->B3SOIFDmxc * T3 * T3 + (1 - pParam->B3SOIFDmxc)*T3;
|
|
T4 = 2 * pParam->B3SOIFDmxc * T3 + (1 - pParam->B3SOIFDmxc);
|
|
dXcsat_dVg = T4 * dT3_dVg;
|
|
dXcsat_dVc = T4 * dT3_dVc;
|
|
|
|
Abeff = Xcsat * Abulk + (1 - Xcsat) * model->B3SOIFDadice;
|
|
T0 = Xcsat * dAbulk_dVg + Abulk * dXcsat_dVg;
|
|
dAbeff_dVg = T0 - model->B3SOIFDadice * dXcsat_dVg;
|
|
dAbeff_dVb = Xcsat * dAbulk_dVb;
|
|
dAbeff_dVc = (Abulk - model->B3SOIFDadice) * dXcsat_dVc;
|
|
here->B3SOIFDabeff = Abeff;
|
|
|
|
/* Mobility calculation */
|
|
if (model->B3SOIFDmobMod == 1)
|
|
{ T0 = Vgsteff + Vth + Vth;
|
|
T2 = ua + uc * Vbseff;
|
|
T3 = T0 / model->B3SOIFDtox;
|
|
T5 = T3 * (T2 + ub * T3);
|
|
dDenomi_dVg = (T2 + 2.0 * ub * T3) / model->B3SOIFDtox;
|
|
dDenomi_dVd = dDenomi_dVg * 2 * dVth_dVd;
|
|
dDenomi_dVb = dDenomi_dVg * 2 * dVth_dVb + uc * T3 ;
|
|
if (selfheat)
|
|
dDenomi_dT = dDenomi_dVg * 2 * dVth_dT
|
|
+ (dua_dT + Vbseff * duc_dT
|
|
+ dub_dT * T3 ) * T3;
|
|
else
|
|
dDenomi_dT = 0.0;
|
|
}
|
|
else if (model->B3SOIFDmobMod == 2)
|
|
{ T5 = Vgsteff / model->B3SOIFDtox * (ua
|
|
+ uc * Vbseff + ub * Vgsteff
|
|
/ model->B3SOIFDtox);
|
|
dDenomi_dVg = (ua + uc * Vbseff
|
|
+ 2.0 * ub * Vgsteff / model->B3SOIFDtox)
|
|
/ model->B3SOIFDtox;
|
|
dDenomi_dVd = 0.0;
|
|
dDenomi_dVb = Vgsteff * uc / model->B3SOIFDtox ;
|
|
if (selfheat)
|
|
dDenomi_dT = Vgsteff / model->B3SOIFDtox
|
|
* (dua_dT + Vbseff * duc_dT + dub_dT
|
|
* Vgsteff / model->B3SOIFDtox);
|
|
else
|
|
dDenomi_dT = 0.0;
|
|
}
|
|
else /* mobMod == 3 */
|
|
{ T0 = Vgsteff + Vth + Vth;
|
|
T2 = 1.0 + uc * Vbseff;
|
|
T3 = T0 / model->B3SOIFDtox;
|
|
T4 = T3 * (ua + ub * T3);
|
|
T5 = T4 * T2;
|
|
dDenomi_dVg = (ua + 2.0 * ub * T3) * T2
|
|
/ model->B3SOIFDtox;
|
|
dDenomi_dVd = dDenomi_dVg * 2.0 * dVth_dVd;
|
|
dDenomi_dVb = dDenomi_dVg * 2.0 * dVth_dVb
|
|
+ uc * T4 ;
|
|
if (selfheat)
|
|
dDenomi_dT = dDenomi_dVg * 2.0 * dVth_dT
|
|
+ (dua_dT + dub_dT * T3) * T3 * T2
|
|
+ T4 * Vbseff * duc_dT;
|
|
else
|
|
dDenomi_dT = 0.0;
|
|
}
|
|
|
|
if (T5 >= -0.8)
|
|
{ Denomi = 1.0 + T5;
|
|
}
|
|
else /* Added to avoid the discontinuity problem caused by ua and ub*/
|
|
{ T9 = 1.0 / (7.0 + 10.0 * T5);
|
|
Denomi = (0.6 + T5) * T9;
|
|
T9 *= T9;
|
|
dDenomi_dVg *= T9;
|
|
dDenomi_dVd *= T9;
|
|
dDenomi_dVb *= T9;
|
|
if (selfheat) dDenomi_dT *= T9;
|
|
else dDenomi_dT = 0.0;
|
|
}
|
|
|
|
here->B3SOIFDueff = ueff = u0temp / Denomi;
|
|
T9 = -ueff / Denomi;
|
|
dueff_dVg = T9 * dDenomi_dVg;
|
|
dueff_dVd = T9 * dDenomi_dVd;
|
|
dueff_dVb = T9 * dDenomi_dVb;
|
|
if (selfheat) dueff_dT = T9 * dDenomi_dT + du0temp_dT / Denomi;
|
|
else dueff_dT = 0.0;
|
|
|
|
/* Saturation Drain Voltage Vdsat */
|
|
WVCox = Weff * vsattemp * model->B3SOIFDcox;
|
|
WVCoxRds = WVCox * Rds;
|
|
|
|
/* dWVCoxRds_dT = WVCox * dRds_dT
|
|
+ Weff * model->B3SOIFDcox * Rds * dvsattemp_dT; */
|
|
|
|
Esat = 2.0 * vsattemp / ueff;
|
|
EsatL = Esat * Leff;
|
|
T0 = -EsatL /ueff;
|
|
dEsatL_dVg = T0 * dueff_dVg;
|
|
dEsatL_dVd = T0 * dueff_dVd;
|
|
dEsatL_dVb = T0 * dueff_dVb;
|
|
if (selfheat)
|
|
dEsatL_dT = T0 * dueff_dT + EsatL / vsattemp * dvsattemp_dT;
|
|
else
|
|
dEsatL_dT = 0.0;
|
|
|
|
/* Sqrt() */
|
|
a1 = pParam->B3SOIFDa1;
|
|
if (a1 == 0.0)
|
|
{ Lambda = pParam->B3SOIFDa2;
|
|
dLambda_dVg = 0.0;
|
|
}
|
|
else if (a1 > 0.0)
|
|
/* Added to avoid the discontinuity problem caused by a1 and a2 (Lambda) */
|
|
{ T0 = 1.0 - pParam->B3SOIFDa2;
|
|
T1 = T0 - pParam->B3SOIFDa1 * Vgsteff - 0.0001;
|
|
T2 = sqrt(T1 * T1 + 0.0004 * T0);
|
|
Lambda = pParam->B3SOIFDa2 + T0 - 0.5 * (T1 + T2);
|
|
dLambda_dVg = 0.5 * pParam->B3SOIFDa1 * (1.0 + T1 / T2);
|
|
}
|
|
else
|
|
{ T1 = pParam->B3SOIFDa2 + pParam->B3SOIFDa1 * Vgsteff - 0.0001;
|
|
T2 = sqrt(T1 * T1 + 0.0004 * pParam->B3SOIFDa2);
|
|
Lambda = 0.5 * (T1 + T2);
|
|
dLambda_dVg = 0.5 * pParam->B3SOIFDa1 * (1.0 + T1 / T2);
|
|
}
|
|
|
|
if (Rds > 0)
|
|
{ tmp2 = dRds_dVg / Rds + dWeff_dVg / Weff;
|
|
tmp3 = dRds_dVb / Rds + dWeff_dVb / Weff;
|
|
}
|
|
else
|
|
{ tmp2 = dWeff_dVg / Weff;
|
|
tmp3 = dWeff_dVb / Weff;
|
|
}
|
|
if ((Rds == 0.0) && (Lambda == 1.0))
|
|
{ T0 = 1.0 / (Abeff * EsatL + Vgst2Vtm);
|
|
tmp1 = 0.0;
|
|
T1 = T0 * T0;
|
|
T2 = Vgst2Vtm * T0;
|
|
T3 = EsatL * Vgst2Vtm;
|
|
Vdsat = T3 * T0;
|
|
|
|
dT0_dVg = -(Abeff * dEsatL_dVg + EsatL * dAbeff_dVg + 1.0) * T1;
|
|
dT0_dVd = -(Abeff * dEsatL_dVd) * T1;
|
|
dT0_dVb = -(Abeff * dEsatL_dVb + EsatL * dAbeff_dVb) * T1;
|
|
dT0_dVc = 0.0;
|
|
if (selfheat)
|
|
dT0_dT = -(Abeff * dEsatL_dT + dVgst2Vtm_dT) * T1;
|
|
else dT0_dT = 0.0;
|
|
|
|
dVdsat_dVg = T3 * dT0_dVg + T2 * dEsatL_dVg + EsatL * T0;
|
|
dVdsat_dVd = T3 * dT0_dVd + T2 * dEsatL_dVd;
|
|
dVdsat_dVb = T3 * dT0_dVb + T2 * dEsatL_dVb;
|
|
dVdsat_dVc = 0.0;
|
|
if (selfheat)
|
|
dVdsat_dT = T3 * dT0_dT + T2 * dEsatL_dT
|
|
+ EsatL * T0 * dVgst2Vtm_dT;
|
|
else dVdsat_dT = 0.0;
|
|
}
|
|
else
|
|
{ tmp1 = dLambda_dVg / (Lambda * Lambda);
|
|
T9 = Abeff * WVCoxRds;
|
|
T8 = Abeff * T9;
|
|
T7 = Vgst2Vtm * T9;
|
|
T6 = Vgst2Vtm * WVCoxRds;
|
|
T0 = 2.0 * Abeff * (T9 - 1.0 + 1.0 / Lambda);
|
|
dT0_dVg = 2.0 * (T8 * tmp2 - Abeff * tmp1
|
|
+ (2.0 * T9 + 1.0 / Lambda - 1.0) * dAbeff_dVg);
|
|
/* dT0_dVb = 2.0 * (T8 * tmp3 this is equivalent to one below, but simpler
|
|
+ (2.0 * T9 + 1.0 / Lambda - 1.0) * dAbeff_dVg); */
|
|
dT0_dVb = 2.0 * (T8 * (2.0 / Abeff * dAbeff_dVb + tmp3)
|
|
+ (1.0 / Lambda - 1.0) * dAbeff_dVb);
|
|
dT0_dVd = 0.0;
|
|
dT0_dVc = 0.0;
|
|
|
|
if (selfheat)
|
|
{
|
|
tmp4 = dRds_dT / Rds + dvsattemp_dT / vsattemp;
|
|
dT0_dT = 2.0 * T8 * tmp4;
|
|
} else tmp4 = dT0_dT = 0.0;
|
|
|
|
T1 = Vgst2Vtm * (2.0 / Lambda - 1.0) + Abeff * EsatL + 3.0 * T7;
|
|
|
|
dT1_dVg = (2.0 / Lambda - 1.0) - 2.0 * Vgst2Vtm * tmp1
|
|
+ Abeff * dEsatL_dVg + EsatL * dAbeff_dVg + 3.0 * (T9
|
|
+ T7 * tmp2 + T6 * dAbeff_dVg);
|
|
dT1_dVb = Abeff * dEsatL_dVb + EsatL * dAbeff_dVb
|
|
+ 3.0 * (T6 * dAbeff_dVb + T7 * tmp3);
|
|
dT1_dVd = Abeff * dEsatL_dVd;
|
|
dT1_dVc = 0.0;
|
|
|
|
|
|
if (selfheat)
|
|
{
|
|
tmp4 += dVgst2Vtm_dT / Vgst2Vtm;
|
|
dT1_dT = (2.0 / Lambda - 1.0) * dVgst2Vtm_dT
|
|
+ Abeff * dEsatL_dT + 3.0 * T7 * tmp4;
|
|
} else dT1_dT = 0.0;
|
|
|
|
T2 = Vgst2Vtm * (EsatL + 2.0 * T6);
|
|
dT2_dVg = EsatL + Vgst2Vtm * dEsatL_dVg
|
|
+ T6 * (4.0 + 2.0 * Vgst2Vtm * tmp2);
|
|
dT2_dVb = Vgst2Vtm * (dEsatL_dVb + 2.0 * T6 * tmp3);
|
|
dT2_dVd = Vgst2Vtm * dEsatL_dVd;
|
|
if (selfheat)
|
|
dT2_dT = Vgst2Vtm * dEsatL_dT + EsatL * dVgst2Vtm_dT
|
|
+ 2.0 * T6 * (dVgst2Vtm_dT + Vgst2Vtm * tmp4);
|
|
else
|
|
dT2_dT = 0.0;
|
|
|
|
T3 = sqrt(T1 * T1 - 2.0 * T0 * T2);
|
|
Vdsat = (T1 - T3) / T0;
|
|
|
|
dVdsat_dVg = (dT1_dVg - (T1 * dT1_dVg - dT0_dVg * T2
|
|
- T0 * dT2_dVg) / T3 - Vdsat * dT0_dVg) / T0;
|
|
dVdsat_dVb = (dT1_dVb - (T1 * dT1_dVb - dT0_dVb * T2
|
|
- T0 * dT2_dVb) / T3 - Vdsat * dT0_dVb) / T0;
|
|
dVdsat_dVd = (dT1_dVd - (T1 * dT1_dVd - T0 * dT2_dVd) / T3) / T0;
|
|
dVdsat_dVc = 0.0;
|
|
if (selfheat)
|
|
dVdsat_dT = (dT1_dT - (T1 * dT1_dT - dT0_dT * T2
|
|
- T0 * dT2_dT) / T3 - Vdsat * dT0_dT) / T0;
|
|
else dVdsat_dT = 0.0;
|
|
}
|
|
here->B3SOIFDvdsat = Vdsat;
|
|
|
|
/* Vdsatii for impact ionization */
|
|
if (pParam->B3SOIFDaii > 0.0)
|
|
{
|
|
if (pParam->B3SOIFDcii != 0.0)
|
|
{
|
|
T0 = pParam->B3SOIFDcii / sqrt(3.0) + pParam->B3SOIFDdii;
|
|
/* Hard limit Vds to T0 => T4 i.e. limit T0 to 3.0 */
|
|
T1 = Vds - T0 - 0.1;
|
|
T2 = sqrt(T1 * T1 + 0.4);
|
|
T3 = T0 + 0.5 * (T1 + T2);
|
|
dT3_dVd = 0.5 * (1.0 + T1/T2);
|
|
|
|
T4 = T3 - pParam->B3SOIFDdii;
|
|
T5 = pParam->B3SOIFDcii / T4;
|
|
T0 = T5 * T5;
|
|
dT0_dVd = - 2 * T0 / T4 * dT3_dVd;
|
|
} else
|
|
{
|
|
T0 = dT0_dVd = 0.0;
|
|
}
|
|
T0 += 1.0;
|
|
|
|
T3 = pParam->B3SOIFDaii + pParam->B3SOIFDbii / Leff;
|
|
T4 = 1.0 / (T0 * Vgsteff + T3 * EsatL);
|
|
T5 = -T4 * T4;
|
|
T6 = Vgsteff * T4;
|
|
T7 = EsatL * Vgsteff;
|
|
Vdsatii = T7 * T4;
|
|
|
|
dT4_dVg = T5 * (T0 + T3 * dEsatL_dVg);
|
|
dT4_dVb = T5 * T3 * dEsatL_dVb;
|
|
dT4_dVd = T5 * (Vgsteff * dT0_dVd + T3 * dEsatL_dVd);
|
|
|
|
if (selfheat) dT4_dT = T5 * (T3 * dEsatL_dT);
|
|
else dT4_dT = 0.0;
|
|
|
|
T8 = T4 * Vgsteff;
|
|
dVdsatii_dVg = T7 * dT4_dVg + T4 * (EsatL + Vgsteff * dEsatL_dVg);
|
|
dVdsatii_dVb = T7 * dT4_dVb + T8 * dEsatL_dVb;
|
|
dVdsatii_dVd = T7 * dT4_dVd + T8 * dEsatL_dVd;
|
|
if (selfheat) dVdsatii_dT = T7 * dT4_dT + T8 * dEsatL_dT;
|
|
else dVdsatii_dT = 0.0;
|
|
} else
|
|
{
|
|
Vdsatii = Vdsat;
|
|
dVdsatii_dVg = dVdsat_dVg;
|
|
dVdsatii_dVb = dVdsat_dVb;
|
|
dVdsatii_dVd = dVdsat_dVd;
|
|
dVdsatii_dT = dVdsat_dT;
|
|
}
|
|
|
|
/* Effective Vds (Vdseff) Calculation */
|
|
T1 = Vdsat - Vds - pParam->B3SOIFDdelta;
|
|
dT1_dVg = dVdsat_dVg;
|
|
dT1_dVd = dVdsat_dVd - 1.0;
|
|
dT1_dVb = dVdsat_dVb;
|
|
dT1_dVc = dVdsat_dVc;
|
|
dT1_dT = dVdsat_dT;
|
|
|
|
T2 = sqrt(T1 * T1 + 4.0 * pParam->B3SOIFDdelta * Vdsat);
|
|
T0 = T1 / T2;
|
|
T3 = 2.0 * pParam->B3SOIFDdelta / T2;
|
|
dT2_dVg = T0 * dT1_dVg + T3 * dVdsat_dVg;
|
|
dT2_dVd = T0 * dT1_dVd + T3 * dVdsat_dVd;
|
|
dT2_dVb = T0 * dT1_dVb + T3 * dVdsat_dVb;
|
|
dT2_dVc = 0.0;
|
|
if (selfheat)
|
|
dT2_dT = T0 * dT1_dT + T3 * dVdsat_dT;
|
|
else dT2_dT = 0.0;
|
|
|
|
Vdseff = Vdsat - 0.5 * (T1 + T2);
|
|
dVdseff_dVg = dVdsat_dVg - 0.5 * (dT1_dVg + dT2_dVg);
|
|
dVdseff_dVd = dVdsat_dVd - 0.5 * (dT1_dVd + dT2_dVd);
|
|
dVdseff_dVb = dVdsat_dVb - 0.5 * (dT1_dVb + dT2_dVb);
|
|
dVdseff_dVc = 0.0;
|
|
if (selfheat)
|
|
dVdseff_dT = dVdsat_dT - 0.5 * (dT1_dT + dT2_dT);
|
|
else dVdseff_dT = 0.0;
|
|
|
|
if (Vdseff > Vds)
|
|
Vdseff = Vds; /* This code is added to fixed the problem
|
|
caused by computer precision when
|
|
Vds is very close to Vdseff. */
|
|
diffVds = Vds - Vdseff;
|
|
|
|
/* Effective Vdsii for Iii calculation */
|
|
T1 = Vdsatii - Vds - pParam->B3SOIFDdelta;
|
|
|
|
T2 = sqrt(T1 * T1 + 4.0 * pParam->B3SOIFDdelta * Vdsatii);
|
|
T0 = T1 / T2;
|
|
T3 = 2.0 * pParam->B3SOIFDdelta / T2;
|
|
T4 = T0 + T3;
|
|
dT2_dVg = T4 * dVdsatii_dVg;
|
|
dT2_dVd = T4 * dVdsatii_dVd - T0;
|
|
dT2_dVb = T4 * dVdsatii_dVb;
|
|
if (selfheat) dT2_dT = T4*dVdsatii_dT;
|
|
else dT2_dT = 0.0;
|
|
|
|
Vdseffii = Vdsatii - 0.5 * (T1 + T2);
|
|
dVdseffii_dVg = 0.5 * (dVdsatii_dVg - dT2_dVg);
|
|
dVdseffii_dVd = 0.5 * (dVdsatii_dVd - dT2_dVd + 1.0);
|
|
dVdseffii_dVb = 0.5 * (dVdsatii_dVb - dT2_dVb);
|
|
if (selfheat)
|
|
dVdseffii_dT = 0.5 * (dVdsatii_dT - dT2_dT);
|
|
else dVdseffii_dT = 0.0;
|
|
diffVdsii = Vds - Vdseffii;
|
|
|
|
/* Calculate VAsat */
|
|
tmp4 = 1.0 - 0.5 * Abeff * Vdsat / Vgst2Vtm;
|
|
T9 = WVCoxRds * Vgsteff;
|
|
T8 = T9 / Vgst2Vtm;
|
|
T0 = EsatL + Vdsat + 2.0 * T9 * tmp4;
|
|
|
|
T7 = 2.0 * WVCoxRds * tmp4;
|
|
dT0_dVg = dEsatL_dVg + dVdsat_dVg + T7 * (1.0 + tmp2 * Vgsteff)
|
|
- T8 * (Abeff * dVdsat_dVg - Abeff * Vdsat / Vgst2Vtm
|
|
+ Vdsat * dAbeff_dVg);
|
|
|
|
dT0_dVb = dEsatL_dVb + dVdsat_dVb + T7 * tmp3 * Vgsteff
|
|
- T8 * (dAbeff_dVb * Vdsat + Abeff * dVdsat_dVb);
|
|
dT0_dVd = dEsatL_dVd + dVdsat_dVd - T8 * Abeff * dVdsat_dVd;
|
|
dT0_dVc = 0.0;
|
|
|
|
if (selfheat)
|
|
{
|
|
tmp4 = dRds_dT / Rds + dvsattemp_dT / vsattemp;
|
|
dT0_dT = dEsatL_dT + dVdsat_dT + T7 * tmp4 * Vgsteff
|
|
- T8 * (Abeff * dVdsat_dT - Abeff * Vdsat * dVgst2Vtm_dT
|
|
/ Vgst2Vtm);
|
|
} else
|
|
dT0_dT = 0.0;
|
|
|
|
T9 = WVCoxRds * Abeff;
|
|
T1 = 2.0 / Lambda - 1.0 + T9;
|
|
dT1_dVg = -2.0 * tmp1 + WVCoxRds * (Abeff * tmp2 + dAbeff_dVg);
|
|
dT1_dVb = dAbeff_dVb * WVCoxRds + T9 * tmp3;
|
|
dT1_dVc = 0.0;
|
|
if (selfheat)
|
|
dT1_dT = T9 * tmp4;
|
|
else
|
|
dT1_dT = 0.0;
|
|
|
|
Vasat = T0 / T1;
|
|
dVasat_dVg = (dT0_dVg - Vasat * dT1_dVg) / T1;
|
|
dVasat_dVb = (dT0_dVb - Vasat * dT1_dVb) / T1;
|
|
dVasat_dVd = dT0_dVd / T1;
|
|
dVasat_dVc = 0.0;
|
|
if (selfheat) dVasat_dT = (dT0_dT - Vasat * dT1_dT) / T1;
|
|
else dVasat_dT = 0.0;
|
|
|
|
/* Calculate VACLM */
|
|
if ((pParam->B3SOIFDpclm > 0.0) && (diffVds > 1.0e-10))
|
|
{ T0 = 1.0 / (pParam->B3SOIFDpclm * Abeff * pParam->B3SOIFDlitl);
|
|
dT0_dVb = -T0 / Abeff * dAbeff_dVb;
|
|
dT0_dVg = -T0 / Abeff * dAbeff_dVg;
|
|
dT0_dVc = 0.0;
|
|
|
|
T2 = Vgsteff / EsatL;
|
|
T1 = Leff * (Abeff + T2);
|
|
dT1_dVg = Leff * ((1.0 - T2 * dEsatL_dVg) / EsatL + dAbeff_dVg);
|
|
dT1_dVb = Leff * (dAbeff_dVb - T2 * dEsatL_dVb / EsatL);
|
|
dT1_dVd = -T2 * dEsatL_dVd / Esat;
|
|
dT1_dVc = 0.0;
|
|
if (selfheat) dT1_dT = -T2 * dEsatL_dT / Esat;
|
|
else dT1_dT = 0.0;
|
|
|
|
T9 = T0 * T1;
|
|
VACLM = T9 * diffVds;
|
|
dVACLM_dVg = T0 * dT1_dVg * diffVds - T9 * dVdseff_dVg
|
|
+ T1 * diffVds * dT0_dVg;
|
|
dVACLM_dVb = (dT0_dVb * T1 + T0 * dT1_dVb) * diffVds
|
|
- T9 * dVdseff_dVb;
|
|
dVACLM_dVd = T0 * dT1_dVd * diffVds + T9 * (1.0 - dVdseff_dVd);
|
|
dVACLM_dVc = 0.0;
|
|
if (selfheat)
|
|
dVACLM_dT = T0 * dT1_dT * diffVds - T9 * dVdseff_dT;
|
|
else dVACLM_dT = 0.0;
|
|
|
|
}
|
|
else
|
|
{ VACLM = MAX_EXP;
|
|
dVACLM_dVd = dVACLM_dVg = dVACLM_dVb = dVACLM_dVc = dVACLM_dT = 0.0;
|
|
}
|
|
|
|
|
|
/* Calculate VADIBL */
|
|
if (pParam->B3SOIFDthetaRout > 0.0)
|
|
{ T8 = Abeff * Vdsat;
|
|
T0 = Vgst2Vtm * T8;
|
|
T1 = Vgst2Vtm + T8;
|
|
dT0_dVg = Vgst2Vtm * Abeff * dVdsat_dVg + T8
|
|
+ Vgst2Vtm * Vdsat * dAbeff_dVg;
|
|
dT1_dVg = 1.0 + Abeff * dVdsat_dVg + Vdsat * dAbeff_dVg;
|
|
dT1_dVb = dAbeff_dVb * Vdsat + Abeff * dVdsat_dVb;
|
|
dT0_dVb = Vgst2Vtm * dT1_dVb;
|
|
dT1_dVd = Abeff * dVdsat_dVd;
|
|
dT0_dVd = Vgst2Vtm * dT1_dVd;
|
|
dT0_dVc = dT1_dVc = 0.0;
|
|
if (selfheat)
|
|
{
|
|
dT0_dT = dVgst2Vtm_dT * T8 + Abeff * Vgst2Vtm * dVdsat_dT;
|
|
dT1_dT = dVgst2Vtm_dT + Abeff * dVdsat_dT;
|
|
} else
|
|
dT0_dT = dT1_dT = 0.0;
|
|
|
|
T9 = T1 * T1;
|
|
T2 = pParam->B3SOIFDthetaRout;
|
|
VADIBL = (Vgst2Vtm - T0 / T1) / T2;
|
|
dVADIBL_dVg = (1.0 - dT0_dVg / T1 + T0 * dT1_dVg / T9) / T2;
|
|
dVADIBL_dVb = (-dT0_dVb / T1 + T0 * dT1_dVb / T9) / T2;
|
|
dVADIBL_dVd = (-dT0_dVd / T1 + T0 * dT1_dVd / T9) / T2;
|
|
dVADIBL_dVc = 0.0;
|
|
if (selfheat)
|
|
dVADIBL_dT = (dVgst2Vtm_dT - dT0_dT/T1 + T0*dT1_dT/T9) / T2;
|
|
else dVADIBL_dT = 0.0;
|
|
|
|
T7 = pParam->B3SOIFDpdiblb * Vbseff;
|
|
if (T7 >= -0.9)
|
|
{ T3 = 1.0 / (1.0 + T7);
|
|
VADIBL *= T3;
|
|
dVADIBL_dVg *= T3;
|
|
dVADIBL_dVb = (dVADIBL_dVb - VADIBL * pParam->B3SOIFDpdiblb)
|
|
* T3;
|
|
dVADIBL_dVd *= T3;
|
|
dVADIBL_dVc *= T3;
|
|
if (selfheat) dVADIBL_dT *= T3;
|
|
else dVADIBL_dT = 0.0;
|
|
}
|
|
else
|
|
/* Added to avoid the discontinuity problem caused by pdiblcb */
|
|
{ T4 = 1.0 / (0.8 + T7);
|
|
T3 = (17.0 + 20.0 * T7) * T4;
|
|
dVADIBL_dVg *= T3;
|
|
dVADIBL_dVb = dVADIBL_dVb * T3
|
|
- VADIBL * pParam->B3SOIFDpdiblb * T4 * T4;
|
|
dVADIBL_dVd *= T3;
|
|
dVADIBL_dVc *= T3;
|
|
if (selfheat) dVADIBL_dT *= T3;
|
|
else dVADIBL_dT = 0.0;
|
|
VADIBL *= T3;
|
|
}
|
|
}
|
|
else
|
|
{ VADIBL = MAX_EXP;
|
|
dVADIBL_dVd = dVADIBL_dVg = dVADIBL_dVb = dVADIBL_dVc
|
|
= dVADIBL_dT = 0.0;
|
|
}
|
|
|
|
/* Calculate VA */
|
|
|
|
T8 = pParam->B3SOIFDpvag / EsatL;
|
|
T9 = T8 * Vgsteff;
|
|
if (T9 > -0.9)
|
|
{ T0 = 1.0 + T9;
|
|
dT0_dVg = T8 * (1.0 - Vgsteff * dEsatL_dVg / EsatL);
|
|
dT0_dVb = -T9 * dEsatL_dVb / EsatL;
|
|
dT0_dVd = -T9 * dEsatL_dVd / EsatL;
|
|
if (selfheat)
|
|
dT0_dT = -T9 * dEsatL_dT / EsatL;
|
|
else
|
|
dT0_dT = 0.0;
|
|
}
|
|
else /* Added to avoid the discontinuity problems caused by pvag */
|
|
{ T1 = 1.0 / (17.0 + 20.0 * T9);
|
|
T0 = (0.8 + T9) * T1;
|
|
T1 *= T1;
|
|
dT0_dVg = T8 * (1.0 - Vgsteff * dEsatL_dVg / EsatL) * T1;
|
|
|
|
T9 *= T1 / EsatL;
|
|
dT0_dVb = -T9 * dEsatL_dVb;
|
|
dT0_dVd = -T9 * dEsatL_dVd;
|
|
if (selfheat)
|
|
dT0_dT = -T9 * dEsatL_dT;
|
|
else
|
|
dT0_dT = 0.0;
|
|
}
|
|
|
|
tmp1 = VACLM * VACLM;
|
|
tmp2 = VADIBL * VADIBL;
|
|
tmp3 = VACLM + VADIBL;
|
|
|
|
T1 = VACLM * VADIBL / tmp3;
|
|
tmp3 *= tmp3;
|
|
dT1_dVg = (tmp1 * dVADIBL_dVg + tmp2 * dVACLM_dVg) / tmp3;
|
|
dT1_dVd = (tmp1 * dVADIBL_dVd + tmp2 * dVACLM_dVd) / tmp3;
|
|
dT1_dVb = (tmp1 * dVADIBL_dVb + tmp2 * dVACLM_dVb) / tmp3;
|
|
dT1_dVc = 0.0;
|
|
if (selfheat)
|
|
dT1_dT = (tmp1 * dVADIBL_dT + tmp2 * dVACLM_dT ) / tmp3;
|
|
else dT1_dT = 0.0;
|
|
|
|
Va = Vasat + T0 * T1;
|
|
dVa_dVg = dVasat_dVg + T1 * dT0_dVg + T0 * dT1_dVg;
|
|
dVa_dVd = dVasat_dVd + T1 * dT0_dVd + T0 * dT1_dVd;
|
|
dVa_dVb = dVasat_dVb + T1 * dT0_dVb + T0 * dT1_dVb;
|
|
dVa_dVc = 0.0;
|
|
if (selfheat)
|
|
dVa_dT = dVasat_dT + T1 * dT0_dT + T0 * dT1_dT;
|
|
else dVa_dT = 0.0;
|
|
|
|
/* Calculate Ids */
|
|
CoxWovL = model->B3SOIFDcox * Weff / Leff;
|
|
beta = ueff * CoxWovL;
|
|
dbeta_dVg = CoxWovL * dueff_dVg + beta * dWeff_dVg / Weff;
|
|
dbeta_dVd = CoxWovL * dueff_dVd;
|
|
dbeta_dVb = CoxWovL * dueff_dVb + beta * dWeff_dVb / Weff;
|
|
if (selfheat) dbeta_dT = CoxWovL * dueff_dT;
|
|
else dbeta_dT = 0.0;
|
|
|
|
T0 = 1.0 - 0.5 * Abeff * Vdseff / Vgst2Vtm;
|
|
dT0_dVg = -0.5 * (Abeff * dVdseff_dVg
|
|
- Abeff * Vdseff / Vgst2Vtm + Vdseff * dAbeff_dVg) / Vgst2Vtm;
|
|
dT0_dVd = -0.5 * Abeff * dVdseff_dVd / Vgst2Vtm;
|
|
dT0_dVb = -0.5 * (Abeff * dVdseff_dVb + dAbeff_dVb * Vdseff)
|
|
/ Vgst2Vtm;
|
|
dT0_dVc = 0.0;
|
|
if (selfheat)
|
|
dT0_dT = -0.5 * (Abeff * dVdseff_dT
|
|
- Abeff * Vdseff / Vgst2Vtm * dVgst2Vtm_dT)
|
|
/ Vgst2Vtm;
|
|
else dT0_dT = 0.0;
|
|
|
|
fgche1 = Vgsteff * T0;
|
|
dfgche1_dVg = Vgsteff * dT0_dVg + T0;
|
|
dfgche1_dVd = Vgsteff * dT0_dVd;
|
|
dfgche1_dVb = Vgsteff * dT0_dVb;
|
|
dfgche1_dVc = 0.0;
|
|
if (selfheat) dfgche1_dT = Vgsteff * dT0_dT;
|
|
else dfgche1_dT = 0.0;
|
|
|
|
T9 = Vdseff / EsatL;
|
|
fgche2 = 1.0 + T9;
|
|
dfgche2_dVg = (dVdseff_dVg - T9 * dEsatL_dVg) / EsatL;
|
|
dfgche2_dVd = (dVdseff_dVd - T9 * dEsatL_dVd) / EsatL;
|
|
dfgche2_dVb = (dVdseff_dVb - T9 * dEsatL_dVb) / EsatL;
|
|
dfgche2_dVc = 0.0;
|
|
if (selfheat) dfgche2_dT = (dVdseff_dT - T9 * dEsatL_dT) / EsatL;
|
|
else dfgche2_dT = 0.0;
|
|
|
|
gche = beta * fgche1 / fgche2;
|
|
dgche_dVg = (beta * dfgche1_dVg + fgche1 * dbeta_dVg
|
|
- gche * dfgche2_dVg) / fgche2;
|
|
dgche_dVd = (beta * dfgche1_dVd + fgche1 * dbeta_dVd
|
|
- gche * dfgche2_dVd) / fgche2;
|
|
dgche_dVb = (beta * dfgche1_dVb + fgche1 * dbeta_dVb
|
|
- gche * dfgche2_dVb) / fgche2;
|
|
dgche_dVc = 0.0;
|
|
if (selfheat)
|
|
dgche_dT = (beta * dfgche1_dT + fgche1 * dbeta_dT
|
|
- gche * dfgche2_dT) / fgche2;
|
|
else dgche_dT = 0.0;
|
|
|
|
T0 = 1.0 + gche * Rds;
|
|
T9 = Vdseff / T0;
|
|
Idl = gche * T9;
|
|
|
|
/* Whoa, these formulas for the derivatives of Idl are convoluted, but I
|
|
verified them to be correct */
|
|
|
|
dIdl_dVg = (gche * dVdseff_dVg + T9 * dgche_dVg) / T0
|
|
- Idl * gche / T0 * dRds_dVg ;
|
|
dIdl_dVd = (gche * dVdseff_dVd + T9 * dgche_dVd) / T0;
|
|
dIdl_dVb = (gche * dVdseff_dVb + T9 * dgche_dVb
|
|
- Idl * dRds_dVb * gche) / T0;
|
|
dIdl_dVc = 0.0;
|
|
if (selfheat)
|
|
dIdl_dT = (gche * dVdseff_dT + T9 * dgche_dT
|
|
- Idl * dRds_dT * gche) / T0;
|
|
else dIdl_dT = 0.0;
|
|
|
|
T9 = diffVds / Va;
|
|
T0 = 1.0 + T9;
|
|
here->B3SOIFDids = Ids = Idl * T0;
|
|
|
|
Gm0 = T0 * dIdl_dVg - Idl * (dVdseff_dVg + T9 * dVa_dVg) / Va;
|
|
Gds0 = T0 * dIdl_dVd + Idl * (1.0 - dVdseff_dVd
|
|
- T9 * dVa_dVd) / Va;
|
|
Gmb0 = T0 * dIdl_dVb - Idl * (dVdseff_dVb + T9 * dVa_dVb) / Va;
|
|
Gmc = 0.0;
|
|
if (selfheat)
|
|
GmT0 = T0 * dIdl_dT - Idl * (dVdseff_dT + T9 * dVa_dT) / Va;
|
|
else GmT0 = 0.0;
|
|
|
|
/* This includes all dependencies from Vgsteff, Vbseff, Vcs */
|
|
|
|
Gm = Gm0 * dVgsteff_dVg + Gmb0 * dVbseff_dVg + Gmc * dVcs_dVg;
|
|
Gmb = Gm0 * dVgsteff_dVb + Gmb0 * dVbseff_dVb + Gmc * dVcs_dVb;
|
|
Gds = Gm0 * dVgsteff_dVd + Gmb0 * dVbseff_dVd + Gmc * dVcs_dVd + Gds0;
|
|
Gme = Gm0 * dVgsteff_dVe + Gmb0 * dVbseff_dVe + Gmc * dVcs_dVe;
|
|
if (selfheat)
|
|
GmT = Gm0 * dVgsteff_dT + Gmb0 * dVbseff_dT + Gmc * dVcs_dT + GmT0;
|
|
else GmT = 0.0;
|
|
|
|
|
|
/* calculate substrate current Iii */
|
|
|
|
Giig = Giib = Giid = Giie = GiiT = 0.0;
|
|
here->B3SOIFDiii = Iii = 0.0;
|
|
|
|
Idgidl = Gdgidld = Gdgidlg = 0.0;
|
|
Isgidl = Gsgidlg = 0;
|
|
|
|
here->B3SOIFDibs = Ibs = 0.0;
|
|
here->B3SOIFDibd = Ibd = 0.0;
|
|
here->B3SOIFDic = Ic = 0.0;
|
|
|
|
Gjsb = Gjsd = GjsT = 0.0;
|
|
Gjdb = Gjdd = GjdT = 0.0;
|
|
Gcd = Gcb = GcT = 0.0;
|
|
|
|
here->B3SOIFDibp = Ibp = 0.0;
|
|
here->B3SOIFDgbpbs = here->B3SOIFDgbpgs = here->B3SOIFDgbpds = 0.0;
|
|
here->B3SOIFDgbpes = here->B3SOIFDgbpps = here->B3SOIFDgbpT = here->B3SOIFDcbodcon = 0.0;
|
|
Gbpbs = Gbpgs = Gbpds = Gbpes = Gbpps = GbpT = 0.0;
|
|
|
|
/* Current going out of drainprime node into the drain of device */
|
|
/* "node" means the SPICE circuit node */
|
|
|
|
here->B3SOIFDcdrain = Ids + Ic;
|
|
here->B3SOIFDcd = Ids + Ic - Ibd + Iii + Idgidl;
|
|
here->B3SOIFDcb = Ibs + Ibd + Ibp - Iii - Idgidl - Isgidl;
|
|
|
|
here->B3SOIFDgds = Gds + Gcd;
|
|
here->B3SOIFDgm = Gm;
|
|
here->B3SOIFDgmbs = Gmb + Gcb;
|
|
here->B3SOIFDgme = Gme;
|
|
if (selfheat)
|
|
here->B3SOIFDgmT = GmT + GcT;
|
|
else
|
|
here->B3SOIFDgmT = 0.0;
|
|
|
|
/* note that sign is switched because power flows out
|
|
of device into the temperature node.
|
|
Currently ommit self-heating due to bipolar current
|
|
because it can cause convergence problem*/
|
|
|
|
here->B3SOIFDgtempg = -Gm * Vds;
|
|
here->B3SOIFDgtempb = -Gmb * Vds;
|
|
here->B3SOIFDgtempe = -Gme * Vds;
|
|
here->B3SOIFDgtempT = -GmT * Vds;
|
|
here->B3SOIFDgtempd = -Gds * Vds - Ids;
|
|
here->B3SOIFDcth = - Ids * Vds - model->B3SOIFDtype *
|
|
(here->B3SOIFDgtempg * Vgs + here->B3SOIFDgtempb * Vbs
|
|
+ here->B3SOIFDgtempe * Ves + here->B3SOIFDgtempd * Vds)
|
|
- here->B3SOIFDgtempT * delTemp;
|
|
|
|
/* Body current which flows into drainprime node from the drain of device */
|
|
|
|
here->B3SOIFDgjdb = Gjdb - Giib;
|
|
here->B3SOIFDgjdd = Gjdd - (Giid + Gdgidld);
|
|
here->B3SOIFDgjdg = - (Giig + Gdgidlg);
|
|
here->B3SOIFDgjde = - Giie;
|
|
if (selfheat) here->B3SOIFDgjdT = GjdT - GiiT;
|
|
else here->B3SOIFDgjdT = 0.0;
|
|
here->B3SOIFDcjd = Ibd - Iii - Idgidl - here->B3SOIFDminIsub/2
|
|
- (here->B3SOIFDgjdb * Vbs + here->B3SOIFDgjdd * Vds
|
|
+ here->B3SOIFDgjdg * Vgs + here->B3SOIFDgjde * Ves
|
|
+ here->B3SOIFDgjdT * delTemp);
|
|
|
|
/* Body current which flows into sourceprime node from the source of device */
|
|
|
|
here->B3SOIFDgjsb = Gjsb;
|
|
here->B3SOIFDgjsd = Gjsd;
|
|
here->B3SOIFDgjsg = - Gsgidlg;
|
|
if (selfheat) here->B3SOIFDgjsT = GjsT;
|
|
else here->B3SOIFDgjsT = 0.0;
|
|
here->B3SOIFDcjs = Ibs - Isgidl - here->B3SOIFDminIsub/2
|
|
- (here->B3SOIFDgjsb * Vbs + here->B3SOIFDgjsd * Vds
|
|
+ here->B3SOIFDgjsg * Vgs + here->B3SOIFDgjsT * delTemp);
|
|
|
|
/* Current flowing into body node */
|
|
|
|
here->B3SOIFDgbbs = Giib - Gjsb - Gjdb - Gbpbs;
|
|
here->B3SOIFDgbgs = Giig + Gdgidlg + Gsgidlg - Gbpgs;
|
|
here->B3SOIFDgbds = Giid + Gdgidld - Gjsd - Gjdd - Gbpds;
|
|
here->B3SOIFDgbes = Giie - Gbpes;
|
|
here->B3SOIFDgbps = - Gbpps;
|
|
if (selfheat) here->B3SOIFDgbT = GiiT - GjsT - GjdT - GbpT;
|
|
else here->B3SOIFDgbT = 0.0;
|
|
here->B3SOIFDcbody = Iii + Idgidl + Isgidl - Ibs - Ibd - Ibp + here->B3SOIFDminIsub
|
|
- (here->B3SOIFDgbbs * Vbs + here->B3SOIFDgbgs * Vgs
|
|
+ here->B3SOIFDgbds * Vds + here->B3SOIFDgbps * Vps
|
|
+ here->B3SOIFDgbes * Ves + here->B3SOIFDgbT * delTemp);
|
|
|
|
/* Calculate Qinv for Noise analysis */
|
|
|
|
T1 = Vgsteff * (1.0 - 0.5 * Abeff * Vdseff / Vgst2Vtm);
|
|
here->B3SOIFDqinv = -model->B3SOIFDcox * pParam->B3SOIFDweff * Leff * T1;
|
|
|
|
/* Begin CV (charge) model */
|
|
|
|
if ((model->B3SOIFDxpart < 0) || (!ChargeComputationNeeded))
|
|
{ qgate = qdrn = qsrc = qbody = 0.0;
|
|
here->B3SOIFDcggb = here->B3SOIFDcgsb = here->B3SOIFDcgdb = 0.0;
|
|
here->B3SOIFDcdgb = here->B3SOIFDcdsb = here->B3SOIFDcddb = 0.0;
|
|
here->B3SOIFDcbgb = here->B3SOIFDcbsb = here->B3SOIFDcbdb = 0.0;
|
|
goto finished;
|
|
}
|
|
else
|
|
{
|
|
CoxWL = model->B3SOIFDcox * pParam->B3SOIFDweffCV
|
|
* pParam->B3SOIFDleffCV;
|
|
|
|
/* By using this Vgsteff,cv, discontinuity in moderate
|
|
inversion charges can be avoid. However, in capMod=3,
|
|
Vdsat from IV is used. The dVdsat_dVg is referred to
|
|
the IV Vgsteff and therefore induces error in the charges
|
|
derivatives. Fortunately, Vgsteff,iv and Vgsteff,cv are
|
|
different only in subthreshold where Qsubs is neglectible.
|
|
So the errors in derivatives is not a serious problem */
|
|
|
|
if ((VgstNVt > -EXP_THRESHOLD) && (VgstNVt < EXP_THRESHOLD))
|
|
{ ExpVgst *= ExpVgst;
|
|
Vgsteff = n * Vtm * log(1.0 + ExpVgst);
|
|
T0 = ExpVgst / (1.0 + ExpVgst);
|
|
T1 = -T0 * (dVth_dVb + Vgst / n * dn_dVb) + Vgsteff / n * dn_dVb;
|
|
dVgsteff_dVd = -T0 * (dVth_dVd + Vgst / n * dn_dVd)
|
|
+ Vgsteff / n * dn_dVd + T1 * dVbseff_dVd;
|
|
dVgsteff_dVg = T0 * dVgs_eff_dVg + T1 * dVbseff_dVg;
|
|
dVgsteff_dVb = T1 * dVbseff_dVb;
|
|
dVgsteff_dVe = T1 * dVbseff_dVe;
|
|
if (selfheat)
|
|
dVgsteff_dT = -T0 * (dVth_dT + Vgst / Temp) + Vgsteff / Temp
|
|
+ T1 * dVbseff_dT;
|
|
else dVgsteff_dT = 0.0;
|
|
}
|
|
|
|
Vfb = Vth - phi - pParam->B3SOIFDk1 * sqrtPhis;
|
|
|
|
dVfb_dVb = dVth_dVb - pParam->B3SOIFDk1 * dsqrtPhis_dVb;
|
|
dVfb_dVd = dVth_dVd;
|
|
dVfb_dT = dVth_dT;
|
|
|
|
if ((model->B3SOIFDcapMod == 2) || (model->B3SOIFDcapMod == 3))
|
|
{
|
|
/* Necessary because charge behaviour very strange at
|
|
Vgsteff = 0 */
|
|
Vgsteff += 1e-4;
|
|
|
|
/* Something common in capMod 2 and 3 */
|
|
V3 = Vfb - Vgs_eff + Vbseff - DELTA_3;
|
|
if (Vfb <= 0.0)
|
|
{ T0 = sqrt(V3 * V3 - 4.0 * DELTA_3 * Vfb);
|
|
T2 = -DELTA_3 / T0;
|
|
}
|
|
else
|
|
{ T0 = sqrt(V3 * V3 + 4.0 * DELTA_3 * Vfb);
|
|
T2 = DELTA_3 / T0;
|
|
}
|
|
|
|
T1 = 0.5 * (1.0 + V3 / T0);
|
|
Vfbeff = Vfb - 0.5 * (V3 + T0);
|
|
dVfbeff_dVd = (1.0 - T1 - T2) * dVfb_dVd;
|
|
dVfbeff_dVb = (1.0 - T1 - T2) * dVfb_dVb - T1;
|
|
dVfbeff_dVrg = T1 * dVgs_eff_dVg;
|
|
if (selfheat) dVfbeff_dT = (1.0 - T1 - T2) * dVfb_dT;
|
|
else dVfbeff_dT = 0.0;
|
|
|
|
Qac0 = -CoxWL * (Vfbeff - Vfb);
|
|
dQac0_dVrg = -CoxWL * dVfbeff_dVrg;
|
|
dQac0_dVd = -CoxWL * (dVfbeff_dVd - dVfb_dVd);
|
|
dQac0_dVb = -CoxWL * (dVfbeff_dVb - dVfb_dVb);
|
|
if (selfheat) dQac0_dT = -CoxWL * (dVfbeff_dT - dVfb_dT);
|
|
else dQac0_dT = 0.0;
|
|
|
|
T0 = 0.5 * K1;
|
|
T3 = Vgs_eff - Vfbeff - Vbseff - Vgsteff;
|
|
if (pParam->B3SOIFDk1 == 0.0)
|
|
{ T1 = 0.0;
|
|
T2 = 0.0;
|
|
}
|
|
else if (T3 < 0.0)
|
|
{ T1 = T0 + T3 / pParam->B3SOIFDk1;
|
|
T2 = CoxWL;
|
|
}
|
|
else
|
|
{ T1 = sqrt(T0 * T0 + T3);
|
|
T2 = CoxWL * T0 / T1;
|
|
}
|
|
|
|
Qsub0 = CoxWL * K1 * (T0 - T1);
|
|
|
|
dQsub0_dVrg = T2 * (dVfbeff_dVrg - dVgs_eff_dVg);
|
|
dQsub0_dVg = T2;
|
|
dQsub0_dVd = T2 * dVfbeff_dVd;
|
|
dQsub0_dVb = T2 * (dVfbeff_dVb + 1);
|
|
if (selfheat) dQsub0_dT = T2 * dVfbeff_dT;
|
|
else dQsub0_dT = 0.0;
|
|
|
|
One_Third_CoxWL = CoxWL / 3.0;
|
|
Two_Third_CoxWL = 2.0 * One_Third_CoxWL;
|
|
AbulkCV = Abulk0 * pParam->B3SOIFDabulkCVfactor;
|
|
dAbulkCV_dVb = pParam->B3SOIFDabulkCVfactor * dAbulk0_dVb;
|
|
|
|
/* This is actually capMod=2 calculation */
|
|
VdsatCV = Vgsteff / AbulkCV;
|
|
dVdsatCV_dVg = 1.0 / AbulkCV;
|
|
dVdsatCV_dVb = -VdsatCV * dAbulkCV_dVb / AbulkCV;
|
|
VdsatCV += 1e-5;
|
|
|
|
V4 = VdsatCV - Vds - DELTA_4;
|
|
T0 = sqrt(V4 * V4 + 4.0 * DELTA_4 * VdsatCV);
|
|
VdseffCV = VdsatCV - 0.5 * (V4 + T0);
|
|
T1 = 0.5 * (1.0 + V4 / T0);
|
|
T2 = DELTA_4 / T0;
|
|
T3 = (1.0 - T1 - T2) / AbulkCV;
|
|
dVdseffCV_dVg = T3;
|
|
dVdseffCV_dVd = T1;
|
|
dVdseffCV_dVb = -T3 * VdsatCV * dAbulkCV_dVb;
|
|
|
|
if (model->B3SOIFDcapMod == 2)
|
|
{
|
|
/* VdsCV Make it compatible with capMod 3 */
|
|
VdsCV = VdseffCV;
|
|
dVdsCV_dVg = dVdseffCV_dVg;
|
|
dVdsCV_dVd = dVdseffCV_dVd;
|
|
dVdsCV_dVb = dVdseffCV_dVb;
|
|
}
|
|
else if (model->B3SOIFDcapMod == 3)
|
|
{
|
|
/* Front gate strong inversion depletion charge */
|
|
/* VdssatCV calculation */
|
|
|
|
T1 = Vgsteff + K1*sqrtPhis + 0.5*K1*K1;
|
|
T2 = Vgsteff + K1*sqrtPhis + Phis + 0.25*K1*K1;
|
|
|
|
dT1_dVb = K1*dsqrtPhis_dVb;
|
|
dT2_dVb = dT1_dVb + dPhis_dVb;
|
|
dT1_dVg = dT2_dVg = 1;
|
|
|
|
/* Note VdsatCV is redefined in capMod = 3 */
|
|
VdsatCV = T1 - K1*sqrt(T2);
|
|
|
|
dVdsatCV_dVb = dT1_dVb - K1/2/sqrt(T2)*dT2_dVb;
|
|
dVdsatCV_dVg = dT1_dVg - K1/2/sqrt(T2)*dT2_dVg;
|
|
|
|
T1 = VdsatCV - Vdsat;
|
|
dT1_dVg = dVdsatCV_dVg - dVdsat_dVg;
|
|
dT1_dVb = dVdsatCV_dVb - dVdsat_dVb;
|
|
dT1_dVd = - dVdsat_dVd;
|
|
dT1_dVc = - dVdsat_dVc;
|
|
dT1_dT = - dVdsat_dT;
|
|
|
|
if (!(T1 == 0.0))
|
|
{ T3 = -0.5 * Vdsat / T1; /* Vdsmax */
|
|
T2 = T3 * Vdsat;
|
|
T4 = T2 + T1 * T3 * T3; /* fmax */
|
|
if ((Vdseff > T2) && (T1 < 0))
|
|
{
|
|
VdsCV = T4;
|
|
T5 = -0.5 / (T1 * T1);
|
|
dT3_dVg = T5 * (T1 * dVdsat_dVg - Vdsat * dT1_dVg);
|
|
dT3_dVb = T5 * (T1 * dVdsat_dVb - Vdsat * dT1_dVb);
|
|
dT3_dVd = T5 * (T1 * dVdsat_dVd - Vdsat * dT1_dVd);
|
|
dT3_dVc=0.0;
|
|
if (selfheat)
|
|
dT3_dT=T5 * (T1 * dVdsat_dT - Vdsat * dT1_dT);
|
|
else dT3_dT=0.0;
|
|
|
|
dVdsCV_dVd = T3 * dVdsat_dVd + Vdsat * dT3_dVd
|
|
+ T3 * (2 * T1 * dT3_dVd + T3 * dT1_dVd);
|
|
dVdsCV_dVg = T3 * dVdsat_dVg + Vdsat * dT3_dVg
|
|
+ T3 * (2 * T1 * dT3_dVg + T3 * dT1_dVg);
|
|
dVdsCV_dVb = T3 * dVdsat_dVb + Vdsat * dT3_dVb
|
|
+ T3 * (2 * T1 * dT3_dVb + T3 * dT1_dVb);
|
|
dVdsCV_dVc = 0.0;
|
|
if (selfheat)
|
|
dVdsCV_dT = T3 * dVdsat_dT + Vdsat * dT3_dT
|
|
+ T3 * (2 * T1 * dT3_dT + T3 * dT1_dT );
|
|
else dVdsCV_dT = 0.0;
|
|
} else
|
|
{ T5 = Vdseff / Vdsat;
|
|
T6 = T5 * T5;
|
|
T7 = 2 * T1 * T5 / Vdsat;
|
|
T8 = T7 / Vdsat;
|
|
VdsCV = Vdseff + T1 * T6;
|
|
dVdsCV_dVd = dVdseff_dVd + T8 *
|
|
( Vdsat * dVdseff_dVd
|
|
- Vdseff * dVdsat_dVd)
|
|
+ T6 * dT1_dVd;
|
|
dVdsCV_dVb = dVdseff_dVb + T8 * ( Vdsat *
|
|
dVdseff_dVb - Vdseff * dVdsat_dVb)
|
|
+ T6 * dT1_dVb;
|
|
dVdsCV_dVg = dVdseff_dVg + T8 *
|
|
( Vdsat * dVdseff_dVg
|
|
- Vdseff * dVdsat_dVg)
|
|
+ T6 * dT1_dVg;
|
|
dVdsCV_dVc = 0.0;
|
|
if (selfheat)
|
|
dVdsCV_dT = dVdseff_dT + T8 *
|
|
( Vdsat * dVdseff_dT
|
|
- Vdseff * dVdsat_dT )
|
|
+ T6 * dT1_dT ;
|
|
else dVdsCV_dT = 0.0;
|
|
}
|
|
} else
|
|
{ VdsCV = Vdseff;
|
|
dVdsCV_dVb = dVdseff_dVb;
|
|
dVdsCV_dVd = dVdseff_dVd;
|
|
dVdsCV_dVg = dVdseff_dVg;
|
|
dVdsCV_dVc = dVdseff_dVc;
|
|
dVdsCV_dT = dVdseff_dT;
|
|
}
|
|
if (VdsCV < 0.0) VdsCV = 0.0;
|
|
VdsCV += 1e-4;
|
|
|
|
if (VdsCV > (VdsatCV - 1e-7))
|
|
{
|
|
VdsCV = VdsatCV - 1e-7;
|
|
}
|
|
Phisd = Phis + VdsCV;
|
|
dPhisd_dVb = dPhis_dVb + dVdsCV_dVb;
|
|
dPhisd_dVd = dVdsCV_dVd;
|
|
dPhisd_dVg = dVdsCV_dVg;
|
|
dPhisd_dVc = dVdsCV_dVc;
|
|
dPhisd_dT = dVdsCV_dT;
|
|
sqrtPhisd = sqrt(Phisd);
|
|
|
|
/* Qdep0 - Depletion charge at Vgs=Vth */
|
|
T10 = CoxWL * K1;
|
|
Qdep0 = T10 * sqrtPhis;
|
|
dQdep0_dVb = T10 * dsqrtPhis_dVb;
|
|
} /* End of if capMod == 3 */
|
|
|
|
/* Something common in both capMod 2 or 3 */
|
|
|
|
/* Backgate charge */
|
|
CboxWL = pParam->B3SOIFDkb3 * Cbox * pParam->B3SOIFDweffCV
|
|
* pParam->B3SOIFDleffCV;
|
|
Cbg = Cbb = Cbd = Cbe = CbT = 0.0;
|
|
Ce2g = Ce2b = Ce2d = Ce2e = Ce2T = 0.0;
|
|
Qbf = Qe2 = 0.0;
|
|
|
|
T2 = - 0.5 * model->B3SOIFDcboxt * pParam->B3SOIFDweffCV
|
|
* pParam->B3SOIFDleffCV;
|
|
Qe1 = T2 * VdsCV - CboxWL * (Vbs0eff - Vesfb);
|
|
Ce1g = T2 * (dVdsCV_dVg * dVgsteff_dVg + dVdsCV_dVb * dVbseff_dVg
|
|
+ dVdsCV_dVc * dVcs_dVg) - CboxWL * dVbs0eff_dVg;
|
|
Ce1d = T2 * (dVdsCV_dVg * dVgsteff_dVd + dVdsCV_dVb * dVbseff_dVd
|
|
+ dVdsCV_dVc * dVcs_dVd + dVdsCV_dVd)
|
|
- CboxWL * dVbs0eff_dVd;
|
|
Ce1b = 0.0;
|
|
Ce1e = T2 * (dVdsCV_dVg * dVgsteff_dVe + dVdsCV_dVb * dVbseff_dVe
|
|
+ dVdsCV_dVc * dVcs_dVe)
|
|
- CboxWL * (dVbs0eff_dVe - 1.0);
|
|
if (selfheat)
|
|
Ce1T = T2 * (dVdsCV_dVg * dVgsteff_dT + dVdsCV_dVb * dVbseff_dT
|
|
+ dVdsCV_dVc * dVcs_dT + dVdsCV_dT)
|
|
- CboxWL * (dVbs0eff_dT + dvfbb_dT);
|
|
else Ce1T = 0.0;
|
|
|
|
/* Total inversion charge */
|
|
T0 = AbulkCV * VdseffCV;
|
|
T1 = 12.0 * (Vgsteff - 0.5 * T0 + 1e-20);
|
|
T2 = VdseffCV / T1;
|
|
T3 = T0 * T2;
|
|
|
|
T4 = (1.0 - 12.0 * T2 * T2 * AbulkCV);
|
|
T5 = (6.0 * T0 * (4.0 * Vgsteff - T0) / (T1 * T1) - 0.5);
|
|
T6 = 12.0 * T2 * T2 * Vgsteff;
|
|
|
|
qinv = CoxWL * (Vgsteff - 0.5 * VdseffCV + T3);
|
|
Cgg1 = CoxWL * (T4 + T5 * dVdseffCV_dVg);
|
|
Cgd1 = CoxWL * T5 * dVdseffCV_dVd;
|
|
Cgb1 = CoxWL * (T5 * dVdseffCV_dVb + T6 * dAbulkCV_dVb);
|
|
|
|
/* Inversion charge partitioning into S / D */
|
|
if (model->B3SOIFDxpart > 0.5)
|
|
{ /* 0/100 Charge partition model */
|
|
T1 = T1 + T1;
|
|
qsrc = -CoxWL * (0.5 * Vgsteff + 0.25 * T0
|
|
- T0 * T0 / T1);
|
|
T7 = (4.0 * Vgsteff - T0) / (T1 * T1);
|
|
T4 = -(0.5 + 24.0 * T0 * T0 / (T1 * T1));
|
|
T5 = -(0.25 * AbulkCV - 12.0 * AbulkCV * T0 * T7);
|
|
T6 = -(0.25 * VdseffCV - 12.0 * T0 * VdseffCV * T7);
|
|
Csg1 = CoxWL * (T4 + T5 * dVdseffCV_dVg);
|
|
Csd1 = CoxWL * T5 * dVdseffCV_dVd;
|
|
Csb1 = CoxWL * (T5 * dVdseffCV_dVb + T6 * dAbulkCV_dVb);
|
|
|
|
}
|
|
else if (model->B3SOIFDxpart < 0.5)
|
|
{ /* 40/60 Charge partition model */
|
|
T1 = T1 / 12.0;
|
|
T2 = 0.5 * CoxWL / (T1 * T1);
|
|
T3 = Vgsteff * (2.0 * T0 * T0 / 3.0 + Vgsteff
|
|
* (Vgsteff - 4.0 * T0 / 3.0))
|
|
- 2.0 * T0 * T0 * T0 / 15.0;
|
|
qsrc = -T2 * T3;
|
|
T7 = 4.0 / 3.0 * Vgsteff * (Vgsteff - T0)
|
|
+ 0.4 * T0 * T0;
|
|
T4 = -2.0 * qsrc / T1 - T2 * (Vgsteff * (3.0
|
|
* Vgsteff - 8.0 * T0 / 3.0)
|
|
+ 2.0 * T0 * T0 / 3.0);
|
|
T5 = (qsrc / T1 + T2 * T7) * AbulkCV;
|
|
T6 = (qsrc / T1 * VdseffCV + T2 * T7 * VdseffCV);
|
|
Csg1 = T4 + T5 * dVdseffCV_dVg;
|
|
Csd1 = T5 * dVdseffCV_dVd;
|
|
Csb1 = T5 * dVdseffCV_dVb + T6 * dAbulkCV_dVb;
|
|
}
|
|
else
|
|
{ /* 50/50 Charge partition model */
|
|
qsrc = - 0.5 * qinv;
|
|
Csg1 = - 0.5 * Cgg1;
|
|
Csb1 = - 0.5 * Cgb1;
|
|
Csd1 = - 0.5 * Cgd1;
|
|
}
|
|
|
|
Csg = Csg1 * dVgsteff_dVg + Csb1 * dVbseff_dVg;
|
|
Csd = Csd1 + Csg1 * dVgsteff_dVd + Csb1 * dVbseff_dVd;
|
|
Csb = Csg1 * dVgsteff_dVb + Csb1 * dVbseff_dVb;
|
|
Cse = Csg1 * dVgsteff_dVe + Csb1 * dVbseff_dVe;
|
|
if (selfheat)
|
|
CsT = Csg1 * dVgsteff_dT + Csb1 * dVbseff_dT;
|
|
else CsT = 0.0;
|
|
|
|
Qex=dQex_dVg=dQex_dVb=dQex_dVd=dQex_dVe=dQex_dT=0.0;
|
|
|
|
qgate = qinv - (Qbf + Qe2);
|
|
qbody = (Qbf - Qe1 + Qex);
|
|
qsub = Qe1 + Qe2 - Qex;
|
|
qdrn = -(qinv + qsrc);
|
|
|
|
Cgg = (Cgg1 * dVgsteff_dVg + Cgb1 * dVbseff_dVg) - Cbg ;
|
|
Cgd = (Cgd1 + Cgg1 * dVgsteff_dVd + Cgb1 * dVbseff_dVd)-Cbd;
|
|
Cgb = (Cgb1 * dVbseff_dVb + Cgg1 * dVgsteff_dVb) - Cbb;
|
|
Cge = (Cgg1 * dVgsteff_dVe + Cgb1 * dVbseff_dVe) - Cbe;
|
|
if (selfheat)
|
|
CgT = (Cgg1 * dVgsteff_dT + Cgb1 * dVbseff_dT ) - CbT;
|
|
else CgT = 0.0;
|
|
|
|
here->B3SOIFDcggb = Cgg - Ce2g;
|
|
here->B3SOIFDcgsb = - (Cgg + Cgd + Cgb + Cge)
|
|
+ (Ce2g + Ce2d + Ce2b + Ce2e);
|
|
here->B3SOIFDcgdb = Cgd - Ce2d;
|
|
here->B3SOIFDcgeb = Cge - Ce2e;
|
|
here->B3SOIFDcgT = CgT - Ce2T;
|
|
|
|
here->B3SOIFDcbgb = Cbg - Ce1g + dQex_dVg;
|
|
here->B3SOIFDcbsb = -(Cbg + Cbd + Cbb + Cbe)
|
|
+ (Ce1g + Ce1d + Ce1b + Ce1e)
|
|
- (dQex_dVg + dQex_dVd + dQex_dVb + dQex_dVe);
|
|
here->B3SOIFDcbdb = Cbd - Ce1d + dQex_dVd;
|
|
here->B3SOIFDcbeb = Cbe - Ce1e + dQex_dVe;
|
|
here->B3SOIFDcbT = CbT - Ce1T + dQex_dT;
|
|
|
|
here->B3SOIFDcegb = Ce1g + Ce2g - dQex_dVg;
|
|
here->B3SOIFDcesb = -(Ce1g + Ce1d + Ce1b + Ce1e)
|
|
-(Ce2g + Ce2d + Ce2b + Ce2e)
|
|
+(dQex_dVg + dQex_dVd + dQex_dVb + dQex_dVe);
|
|
here->B3SOIFDcedb = Ce1d + Ce2d - dQex_dVd;
|
|
here->B3SOIFDceeb = Ce1e + Ce2e - dQex_dVe;
|
|
here->B3SOIFDceT = Ce1T + Ce2T - dQex_dT;
|
|
|
|
here->B3SOIFDcdgb = -(Cgg + Cbg + Csg);
|
|
here->B3SOIFDcddb = -(Cgd + Cbd + Csd);
|
|
here->B3SOIFDcdeb = -(Cge + Cbe + Cse);
|
|
here->B3SOIFDcdT = -(CgT + CbT + CsT);
|
|
here->B3SOIFDcdsb = (Cgg + Cgd + Cgb + Cge
|
|
+ Cbg + Cbd + Cbb + Cbe
|
|
+ Csg + Csd + Csb + Cse);
|
|
|
|
} /* End of if capMod == 2 or capMod ==3 */
|
|
}
|
|
|
|
finished: /* returning Values to Calling Routine */
|
|
/*
|
|
* COMPUTE EQUIVALENT DRAIN CURRENT SOURCE
|
|
*/
|
|
if (ChargeComputationNeeded)
|
|
{
|
|
qjs = qjd = 0.0;
|
|
gcjdds = gcjdbs = gcjdT = 0.0;
|
|
gcjsbs = gcjsT = 0.0;
|
|
|
|
qdrn -= qjd;
|
|
qbody += (qjs + qjd);
|
|
qsrc = -(qgate + qbody + qdrn + qsub);
|
|
|
|
/* Update the conductance */
|
|
here->B3SOIFDcddb -= gcjdds;
|
|
here->B3SOIFDcdT -= gcjdT;
|
|
here->B3SOIFDcdsb += gcjdds + gcjdbs;
|
|
|
|
here->B3SOIFDcbdb += (gcjdds);
|
|
here->B3SOIFDcbT += (gcjdT + gcjsT);
|
|
here->B3SOIFDcbsb -= (gcjdds + gcjdbs + gcjsbs);
|
|
|
|
/* Extrinsic Bottom S/D to substrate charge */
|
|
T10 = -model->B3SOIFDtype * ves;
|
|
/* T10 is vse without type conversion */
|
|
if ( ((pParam->B3SOIFDnsub > 0) && (model->B3SOIFDtype > 0)) ||
|
|
((pParam->B3SOIFDnsub < 0) && (model->B3SOIFDtype < 0)) )
|
|
{
|
|
if (T10 < pParam->B3SOIFDvsdfb)
|
|
{ here->B3SOIFDqse = here->B3SOIFDcsbox * (T10 - pParam->B3SOIFDvsdfb);
|
|
here->B3SOIFDgcse = here->B3SOIFDcsbox;
|
|
}
|
|
else if (T10 < pParam->B3SOIFDsdt1)
|
|
{ T0 = T10 - pParam->B3SOIFDvsdfb;
|
|
T1 = T0 * T0;
|
|
here->B3SOIFDqse = T0 * (here->B3SOIFDcsbox -
|
|
pParam->B3SOIFDst2 / 3 * T1) ;
|
|
here->B3SOIFDgcse = here->B3SOIFDcsbox - pParam->B3SOIFDst2 * T1;
|
|
}
|
|
else if (T10 < pParam->B3SOIFDvsdth)
|
|
{ T0 = T10 - pParam->B3SOIFDvsdth;
|
|
T1 = T0 * T0;
|
|
here->B3SOIFDqse = here->B3SOIFDcsmin * T10 + here->B3SOIFDst4 +
|
|
pParam->B3SOIFDst3 / 3 * T0 * T1;
|
|
here->B3SOIFDgcse = here->B3SOIFDcsmin + pParam->B3SOIFDst3 * T1;
|
|
}
|
|
else
|
|
{ here->B3SOIFDqse = here->B3SOIFDcsmin * T10 + here->B3SOIFDst4;
|
|
here->B3SOIFDgcse = here->B3SOIFDcsmin;
|
|
}
|
|
} else
|
|
{
|
|
if (T10 < pParam->B3SOIFDvsdth)
|
|
{ here->B3SOIFDqse = here->B3SOIFDcsmin * (T10 - pParam->B3SOIFDvsdth);
|
|
here->B3SOIFDgcse = here->B3SOIFDcsmin;
|
|
}
|
|
else if (T10 < pParam->B3SOIFDsdt1)
|
|
{ T0 = T10 - pParam->B3SOIFDvsdth;
|
|
T1 = T0 * T0;
|
|
here->B3SOIFDqse = T0 * (here->B3SOIFDcsmin - pParam->B3SOIFDst2 / 3 * T1) ;
|
|
here->B3SOIFDgcse = here->B3SOIFDcsmin - pParam->B3SOIFDst2 * T1;
|
|
}
|
|
else if (T10 < pParam->B3SOIFDvsdfb)
|
|
{ T0 = T10 - pParam->B3SOIFDvsdfb;
|
|
T1 = T0 * T0;
|
|
here->B3SOIFDqse = here->B3SOIFDcsbox * T10 + here->B3SOIFDst4 +
|
|
pParam->B3SOIFDst3 / 3 * T0 * T1;
|
|
here->B3SOIFDgcse = here->B3SOIFDcsbox + pParam->B3SOIFDst3 * T1;
|
|
}
|
|
else
|
|
{ here->B3SOIFDqse = here->B3SOIFDcsbox * T10 + here->B3SOIFDst4;
|
|
here->B3SOIFDgcse = here->B3SOIFDcsbox;
|
|
}
|
|
}
|
|
|
|
/* T11 is vde without type conversion */
|
|
T11 = model->B3SOIFDtype * (vds - ves);
|
|
if ( ((pParam->B3SOIFDnsub > 0) && (model->B3SOIFDtype > 0)) ||
|
|
((pParam->B3SOIFDnsub < 0) && (model->B3SOIFDtype < 0)) )
|
|
{
|
|
if (T11 < pParam->B3SOIFDvsdfb)
|
|
{ here->B3SOIFDqde = here->B3SOIFDcdbox * (T11 - pParam->B3SOIFDvsdfb);
|
|
here->B3SOIFDgcde = here->B3SOIFDcdbox;
|
|
}
|
|
else if (T11 < pParam->B3SOIFDsdt1)
|
|
{ T0 = T11 - pParam->B3SOIFDvsdfb;
|
|
T1 = T0 * T0;
|
|
here->B3SOIFDqde = T0 * (here->B3SOIFDcdbox - pParam->B3SOIFDdt2 / 3 * T1) ;
|
|
here->B3SOIFDgcde = here->B3SOIFDcdbox - pParam->B3SOIFDdt2 * T1;
|
|
}
|
|
else if (T11 < pParam->B3SOIFDvsdth)
|
|
{ T0 = T11 - pParam->B3SOIFDvsdth;
|
|
T1 = T0 * T0;
|
|
here->B3SOIFDqde = here->B3SOIFDcdmin * T11 + here->B3SOIFDdt4 +
|
|
pParam->B3SOIFDdt3 / 3 * T0 * T1;
|
|
here->B3SOIFDgcde = here->B3SOIFDcdmin + pParam->B3SOIFDdt3 * T1;
|
|
}
|
|
else
|
|
{ here->B3SOIFDqde = here->B3SOIFDcdmin * T11 + here->B3SOIFDdt4;
|
|
here->B3SOIFDgcde = here->B3SOIFDcdmin;
|
|
}
|
|
} else
|
|
{
|
|
if (T11 < pParam->B3SOIFDvsdth)
|
|
{ here->B3SOIFDqde = here->B3SOIFDcdmin * (T11 - pParam->B3SOIFDvsdth);
|
|
here->B3SOIFDgcde = here->B3SOIFDcdmin;
|
|
}
|
|
else if (T11 < pParam->B3SOIFDsdt1)
|
|
{ T0 = T11 - pParam->B3SOIFDvsdth;
|
|
T1 = T0 * T0;
|
|
here->B3SOIFDqde = T0 * (here->B3SOIFDcdmin - pParam->B3SOIFDdt2 / 3 * T1) ;
|
|
here->B3SOIFDgcde = here->B3SOIFDcdmin - pParam->B3SOIFDdt2 * T1;
|
|
}
|
|
else if (T11 < pParam->B3SOIFDvsdfb)
|
|
{ T0 = T11 - pParam->B3SOIFDvsdfb;
|
|
T1 = T0 * T0;
|
|
here->B3SOIFDqde = here->B3SOIFDcdbox * T11 + here->B3SOIFDdt4 +
|
|
pParam->B3SOIFDdt3 / 3 * T0 * T1;
|
|
here->B3SOIFDgcde = here->B3SOIFDcdbox + pParam->B3SOIFDdt3 * T1;
|
|
}
|
|
else
|
|
{ here->B3SOIFDqde = here->B3SOIFDcdbox * T11 + here->B3SOIFDdt4;
|
|
here->B3SOIFDgcde = here->B3SOIFDcdbox;
|
|
}
|
|
}
|
|
|
|
/* Extrinsic : Sidewall fringing S/D charge */
|
|
here->B3SOIFDqse += pParam->B3SOIFDcsesw * T10;
|
|
here->B3SOIFDgcse += pParam->B3SOIFDcsesw;
|
|
here->B3SOIFDqde += pParam->B3SOIFDcdesw * T11;
|
|
here->B3SOIFDgcde += pParam->B3SOIFDcdesw;
|
|
|
|
/* All charge are mutliplied with type at the end, but qse and qde
|
|
have true polarity => so pre-mutliplied with type */
|
|
here->B3SOIFDqse *= model->B3SOIFDtype;
|
|
here->B3SOIFDqde *= model->B3SOIFDtype;
|
|
}
|
|
|
|
here->B3SOIFDxc = Xc;
|
|
here->B3SOIFDcbb = Cbb;
|
|
here->B3SOIFDcbd = Cbd;
|
|
here->B3SOIFDcbg = Cbg;
|
|
here->B3SOIFDqbf = Qbf;
|
|
here->B3SOIFDqjs = qjs;
|
|
here->B3SOIFDqjd = qjd;
|
|
|
|
if (here->B3SOIFDdebugMod == -1)
|
|
ChargeComputationNeeded = 0;
|
|
|
|
/*
|
|
* check convergence
|
|
*/
|
|
if ((here->B3SOIFDoff == 0) || (!(ckt->CKTmode & MODEINITFIX)))
|
|
{ if (Check == 1)
|
|
{ ckt->CKTnoncon++;
|
|
if (here->B3SOIFDdebugMod > 2)
|
|
fprintf(fpdebug, "Check is on, noncon=%d\n", ckt->CKTnoncon++);
|
|
|
|
}
|
|
}
|
|
|
|
*(ckt->CKTstate0 + here->B3SOIFDvg) = vg;
|
|
*(ckt->CKTstate0 + here->B3SOIFDvd) = vd;
|
|
*(ckt->CKTstate0 + here->B3SOIFDvs) = vs;
|
|
*(ckt->CKTstate0 + here->B3SOIFDvp) = vp;
|
|
*(ckt->CKTstate0 + here->B3SOIFDve) = ve;
|
|
|
|
*(ckt->CKTstate0 + here->B3SOIFDvbs) = vbs;
|
|
*(ckt->CKTstate0 + here->B3SOIFDvbd) = vbd;
|
|
*(ckt->CKTstate0 + here->B3SOIFDvgs) = vgs;
|
|
*(ckt->CKTstate0 + here->B3SOIFDvds) = vds;
|
|
*(ckt->CKTstate0 + here->B3SOIFDves) = ves;
|
|
*(ckt->CKTstate0 + here->B3SOIFDvps) = vps;
|
|
*(ckt->CKTstate0 + here->B3SOIFDdeltemp) = delTemp;
|
|
|
|
/* bulk and channel charge plus overlaps */
|
|
|
|
if (!ChargeComputationNeeded)
|
|
goto line850;
|
|
|
|
line755:
|
|
ag0 = ckt->CKTag[0];
|
|
|
|
T0 = vgd + DELTA_1;
|
|
T1 = sqrt(T0 * T0 + 4.0 * DELTA_1);
|
|
T2 = 0.5 * (T0 - T1);
|
|
|
|
T3 = pParam->B3SOIFDweffCV * pParam->B3SOIFDcgdl;
|
|
T4 = sqrt(1.0 - 4.0 * T2 / pParam->B3SOIFDckappa);
|
|
cgdo = pParam->B3SOIFDcgdo + T3 - T3 * (1.0 - 1.0 / T4)
|
|
* (0.5 - 0.5 * T0 / T1);
|
|
qgdo = (pParam->B3SOIFDcgdo + T3) * vgd - T3 * (T2
|
|
+ 0.5 * pParam->B3SOIFDckappa * (T4 - 1.0));
|
|
|
|
T0 = vgs + DELTA_1;
|
|
T1 = sqrt(T0 * T0 + 4.0 * DELTA_1);
|
|
T2 = 0.5 * (T0 - T1);
|
|
T3 = pParam->B3SOIFDweffCV * pParam->B3SOIFDcgsl;
|
|
T4 = sqrt(1.0 - 4.0 * T2 / pParam->B3SOIFDckappa);
|
|
cgso = pParam->B3SOIFDcgso + T3 - T3 * (1.0 - 1.0 / T4)
|
|
* (0.5 - 0.5 * T0 / T1);
|
|
qgso = (pParam->B3SOIFDcgso + T3) * vgs - T3 * (T2
|
|
+ 0.5 * pParam->B3SOIFDckappa * (T4 - 1.0));
|
|
|
|
|
|
if (here->B3SOIFDmode > 0)
|
|
{ gcdgb = (here->B3SOIFDcdgb - cgdo) * ag0;
|
|
gcddb = (here->B3SOIFDcddb + cgdo + here->B3SOIFDgcde) * ag0;
|
|
gcdsb = here->B3SOIFDcdsb * ag0;
|
|
gcdeb = (here->B3SOIFDcdeb - here->B3SOIFDgcde) * ag0;
|
|
gcdT = model->B3SOIFDtype * here->B3SOIFDcdT * ag0;
|
|
|
|
gcsgb = -(here->B3SOIFDcggb + here->B3SOIFDcbgb + here->B3SOIFDcdgb
|
|
+ here->B3SOIFDcegb + cgso) * ag0;
|
|
gcsdb = -(here->B3SOIFDcgdb + here->B3SOIFDcbdb + here->B3SOIFDcddb
|
|
+ here->B3SOIFDcedb) * ag0;
|
|
gcssb = (cgso + here->B3SOIFDgcse - (here->B3SOIFDcgsb + here->B3SOIFDcbsb
|
|
+ here->B3SOIFDcdsb + here->B3SOIFDcesb)) * ag0;
|
|
gcseb = -(here->B3SOIFDgcse + here->B3SOIFDcgeb + here->B3SOIFDcbeb + here->B3SOIFDcdeb
|
|
+ here->B3SOIFDceeb) * ag0;
|
|
gcsT = - model->B3SOIFDtype * (here->B3SOIFDcgT + here->B3SOIFDcbT + here->B3SOIFDcdT
|
|
+ here->B3SOIFDceT) * ag0;
|
|
|
|
gcggb = (here->B3SOIFDcggb + cgdo + cgso + pParam->B3SOIFDcgeo) * ag0;
|
|
gcgdb = (here->B3SOIFDcgdb - cgdo) * ag0;
|
|
gcgsb = (here->B3SOIFDcgsb - cgso) * ag0;
|
|
gcgeb = (here->B3SOIFDcgeb - pParam->B3SOIFDcgeo) * ag0;
|
|
gcgT = model->B3SOIFDtype * here->B3SOIFDcgT * ag0;
|
|
|
|
gcbgb = here->B3SOIFDcbgb * ag0;
|
|
gcbdb = here->B3SOIFDcbdb * ag0;
|
|
gcbsb = here->B3SOIFDcbsb * ag0;
|
|
gcbeb = here->B3SOIFDcbeb * ag0;
|
|
gcbT = model->B3SOIFDtype * here->B3SOIFDcbT * ag0;
|
|
|
|
gcegb = (here->B3SOIFDcegb - pParam->B3SOIFDcgeo) * ag0;
|
|
gcedb = (here->B3SOIFDcedb - here->B3SOIFDgcde) * ag0;
|
|
gcesb = (here->B3SOIFDcesb - here->B3SOIFDgcse) * ag0;
|
|
gceeb = (here->B3SOIFDgcse + here->B3SOIFDgcde +
|
|
here->B3SOIFDceeb + pParam->B3SOIFDcgeo) * ag0;
|
|
|
|
gceT = model->B3SOIFDtype * here->B3SOIFDceT * ag0;
|
|
|
|
gcTt = pParam->B3SOIFDcth * ag0;
|
|
|
|
sxpart = 0.6;
|
|
dxpart = 0.4;
|
|
|
|
/* Lump the overlap capacitance and S/D parasitics */
|
|
qgd = qgdo;
|
|
qgs = qgso;
|
|
qge = pParam->B3SOIFDcgeo * vge;
|
|
qgate += qgd + qgs + qge;
|
|
qdrn += here->B3SOIFDqde - qgd;
|
|
qsub -= qge + here->B3SOIFDqse + here->B3SOIFDqde;
|
|
qsrc = -(qgate + qbody + qdrn + qsub);
|
|
}
|
|
else
|
|
{ gcsgb = (here->B3SOIFDcdgb - cgso) * ag0;
|
|
gcssb = (here->B3SOIFDcddb + cgso + here->B3SOIFDgcse) * ag0;
|
|
gcsdb = here->B3SOIFDcdsb * ag0;
|
|
gcseb = (here->B3SOIFDcdeb - here->B3SOIFDgcse) * ag0;
|
|
gcsT = model->B3SOIFDtype * here->B3SOIFDcdT * ag0;
|
|
|
|
gcdgb = -(here->B3SOIFDcggb + here->B3SOIFDcbgb + here->B3SOIFDcdgb
|
|
+ here->B3SOIFDcegb + cgdo) * ag0;
|
|
gcdsb = -(here->B3SOIFDcgdb + here->B3SOIFDcbdb + here->B3SOIFDcddb
|
|
+ here->B3SOIFDcedb) * ag0;
|
|
gcddb = (cgdo + here->B3SOIFDgcde - (here->B3SOIFDcgsb + here->B3SOIFDcbsb
|
|
+ here->B3SOIFDcdsb + here->B3SOIFDcesb)) * ag0;
|
|
gcdeb = -(here->B3SOIFDgcde + here->B3SOIFDcgeb + here->B3SOIFDcbeb + here->B3SOIFDcdeb
|
|
+ here->B3SOIFDceeb) * ag0;
|
|
gcdT = - model->B3SOIFDtype * (here->B3SOIFDcgT + here->B3SOIFDcbT
|
|
+ here->B3SOIFDcdT + here->B3SOIFDceT) * ag0;
|
|
|
|
gcggb = (here->B3SOIFDcggb + cgdo + cgso + pParam->B3SOIFDcgeo) * ag0;
|
|
gcgsb = (here->B3SOIFDcgdb - cgso) * ag0;
|
|
gcgdb = (here->B3SOIFDcgsb - cgdo) * ag0;
|
|
gcgeb = (here->B3SOIFDcgeb - pParam->B3SOIFDcgeo) * ag0;
|
|
gcgT = model->B3SOIFDtype * here->B3SOIFDcgT * ag0;
|
|
|
|
gcbgb = here->B3SOIFDcbgb * ag0;
|
|
gcbsb = here->B3SOIFDcbdb * ag0;
|
|
gcbdb = here->B3SOIFDcbsb * ag0;
|
|
gcbeb = here->B3SOIFDcbeb * ag0;
|
|
gcbT = model->B3SOIFDtype * here->B3SOIFDcbT * ag0;
|
|
|
|
gcegb = (here->B3SOIFDcegb - pParam->B3SOIFDcgeo) * ag0;
|
|
gcesb = (here->B3SOIFDcedb - here->B3SOIFDgcse) * ag0;
|
|
gcedb = (here->B3SOIFDcesb - here->B3SOIFDgcde) * ag0;
|
|
gceeb = (here->B3SOIFDceeb + pParam->B3SOIFDcgeo +
|
|
here->B3SOIFDgcse + here->B3SOIFDgcde) * ag0;
|
|
gceT = model->B3SOIFDtype * here->B3SOIFDceT * ag0;
|
|
|
|
gcTt = pParam->B3SOIFDcth * ag0;
|
|
|
|
dxpart = 0.6;
|
|
sxpart = 0.4;
|
|
|
|
/* Lump the overlap capacitance */
|
|
qgd = qgdo;
|
|
qgs = qgso;
|
|
qge = pParam->B3SOIFDcgeo * vge;
|
|
qgate += qgd + qgs + qge;
|
|
qsrc = qdrn - qgs + here->B3SOIFDqse;
|
|
qsub -= qge + here->B3SOIFDqse + here->B3SOIFDqde;
|
|
qdrn = -(qgate + qbody + qsrc + qsub);
|
|
}
|
|
|
|
here->B3SOIFDcgdo = cgdo;
|
|
here->B3SOIFDcgso = cgso;
|
|
|
|
if (ByPass) goto line860;
|
|
|
|
*(ckt->CKTstate0 + here->B3SOIFDqe) = qsub;
|
|
*(ckt->CKTstate0 + here->B3SOIFDqg) = qgate;
|
|
*(ckt->CKTstate0 + here->B3SOIFDqd) = qdrn;
|
|
*(ckt->CKTstate0 + here->B3SOIFDqb) = qbody;
|
|
if ((model->B3SOIFDshMod == 1) && (here->B3SOIFDrth0!=0.0))
|
|
*(ckt->CKTstate0 + here->B3SOIFDqth) = pParam->B3SOIFDcth * delTemp;
|
|
|
|
|
|
/* store small signal parameters */
|
|
if (ckt->CKTmode & MODEINITSMSIG)
|
|
{ goto line1000;
|
|
}
|
|
if (!ChargeComputationNeeded)
|
|
goto line850;
|
|
|
|
|
|
if (ckt->CKTmode & MODEINITTRAN)
|
|
{ *(ckt->CKTstate1 + here->B3SOIFDqb) =
|
|
*(ckt->CKTstate0 + here->B3SOIFDqb);
|
|
*(ckt->CKTstate1 + here->B3SOIFDqg) =
|
|
*(ckt->CKTstate0 + here->B3SOIFDqg);
|
|
*(ckt->CKTstate1 + here->B3SOIFDqd) =
|
|
*(ckt->CKTstate0 + here->B3SOIFDqd);
|
|
*(ckt->CKTstate1 + here->B3SOIFDqe) =
|
|
*(ckt->CKTstate0 + here->B3SOIFDqe);
|
|
*(ckt->CKTstate1 + here->B3SOIFDqth) =
|
|
*(ckt->CKTstate0 + here->B3SOIFDqth);
|
|
}
|
|
|
|
error = NIintegrate(ckt, &geq, &ceq,0.0,here->B3SOIFDqb);
|
|
if (error) return(error);
|
|
error = NIintegrate(ckt, &geq, &ceq, 0.0, here->B3SOIFDqg);
|
|
if (error) return(error);
|
|
error = NIintegrate(ckt,&geq, &ceq, 0.0, here->B3SOIFDqd);
|
|
if (error) return(error);
|
|
error = NIintegrate(ckt,&geq, &ceq, 0.0, here->B3SOIFDqe);
|
|
if (error) return(error);
|
|
if ((model->B3SOIFDshMod == 1) && (here->B3SOIFDrth0!=0.0))
|
|
{
|
|
error = NIintegrate(ckt, &geq, &ceq, 0.0, here->B3SOIFDqth);
|
|
if (error) return (error);
|
|
}
|
|
|
|
goto line860;
|
|
|
|
line850:
|
|
/* initialize to zero charge conductance and current */
|
|
ceqqe = ceqqg = ceqqb = ceqqd = ceqqth= 0.0;
|
|
|
|
gcdgb = gcddb = gcdsb = gcdeb = gcdT = 0.0;
|
|
gcsgb = gcsdb = gcssb = gcseb = gcsT = 0.0;
|
|
gcggb = gcgdb = gcgsb = gcgeb = gcgT = 0.0;
|
|
gcbgb = gcbdb = gcbsb = gcbeb = gcbT = 0.0;
|
|
gcegb = gcedb = gceeb = gcesb = gceT = 0.0;
|
|
gcTt = 0.0;
|
|
|
|
sxpart = (1.0 - (dxpart = (here->B3SOIFDmode > 0) ? 0.4 : 0.6));
|
|
|
|
goto line900;
|
|
|
|
line860:
|
|
/* evaluate equivalent charge current */
|
|
|
|
cqgate = *(ckt->CKTstate0 + here->B3SOIFDcqg);
|
|
cqbody = *(ckt->CKTstate0 + here->B3SOIFDcqb);
|
|
cqdrn = *(ckt->CKTstate0 + here->B3SOIFDcqd);
|
|
cqsub = *(ckt->CKTstate0 + here->B3SOIFDcqe);
|
|
cqtemp = *(ckt->CKTstate0 + here->B3SOIFDcqth);
|
|
|
|
here->B3SOIFDcb += cqbody;
|
|
here->B3SOIFDcd += cqdrn;
|
|
|
|
ceqqg = cqgate - gcggb * vgb + gcgdb * vbd + gcgsb * vbs
|
|
- gcgeb * veb - gcgT * delTemp;
|
|
ceqqb = cqbody - gcbgb * vgb + gcbdb * vbd + gcbsb * vbs
|
|
- gcbeb * veb - gcbT * delTemp;
|
|
ceqqd = cqdrn - gcdgb * vgb + gcddb * vbd + gcdsb * vbs
|
|
- gcdeb * veb - gcdT * delTemp;
|
|
ceqqe = cqsub - gcegb * vgb + gcedb * vbd + gcesb * vbs
|
|
- gceeb * veb - gceT * delTemp;;
|
|
ceqqth = cqtemp - gcTt * delTemp;
|
|
|
|
if (ckt->CKTmode & MODEINITTRAN)
|
|
{ *(ckt->CKTstate1 + here->B3SOIFDcqe) =
|
|
*(ckt->CKTstate0 + here->B3SOIFDcqe);
|
|
*(ckt->CKTstate1 + here->B3SOIFDcqb) =
|
|
*(ckt->CKTstate0 + here->B3SOIFDcqb);
|
|
*(ckt->CKTstate1 + here->B3SOIFDcqg) =
|
|
*(ckt->CKTstate0 + here->B3SOIFDcqg);
|
|
*(ckt->CKTstate1 + here->B3SOIFDcqd) =
|
|
*(ckt->CKTstate0 + here->B3SOIFDcqd);
|
|
*(ckt->CKTstate1 + here->B3SOIFDcqth) =
|
|
*(ckt->CKTstate0 + here->B3SOIFDcqth);
|
|
}
|
|
|
|
/*
|
|
* load current vector
|
|
*/
|
|
line900:
|
|
|
|
m = here->B3SOIFDm;
|
|
|
|
if (here->B3SOIFDmode >= 0)
|
|
{ Gm = here->B3SOIFDgm;
|
|
Gmbs = here->B3SOIFDgmbs;
|
|
Gme = here->B3SOIFDgme;
|
|
GmT = model->B3SOIFDtype * here->B3SOIFDgmT;
|
|
FwdSum = Gm + Gmbs + Gme;
|
|
RevSum = 0.0;
|
|
cdreq = model->B3SOIFDtype * (here->B3SOIFDcdrain - here->B3SOIFDgds * vds
|
|
- Gm * vgs - Gmbs * vbs - Gme * ves - GmT * delTemp);
|
|
/* ceqbs now is compatible with cdreq, ie. going in is +ve */
|
|
/* Equivalent current source from the diode */
|
|
ceqbs = here->B3SOIFDcjs;
|
|
ceqbd = here->B3SOIFDcjd;
|
|
/* Current going in is +ve */
|
|
ceqbody = -here->B3SOIFDcbody;
|
|
ceqth = here->B3SOIFDcth;
|
|
ceqbodcon = here->B3SOIFDcbodcon;
|
|
|
|
gbbg = -here->B3SOIFDgbgs;
|
|
gbbdp = -here->B3SOIFDgbds;
|
|
gbbb = -here->B3SOIFDgbbs;
|
|
gbbe = -here->B3SOIFDgbes;
|
|
gbbp = -here->B3SOIFDgbps;
|
|
gbbT = -model->B3SOIFDtype * here->B3SOIFDgbT;
|
|
gbbsp = - ( gbbg + gbbdp + gbbb + gbbe + gbbp);
|
|
|
|
gddpg = -here->B3SOIFDgjdg;
|
|
gddpdp = -here->B3SOIFDgjdd;
|
|
gddpb = -here->B3SOIFDgjdb;
|
|
gddpe = -here->B3SOIFDgjde;
|
|
gddpT = -model->B3SOIFDtype * here->B3SOIFDgjdT;
|
|
gddpsp = - ( gddpg + gddpdp + gddpb + gddpe);
|
|
|
|
gsspg = -here->B3SOIFDgjsg;
|
|
gsspdp = -here->B3SOIFDgjsd;
|
|
gsspb = -here->B3SOIFDgjsb;
|
|
gsspe = 0.0;
|
|
gsspT = -model->B3SOIFDtype * here->B3SOIFDgjsT;
|
|
gsspsp = - (gsspg + gsspdp + gsspb + gsspe);
|
|
|
|
gppg = -here->B3SOIFDgbpgs;
|
|
gppdp = -here->B3SOIFDgbpds;
|
|
gppb = -here->B3SOIFDgbpbs;
|
|
gppe = -here->B3SOIFDgbpes;
|
|
gppp = -here->B3SOIFDgbpps;
|
|
gppT = -model->B3SOIFDtype * here->B3SOIFDgbpT;
|
|
gppsp = - (gppg + gppdp + gppb + gppe + gppp);
|
|
|
|
gTtg = here->B3SOIFDgtempg;
|
|
gTtb = here->B3SOIFDgtempb;
|
|
gTte = here->B3SOIFDgtempe;
|
|
gTtdp = here->B3SOIFDgtempd;
|
|
gTtt = here->B3SOIFDgtempT;
|
|
gTtsp = - (gTtg + gTtb + gTte + gTtdp);
|
|
}
|
|
else
|
|
{ Gm = -here->B3SOIFDgm;
|
|
Gmbs = -here->B3SOIFDgmbs;
|
|
Gme = -here->B3SOIFDgme;
|
|
GmT = -model->B3SOIFDtype * here->B3SOIFDgmT;
|
|
FwdSum = 0.0;
|
|
RevSum = -(Gm + Gmbs + Gme);
|
|
cdreq = -model->B3SOIFDtype * (here->B3SOIFDcdrain + here->B3SOIFDgds*vds
|
|
+ Gm * vgd + Gmbs * vbd + Gme * (ves - vds) + GmT * delTemp);
|
|
ceqbs = here->B3SOIFDcjd;
|
|
ceqbd = here->B3SOIFDcjs;
|
|
/* Current going in is +ve */
|
|
ceqbody = -here->B3SOIFDcbody;
|
|
ceqth = here->B3SOIFDcth;
|
|
ceqbodcon = here->B3SOIFDcbodcon;
|
|
|
|
gbbg = -here->B3SOIFDgbgs;
|
|
gbbb = -here->B3SOIFDgbbs;
|
|
gbbe = -here->B3SOIFDgbes;
|
|
gbbp = -here->B3SOIFDgbps;
|
|
gbbsp = -here->B3SOIFDgbds;
|
|
gbbT = -model->B3SOIFDtype * here->B3SOIFDgbT;
|
|
gbbdp = - ( gbbg + gbbsp + gbbb + gbbe + gbbp);
|
|
|
|
gddpg = -here->B3SOIFDgjsg;
|
|
gddpsp = -here->B3SOIFDgjsd;
|
|
gddpb = -here->B3SOIFDgjsb;
|
|
gddpe = 0.0;
|
|
gddpT = -model->B3SOIFDtype * here->B3SOIFDgjsT;
|
|
gddpdp = - (gddpg + gddpsp + gddpb + gddpe);
|
|
|
|
gsspg = -here->B3SOIFDgjdg;
|
|
gsspsp = -here->B3SOIFDgjdd;
|
|
gsspb = -here->B3SOIFDgjdb;
|
|
gsspe = -here->B3SOIFDgjde;
|
|
gsspT = -model->B3SOIFDtype * here->B3SOIFDgjdT;
|
|
gsspdp = - ( gsspg + gsspsp + gsspb + gsspe);
|
|
|
|
gppg = -here->B3SOIFDgbpgs;
|
|
gppsp = -here->B3SOIFDgbpds;
|
|
gppb = -here->B3SOIFDgbpbs;
|
|
gppe = -here->B3SOIFDgbpes;
|
|
gppp = -here->B3SOIFDgbpps;
|
|
gppT = -model->B3SOIFDtype * here->B3SOIFDgbpT;
|
|
gppdp = - (gppg + gppsp + gppb + gppe + gppp);
|
|
|
|
gTtg = here->B3SOIFDgtempg;
|
|
gTtb = here->B3SOIFDgtempb;
|
|
gTte = here->B3SOIFDgtempe;
|
|
gTtsp = here->B3SOIFDgtempd;
|
|
gTtt = here->B3SOIFDgtempT;
|
|
gTtdp = - (gTtg + gTtb + gTte + gTtsp);
|
|
}
|
|
|
|
if (model->B3SOIFDtype > 0)
|
|
{
|
|
ceqqg = ceqqg;
|
|
ceqqb = ceqqb;
|
|
ceqqe = ceqqe;
|
|
ceqqd = ceqqd;
|
|
}
|
|
else
|
|
{
|
|
ceqbodcon = -ceqbodcon;
|
|
ceqbody = -ceqbody;
|
|
ceqbs = -ceqbs;
|
|
ceqbd = -ceqbd;
|
|
ceqqg = -ceqqg;
|
|
ceqqb = -ceqqb;
|
|
ceqqd = -ceqqd;
|
|
ceqqe = -ceqqe;
|
|
}
|
|
|
|
(*(ckt->CKTrhs + here->B3SOIFDgNode) -= m * ceqqg);
|
|
(*(ckt->CKTrhs + here->B3SOIFDdNodePrime) += m * (ceqbd - cdreq - ceqqd));
|
|
(*(ckt->CKTrhs + here->B3SOIFDsNodePrime) += m * (cdreq + ceqbs + ceqqg
|
|
+ ceqqb + ceqqd + ceqqe));
|
|
(*(ckt->CKTrhs + here->B3SOIFDeNode) -= m * ceqqe);
|
|
|
|
if (here->B3SOIFDbodyMod == 1) {
|
|
(*(ckt->CKTrhs + here->B3SOIFDpNode) += m * ceqbodcon);
|
|
}
|
|
|
|
if (selfheat) {
|
|
(*(ckt->CKTrhs + here->B3SOIFDtempNode) -= m * (ceqth + ceqqth));
|
|
}
|
|
|
|
|
|
|
|
if ((here->B3SOIFDdebugMod > 1) || (here->B3SOIFDdebugMod == -1))
|
|
{
|
|
*(ckt->CKTrhs + here->B3SOIFDvbsNode) = here->B3SOIFDvbsdio;
|
|
*(ckt->CKTrhs + here->B3SOIFDidsNode) = here->B3SOIFDids;
|
|
*(ckt->CKTrhs + here->B3SOIFDicNode) = here->B3SOIFDic;
|
|
*(ckt->CKTrhs + here->B3SOIFDibsNode) = here->B3SOIFDibs;
|
|
*(ckt->CKTrhs + here->B3SOIFDibdNode) = here->B3SOIFDibd;
|
|
*(ckt->CKTrhs + here->B3SOIFDiiiNode) = here->B3SOIFDiii;
|
|
*(ckt->CKTrhs + here->B3SOIFDigidlNode) = here->B3SOIFDigidl;
|
|
*(ckt->CKTrhs + here->B3SOIFDitunNode) = here->B3SOIFDitun;
|
|
*(ckt->CKTrhs + here->B3SOIFDibpNode) = here->B3SOIFDibp;
|
|
*(ckt->CKTrhs + here->B3SOIFDabeffNode) = here->B3SOIFDabeff;
|
|
*(ckt->CKTrhs + here->B3SOIFDvbs0effNode) = here->B3SOIFDvbs0eff;
|
|
*(ckt->CKTrhs + here->B3SOIFDvbseffNode) = here->B3SOIFDvbseff;
|
|
*(ckt->CKTrhs + here->B3SOIFDxcNode) = here->B3SOIFDxc;
|
|
*(ckt->CKTrhs + here->B3SOIFDcbbNode) = here->B3SOIFDcbb;
|
|
*(ckt->CKTrhs + here->B3SOIFDcbdNode) = here->B3SOIFDcbd;
|
|
*(ckt->CKTrhs + here->B3SOIFDcbgNode) = here->B3SOIFDcbg;
|
|
*(ckt->CKTrhs + here->B3SOIFDqbfNode) = here->B3SOIFDqbf;
|
|
*(ckt->CKTrhs + here->B3SOIFDqjsNode) = here->B3SOIFDqjs;
|
|
*(ckt->CKTrhs + here->B3SOIFDqjdNode) = here->B3SOIFDqjd;
|
|
|
|
/* clean up last */
|
|
*(ckt->CKTrhs + here->B3SOIFDgmNode) = Gm;
|
|
*(ckt->CKTrhs + here->B3SOIFDgmbsNode) = Gmbs;
|
|
*(ckt->CKTrhs + here->B3SOIFDgdsNode) = Gds;
|
|
*(ckt->CKTrhs + here->B3SOIFDgmeNode) = Gme;
|
|
*(ckt->CKTrhs + here->B3SOIFDqdNode) = qdrn;
|
|
*(ckt->CKTrhs + here->B3SOIFDcbeNode) = Cbe;
|
|
*(ckt->CKTrhs + here->B3SOIFDvbs0teffNode) = Vbs0teff;
|
|
*(ckt->CKTrhs + here->B3SOIFDvthNode) = here->B3SOIFDvon;
|
|
*(ckt->CKTrhs + here->B3SOIFDvgsteffNode) = Vgsteff;
|
|
*(ckt->CKTrhs + here->B3SOIFDxcsatNode) = Xcsat;
|
|
*(ckt->CKTrhs + here->B3SOIFDqaccNode) = -Qac0;
|
|
*(ckt->CKTrhs + here->B3SOIFDqsub0Node) = Qsub0;
|
|
*(ckt->CKTrhs + here->B3SOIFDqsubs1Node) = Qsubs1;
|
|
*(ckt->CKTrhs + here->B3SOIFDqsubs2Node) = Qsubs2;
|
|
*(ckt->CKTrhs + here->B3SOIFDvdscvNode) = VdsCV;
|
|
*(ckt->CKTrhs + here->B3SOIFDvcscvNode) = VcsCV;
|
|
*(ckt->CKTrhs + here->B3SOIFDqgNode) = qgate;
|
|
*(ckt->CKTrhs + here->B3SOIFDqbNode) = qbody;
|
|
*(ckt->CKTrhs + here->B3SOIFDqeNode) = qsub;
|
|
*(ckt->CKTrhs + here->B3SOIFDdum1Node) = here->B3SOIFDdum1;
|
|
*(ckt->CKTrhs + here->B3SOIFDdum2Node) = here->B3SOIFDdum2;
|
|
*(ckt->CKTrhs + here->B3SOIFDdum3Node) = here->B3SOIFDdum3;
|
|
*(ckt->CKTrhs + here->B3SOIFDdum4Node) = here->B3SOIFDdum4;
|
|
*(ckt->CKTrhs + here->B3SOIFDdum5Node) = here->B3SOIFDdum5;
|
|
/* end clean up last */
|
|
}
|
|
|
|
|
|
/*
|
|
* load y matrix
|
|
*/
|
|
(*(here->B3SOIFDEgPtr) += m * gcegb);
|
|
(*(here->B3SOIFDEdpPtr) += m * gcedb);
|
|
(*(here->B3SOIFDEspPtr) += m * gcesb);
|
|
(*(here->B3SOIFDGePtr) += m * gcgeb);
|
|
(*(here->B3SOIFDDPePtr) += m * (Gme + gddpe + gcdeb));
|
|
(*(here->B3SOIFDSPePtr) += m * (gsspe - Gme + gcseb));
|
|
|
|
(*(here->B3SOIFDEePtr) += m * gceeb);
|
|
|
|
(*(here->B3SOIFDGgPtr) += m * (gcggb + ckt->CKTgmin));
|
|
(*(here->B3SOIFDGdpPtr) += m * (gcgdb - ckt->CKTgmin));
|
|
(*(here->B3SOIFDGspPtr) += m * gcgsb );
|
|
|
|
(*(here->B3SOIFDDPgPtr) += m * ((Gm + gcdgb) + gddpg - ckt->CKTgmin));
|
|
(*(here->B3SOIFDDPdpPtr) += m * ((here->B3SOIFDdrainConductance
|
|
+ here->B3SOIFDgds + gddpdp
|
|
+ RevSum + gcddb) + ckt->CKTgmin));
|
|
(*(here->B3SOIFDDPspPtr) -= m * (-gddpsp + here->B3SOIFDgds + FwdSum - gcdsb));
|
|
|
|
(*(here->B3SOIFDDPdPtr) -= m * here->B3SOIFDdrainConductance);
|
|
|
|
(*(here->B3SOIFDSPgPtr) += m * (gcsgb - Gm + gsspg));
|
|
(*(here->B3SOIFDSPdpPtr) -= m * (here->B3SOIFDgds - gsspdp + RevSum - gcsdb));
|
|
(*(here->B3SOIFDSPspPtr) += m * (here->B3SOIFDsourceConductance
|
|
+ here->B3SOIFDgds + gsspsp
|
|
+ FwdSum + gcssb));
|
|
(*(here->B3SOIFDSPsPtr) -= m * here->B3SOIFDsourceConductance);
|
|
|
|
|
|
(*(here->B3SOIFDDdPtr) += m * here->B3SOIFDdrainConductance);
|
|
(*(here->B3SOIFDDdpPtr) -= m * here->B3SOIFDdrainConductance);
|
|
|
|
|
|
(*(here->B3SOIFDSsPtr) += m * here->B3SOIFDsourceConductance);
|
|
(*(here->B3SOIFDSspPtr) -= m * here->B3SOIFDsourceConductance);
|
|
|
|
if (here->B3SOIFDbodyMod == 1) {
|
|
(*(here->B3SOIFDBpPtr) -= m * gppp);
|
|
(*(here->B3SOIFDPbPtr) += m * gppb);
|
|
(*(here->B3SOIFDPpPtr) += m * gppp);
|
|
(*(here->B3SOIFDPgPtr) += m * gppg);
|
|
(*(here->B3SOIFDPdpPtr) += m * gppdp);
|
|
(*(here->B3SOIFDPspPtr) += m * gppsp);
|
|
(*(here->B3SOIFDPePtr) += m * gppe);
|
|
}
|
|
|
|
if (selfheat)
|
|
{
|
|
(*(here->B3SOIFDDPtempPtr) += m * (GmT + gddpT + gcdT));
|
|
(*(here->B3SOIFDSPtempPtr) += m * (-GmT + gsspT + gcsT));
|
|
(*(here->B3SOIFDBtempPtr) += m * (gbbT + gcbT));
|
|
(*(here->B3SOIFDEtempPtr) += m * gceT);
|
|
(*(here->B3SOIFDGtempPtr) += m * gcgT);
|
|
if (here->B3SOIFDbodyMod == 1) {
|
|
(*(here->B3SOIFDPtempPtr) += m * gppT);
|
|
}
|
|
(*(here->B3SOIFDTemptempPtr) += m * (gTtt + 1/pParam->B3SOIFDrth + gcTt));
|
|
(*(here->B3SOIFDTempgPtr) += m * gTtg);
|
|
(*(here->B3SOIFDTempbPtr) += m * gTtb);
|
|
(*(here->B3SOIFDTempePtr) += m * gTte);
|
|
(*(here->B3SOIFDTempdpPtr) += m * gTtdp);
|
|
(*(here->B3SOIFDTempspPtr) += m * gTtsp);
|
|
}
|
|
|
|
if ((here->B3SOIFDdebugMod > 1) || (here->B3SOIFDdebugMod == -1))
|
|
{
|
|
*(here->B3SOIFDVbsPtr) += m * 1;
|
|
*(here->B3SOIFDIdsPtr) += m * 1;
|
|
*(here->B3SOIFDIcPtr) += m * 1;
|
|
*(here->B3SOIFDIbsPtr) += m * 1;
|
|
*(here->B3SOIFDIbdPtr) += m * 1;
|
|
*(here->B3SOIFDIiiPtr) += m * 1;
|
|
*(here->B3SOIFDIgidlPtr) += m * 1;
|
|
*(here->B3SOIFDItunPtr) += m * 1;
|
|
*(here->B3SOIFDIbpPtr) += m * 1;
|
|
*(here->B3SOIFDAbeffPtr) += m * 1;
|
|
*(here->B3SOIFDVbs0effPtr) += m * 1;
|
|
*(here->B3SOIFDVbseffPtr) += m * 1;
|
|
*(here->B3SOIFDXcPtr) += m * 1;
|
|
*(here->B3SOIFDCbgPtr) += m * 1;
|
|
*(here->B3SOIFDCbbPtr) += m * 1;
|
|
*(here->B3SOIFDCbdPtr) += m * 1;
|
|
*(here->B3SOIFDqbPtr) += m * 1;
|
|
*(here->B3SOIFDQbfPtr) += m * 1;
|
|
*(here->B3SOIFDQjsPtr) += m * 1;
|
|
*(here->B3SOIFDQjdPtr) += m * 1;
|
|
|
|
/* clean up last */
|
|
*(here->B3SOIFDGmPtr) += m * 1;
|
|
*(here->B3SOIFDGmbsPtr) += m * 1;
|
|
*(here->B3SOIFDGdsPtr) += m * 1;
|
|
*(here->B3SOIFDGmePtr) += m * 1;
|
|
*(here->B3SOIFDVbs0teffPtr) += m * 1;
|
|
*(here->B3SOIFDVgsteffPtr) += m * 1;
|
|
*(here->B3SOIFDCbePtr) += m * 1;
|
|
*(here->B3SOIFDVthPtr) += m * 1;
|
|
*(here->B3SOIFDXcsatPtr) += m * 1;
|
|
*(here->B3SOIFDVdscvPtr) += m * 1;
|
|
*(here->B3SOIFDVcscvPtr) += m * 1;
|
|
*(here->B3SOIFDQaccPtr) += m * 1;
|
|
*(here->B3SOIFDQsub0Ptr) += m * 1;
|
|
*(here->B3SOIFDQsubs1Ptr) += m * 1;
|
|
*(here->B3SOIFDQsubs2Ptr) += m * 1;
|
|
*(here->B3SOIFDqgPtr) += m * 1;
|
|
*(here->B3SOIFDqdPtr) += m * 1;
|
|
*(here->B3SOIFDqePtr) += m * 1;
|
|
*(here->B3SOIFDDum1Ptr) += m * 1;
|
|
*(here->B3SOIFDDum2Ptr) += m * 1;
|
|
*(here->B3SOIFDDum3Ptr) += m * 1;
|
|
*(here->B3SOIFDDum4Ptr) += m * 1;
|
|
*(here->B3SOIFDDum5Ptr) += m * 1;
|
|
/* end clean up last */
|
|
}
|
|
|
|
line1000: ;
|
|
|
|
/* Here NaN will be detected in any conductance or equivalent current. Note
|
|
that nandetect is initialized within the "if" statements */
|
|
|
|
if ((nandetect = isnan (*(here->B3SOIFDGgPtr))))
|
|
{ strcpy (nanmessage, "GgPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDGdpPtr))))
|
|
{ strcpy (nanmessage, "GdpPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDGspPtr))))
|
|
{ strcpy (nanmessage, "GspPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDDPgPtr))))
|
|
{ strcpy (nanmessage, "DPgPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDDPdpPtr))))
|
|
{ strcpy (nanmessage, "DPdpPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDDPspPtr))))
|
|
{ strcpy (nanmessage, "DPspPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDSPgPtr))))
|
|
{ strcpy (nanmessage, "SPgPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDSPdpPtr))))
|
|
{ strcpy (nanmessage, "SPdpPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDSPspPtr))))
|
|
{ strcpy (nanmessage, "SPspPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDEePtr))))
|
|
{ strcpy (nanmessage, "EePtr"); }
|
|
|
|
/* At this point, nandetect = 0 if none of the
|
|
conductances checked so far are NaN */
|
|
|
|
if (nandetect == 0)
|
|
{
|
|
if ((nandetect = isnan (*(here->B3SOIFDEgPtr))))
|
|
{ strcpy (nanmessage, "EgPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDEdpPtr))))
|
|
{ strcpy (nanmessage, "EdpPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDEspPtr))))
|
|
{ strcpy (nanmessage, "EspPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDGePtr))))
|
|
{ strcpy (nanmessage, "GePtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDDPePtr))))
|
|
{ strcpy (nanmessage, "DPePtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDSPePtr))))
|
|
{ strcpy (nanmessage, "SPePtr"); } }
|
|
|
|
/* Now check if self-heating caused NaN if nothing else
|
|
has so far (check tempnode current also) */
|
|
|
|
if (selfheat && nandetect == 0)
|
|
{
|
|
if ((nandetect = isnan (*(here->B3SOIFDTemptempPtr))))
|
|
{ strcpy (nanmessage, "TemptempPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDTempgPtr))))
|
|
{ strcpy (nanmessage, "TempgPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDTempbPtr))))
|
|
{ strcpy (nanmessage, "TempbPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDTempePtr))))
|
|
{ strcpy (nanmessage, "TempePtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDTempdpPtr))))
|
|
{ strcpy (nanmessage, "TempdpPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDTempspPtr))))
|
|
{ strcpy (nanmessage, "TempspPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDGtempPtr))))
|
|
{ strcpy (nanmessage, "GtempPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDDPtempPtr))))
|
|
{ strcpy (nanmessage, "DPtempPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDSPtempPtr))))
|
|
{ strcpy (nanmessage, "SPtempPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDEtempPtr))))
|
|
{ strcpy (nanmessage, "EtempPtr"); }
|
|
else if ((nandetect = isnan (*(here->B3SOIFDBtempPtr))))
|
|
{ strcpy (nanmessage, "BtempPtr"); }
|
|
else if ((nandetect = isnan (*(ckt->CKTrhs + here->B3SOIFDtempNode))))
|
|
{ strcpy (nanmessage, "tempNode"); }
|
|
}
|
|
|
|
/* Lastly, check all equivalent currents (tempnode is
|
|
checked above */
|
|
|
|
if (nandetect == 0)
|
|
{
|
|
if ((nandetect = isnan (*(ckt->CKTrhs
|
|
+ here->B3SOIFDgNode))))
|
|
{ strcpy (nanmessage, "gNode"); }
|
|
else if ((nandetect = isnan (*(ckt->CKTrhs
|
|
+ here->B3SOIFDbNode))))
|
|
{ strcpy (nanmessage, "bNode"); }
|
|
else if ((nandetect = isnan (*(ckt->CKTrhs
|
|
+ here->B3SOIFDdNodePrime))))
|
|
{ strcpy (nanmessage, "dpNode"); }
|
|
else if ((nandetect = isnan (*(ckt->CKTrhs
|
|
+ here->B3SOIFDsNodePrime))))
|
|
{ strcpy (nanmessage, "spNode"); }
|
|
else if ((nandetect = isnan (*(ckt->CKTrhs
|
|
+ here->B3SOIFDeNode))))
|
|
{ strcpy (nanmessage, "eNode"); }
|
|
}
|
|
|
|
/* Now print error message if NaN detected. Note that
|
|
error will only be printed once (the first time it is
|
|
encountered) each time SPICE is run since nanfound is
|
|
static variable */
|
|
|
|
if (nanfound == 0 && nandetect)
|
|
{
|
|
fprintf(stderr, "Alberto says: YOU TURKEY! %s is NaN for instance %s at time %g!\n", nanmessage, here->B3SOIFDname, ckt->CKTtime);
|
|
nanfound = nandetect;
|
|
fprintf(stderr, " The program exit!\n");
|
|
exit(-1);
|
|
}
|
|
|
|
if (here->B3SOIFDdebugMod > 2)
|
|
{
|
|
fprintf(fpdebug, "Ids = %.4e, Ic = %.4e, cqdrn = %.4e, gmin=%.3e\n",
|
|
Ids, Ic, cqdrn, ckt->CKTgmin);
|
|
fprintf(fpdebug, "Iii = %.4e, Idgidl = %.4e, Ibs = %.14e\n",
|
|
Iii, Idgidl, Ibs);
|
|
fprintf(fpdebug, "Ibd = %.4e, Ibp = %.4e\n", Ibd, Ibp);
|
|
fprintf(fpdebug, "qbody = %.5e, qbf = %.5e, qbe = %.5e\n",
|
|
qbody, Qbf, -(Qe1+Qe2));
|
|
fprintf(fpdebug, "qbs = %.5e, qbd = %.5e\n", qjs, qjd);
|
|
fprintf(fpdebug, "qdrn = %.5e, qinv = %.5e\n", qdrn, qinv);
|
|
|
|
|
|
|
|
|
|
/* I am trying to debug the convergence problems here by printing out
|
|
the entire Jacobian and equivalent current matrix */
|
|
|
|
if (here->B3SOIFDdebugMod > 4) {
|
|
fprintf(fpdebug, "Ibtot = %.6e;\t Cbtot = %.6e;\n", Ibs+Ibp+Ibd-Iii-Idgidl-Isgidl, cqbody);
|
|
fprintf(fpdebug, "ceqg = %.6e;\t ceqb = %.6e;\t ceqdp = %.6e;\t ceqsp = %.6e;\n",
|
|
*(ckt->CKTrhs + here->B3SOIFDgNode),
|
|
*(ckt->CKTrhs + here->B3SOIFDbNode),
|
|
*(ckt->CKTrhs + here->B3SOIFDdNodePrime),
|
|
*(ckt->CKTrhs + here->B3SOIFDsNodePrime));
|
|
fprintf(fpdebug, "ceqe = %.6e;\t ceqp = %.6e;\t ceqth = %.6e;\n",
|
|
*(ckt->CKTrhs + here->B3SOIFDeNode),
|
|
*(ckt->CKTrhs + here->B3SOIFDpNode),
|
|
*(ckt->CKTrhs + here->B3SOIFDtempNode));
|
|
|
|
fprintf(fpdebug, "Eg = %.5e;\t Edp = %.5e;\t Esp = %.5e;\t Eb = %.5e;\n",
|
|
*(here->B3SOIFDEgPtr),
|
|
*(here->B3SOIFDEdpPtr),
|
|
*(here->B3SOIFDEspPtr),
|
|
*(here->B3SOIFDEbPtr));
|
|
fprintf(fpdebug, "Ee = %.5e;\t Gg = %.5e;\t Gdp = %.5e;\t Gsp = %.5e;\n",
|
|
*(here->B3SOIFDEePtr),
|
|
*(here->B3SOIFDGgPtr),
|
|
*(here->B3SOIFDGdpPtr),
|
|
*(here->B3SOIFDGspPtr));
|
|
fprintf(fpdebug, "Gb = %.5e;\t Ge = %.5e;\t DPg = %.5e;\t DPdp = %.5e;\n",
|
|
*(here->B3SOIFDGbPtr),
|
|
*(here->B3SOIFDGePtr),
|
|
*(here->B3SOIFDDPgPtr),
|
|
*(here->B3SOIFDDPdpPtr));
|
|
fprintf(fpdebug, "DPsp = %.5e;\t DPb = %.5e;\t DPe = %.5e;\t\n",
|
|
*(here->B3SOIFDDPspPtr),
|
|
*(here->B3SOIFDDPbPtr),
|
|
*(here->B3SOIFDDPePtr));
|
|
fprintf(fpdebug, "DPd = %.5e;\t SPg = %.5e;\t SPdp = %.5e;\t SPsp = %.5e;\n",
|
|
*(here->B3SOIFDDPdPtr),
|
|
*(here->B3SOIFDSPgPtr),
|
|
*(here->B3SOIFDSPdpPtr),
|
|
*(here->B3SOIFDSPspPtr));
|
|
fprintf(fpdebug, "SPb = %.5e;\t SPe = %.5e;\t SPs = %.5e;\n",
|
|
*(here->B3SOIFDSPbPtr),
|
|
*(here->B3SOIFDSPePtr),
|
|
*(here->B3SOIFDSPsPtr));
|
|
fprintf(fpdebug, "Dd = %.5e;\t Ddp = %.5e;\t Ss = %.5e;\t Ssp = %.5e;\n",
|
|
*(here->B3SOIFDDdPtr),
|
|
*(here->B3SOIFDDdpPtr),
|
|
*(here->B3SOIFDSsPtr),
|
|
*(here->B3SOIFDSspPtr));
|
|
fprintf(fpdebug, "Bg = %.5e;\t Bdp = %.5e;\t Bsp = %.5e;\t Bb = %.5e;\n",
|
|
*(here->B3SOIFDBgPtr),
|
|
*(here->B3SOIFDBdpPtr),
|
|
*(here->B3SOIFDBspPtr),
|
|
*(here->B3SOIFDBbPtr));
|
|
fprintf(fpdebug, "Be = %.5e;\t Btot = %.5e;\t DPtot = %.5e;\n",
|
|
*(here->B3SOIFDBePtr),
|
|
*(here->B3SOIFDBgPtr) + *(here->B3SOIFDBdpPtr)
|
|
+ *(here->B3SOIFDBspPtr) + *(here->B3SOIFDBbPtr)
|
|
+ *(here->B3SOIFDBePtr),
|
|
*(here->B3SOIFDDPePtr)
|
|
+ *(here->B3SOIFDDPgPtr) + *(here->B3SOIFDDPdpPtr)
|
|
+ *(here->B3SOIFDDPspPtr) + *(here->B3SOIFDDPbPtr));
|
|
|
|
if (selfheat) {
|
|
fprintf (fpdebug, "DPtemp = %.5e;\t SPtemp = %.5e;\t Btemp = %.5e;\n",
|
|
*(here->B3SOIFDDPtempPtr), *(here->B3SOIFDSPtempPtr),
|
|
*(here->B3SOIFDBtempPtr));
|
|
fprintf (fpdebug, "Gtemp = %.5e;\t Etemp = %.5e;\n",
|
|
*(here->B3SOIFDGtempPtr), *(here->B3SOIFDEtempPtr));
|
|
fprintf (fpdebug, "Tempg = %.5e;\t Tempdp = %.5e;\t Tempsp = %.5e;\t Tempb = %.5e;\n",
|
|
*(here->B3SOIFDTempgPtr), *(here->B3SOIFDTempdpPtr),
|
|
*(here->B3SOIFDTempspPtr), *(here->B3SOIFDTempbPtr));
|
|
fprintf (fpdebug, "Tempe = %.5e;\t TempT = %.5e;\t Temptot = %.5e;\n",
|
|
*(here->B3SOIFDTempePtr), *(here->B3SOIFDTemptempPtr),
|
|
*(here->B3SOIFDTempgPtr) + *(here->B3SOIFDTempdpPtr)
|
|
+ *(here->B3SOIFDTempspPtr)+ *(here->B3SOIFDTempbPtr)
|
|
+ *(here->B3SOIFDTempePtr));
|
|
}
|
|
|
|
if (here->B3SOIFDbodyMod == 1)
|
|
{
|
|
fprintf(fpdebug, "ceqbodcon=%.5e;\t", ceqbodcon);
|
|
fprintf(fpdebug, "Bp = %.5e;\t Pb = %.5e;\t Pp = %.5e;\n", -gppp, gppb, gppp);
|
|
fprintf(fpdebug, "Pg=%.5e;\t Pdp=%.5e;\t Psp=%.5e;\t Pe=%.5e;\n",
|
|
gppg, gppdp, gppsp, gppe);
|
|
}
|
|
}
|
|
|
|
if (here->B3SOIFDdebugMod > 3)
|
|
{
|
|
fprintf(fpdebug, "Vth = %.4f, Vbs0eff = %.8f, Vdsat = %.4f\n",
|
|
Vth, Vbs0eff, Vdsat);
|
|
fprintf(fpdebug, "ueff = %g, Vgsteff = %.4f, Vdseff = %.4f\n",
|
|
ueff, Vgsteff, Vdseff);
|
|
fprintf(fpdebug, "Vthfd = %.4f, Vbs0mos = %.4f, Vbs0 = %.4f\n",
|
|
Vthfd, Vbs0mos, Vbs0);
|
|
fprintf(fpdebug, "Vbs0t = %.4f, Vbsdio = %.8f\n",
|
|
Vbs0t, Vbsdio);
|
|
}
|
|
|
|
fclose(fpdebug);
|
|
}
|
|
|
|
here->B3SOIFDiterations++; /* increment the iteration counter */
|
|
|
|
} /* End of Mosfet Instance */
|
|
} /* End of Model Instance */
|
|
|
|
|
|
return(OK);
|
|
}
|
|
|