diff --git a/src/spicelib/devices/bsim3/b3check.c b/src/spicelib/devices/bsim3/b3check.c index eb00bc5ad..2c8e9ced2 100644 --- a/src/spicelib/devices/bsim3/b3check.c +++ b/src/spicelib/devices/bsim3/b3check.c @@ -3,11 +3,12 @@ /********** * Copyright 2004 Regents of the University of California. All rights reserved. * File: b3check.c of BSIM3v3.3.0 - * Author: 1995 Min-Chie Jeng + * Author: 1995 Min-Chie Jeng * Author: 1997-1999 Weidong Liu. * Author: 2001 Xuemei Xi. * Modified by Xuemei Xi, 10/05, 12/14, 2001. * Modified by Xuemei Xi, 07/29/2005. + * Modified by Dietmar Warning, 12/21/2020. **********/ #include "ngspice/ngspice.h" @@ -18,438 +19,378 @@ #include "ngspice/sperror.h" #include "ngspice/devdefs.h" #include "ngspice/suffix.h" +#include "ngspice/wordlist.h" +#include "ngspice/cpextern.h" + +/* Check for correctness of the BSIM3.3 parameters: + If parameter excursions are found, put the warning or error message into a wordlist. + Only then open a file b3v33check.log and print the data into the file. */ int -BSIM3checkModel( -BSIM3model *model, -BSIM3instance *here, -CKTcircuit *ckt) +BSIM3checkModel (BSIM3model *model, BSIM3instance *here, CKTcircuit *ckt) { -struct bsim3SizeDependParam *pParam; -int Fatal_Flag = 0; -FILE *fplog; - + struct bsim3SizeDependParam *pParam; + int Fatal_Flag = 0; + FILE *fplog; + wordlist* wl, *wlstart; + NG_IGNORE(ckt); - if ((fplog = fopen("b3v33check.log", "w")) != NULL) - { pParam = here->pParam; - fprintf(fplog, "BSIM3v3.3.0 Parameter Checking.\n"); - if ((strncmp(model->BSIM3version, "3.3.0", 5)) && (strncmp(model->BSIM3version, "3.30", 4)) && (strncmp(model->BSIM3version, "3.3", 3))) - { fprintf(fplog, "Warning: This model is BSIM3v3.3.0; you specified a wrong version number.\n"); - printf("Warning: This model is BSIM3v3.3.0; you specified a wrong version number.\n"); - } - fprintf(fplog, "Model = %s\n", model->BSIM3modName); - - if (pParam->BSIM3nlx < -pParam->BSIM3leff) - { fprintf(fplog, "Fatal: Nlx = %g is less than -Leff.\n", - pParam->BSIM3nlx); - printf("Fatal: Nlx = %g is less than -Leff.\n", - pParam->BSIM3nlx); - Fatal_Flag = 1; - } + pParam = here->pParam; - if (model->BSIM3tox <= 0.0) - { fprintf(fplog, "Fatal: Tox = %g is not positive.\n", - model->BSIM3tox); - printf("Fatal: Tox = %g is not positive.\n", model->BSIM3tox); - Fatal_Flag = 1; - } - - if (model->BSIM3toxm <= 0.0) - { fprintf(fplog, "Fatal: Toxm = %g is not positive.\n", - model->BSIM3toxm); - printf("Fatal: Toxm = %g is not positive.\n", model->BSIM3toxm); - Fatal_Flag = 1; - } - - if (model->BSIM3lintnoi > pParam->BSIM3leff/2) - { fprintf(fplog, "Fatal: Lintnoi = %g is too large - Leff for noise is negative.\n", - model->BSIM3lintnoi); - printf("Fatal: Lintnoi = %g is too large - Leff for noise is negative.\n", - model->BSIM3lintnoi); - Fatal_Flag = 1; - } - - if (pParam->BSIM3npeak <= 0.0) - { fprintf(fplog, "Fatal: Nch = %g is not positive.\n", - pParam->BSIM3npeak); - printf("Fatal: Nch = %g is not positive.\n", - pParam->BSIM3npeak); - Fatal_Flag = 1; - } - if (pParam->BSIM3nsub <= 0.0) - { fprintf(fplog, "Fatal: Nsub = %g is not positive.\n", - pParam->BSIM3nsub); - printf("Fatal: Nsub = %g is not positive.\n", - pParam->BSIM3nsub); - Fatal_Flag = 1; - } - if (pParam->BSIM3ngate < 0.0) - { fprintf(fplog, "Fatal: Ngate = %g is not positive.\n", - pParam->BSIM3ngate); - printf("Fatal: Ngate = %g Ngate is not positive.\n", - pParam->BSIM3ngate); - Fatal_Flag = 1; - } - if (pParam->BSIM3ngate > 1.e25) - { fprintf(fplog, "Fatal: Ngate = %g is too high.\n", - pParam->BSIM3ngate); - printf("Fatal: Ngate = %g Ngate is too high\n", - pParam->BSIM3ngate); - Fatal_Flag = 1; - } - if (pParam->BSIM3xj <= 0.0) - { fprintf(fplog, "Fatal: Xj = %g is not positive.\n", - pParam->BSIM3xj); - printf("Fatal: Xj = %g is not positive.\n", pParam->BSIM3xj); - Fatal_Flag = 1; - } - - if (pParam->BSIM3dvt1 < 0.0) - { fprintf(fplog, "Fatal: Dvt1 = %g is negative.\n", - pParam->BSIM3dvt1); - printf("Fatal: Dvt1 = %g is negative.\n", pParam->BSIM3dvt1); - Fatal_Flag = 1; - } - - if (pParam->BSIM3dvt1w < 0.0) - { fprintf(fplog, "Fatal: Dvt1w = %g is negative.\n", - pParam->BSIM3dvt1w); - printf("Fatal: Dvt1w = %g is negative.\n", pParam->BSIM3dvt1w); - Fatal_Flag = 1; - } - - if (pParam->BSIM3w0 == -pParam->BSIM3weff) - { fprintf(fplog, "Fatal: (W0 + Weff) = 0 causing divided-by-zero.\n"); - printf("Fatal: (W0 + Weff) = 0 causing divided-by-zero.\n"); - Fatal_Flag = 1; - } - - if (pParam->BSIM3dsub < 0.0) - { fprintf(fplog, "Fatal: Dsub = %g is negative.\n", pParam->BSIM3dsub); - printf("Fatal: Dsub = %g is negative.\n", pParam->BSIM3dsub); - Fatal_Flag = 1; - } - if (pParam->BSIM3b1 == -pParam->BSIM3weff) - { fprintf(fplog, "Fatal: (B1 + Weff) = 0 causing divided-by-zero.\n"); - printf("Fatal: (B1 + Weff) = 0 causing divided-by-zero.\n"); - Fatal_Flag = 1; - } - if (pParam->BSIM3u0temp <= 0.0) - { fprintf(fplog, "Fatal: u0 at current temperature = %g is not positive.\n", pParam->BSIM3u0temp); - printf("Fatal: u0 at current temperature = %g is not positive.\n", - pParam->BSIM3u0temp); - Fatal_Flag = 1; - } - -/* Check delta parameter */ - if (pParam->BSIM3delta < 0.0) - { fprintf(fplog, "Fatal: Delta = %g is less than zero.\n", - pParam->BSIM3delta); - printf("Fatal: Delta = %g is less than zero.\n", pParam->BSIM3delta); - Fatal_Flag = 1; - } - - if (pParam->BSIM3vsattemp <= 0.0) - { fprintf(fplog, "Fatal: Vsat at current temperature = %g is not positive.\n", pParam->BSIM3vsattemp); - printf("Fatal: Vsat at current temperature = %g is not positive.\n", - pParam->BSIM3vsattemp); - Fatal_Flag = 1; - } + if (cp_getvar("ng_nomodcheck", CP_BOOL, NULL, 0)) + return(0); + + wl = wlstart = TMALLOC(wordlist, 1); + wl->wl_prev = NULL; + wl->wl_next = NULL; + wl->wl_word = tprintf("\nChecking parameters for BSIM 3.3 model %s\n", model->BSIM3modName); + + if ((strncmp(model->BSIM3version, "3.3.0", 5)) && (strncmp(model->BSIM3version, "3.30", 4)) && (strncmp(model->BSIM3version, "3.3", 3))) + { + printf("Warning: This model is BSIM3v3.3.0; you specified a wrong version number.\n"); + wl_append_word(&wl, &wl, tprintf("Warning: This model is BSIM3v3.3.0; you specified a wrong version number.\n")); + } + + if (pParam->BSIM3nlx < -pParam->BSIM3leff) + { wl_append_word(&wl, &wl, tprintf("Fatal: Nlx = %g is less than -Leff.\n", + pParam->BSIM3nlx)); + Fatal_Flag = 1; + } + + if (model->BSIM3tox <= 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Tox = %g is not positive.\n", + model->BSIM3tox)); + Fatal_Flag = 1; + } + + if (model->BSIM3toxm <= 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Toxm = %g is not positive.\n", + model->BSIM3toxm)); + Fatal_Flag = 1; + } + + if (model->BSIM3lintnoi > pParam->BSIM3leff/2) + { wl_append_word(&wl, &wl, tprintf("Fatal: Lintnoi = %g is too large - Leff for noise is negative.\n", + model->BSIM3lintnoi)); + Fatal_Flag = 1; + } + + if (pParam->BSIM3npeak <= 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Nch = %g is not positive.\n", + pParam->BSIM3npeak)); + Fatal_Flag = 1; + } + if (pParam->BSIM3nsub <= 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Nsub = %g is not positive.\n", + pParam->BSIM3nsub)); + Fatal_Flag = 1; + } + if (pParam->BSIM3ngate < 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Ngate = %g is not positive.\n", + pParam->BSIM3ngate)); + Fatal_Flag = 1; + } + if (pParam->BSIM3ngate > 1.e25) + { wl_append_word(&wl, &wl, tprintf("Fatal: Ngate = %g is too high.\n", + pParam->BSIM3ngate)); + Fatal_Flag = 1; + } + if (pParam->BSIM3xj <= 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Xj = %g is not positive.\n", + pParam->BSIM3xj)); + Fatal_Flag = 1; + } + + if (pParam->BSIM3dvt1 < 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Dvt1 = %g is negative.\n", + pParam->BSIM3dvt1)); + Fatal_Flag = 1; + } + + if (pParam->BSIM3dvt1w < 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Dvt1w = %g is negative.\n", + pParam->BSIM3dvt1w)); + Fatal_Flag = 1; + } + + if (pParam->BSIM3w0 == -pParam->BSIM3weff) + { wl_append_word(&wl, &wl, tprintf("Fatal: (W0 + Weff) = 0 causing divided-by-zero.\n")); + Fatal_Flag = 1; + } + + if (pParam->BSIM3dsub < 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Dsub = %g is negative.\n", pParam->BSIM3dsub)); + Fatal_Flag = 1; + } + if (pParam->BSIM3b1 == -pParam->BSIM3weff) + { wl_append_word(&wl, &wl, tprintf("Fatal: (B1 + Weff) = 0 causing divided-by-zero.\n")); + Fatal_Flag = 1; + } + if (pParam->BSIM3u0temp <= 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: u0 at current temperature = %g is not positive.\n", pParam->BSIM3u0temp)); + Fatal_Flag = 1; + } + +/* Check delta parameter */ + if (pParam->BSIM3delta < 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Delta = %g is less than zero.\n", + pParam->BSIM3delta)); + Fatal_Flag = 1; + } + + if (pParam->BSIM3vsattemp <= 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Vsat at current temperature = %g is not positive.\n", pParam->BSIM3vsattemp)); + Fatal_Flag = 1; + } /* Check Rout parameters */ - if (pParam->BSIM3pclm <= 0.0) - { fprintf(fplog, "Fatal: Pclm = %g is not positive.\n", pParam->BSIM3pclm); - printf("Fatal: Pclm = %g is not positive.\n", pParam->BSIM3pclm); - Fatal_Flag = 1; - } - - if (pParam->BSIM3drout < 0.0) - { fprintf(fplog, "Fatal: Drout = %g is negative.\n", pParam->BSIM3drout); - printf("Fatal: Drout = %g is negative.\n", pParam->BSIM3drout); - Fatal_Flag = 1; - } - - if (pParam->BSIM3pscbe2 <= 0.0) - { fprintf(fplog, "Warning: Pscbe2 = %g is not positive.\n", - pParam->BSIM3pscbe2); - printf("Warning: Pscbe2 = %g is not positive.\n", pParam->BSIM3pscbe2); - } - - /* ACM model */ - if (model->BSIM3acmMod == 0) { - if (model->BSIM3unitLengthSidewallJctCap > 0.0 || - model->BSIM3unitLengthGateSidewallJctCap > 0.0) - { - if (here->BSIM3drainPerimeter < pParam->BSIM3weff) - { fprintf(fplog, "Warning: Pd = %g is less than W.\n", - here->BSIM3drainPerimeter); - printf("Warning: Pd = %g is less than W.\n", - here->BSIM3drainPerimeter); - } - if (here->BSIM3sourcePerimeter < pParam->BSIM3weff) - { fprintf(fplog, "Warning: Ps = %g is less than W.\n", - here->BSIM3sourcePerimeter); - printf("Warning: Ps = %g is less than W.\n", - here->BSIM3sourcePerimeter); - } + if (pParam->BSIM3pclm <= 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Pclm = %g is not positive.\n", pParam->BSIM3pclm)); + Fatal_Flag = 1; + } + + if (pParam->BSIM3drout < 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Drout = %g is negative.\n", pParam->BSIM3drout)); + Fatal_Flag = 1; + } + + if (pParam->BSIM3pscbe2 <= 0.0) + { wl_append_word(&wl, &wl, tprintf("Warning: Pscbe2 = %g is not positive.\n", + pParam->BSIM3pscbe2)); + } + + /* ACM model */ + if (model->BSIM3acmMod == 0) { + if (model->BSIM3unitLengthSidewallJctCap > 0.0 || + model->BSIM3unitLengthGateSidewallJctCap > 0.0) + { + if (here->BSIM3drainPerimeter < pParam->BSIM3weff) + { wl_append_word(&wl, &wl, tprintf("Warning: Pd = %g is less than W.\n", + here->BSIM3drainPerimeter)); + } + if (here->BSIM3sourcePerimeter < pParam->BSIM3weff) + { wl_append_word(&wl, &wl, tprintf("Warning: Ps = %g is less than W.\n", + here->BSIM3sourcePerimeter)); } } + } - if ((model->BSIM3calcacm > 0) && (model->BSIM3acmMod != 12)) - { fprintf(fplog, "Warning: CALCACM = %d is wrong. Set back to 0.\n", - model->BSIM3calcacm); - printf("Warning: CALCACM = %d is wrong. Set back to 0.\n", model->BSIM3calcacm); - model->BSIM3calcacm = 0; - } + if ((model->BSIM3calcacm > 0) && (model->BSIM3acmMod != 12)) + { wl_append_word(&wl, &wl, tprintf("Warning: CALCACM = %d is wrong. Set back to 0.\n", + model->BSIM3calcacm)); + model->BSIM3calcacm = 0; + } - if (pParam->BSIM3noff < 0.1) - { fprintf(fplog, "Warning: Noff = %g is too small.\n", - pParam->BSIM3noff); - printf("Warning: Noff = %g is too small.\n", pParam->BSIM3noff); - } - if (pParam->BSIM3noff > 4.0) - { fprintf(fplog, "Warning: Noff = %g is too large.\n", - pParam->BSIM3noff); - printf("Warning: Noff = %g is too large.\n", pParam->BSIM3noff); - } + if (pParam->BSIM3noff < 0.1) + { wl_append_word(&wl, &wl, tprintf("Warning: Noff = %g is too small.\n", + pParam->BSIM3noff)); + } + if (pParam->BSIM3noff > 4.0) + { wl_append_word(&wl, &wl, tprintf("Warning: Noff = %g is too large.\n", + pParam->BSIM3noff)); + } + + if (pParam->BSIM3voffcv < -0.5) + { wl_append_word(&wl, &wl, tprintf("Warning: Voffcv = %g is too small.\n", + pParam->BSIM3voffcv)); + } + if (pParam->BSIM3voffcv > 0.5) + { wl_append_word(&wl, &wl, tprintf("Warning: Voffcv = %g is too large.\n", + pParam->BSIM3voffcv)); + } + + if (model->BSIM3ijth < 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Ijth = %g cannot be negative.\n", + model->BSIM3ijth)); + Fatal_Flag = 1; + } - if (pParam->BSIM3voffcv < -0.5) - { fprintf(fplog, "Warning: Voffcv = %g is too small.\n", - pParam->BSIM3voffcv); - printf("Warning: Voffcv = %g is too small.\n", pParam->BSIM3voffcv); +/* Check capacitance parameters */ + if (pParam->BSIM3clc < 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Clc = %g is negative.\n", pParam->BSIM3clc)); + Fatal_Flag = 1; + } + + if (pParam->BSIM3moin < 5.0) + { wl_append_word(&wl, &wl, tprintf("Warning: Moin = %g is too small.\n", + pParam->BSIM3moin)); + } + if (pParam->BSIM3moin > 25.0) + { wl_append_word(&wl, &wl, tprintf("Warning: Moin = %g is too large.\n", + pParam->BSIM3moin)); + } + + if(model->BSIM3capMod ==3) { + if (pParam->BSIM3acde < 0.4) + { wl_append_word(&wl, &wl, tprintf("Warning: Acde = %g is too small.\n", + pParam->BSIM3acde)); } - if (pParam->BSIM3voffcv > 0.5) - { fprintf(fplog, "Warning: Voffcv = %g is too large.\n", - pParam->BSIM3voffcv); - printf("Warning: Voffcv = %g is too large.\n", pParam->BSIM3voffcv); + if (pParam->BSIM3acde > 1.6) + { wl_append_word(&wl, &wl, tprintf("Warning: Acde = %g is too large.\n", + pParam->BSIM3acde)); } + } - if (model->BSIM3ijth < 0.0) - { fprintf(fplog, "Fatal: Ijth = %g cannot be negative.\n", - model->BSIM3ijth); - printf("Fatal: Ijth = %g cannot be negative.\n", model->BSIM3ijth); - Fatal_Flag = 1; + if (model->BSIM3paramChk ==1) + { + /* Check L and W parameters */ + if (pParam->BSIM3leff <= 5.0e-8) + { wl_append_word(&wl, &wl, tprintf("Warning: Leff = %g may be too small.\n", + pParam->BSIM3leff)); } -/* Check capacitance parameters */ - if (pParam->BSIM3clc < 0.0) - { fprintf(fplog, "Fatal: Clc = %g is negative.\n", pParam->BSIM3clc); - printf("Fatal: Clc = %g is negative.\n", pParam->BSIM3clc); - Fatal_Flag = 1; - } - - if (pParam->BSIM3moin < 5.0) - { fprintf(fplog, "Warning: Moin = %g is too small.\n", - pParam->BSIM3moin); - printf("Warning: Moin = %g is too small.\n", pParam->BSIM3moin); - } - if (pParam->BSIM3moin > 25.0) - { fprintf(fplog, "Warning: Moin = %g is too large.\n", - pParam->BSIM3moin); - printf("Warning: Moin = %g is too large.\n", pParam->BSIM3moin); - } - - if(model->BSIM3capMod ==3) { - if (pParam->BSIM3acde < 0.4) - { fprintf(fplog, "Warning: Acde = %g is too small.\n", - pParam->BSIM3acde); - printf("Warning: Acde = %g is too small.\n", pParam->BSIM3acde); - } - if (pParam->BSIM3acde > 1.6) - { fprintf(fplog, "Warning: Acde = %g is too large.\n", - pParam->BSIM3acde); - printf("Warning: Acde = %g is too large.\n", pParam->BSIM3acde); - } - } - - if (model->BSIM3paramChk ==1) - { -/* Check L and W parameters */ - if (pParam->BSIM3leff <= 5.0e-8) - { fprintf(fplog, "Warning: Leff = %g may be too small.\n", - pParam->BSIM3leff); - printf("Warning: Leff = %g may be too small.\n", - pParam->BSIM3leff); - } - if (pParam->BSIM3leffCV <= 5.0e-8) - { fprintf(fplog, "Warning: Leff for CV = %g may be too small.\n", - pParam->BSIM3leffCV); - printf("Warning: Leff for CV = %g may be too small.\n", - pParam->BSIM3leffCV); - } - + { wl_append_word(&wl, &wl, tprintf("Warning: Leff for CV = %g may be too small.\n", + pParam->BSIM3leffCV)); + } + if (pParam->BSIM3weff <= 1.0e-7) - { fprintf(fplog, "Warning: Weff = %g may be too small.\n", - pParam->BSIM3weff); - printf("Warning: Weff = %g may be too small.\n", - pParam->BSIM3weff); - } - + { wl_append_word(&wl, &wl, tprintf("Warning: Weff = %g may be too small.\n", + pParam->BSIM3weff)); + } + if (pParam->BSIM3weffCV <= 1.0e-7) - { fprintf(fplog, "Warning: Weff for CV = %g may be too small.\n", - pParam->BSIM3weffCV); - printf("Warning: Weff for CV = %g may be too small.\n", - pParam->BSIM3weffCV); - } - -/* Check threshold voltage parameters */ + { wl_append_word(&wl, &wl, tprintf("Warning: Weff for CV = %g may be too small.\n", + pParam->BSIM3weffCV)); + } + + /* Check threshold voltage parameters */ if (pParam->BSIM3nlx < 0.0) - { fprintf(fplog, "Warning: Nlx = %g is negative.\n", pParam->BSIM3nlx); - printf("Warning: Nlx = %g is negative.\n", pParam->BSIM3nlx); + { wl_append_word(&wl, &wl, tprintf("Warning: Nlx = %g is negative.\n", pParam->BSIM3nlx)); } if (model->BSIM3tox < 1.0e-9) - { fprintf(fplog, "Warning: Tox = %g is less than 10A.\n", - model->BSIM3tox); - printf("Warning: Tox = %g is less than 10A.\n", model->BSIM3tox); + { wl_append_word(&wl, &wl, tprintf("Warning: Tox = %g is less than 10A.\n", + model->BSIM3tox)); } if (pParam->BSIM3npeak <= 1.0e15) - { fprintf(fplog, "Warning: Nch = %g may be too small.\n", - pParam->BSIM3npeak); - printf("Warning: Nch = %g may be too small.\n", - pParam->BSIM3npeak); + { wl_append_word(&wl, &wl, tprintf("Warning: Nch = %g may be too small.\n", + pParam->BSIM3npeak)); } else if (pParam->BSIM3npeak >= 1.0e21) - { fprintf(fplog, "Warning: Nch = %g may be too large.\n", - pParam->BSIM3npeak); - printf("Warning: Nch = %g may be too large.\n", - pParam->BSIM3npeak); + { wl_append_word(&wl, &wl, tprintf("Warning: Nch = %g may be too large.\n", + pParam->BSIM3npeak)); } - if (pParam->BSIM3nsub <= 1.0e14) - { fprintf(fplog, "Warning: Nsub = %g may be too small.\n", - pParam->BSIM3nsub); - printf("Warning: Nsub = %g may be too small.\n", - pParam->BSIM3nsub); + if (pParam->BSIM3nsub <= 1.0e14) + { wl_append_word(&wl, &wl, tprintf("Warning: Nsub = %g may be too small.\n", + pParam->BSIM3nsub)); } else if (pParam->BSIM3nsub >= 1.0e21) - { fprintf(fplog, "Warning: Nsub = %g may be too large.\n", - pParam->BSIM3nsub); - printf("Warning: Nsub = %g may be too large.\n", - pParam->BSIM3nsub); + { wl_append_word(&wl, &wl, tprintf("Warning: Nsub = %g may be too large.\n", + pParam->BSIM3nsub)); } if ((pParam->BSIM3ngate > 0.0) && (pParam->BSIM3ngate <= 1.e18)) - { fprintf(fplog, "Warning: Ngate = %g is less than 1.E18cm^-3.\n", - pParam->BSIM3ngate); - printf("Warning: Ngate = %g is less than 1.E18cm^-3.\n", - pParam->BSIM3ngate); + { wl_append_word(&wl, &wl, tprintf("Warning: Ngate = %g is less than 1.E18cm^-3.\n", + pParam->BSIM3ngate)); } - + if (pParam->BSIM3dvt0 < 0.0) - { fprintf(fplog, "Warning: Dvt0 = %g is negative.\n", - pParam->BSIM3dvt0); - printf("Warning: Dvt0 = %g is negative.\n", pParam->BSIM3dvt0); + { wl_append_word(&wl, &wl, tprintf("Warning: Dvt0 = %g is negative.\n", + pParam->BSIM3dvt0)); } - + if (fabs(1.0e-6 / (pParam->BSIM3w0 + pParam->BSIM3weff)) > 10.0) - { fprintf(fplog, "Warning: (W0 + Weff) may be too small.\n"); - printf("Warning: (W0 + Weff) may be too small.\n"); + { wl_append_word(&wl, &wl, tprintf("Warning: (W0 + Weff) may be too small.\n")); } -/* Check subthreshold parameters */ + /* Check subthreshold parameters */ if (pParam->BSIM3nfactor < 0.0) - { fprintf(fplog, "Warning: Nfactor = %g is negative.\n", - pParam->BSIM3nfactor); - printf("Warning: Nfactor = %g is negative.\n", pParam->BSIM3nfactor); + { wl_append_word(&wl, &wl, tprintf("Warning: Nfactor = %g is negative.\n", + pParam->BSIM3nfactor)); } if (pParam->BSIM3cdsc < 0.0) - { fprintf(fplog, "Warning: Cdsc = %g is negative.\n", - pParam->BSIM3cdsc); - printf("Warning: Cdsc = %g is negative.\n", pParam->BSIM3cdsc); + { wl_append_word(&wl, &wl, tprintf("Warning: Cdsc = %g is negative.\n", + pParam->BSIM3cdsc)); } if (pParam->BSIM3cdscd < 0.0) - { fprintf(fplog, "Warning: Cdscd = %g is negative.\n", - pParam->BSIM3cdscd); - printf("Warning: Cdscd = %g is negative.\n", pParam->BSIM3cdscd); + { wl_append_word(&wl, &wl, tprintf("Warning: Cdscd = %g is negative.\n", + pParam->BSIM3cdscd)); } -/* Check DIBL parameters */ + /* Check DIBL parameters */ if (pParam->BSIM3eta0 < 0.0) - { fprintf(fplog, "Warning: Eta0 = %g is negative.\n", - pParam->BSIM3eta0); - printf("Warning: Eta0 = %g is negative.\n", pParam->BSIM3eta0); - } - -/* Check Abulk parameters */ - if (fabs(1.0e-6 / (pParam->BSIM3b1 + pParam->BSIM3weff)) > 10.0) - { fprintf(fplog, "Warning: (B1 + Weff) may be too small.\n"); - printf("Warning: (B1 + Weff) may be too small.\n"); - } - - -/* Check Saturation parameters */ - if (pParam->BSIM3a2 < 0.01) - { fprintf(fplog, "Warning: A2 = %g is too small. Set to 0.01.\n", pParam->BSIM3a2); - printf("Warning: A2 = %g is too small. Set to 0.01.\n", - pParam->BSIM3a2); + { wl_append_word(&wl, &wl, tprintf("Warning: Eta0 = %g is negative.\n", + pParam->BSIM3eta0)); + } + + /* Check Abulk parameters */ + if (fabs(1.0e-6 / (pParam->BSIM3b1 + pParam->BSIM3weff)) > 10.0) + { wl_append_word(&wl, &wl, tprintf("Warning: (B1 + Weff) may be too small.\n")); + } + + /* Check Saturation parameters */ + if (pParam->BSIM3a2 < 0.01) + { wl_append_word(&wl, &wl, tprintf("Warning: A2 = %g is too small. Set to 0.01.\n", pParam->BSIM3a2)); pParam->BSIM3a2 = 0.01; } else if (pParam->BSIM3a2 > 1.0) - { fprintf(fplog, "Warning: A2 = %g is larger than 1. A2 is set to 1 and A1 is set to 0.\n", - pParam->BSIM3a2); - printf("Warning: A2 = %g is larger than 1. A2 is set to 1 and A1 is set to 0.\n", - pParam->BSIM3a2); + { wl_append_word(&wl, &wl, tprintf("Warning: A2 = %g is larger than 1. A2 is set to 1 and A1 is set to 0.\n", + pParam->BSIM3a2)); pParam->BSIM3a2 = 1.0; pParam->BSIM3a1 = 0.0; + } if (pParam->BSIM3rdsw < 0.0) - { fprintf(fplog, "Warning: Rdsw = %g is negative. Set to zero.\n", - pParam->BSIM3rdsw); - printf("Warning: Rdsw = %g is negative. Set to zero.\n", - pParam->BSIM3rdsw); + { wl_append_word(&wl, &wl, tprintf("Warning: Rdsw = %g is negative. Set to zero.\n", + pParam->BSIM3rdsw)); pParam->BSIM3rdsw = 0.0; pParam->BSIM3rds0 = 0.0; } - if (pParam->BSIM3rds0 < 0.0) - { fprintf(fplog, "Warning: Rds at current temperature = %g is negative. Set to zero.\n", - pParam->BSIM3rds0); - printf("Warning: Rds at current temperature = %g is negative. Set to zero.\n", - pParam->BSIM3rds0); + else if ((pParam->BSIM3rds0 > 0.0) && (pParam->BSIM3rds0 < 0.001)) + { wl_append_word(&wl, &wl, tprintf("Warning: Rds at current temperature = %g is less than 0.001 ohm. Set to zero.\n", + pParam->BSIM3rds0)); pParam->BSIM3rds0 = 0.0; } - - if (pParam->BSIM3vsattemp < 1.0e3) - { fprintf(fplog, "Warning: Vsat at current temperature = %g may be too small.\n", pParam->BSIM3vsattemp); - printf("Warning: Vsat at current temperature = %g may be too small.\n", pParam->BSIM3vsattemp); + if (pParam->BSIM3vsattemp < 1.0e3) + { wl_append_word(&wl, &wl, tprintf("Warning: Vsat at current temperature = %g may be too small.\n", pParam->BSIM3vsattemp)); } if (pParam->BSIM3pdibl1 < 0.0) - { fprintf(fplog, "Warning: Pdibl1 = %g is negative.\n", - pParam->BSIM3pdibl1); - printf("Warning: Pdibl1 = %g is negative.\n", pParam->BSIM3pdibl1); + { wl_append_word(&wl, &wl, tprintf("Warning: Pdibl1 = %g is negative.\n", + pParam->BSIM3pdibl1)); } if (pParam->BSIM3pdibl2 < 0.0) - { fprintf(fplog, "Warning: Pdibl2 = %g is negative.\n", - pParam->BSIM3pdibl2); - printf("Warning: Pdibl2 = %g is negative.\n", pParam->BSIM3pdibl2); + { wl_append_word(&wl, &wl, tprintf("Warning: Pdibl2 = %g is negative.\n", + pParam->BSIM3pdibl2)); } -/* Check overlap capacitance parameters */ + /* Check overlap capacitance parameters */ if (model->BSIM3cgdo < 0.0) - { fprintf(fplog, "Warning: cgdo = %g is negative. Set to zero.\n", model->BSIM3cgdo); - printf("Warning: cgdo = %g is negative. Set to zero.\n", model->BSIM3cgdo); + { wl_append_word(&wl, &wl, tprintf("Warning: cgdo = %g is negative. Set to zero.\n", model->BSIM3cgdo)); model->BSIM3cgdo = 0.0; - } + } if (model->BSIM3cgso < 0.0) - { fprintf(fplog, "Warning: cgso = %g is negative. Set to zero.\n", model->BSIM3cgso); - printf("Warning: cgso = %g is negative. Set to zero.\n", model->BSIM3cgso); + { wl_append_word(&wl, &wl, tprintf("Warning: cgso = %g is negative. Set to zero.\n", model->BSIM3cgso)); model->BSIM3cgso = 0.0; - } + } if (model->BSIM3cgbo < 0.0) - { fprintf(fplog, "Warning: cgbo = %g is negative. Set to zero.\n", model->BSIM3cgbo); - printf("Warning: cgbo = %g is negative. Set to zero.\n", model->BSIM3cgbo); + { wl_append_word(&wl, &wl, tprintf("Warning: cgbo = %g is negative. Set to zero.\n", model->BSIM3cgbo)); model->BSIM3cgbo = 0.0; } - }/* loop for the parameter check for warning messages */ - fclose(fplog); } - else - { fprintf(stderr, "Warning: Can't open log file. Parameter checking skipped.\n"); + + if (wlstart->wl_next) { + if ((fplog = fopen("b3v33check.log", "w")) != NULL) { + while (wlstart) { + fprintf(fplog, "%s", wlstart->wl_word); + fprintf(stderr, "%s", wlstart->wl_word); + wlstart = wlstart->wl_next; + } + fclose(fplog); + } + else { + while (wlstart) { + fprintf(stderr, "%s", wlstart->wl_word); + wlstart = wlstart->wl_next; + } + } } + wl_free(wlstart); + return(Fatal_Flag); } diff --git a/src/spicelib/devices/bsim3v32/b3v32check.c b/src/spicelib/devices/bsim3v32/b3v32check.c index 18eca0401..b18ddb509 100644 --- a/src/spicelib/devices/bsim3v32/b3v32check.c +++ b/src/spicelib/devices/bsim3v32/b3v32check.c @@ -8,6 +8,7 @@ * Author: 2001 Xuemei Xi * Modified by Xuemei Xi, 10/05, 12/14, 2001. * Modified by Paolo Nenzi 2002 and Dietmar Warning 2003 + * Modified by Dietmar Warning, 12/21/2020. **********/ #include "ngspice/ngspice.h" @@ -18,441 +19,377 @@ #include "ngspice/sperror.h" #include "ngspice/devdefs.h" #include "ngspice/suffix.h" +#include "ngspice/wordlist.h" +#include "ngspice/cpextern.h" + +/* Check for correctness of the BSIM3.2 parameters: + If parameter excursions are found, put the warning or error message into a wordlist. + Only then open a file b3v32check.log and print the data into the file. */ int BSIM3v32checkModel (BSIM3v32model *model, BSIM3v32instance *here, CKTcircuit *ckt) { -struct bsim3v32SizeDependParam *pParam; -int Fatal_Flag = 0; -FILE *fplog; - - NG_IGNORE(ckt); - - if ((fplog = fopen("b3v32check.log", "w")) != NULL) - { pParam = here->pParam; - - fprintf (fplog, - "BSIM3 Model (Supports: v3.2, v3.2.2, v3.2.3, v3.2.4)\n"); - fprintf (fplog, "Parameter Checking.\n"); - fprintf (fplog, "Model = %s\n", model->BSIM3v32modName); - fprintf (fplog, "W = %g, L = %g, M = %g\n", here->BSIM3v32w, - here->BSIM3v32l, here->BSIM3v32m); - - if ((strcmp(model->BSIM3v32version, "3.2.4")) && (strncmp(model->BSIM3v32version, "3.24", 4)) - && (strcmp(model->BSIM3v32version, "3.2.3")) && (strncmp(model->BSIM3v32version, "3.23", 4)) - && (strcmp(model->BSIM3v32version, "3.2.2")) && (strncmp(model->BSIM3v32version, "3.22", 4)) - && (strncmp(model->BSIM3v32version, "3.2", 3)) && (strncmp(model->BSIM3v32version, "3.20", 4))) + struct bsim3v32SizeDependParam *pParam; + int Fatal_Flag = 0; + FILE *fplog; + wordlist* wl, *wlstart; + + NG_IGNORE(ckt); + + pParam = here->pParam; + + if (cp_getvar("ng_nomodcheck", CP_BOOL, NULL, 0)) + return(0); + + wl = wlstart = TMALLOC(wordlist, 1); + wl->wl_prev = NULL; + wl->wl_next = NULL; + wl->wl_word = tprintf("\nChecking parameters for BSIM 3.2 model %s\n", model->BSIM3v32modName); + + if ((strcmp(model->BSIM3v32version, "3.2.4")) && (strncmp(model->BSIM3v32version, "3.24", 4)) + && (strcmp(model->BSIM3v32version, "3.2.3")) && (strncmp(model->BSIM3v32version, "3.23", 4)) + && (strcmp(model->BSIM3v32version, "3.2.2")) && (strncmp(model->BSIM3v32version, "3.22", 4)) + && (strncmp(model->BSIM3v32version, "3.2", 3)) && (strncmp(model->BSIM3v32version, "3.20", 4))) + { + printf("Warning: This model supports BSIM3v3.2, BSIM3v3.2.2, BSIM3v3.2.3, BSIM3v3.2.4\n"); + printf("You specified a wrong version number. Working now with BSIM3v3.2.4.\n"); + wl_append_word(&wl, &wl, tprintf("Warning: This model supports BSIM3v3.2, BSIM3v3.2.2, BSIM3v3.2.3, BSIM3v3.2.4\n")); + wl_append_word(&wl, &wl, tprintf("You specified a wrong version number. Working now with BSIM3v3.2.4.\n")); + } + + if (pParam->BSIM3v32nlx < -pParam->BSIM3v32leff) + { wl_append_word(&wl, &wl, tprintf("Fatal: Nlx = %g is less than -Leff.\n", + pParam->BSIM3v32nlx)); + Fatal_Flag = 1; + } + + if (model->BSIM3v32tox <= 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Tox = %g is not positive.\n", + model->BSIM3v32tox)); + Fatal_Flag = 1; + } + + if (model->BSIM3v32toxm <= 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Toxm = %g is not positive.\n", + model->BSIM3v32toxm)); + Fatal_Flag = 1; + } + + if (pParam->BSIM3v32npeak <= 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Nch = %g is not positive.\n", + pParam->BSIM3v32npeak)); + Fatal_Flag = 1; + } + if (pParam->BSIM3v32nsub <= 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Nsub = %g is not positive.\n", + pParam->BSIM3v32nsub)); + Fatal_Flag = 1; + } + if (pParam->BSIM3v32ngate < 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Ngate = %g is not positive.\n", + pParam->BSIM3v32ngate)); + Fatal_Flag = 1; + } + if (pParam->BSIM3v32ngate > 1.e25) + { wl_append_word(&wl, &wl, tprintf("Fatal: Ngate = %g is too high.\n", + pParam->BSIM3v32ngate)); + Fatal_Flag = 1; + } + if (pParam->BSIM3v32xj <= 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Xj = %g is not positive.\n", + pParam->BSIM3v32xj)); + Fatal_Flag = 1; + } + + if (pParam->BSIM3v32dvt1 < 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Dvt1 = %g is negative.\n", + pParam->BSIM3v32dvt1)); + Fatal_Flag = 1; + } + + if (pParam->BSIM3v32dvt1w < 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Dvt1w = %g is negative.\n", + pParam->BSIM3v32dvt1w)); + Fatal_Flag = 1; + } + + if (pParam->BSIM3v32w0 == -pParam->BSIM3v32weff) + { wl_append_word(&wl, &wl, tprintf("Fatal: (W0 + Weff) = 0 causing divided-by-zero.\n")); + Fatal_Flag = 1; + } + + if (pParam->BSIM3v32dsub < 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Dsub = %g is negative.\n", pParam->BSIM3v32dsub)); + Fatal_Flag = 1; + } + if (pParam->BSIM3v32b1 == -pParam->BSIM3v32weff) + { wl_append_word(&wl, &wl, tprintf("Fatal: (B1 + Weff) = 0 causing divided-by-zero.\n")); + Fatal_Flag = 1; + } + if (pParam->BSIM3v32u0temp <= 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: u0 at current temperature = %g is not positive.\n", pParam->BSIM3v32u0temp)); + Fatal_Flag = 1; + } + +/* Check delta parameter */ + if (pParam->BSIM3v32delta < 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Delta = %g is less than zero.\n", + pParam->BSIM3v32delta)); + Fatal_Flag = 1; + } + + if (pParam->BSIM3v32vsattemp <= 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Vsat at current temperature = %g is not positive.\n", pParam->BSIM3v32vsattemp)); + Fatal_Flag = 1; + } +/* Check Rout parameters */ + if (pParam->BSIM3v32pclm <= 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Pclm = %g is not positive.\n", pParam->BSIM3v32pclm)); + Fatal_Flag = 1; + } + + if (pParam->BSIM3v32drout < 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Drout = %g is negative.\n", pParam->BSIM3v32drout)); + Fatal_Flag = 1; + } + + if (pParam->BSIM3v32pscbe2 <= 0.0) + { wl_append_word(&wl, &wl, tprintf("Warning: Pscbe2 = %g is not positive.\n", + pParam->BSIM3v32pscbe2)); + } + + /* ACM model */ + if (model->BSIM3v32acmMod == 0) { + if (model->BSIM3v32unitLengthSidewallJctCap > 0.0 || + model->BSIM3v32unitLengthGateSidewallJctCap > 0.0) { - fprintf (fplog, - "Warning: This model supports BSIM3v3.2, BSIM3v3.2.2, BSIM3v3.2.3, BSIM3v3.2.4\n"); - fprintf (fplog, - "You specified a wrong version number. Working now with BSIM3v3.2.4.\n"); - printf ("Warning: This model supports BSIM3v3.2, BSIM3v3.2.2, BSIM3v3.2.3, BSIM3v3.2.4\n"); - printf ("You specified a wrong version number. Working now with BSIM3v3.2.4.\n"); - } - - if (pParam->BSIM3v32nlx < -pParam->BSIM3v32leff) - { fprintf(fplog, "Fatal: Nlx = %g is less than -Leff.\n", - pParam->BSIM3v32nlx); - printf("Fatal: Nlx = %g is less than -Leff.\n", - pParam->BSIM3v32nlx); - Fatal_Flag = 1; - } - - if (model->BSIM3v32tox <= 0.0) - { fprintf(fplog, "Fatal: Tox = %g is not positive.\n", - model->BSIM3v32tox); - printf("Fatal: Tox = %g is not positive.\n", model->BSIM3v32tox); - Fatal_Flag = 1; - } - - if (model->BSIM3v32toxm <= 0.0) - { fprintf(fplog, "Fatal: Toxm = %g is not positive.\n", - model->BSIM3v32toxm); - printf("Fatal: Toxm = %g is not positive.\n", model->BSIM3v32toxm); - Fatal_Flag = 1; - } - - if (pParam->BSIM3v32npeak <= 0.0) - { fprintf(fplog, "Fatal: Nch = %g is not positive.\n", - pParam->BSIM3v32npeak); - printf("Fatal: Nch = %g is not positive.\n", - pParam->BSIM3v32npeak); - Fatal_Flag = 1; - } - if (pParam->BSIM3v32nsub <= 0.0) - { fprintf(fplog, "Fatal: Nsub = %g is not positive.\n", - pParam->BSIM3v32nsub); - printf("Fatal: Nsub = %g is not positive.\n", - pParam->BSIM3v32nsub); - Fatal_Flag = 1; - } - if (pParam->BSIM3v32ngate < 0.0) - { fprintf(fplog, "Fatal: Ngate = %g is not positive.\n", - pParam->BSIM3v32ngate); - printf("Fatal: Ngate = %g Ngate is not positive.\n", - pParam->BSIM3v32ngate); - Fatal_Flag = 1; - } - if (pParam->BSIM3v32ngate > 1.e25) - { fprintf(fplog, "Fatal: Ngate = %g is too high.\n", - pParam->BSIM3v32ngate); - printf("Fatal: Ngate = %g Ngate is too high\n", - pParam->BSIM3v32ngate); - Fatal_Flag = 1; - } - if (pParam->BSIM3v32xj <= 0.0) - { fprintf(fplog, "Fatal: Xj = %g is not positive.\n", - pParam->BSIM3v32xj); - printf("Fatal: Xj = %g is not positive.\n", pParam->BSIM3v32xj); - Fatal_Flag = 1; - } - - if (pParam->BSIM3v32dvt1 < 0.0) - { fprintf(fplog, "Fatal: Dvt1 = %g is negative.\n", - pParam->BSIM3v32dvt1); - printf("Fatal: Dvt1 = %g is negative.\n", pParam->BSIM3v32dvt1); - Fatal_Flag = 1; - } - - if (pParam->BSIM3v32dvt1w < 0.0) - { fprintf(fplog, "Fatal: Dvt1w = %g is negative.\n", - pParam->BSIM3v32dvt1w); - printf("Fatal: Dvt1w = %g is negative.\n", pParam->BSIM3v32dvt1w); - Fatal_Flag = 1; - } - - if (pParam->BSIM3v32w0 == -pParam->BSIM3v32weff) - { fprintf(fplog, "Fatal: (W0 + Weff) = 0 causing divided-by-zero.\n"); - printf("Fatal: (W0 + Weff) = 0 causing divided-by-zero.\n"); - Fatal_Flag = 1; - } - - if (pParam->BSIM3v32dsub < 0.0) - { fprintf(fplog, "Fatal: Dsub = %g is negative.\n", pParam->BSIM3v32dsub); - printf("Fatal: Dsub = %g is negative.\n", pParam->BSIM3v32dsub); - Fatal_Flag = 1; - } - if (pParam->BSIM3v32b1 == -pParam->BSIM3v32weff) - { fprintf(fplog, "Fatal: (B1 + Weff) = 0 causing divided-by-zero.\n"); - printf("Fatal: (B1 + Weff) = 0 causing divided-by-zero.\n"); - Fatal_Flag = 1; - } - if (pParam->BSIM3v32u0temp <= 0.0) - { fprintf(fplog, "Fatal: u0 at current temperature = %g is not positive.\n", pParam->BSIM3v32u0temp); - printf("Fatal: u0 at current temperature = %g is not positive.\n", - pParam->BSIM3v32u0temp); - Fatal_Flag = 1; + if (here->BSIM3v32drainPerimeter < pParam->BSIM3v32weff) + { wl_append_word(&wl, &wl, tprintf("Warning: Pd = %g is less than W.\n", + here->BSIM3v32drainPerimeter)); + } + if (here->BSIM3v32sourcePerimeter < pParam->BSIM3v32weff) + { wl_append_word(&wl, &wl, tprintf("Warning: Ps = %g is less than W.\n", + here->BSIM3v32sourcePerimeter)); + } } + } + + if ((model->BSIM3v32calcacm > 0) && (model->BSIM3v32acmMod != 12)) + { wl_append_word(&wl, &wl, tprintf("Warning: CALCACM = %d is wrong. Set back to 0.\n", + model->BSIM3v32calcacm)); + model->BSIM3v32calcacm = 0; + } + + if (pParam->BSIM3v32noff < 0.1) + { wl_append_word(&wl, &wl, tprintf("Warning: Noff = %g is too small.\n", + pParam->BSIM3v32noff)); + } + if (pParam->BSIM3v32noff > 4.0) + { wl_append_word(&wl, &wl, tprintf("Warning: Noff = %g is too large.\n", + pParam->BSIM3v32noff)); + } + + if (pParam->BSIM3v32voffcv < -0.5) + { wl_append_word(&wl, &wl, tprintf("Warning: Voffcv = %g is too small.\n", + pParam->BSIM3v32voffcv)); + } + if (pParam->BSIM3v32voffcv > 0.5) + { wl_append_word(&wl, &wl, tprintf("Warning: Voffcv = %g is too large.\n", + pParam->BSIM3v32voffcv)); + } + + if (model->BSIM3v32ijth < 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Ijth = %g cannot be negative.\n", + model->BSIM3v32ijth)); + Fatal_Flag = 1; + } -/* Check delta parameter */ - if (pParam->BSIM3v32delta < 0.0) - { fprintf(fplog, "Fatal: Delta = %g is less than zero.\n", - pParam->BSIM3v32delta); - printf("Fatal: Delta = %g is less than zero.\n", pParam->BSIM3v32delta); - Fatal_Flag = 1; +/* Check capacitance parameters */ + if (pParam->BSIM3v32clc < 0.0) + { wl_append_word(&wl, &wl, tprintf("Fatal: Clc = %g is negative.\n", pParam->BSIM3v32clc)); + Fatal_Flag = 1; + } + + if (pParam->BSIM3v32moin < 5.0) + { wl_append_word(&wl, &wl, tprintf("Warning: Moin = %g is too small.\n", + pParam->BSIM3v32moin)); + } + if (pParam->BSIM3v32moin > 25.0) + { wl_append_word(&wl, &wl, tprintf("Warning: Moin = %g is too large.\n", + pParam->BSIM3v32moin)); + } + + if(model->BSIM3v32capMod ==3) { + if (pParam->BSIM3v32acde < 0.4) + { wl_append_word(&wl, &wl, tprintf("Warning: Acde = %g is too small.\n", + pParam->BSIM3v32acde)); + } + if (pParam->BSIM3v32acde > 1.6) + { wl_append_word(&wl, &wl, tprintf("Warning: Acde = %g is too large.\n", + pParam->BSIM3v32acde)); + } + } + + if (model->BSIM3v32paramChk ==1) + { + /* Check L and W parameters */ + if (pParam->BSIM3v32leff <= 5.0e-8) + { wl_append_word(&wl, &wl, tprintf("Warning: Leff = %g may be too small.\n", + pParam->BSIM3v32leff)); } - if (pParam->BSIM3v32vsattemp <= 0.0) - { fprintf(fplog, "Fatal: Vsat at current temperature = %g is not positive.\n", pParam->BSIM3v32vsattemp); - printf("Fatal: Vsat at current temperature = %g is not positive.\n", - pParam->BSIM3v32vsattemp); - Fatal_Flag = 1; + if (pParam->BSIM3v32leffCV <= 5.0e-8) + { wl_append_word(&wl, &wl, tprintf("Warning: Leff for CV = %g may be too small.\n", + pParam->BSIM3v32leffCV)); } -/* Check Rout parameters */ - if (pParam->BSIM3v32pclm <= 0.0) - { fprintf(fplog, "Fatal: Pclm = %g is not positive.\n", pParam->BSIM3v32pclm); - printf("Fatal: Pclm = %g is not positive.\n", pParam->BSIM3v32pclm); - Fatal_Flag = 1; - } - - if (pParam->BSIM3v32drout < 0.0) - { fprintf(fplog, "Fatal: Drout = %g is negative.\n", pParam->BSIM3v32drout); - printf("Fatal: Drout = %g is negative.\n", pParam->BSIM3v32drout); - Fatal_Flag = 1; - } - - if (pParam->BSIM3v32pscbe2 <= 0.0) - { fprintf(fplog, "Warning: Pscbe2 = %g is not positive.\n", - pParam->BSIM3v32pscbe2); - printf("Warning: Pscbe2 = %g is not positive.\n", pParam->BSIM3v32pscbe2); - } - - /* ACM model */ - if (model->BSIM3v32acmMod == 0) { - if (model->BSIM3v32unitLengthSidewallJctCap > 0.0 || - model->BSIM3v32unitLengthGateSidewallJctCap > 0.0) - { - if (here->BSIM3v32drainPerimeter < pParam->BSIM3v32weff) - { fprintf(fplog, "Warning: Pd = %g is less than W.\n", - here->BSIM3v32drainPerimeter); - printf("Warning: Pd = %g is less than W.\n", - here->BSIM3v32drainPerimeter); - } - if (here->BSIM3v32sourcePerimeter < pParam->BSIM3v32weff) - { fprintf(fplog, "Warning: Ps = %g is less than W.\n", - here->BSIM3v32sourcePerimeter); - printf("Warning: Ps = %g is less than W.\n", - here->BSIM3v32sourcePerimeter); - } - } + + if (pParam->BSIM3v32weff <= 1.0e-7) + { wl_append_word(&wl, &wl, tprintf("Warning: Weff = %g may be too small.\n", + pParam->BSIM3v32weff)); } - if ((model->BSIM3v32calcacm > 0) && (model->BSIM3v32acmMod != 12)) - { fprintf(fplog, "Warning: CALCACM = %d is wrong. Set back to 0.\n", - model->BSIM3v32calcacm); - printf("Warning: CALCACM = %d is wrong. Set back to 0.\n", model->BSIM3v32calcacm); - model->BSIM3v32calcacm = 0; + if (pParam->BSIM3v32weffCV <= 1.0e-7) + { wl_append_word(&wl, &wl, tprintf("Warning: Weff for CV = %g may be too small.\n", + pParam->BSIM3v32weffCV)); } - if (pParam->BSIM3v32noff < 0.1) - { fprintf(fplog, "Warning: Noff = %g is too small.\n", - pParam->BSIM3v32noff); - printf("Warning: Noff = %g is too small.\n", pParam->BSIM3v32noff); + /* Check threshold voltage parameters */ + if (pParam->BSIM3v32nlx < 0.0) + { wl_append_word(&wl, &wl, tprintf("Warning: Nlx = %g is negative.\n", pParam->BSIM3v32nlx)); } - if (pParam->BSIM3v32noff > 4.0) - { fprintf(fplog, "Warning: Noff = %g is too large.\n", - pParam->BSIM3v32noff); - printf("Warning: Noff = %g is too large.\n", pParam->BSIM3v32noff); + if (model->BSIM3v32tox < 1.0e-9) + { wl_append_word(&wl, &wl, tprintf("Warning: Tox = %g is less than 10A.\n", + model->BSIM3v32tox)); } - if (pParam->BSIM3v32voffcv < -0.5) - { fprintf(fplog, "Warning: Voffcv = %g is too small.\n", - pParam->BSIM3v32voffcv); - printf("Warning: Voffcv = %g is too small.\n", pParam->BSIM3v32voffcv); + if (pParam->BSIM3v32npeak <= 1.0e15) + { wl_append_word(&wl, &wl, tprintf("Warning: Nch = %g may be too small.\n", + pParam->BSIM3v32npeak)); } - if (pParam->BSIM3v32voffcv > 0.5) - { fprintf(fplog, "Warning: Voffcv = %g is too large.\n", - pParam->BSIM3v32voffcv); - printf("Warning: Voffcv = %g is too large.\n", pParam->BSIM3v32voffcv); + else if (pParam->BSIM3v32npeak >= 1.0e21) + { wl_append_word(&wl, &wl, tprintf("Warning: Nch = %g may be too large.\n", + pParam->BSIM3v32npeak)); } - if (model->BSIM3v32ijth < 0.0) - { fprintf(fplog, "Fatal: Ijth = %g cannot be negative.\n", - model->BSIM3v32ijth); - printf("Fatal: Ijth = %g cannot be negative.\n", model->BSIM3v32ijth); - Fatal_Flag = 1; + if (pParam->BSIM3v32nsub <= 1.0e14) + { wl_append_word(&wl, &wl, tprintf("Warning: Nsub = %g may be too small.\n", + pParam->BSIM3v32nsub)); + } + else if (pParam->BSIM3v32nsub >= 1.0e21) + { wl_append_word(&wl, &wl, tprintf("Warning: Nsub = %g may be too large.\n", + pParam->BSIM3v32nsub)); } -/* Check capacitance parameters */ - if (pParam->BSIM3v32clc < 0.0) - { fprintf(fplog, "Fatal: Clc = %g is negative.\n", pParam->BSIM3v32clc); - printf("Fatal: Clc = %g is negative.\n", pParam->BSIM3v32clc); - Fatal_Flag = 1; + if ((pParam->BSIM3v32ngate > 0.0) && + (pParam->BSIM3v32ngate <= 1.e18)) + { wl_append_word(&wl, &wl, tprintf("Warning: Ngate = %g is less than 1.E18cm^-3.\n", + pParam->BSIM3v32ngate)); } - if (pParam->BSIM3v32moin < 5.0) - { fprintf(fplog, "Warning: Moin = %g is too small.\n", - pParam->BSIM3v32moin); - printf("Warning: Moin = %g is too small.\n", pParam->BSIM3v32moin); + if (pParam->BSIM3v32dvt0 < 0.0) + { wl_append_word(&wl, &wl, tprintf("Warning: Dvt0 = %g is negative.\n", + pParam->BSIM3v32dvt0)); } - if (pParam->BSIM3v32moin > 25.0) - { fprintf(fplog, "Warning: Moin = %g is too large.\n", - pParam->BSIM3v32moin); - printf("Warning: Moin = %g is too large.\n", pParam->BSIM3v32moin); + + if (fabs(1.0e-6 / (pParam->BSIM3v32w0 + pParam->BSIM3v32weff)) > 10.0) + { wl_append_word(&wl, &wl, tprintf("Warning: (W0 + Weff) may be too small.\n")); } - if(model->BSIM3v32capMod ==3) { - if (pParam->BSIM3v32acde < 0.4) - { fprintf(fplog, "Warning: Acde = %g is too small.\n", - pParam->BSIM3v32acde); - printf("Warning: Acde = %g is too small.\n", pParam->BSIM3v32acde); - } - if (pParam->BSIM3v32acde > 1.6) - { fprintf(fplog, "Warning: Acde = %g is too large.\n", - pParam->BSIM3v32acde); - printf("Warning: Acde = %g is too large.\n", pParam->BSIM3v32acde); - } + /* Check subthreshold parameters */ + if (pParam->BSIM3v32nfactor < 0.0) + { wl_append_word(&wl, &wl, tprintf("Warning: Nfactor = %g is negative.\n", + pParam->BSIM3v32nfactor)); + } + if (pParam->BSIM3v32cdsc < 0.0) + { wl_append_word(&wl, &wl, tprintf("Warning: Cdsc = %g is negative.\n", + pParam->BSIM3v32cdsc)); + } + if (pParam->BSIM3v32cdscd < 0.0) + { wl_append_word(&wl, &wl, tprintf("Warning: Cdscd = %g is negative.\n", + pParam->BSIM3v32cdscd)); + } + /* Check DIBL parameters */ + if (pParam->BSIM3v32eta0 < 0.0) + { wl_append_word(&wl, &wl, tprintf("Warning: Eta0 = %g is negative.\n", + pParam->BSIM3v32eta0)); } - if (model->BSIM3v32paramChk ==1) - { - /* Check L and W parameters */ - if (pParam->BSIM3v32leff <= 5.0e-8) - { fprintf(fplog, "Warning: Leff = %g may be too small.\n", - pParam->BSIM3v32leff); - printf("Warning: Leff = %g may be too small.\n", - pParam->BSIM3v32leff); - } - - if (pParam->BSIM3v32leffCV <= 5.0e-8) - { fprintf(fplog, "Warning: Leff for CV = %g may be too small.\n", - pParam->BSIM3v32leffCV); - printf("Warning: Leff for CV = %g may be too small.\n", - pParam->BSIM3v32leffCV); - } - - if (pParam->BSIM3v32weff <= 1.0e-7) - { fprintf(fplog, "Warning: Weff = %g may be too small.\n", - pParam->BSIM3v32weff); - printf("Warning: Weff = %g may be too small.\n", - pParam->BSIM3v32weff); - } - - if (pParam->BSIM3v32weffCV <= 1.0e-7) - { fprintf(fplog, "Warning: Weff for CV = %g may be too small.\n", - pParam->BSIM3v32weffCV); - printf("Warning: Weff for CV = %g may be too small.\n", - pParam->BSIM3v32weffCV); - } - - /* Check threshold voltage parameters */ - if (pParam->BSIM3v32nlx < 0.0) - { fprintf(fplog, "Warning: Nlx = %g is negative.\n", pParam->BSIM3v32nlx); - printf("Warning: Nlx = %g is negative.\n", pParam->BSIM3v32nlx); - } - if (model->BSIM3v32tox < 1.0e-9) - { fprintf(fplog, "Warning: Tox = %g is less than 10A.\n", - model->BSIM3v32tox); - printf("Warning: Tox = %g is less than 10A.\n", model->BSIM3v32tox); - } - - if (pParam->BSIM3v32npeak <= 1.0e15) - { fprintf(fplog, "Warning: Nch = %g may be too small.\n", - pParam->BSIM3v32npeak); - printf("Warning: Nch = %g may be too small.\n", - pParam->BSIM3v32npeak); - } - else if (pParam->BSIM3v32npeak >= 1.0e21) - { fprintf(fplog, "Warning: Nch = %g may be too large.\n", - pParam->BSIM3v32npeak); - printf("Warning: Nch = %g may be too large.\n", - pParam->BSIM3v32npeak); - } - - if (pParam->BSIM3v32nsub <= 1.0e14) - { fprintf(fplog, "Warning: Nsub = %g may be too small.\n", - pParam->BSIM3v32nsub); - printf("Warning: Nsub = %g may be too small.\n", - pParam->BSIM3v32nsub); - } - else if (pParam->BSIM3v32nsub >= 1.0e21) - { fprintf(fplog, "Warning: Nsub = %g may be too large.\n", - pParam->BSIM3v32nsub); - printf("Warning: Nsub = %g may be too large.\n", - pParam->BSIM3v32nsub); - } - - if ((pParam->BSIM3v32ngate > 0.0) && - (pParam->BSIM3v32ngate <= 1.e18)) - { fprintf(fplog, "Warning: Ngate = %g is less than 1.E18cm^-3.\n", - pParam->BSIM3v32ngate); - printf("Warning: Ngate = %g is less than 1.E18cm^-3.\n", - pParam->BSIM3v32ngate); - } - - if (pParam->BSIM3v32dvt0 < 0.0) - { fprintf(fplog, "Warning: Dvt0 = %g is negative.\n", - pParam->BSIM3v32dvt0); - printf("Warning: Dvt0 = %g is negative.\n", pParam->BSIM3v32dvt0); - } - - if (fabs(1.0e-6 / (pParam->BSIM3v32w0 + pParam->BSIM3v32weff)) > 10.0) - { fprintf(fplog, "Warning: (W0 + Weff) may be too small.\n"); - printf("Warning: (W0 + Weff) may be too small.\n"); - } - - /* Check subthreshold parameters */ - if (pParam->BSIM3v32nfactor < 0.0) - { fprintf(fplog, "Warning: Nfactor = %g is negative.\n", - pParam->BSIM3v32nfactor); - printf("Warning: Nfactor = %g is negative.\n", pParam->BSIM3v32nfactor); - } - if (pParam->BSIM3v32cdsc < 0.0) - { fprintf(fplog, "Warning: Cdsc = %g is negative.\n", - pParam->BSIM3v32cdsc); - printf("Warning: Cdsc = %g is negative.\n", pParam->BSIM3v32cdsc); - } - if (pParam->BSIM3v32cdscd < 0.0) - { fprintf(fplog, "Warning: Cdscd = %g is negative.\n", - pParam->BSIM3v32cdscd); - printf("Warning: Cdscd = %g is negative.\n", pParam->BSIM3v32cdscd); - } - /* Check DIBL parameters */ - if (pParam->BSIM3v32eta0 < 0.0) - { fprintf(fplog, "Warning: Eta0 = %g is negative.\n", - pParam->BSIM3v32eta0); - printf("Warning: Eta0 = %g is negative.\n", pParam->BSIM3v32eta0); - } - - /* Check Abulk parameters */ - if (fabs(1.0e-6 / (pParam->BSIM3v32b1 + pParam->BSIM3v32weff)) > 10.0) - { fprintf(fplog, "Warning: (B1 + Weff) may be too small.\n"); - printf("Warning: (B1 + Weff) may be too small.\n"); - } - - - /* Check Saturation parameters */ - if (pParam->BSIM3v32a2 < 0.01) - { fprintf(fplog, "Warning: A2 = %g is too small. Set to 0.01.\n", pParam->BSIM3v32a2); - printf("Warning: A2 = %g is too small. Set to 0.01.\n", - pParam->BSIM3v32a2); - pParam->BSIM3v32a2 = 0.01; - } - else if (pParam->BSIM3v32a2 > 1.0) - { fprintf(fplog, "Warning: A2 = %g is larger than 1. A2 is set to 1 and A1 is set to 0.\n", - pParam->BSIM3v32a2); - printf("Warning: A2 = %g is larger than 1. A2 is set to 1 and A1 is set to 0.\n", - pParam->BSIM3v32a2); - pParam->BSIM3v32a2 = 1.0; - pParam->BSIM3v32a1 = 0.0; - - } - - if (pParam->BSIM3v32rdsw < 0.0) - { fprintf(fplog, "Warning: Rdsw = %g is negative. Set to zero.\n", - pParam->BSIM3v32rdsw); - printf("Warning: Rdsw = %g is negative. Set to zero.\n", - pParam->BSIM3v32rdsw); - pParam->BSIM3v32rdsw = 0.0; - pParam->BSIM3v32rds0 = 0.0; - } - else if ((pParam->BSIM3v32rds0 > 0.0) && (pParam->BSIM3v32rds0 < 0.001)) - { fprintf(fplog, "Warning: Rds at current temperature = %g is less than 0.001 ohm. Set to zero.\n", - pParam->BSIM3v32rds0); - printf("Warning: Rds at current temperature = %g is less than 0.001 ohm. Set to zero.\n", - pParam->BSIM3v32rds0); - pParam->BSIM3v32rds0 = 0.0; - } - if (pParam->BSIM3v32vsattemp < 1.0e3) - { fprintf(fplog, "Warning: Vsat at current temperature = %g may be too small.\n", pParam->BSIM3v32vsattemp); - printf("Warning: Vsat at current temperature = %g may be too small.\n", pParam->BSIM3v32vsattemp); - } - - if (pParam->BSIM3v32pdibl1 < 0.0) - { fprintf(fplog, "Warning: Pdibl1 = %g is negative.\n", - pParam->BSIM3v32pdibl1); - printf("Warning: Pdibl1 = %g is negative.\n", pParam->BSIM3v32pdibl1); - } - if (pParam->BSIM3v32pdibl2 < 0.0) - { fprintf(fplog, "Warning: Pdibl2 = %g is negative.\n", - pParam->BSIM3v32pdibl2); - printf("Warning: Pdibl2 = %g is negative.\n", pParam->BSIM3v32pdibl2); - } - /* Check overlap capacitance parameters */ - if (model->BSIM3v32cgdo < 0.0) - { fprintf(fplog, "Warning: cgdo = %g is negative. Set to zero.\n", model->BSIM3v32cgdo); - printf("Warning: cgdo = %g is negative. Set to zero.\n", model->BSIM3v32cgdo); - model->BSIM3v32cgdo = 0.0; - } - if (model->BSIM3v32cgso < 0.0) - { fprintf(fplog, "Warning: cgso = %g is negative. Set to zero.\n", model->BSIM3v32cgso); - printf("Warning: cgso = %g is negative. Set to zero.\n", model->BSIM3v32cgso); - model->BSIM3v32cgso = 0.0; - } - if (model->BSIM3v32cgbo < 0.0) - { fprintf(fplog, "Warning: cgbo = %g is negative. Set to zero.\n", model->BSIM3v32cgbo); - printf("Warning: cgbo = %g is negative. Set to zero.\n", model->BSIM3v32cgbo); - model->BSIM3v32cgbo = 0.0; - } - - }/* loop for the parameter check for warning messages */ - fclose(fplog); - } - else - { fprintf(stderr, "Warning: Can't open log file. Parameter checking skipped.\n"); - } - - return(Fatal_Flag); + /* Check Abulk parameters */ + if (fabs(1.0e-6 / (pParam->BSIM3v32b1 + pParam->BSIM3v32weff)) > 10.0) + { wl_append_word(&wl, &wl, tprintf("Warning: (B1 + Weff) may be too small.\n")); + } + + /* Check Saturation parameters */ + if (pParam->BSIM3v32a2 < 0.01) + { wl_append_word(&wl, &wl, tprintf("Warning: A2 = %g is too small. Set to 0.01.\n", pParam->BSIM3v32a2)); + pParam->BSIM3v32a2 = 0.01; + } + else if (pParam->BSIM3v32a2 > 1.0) + { wl_append_word(&wl, &wl, tprintf("Warning: A2 = %g is larger than 1. A2 is set to 1 and A1 is set to 0.\n", + pParam->BSIM3v32a2)); + pParam->BSIM3v32a2 = 1.0; + pParam->BSIM3v32a1 = 0.0; + + } + + if (pParam->BSIM3v32rdsw < 0.0) + { wl_append_word(&wl, &wl, tprintf("Warning: Rdsw = %g is negative. Set to zero.\n", + pParam->BSIM3v32rdsw)); + pParam->BSIM3v32rdsw = 0.0; + pParam->BSIM3v32rds0 = 0.0; + } + else if ((pParam->BSIM3v32rds0 > 0.0) && (pParam->BSIM3v32rds0 < 0.001)) + { wl_append_word(&wl, &wl, tprintf("Warning: Rds at current temperature = %g is less than 0.001 ohm. Set to zero.\n", + pParam->BSIM3v32rds0)); + pParam->BSIM3v32rds0 = 0.0; + } + if (pParam->BSIM3v32vsattemp < 1.0e3) + { wl_append_word(&wl, &wl, tprintf("Warning: Vsat at current temperature = %g may be too small.\n", pParam->BSIM3v32vsattemp)); + } + + if (pParam->BSIM3v32pdibl1 < 0.0) + { wl_append_word(&wl, &wl, tprintf("Warning: Pdibl1 = %g is negative.\n", + pParam->BSIM3v32pdibl1)); + } + if (pParam->BSIM3v32pdibl2 < 0.0) + { wl_append_word(&wl, &wl, tprintf("Warning: Pdibl2 = %g is negative.\n", + pParam->BSIM3v32pdibl2)); + } + /* Check overlap capacitance parameters */ + if (model->BSIM3v32cgdo < 0.0) + { wl_append_word(&wl, &wl, tprintf("Warning: cgdo = %g is negative. Set to zero.\n", model->BSIM3v32cgdo)); + model->BSIM3v32cgdo = 0.0; + } + if (model->BSIM3v32cgso < 0.0) + { wl_append_word(&wl, &wl, tprintf("Warning: cgso = %g is negative. Set to zero.\n", model->BSIM3v32cgso)); + model->BSIM3v32cgso = 0.0; + } + if (model->BSIM3v32cgbo < 0.0) + { wl_append_word(&wl, &wl, tprintf("Warning: cgbo = %g is negative. Set to zero.\n", model->BSIM3v32cgbo)); + model->BSIM3v32cgbo = 0.0; + } + + } + + if (wlstart->wl_next) { + if ((fplog = fopen("b3v32check.log", "w")) != NULL) { + while (wlstart) { + fprintf(fplog, "%s", wlstart->wl_word); + fprintf(stderr, "%s", wlstart->wl_word); + wlstart = wlstart->wl_next; + } + fclose(fplog); + } + else { + while (wlstart) { + fprintf(stderr, "%s", wlstart->wl_word); + wlstart = wlstart->wl_next; + } + } + } + + wl_free(wlstart); + + return(Fatal_Flag); }