linux env doing model card tunning

This code was contributed by charleshsiao on 2014-04-01 .. formatting is tidied up, now we just need some explanatory context :-).

HJG So... executing sed and awk means this needs to be run on a unix-system. Using sed plus awk separately is... well, strange. (awk could do both steps of the processing) Doing that inside tcl, which has perfectly good string-operations, is... well, stranger. (tcl could do it alone)
Reading a bunch of files means we need to know the contents of theses files to get this program running. FS_NVM_V2.lib - no idea...

This code is for semiconductor device model tunning. We can do its tunning if no money to buying modeling tool. We maybe can get model card from tsmc/UMC... like this form.

+ u0 = 0.012
+ ua = ...
+ ub = ...
+ dsub = ...
+ drout = ...
+ pvag = ...
+ cdsc = ...
+ cdscb = ...
+ cdscd = ...

When doing curve fitting, we need do sp runset writing in advance. Maybe named vd-id.sp and need hspice tool. so, judge.tcl would looks like:

exec hspice vd-id.sp > OUT
foreach ip {46 47 48 49 50 51 52} {
set line exec sed -n "${ip}p" OUT
if regexp {^\s*[0-9} $line x] break
}; #dynamic check which lines are our needing data
exec cat OUT | sed -n "${ip},expr $ip+55p" | awk \{print $2} > gc
...
set run 1
proc base {file array} {
    global $array
    array set $array {}; set ww 1
    set hand [open $file r]
    set yy [read $hand]
    foreach j $yy {
        array set $array "$ww $j"
        incr ww
    }
    incr ww -1
    close $hand
    set sum 0
    for {set w 1} {$w<=$ww} {incr w} {set sum [expr $sum+[set ${array}($w)]*[set ${array}($w)]]}
    set ${array}(sum) $sum
}

foreach {rr} {ga gb da db dat dbt dyt dzt} {
    base $rr $rr
}

proc comp {a b} {
    global $a $b
    if {[llength [array get $a]]!=[llength [array get $b]]} return
    set num [llength [array names $a]]
    incr num -1
    set dsum 0
    for {set i 1} {$i<=$num} {incr i} {
        set dsum [expr $dsum+([set ${a}($i)]-[set ${b}($i)])*([set ${a}($i)]-[set ${b}($i)])]
    }
    return "[expr sqrt($dsum/$num)/sqrt([set ${a}(sum)]/$num)]"
}
#puts [comp ga gc]

proc rms {} {
    foreach rr {gc gd dd de dd2 de2 dw2 dx2} {global $rr; base $rr $rr}
    return [expr 1.0*[comp ga gc]]
    return [expr ([comp ga gc]+[comp gb gd])*0.5]
   #return [expr ([comp ga gc]+[comp gb gd]+[comp da dd]+[comp db de])*0.25]
   #return [expr ([comp ga gc]+[comp gb gd]+[comp da dd]+[comp db de]+[comp dat dd2]+[comp dbt de2]+[comp dyt dw2]+[comp dzt dx2])*0.125]
}

proc altr {l c inter} {
    global run
    set vsat [exec sed -n ${l}p l55sp_v151_FS_NVM_V2.lib | awk "{print \$[set c]}"]
   #set vsat [exec sed -n  763p l55sp_v151.lib | awk "{print \$4}"]
    set line [exec sed -n ${l}p l55sp_v151_FS_NVM_V2.lib]
   #set line [exec sed -n  763p l55sp_v151.lib]
    exec tclsh judge.tcl &
    exec tclsh judge2.tcl &
    after 4000
    #puts [rms]
    #puts "[comp ga gc]+[comp gb gd]+[comp da dd]+[comp db de]+[comp dat dd2]+[comp dbt de2]+[comp dyt dw2]+[comp dzt dx2]"
    set now [rms]
    set last 1000
    set plus [expr $vsat+$inter]
    set minus [expr $vsat-$inter]
    set line [lreplace $line [expr $c-1] [expr $c-1] $plus]
    exec sed -i "${l}c $line" l55sp_v151_FS_NVM_V2.lib
    exec tclsh judge.tcl &
    exec tclsh judge2.tcl &
    after 4000
    set plus [rms]
    set line [lreplace $line [expr $c-1] [expr $c-1] $minus]
    exec sed -i "${l}c $line" l55sp_v151_FS_NVM_V2.lib
    exec tclsh judge.tcl &
    exec tclsh judge2.tcl &
    after 4000
    set minus [rms]
    if {$minus<$now} {set dir -1} elseif {$plus<$now} {set dir 1} else {
        global var brms; set var "temp best!!"; set brms "$now"
        set line [lreplace $line [expr $c-1] [expr $c-1] $vsat]
        exec sed -i "${l}c $line" l55sp_v151_FS_NVM_V2.lib
        return
    }
    #puts $dir
    while {$last>=$now && $run==1} {
        set last $now
        set vsat [expr $vsat+$inter*$dir]
        set line [lreplace $line [expr $c-1] [expr $c-1] $vsat]
        exec sed -i "${l}c $line" l55sp_v151_FS_NVM_V2.lib
        exec tclsh judge.tcl &
        exec tclsh judge2.tcl &
        after 4000
        set now [rms]
        puts $vsat
        puts $now
    }
    set vsat [expr $vsat-$inter*$dir]
    set line [lreplace $line [expr $c-1] [expr $c-1] $vsat]
    exec sed -i "${l}c $line" l55sp_v151_FS_NVM_V2.lib
    exec tclsh judge.tcl &
    exec tclsh judge2.tcl &
    after 4000
    set now [rms]
    global var brms
    set var $vsat
    set brms $now
}

bind . <Control-Key-c> {set run 0}

#altr 214 4 0.0001
label .a  -text "item default value:" -width 50
label .b  -textvariable va -fg blue
label .aa -text "now value:"
label .bb -textvariable varl -fg red
label .c  -textvariable item -fg red
#button .a -text [exec sed -n 214p default.lib | awk "{print \$4}"] -command {}
#button .b -text U0 -command {altr 214 4 0.0001}
#button .c -text UA -command {altr 215 4 0.1e-9}
#button .d -text UB -command {altr 216 4 0.1e-18}
menu .m
. configure -menu .m
.m add command -label U0   -command {set l 214; set c 4; set varl [exec sed -n ${l}p l55sp_v151_FS_NVM_V2.lib | awk "{print \$[set c]}"]; set item U0; set va [exec sed -n ${l}p default.lib | awk "{print \$[set c]}"]; set brms ""; set var ""; set int [exp $va]}
.m add command -label UA   -command {set l 215; set c 4; set varl [exec sed -n ${l}p l55sp_v151_FS_NVM_V2.lib | awk "{print \$[set c]}"]; set item UA; set va [exec sed -n ${l}p default.lib | awk "{print \$[set c]}"]; set brms ""; set var ""; set int [exp $va]}
.m add command -label UB   -command {set l 216; set c 4; set varl [exec sed -n ${l}p l55sp_v151_FS_NVM_V2.lib | awk "{print \$[set c]}"]; set item UB; set va [exec sed -n ${l}p default.lib | awk "{print \$[set c]}"]; set brms ""; set var ""; set int [exp $va]}
.m add command -label DSUB  -command {set l 217; set c 4; set item DSUB; foreach {varl va int} [settle $l $c] break; set brms ""; set var ""}
.m add command -label DROUT -command {set l 218; set c 4; set item DROUT; foreach {varl va int} [settle $l $c] break; set brms ""; set var ""}
.m add command -label PVAG  -command {set l 219; set c 4; set item PVAG; foreach {varl va int} [settle $l $c] break; set brms ""; set var ""}
.m add command -label CDSC  -command {set l 220; set c 4; set item CDSC; foreach {varl va int} [settle $l $c] break; set brms ""; set var ""}
.m add command -label CDSCB -command {set l 221; set c 4; set item CDSCB; foreach {varl va int} [settle $l $c] break; set brms ""; set var ""}
.m add command -label CDSCD -command {set l 222; set c 4; set item CDSCD; foreach {varl va int} [settle $l $c] break; set brms ""; set var ""}

entry  .cc -textvariable int
button .d -text RUN -command {if {$int!=""} {altr $l $c $int}}
label  .e -text "temp the best arg-rms"
entry  .f -textvariable brms
label  .g -text "temp the fittest value"
entry  .i -textvariable var
#button .e -text minv -command {altr 752 7 1}
pack .a .b .aa .bb .c .cc .d .e .f .g .i



proc exp val {
#if {$val>0} {set op 1} else {set op 0}
    if {[expr log10(abs($val))]>=0} {set pw 1} else {set pw 0}
    if $pw {set new "1e[expr int(log10(abs($val)))-2]"} else {set new "1e[expr int(log10(abs($val)))-3]"}
    #if $op {return $new} else {return [expr -$new]}
    return $new
}

proc settle {l c} {
    set varl [exec sed -n ${l}p l55sp_v151_FS_NVM_V2.lib | awk "{print \$[set c]}"]
    set va   [exec sed -n ${l}p default.lib | awk "{print \$[set c]}"]
    set int [exp $va]
    return "$varl $va $int"
}