divert(-1) liblog.m4 Logic gates * Circuit_macros Version 10.7, copyright (c) 2024 J. D. Aplevich under * * the LaTeX Project Public Licence in file Licence.txt. The files of * * this distribution may be redistributed or modified provided that this * * copyright notice is included and provided that modifications are clearly * * marked to distinguish them from this distribution. There is no warranty * * whatsoever for these files. * define(`liblog_') ifdef(`libgen_',,`include(libgen.m4)divert(-1)') ifdef(`libcct_',,`include(libcct.m4)divert(-1)') `Notes: ================================================================== Gates other than BUFFER and NOT have an optional integer argument N that sets the number of input locations, which then have labels In1 to InN. BUFFER and NOT gates have In1 only. If there is a first argument, it is a line specification and the gate is drawn along the line as for a two-terminal element. NEGATED inputs are obtained if the second argument is N (uppercase n). DEBUGGING: The statement print "`$0'($@)" ; inserted into a macro will display the macro name and current arguments =========================================================================' `Nonzero parameter for logic size and grid mesh' define(`L_unit',`(linewid/10)') `Dimensions in L_units, also for external use:' define(`G_hht',3) `gate half-height' define(`AND_ht',`(2*G_hht)') `gate heights and widths ...' define(`AND_wd',7) define(`BUF_ht',4) define(`BUF_wd',3.5) define(`OR_rad',7) `OR input radius' define(`XOR_off',1) `XOR and NXOR parameter' define(`N_diam',(3/2)) `not-circle diameter' define(`N_rad',`(N_diam/2)') `not-circle radius' define(`NOT_rad',`(N_rad*L_unit)') `scaled radius eg line chop NOT_rad' define(`N_attr') `not-circle attribs eg shaded "green"' define(`H_ht',2) `Hysteresis symbol dimen' define(`Mx_pins',6) `max number of gate input pins without extensions Possibly 4 is better for negated inputs' define(`FF_wid',12) `Bistable (flipflop)' define(`FF_ht',18) define(`Mux_wid',8) `Multiplexer defaults' define(`Mux_ht',18) define(`Lg_body') `Gate body attributes, e.g., shaded "color"' define(`lg_plen',4) `Logic pin' define(`lg_pinsep',`(3*L_unit)') `logic pin separation in logic units' define(`lg_chipwd',`(18*L_unit)') `default chip width' define(`lg_pintxt', `"ifxfig(`$1',`ifsvg(`svg_small(`$1',75)',`sp_{\scriptsize `$1'}sp_')')"') `Logic pin text with bar where possible' define(`lg_bartxt',`iflatex(`$\overline{\hbox{`$1'}}$', `ifsvg(`svg_ol(`$1')',`$1')')') `The comma has to be at the top level' define(`m4_pattocomma',`patsubst(`$1',`$2',`,')') `Scale grid coordinates to world coordinates' define(`grid_',`(vscal_(L_unit,`$1',`$2'))') `Scale and rotate grid coords to world coords' define(`svec_',`vec_(vscal_(L_unit,`$1',`$2'))') `Relative svec_' define(`rsvec_',`Here+svec_(`$1',`$2')') `NOT_circle convenience for drawing NOT circles' define(`NOT_circle',`circle diam N_diam*L_unit N_attr') `LH_symbol([U|D|L|R|degrees][I],keys) I=inverted logical hysteresis symbol keys: hght=expr; (H_ht) wdth=fraction; (body wdth=frac*hght) ' define(`LH_symbol',`[ define(`m4LH',patsubst(`$1',I))dnl pushkeys_(`$2',`hght:H_ht; wdth:0.6;')dnl define(`m4Hs',ifinstr(`$1',I,-)H_ht)setdir_(m4LH,R)dnl line to svec_(H_ht,0) \ then to svec_(1.1*H_ht,m4Hs) line from rsvec_((1-m4wdth)*H_ht,0) \ to rsvec_(-m4wdth*H_ht,0) \ then to rsvec_(-(m4wdth+0.1)*H_ht,-(m4Hs)) `$2'; resetdir_ popdef(`m4hght',`m4wdth') ]') `LT_symbol(U|D|L|R|degrees,keys) triangle_symbol keys: wdth=expr; (H_ht) ' define(`LT_symbol', `[ setdir_(`$1',R) pushkeys_(`$2',`wdth:H_ht') line to svec_(0,m4wdth*5/8) then to svec_(m4wdth,0) \ then to svec_(0,-m4wdth*5/8) then to Here `$2'; resetdir_ popdef(`m4wdth') ] ') `BOX_gate(inputs,output,swid,sht,label, attributes) drawn in the current direction args 3 and 4 are L_unit values inputs=[P|N]* output=P|N' define(`BOX_gate',`dnl define(`m4m',`ifelse(`$1',,2,len(`$1'))')define(`m4a',`$1')dnl define(`m4h',`ifelse(`$3',,AND_wd,`$3')')dnl define(`m4v',`ifelse(`$4',,AND_wd,`$4')')dnl [ Box: [Outline: line to svec_(0,m4v/2) then to svec_(m4h,m4v/2) \ then to svec_(m4h,-m4v/2) then to svec_(0,-m4v/2) then to (0,0) \ Lg_body `$6'] \ with .Outline.start at (0,0) ifelse(`$5',,,`{ move to Box.n+(0,-5pt__) m4lstring($5,"ifsvg(`svg_small($5,75)',`{\scriptsize$ $5 $}')") }') IOdefs(from svec_(0,m4v/2) to svec_(0,-m4v/2),In,`$1',R) Out: svec_(m4h,0) ifelse(`$2',N,`+svec_(N_diam,0) N_Out: NOT_circle \ at Out-svec_(N_rad,0)'); `$7']') `AND_gate(n,[N][B],[wid,[ht]],attributes) drawn in the current direction 0 <= n <= 16; N=negated inputs, B=box shape or AND_gate(chars,[B],wid,ht,attributes) arg1 is one or more of N|P ' define(`AND_gate',`define(`m4arg1',ifelse(`$1',,2,`$1'))dnl ifelse(ifinstr(m4arg1,N,1)`'ifinstr(m4arg1,P,1),, `m4dupstr(ifinstr(`$2',N,N,P),m4arg1,`m4PN')',`define(`m4PN',m4arg1)')dnl ifinstr(`$2',B, `BOX_gate(m4PN,ifinstr(`$2',N,N,P),ifelse(`$3',,,`($3)/(L_unit)'), ifelse(`$4',,,`($4)/(L_unit)'),ifsvg(`&```amp''';',`\&'),Lg_body `$5',`$6')', `AND_gen(m4PN,ifelse(`$2',N,N)IBAONESEC,`$3',`$4',Lg_body `$5',`$6')')') `NAND_gate(n,[N][B],[wid,[ht]],attributes) 0 <= n <= 16; N=negated inputs, B=box shape or NAND_gate(chars,[B],wid,ht) arg1 is one or more of N|P ' define(`NAND_gate',`define(`m4arg1',ifelse(`$1',,2,`$1'))dnl ifelse(ifinstr(m4arg1,N,1)`'ifinstr(m4arg1,P,1),, `m4dupstr(ifinstr(`$2',N,N,P),m4arg1,`m4PN')',`define(`m4PN',m4arg1)')dnl ifinstr(`$2',B, `BOX_gate(m4PN,ifelse(`$2',N,N,P),ifelse(`$3',,,`($3)/(L_unit)'), ifelse(`$4',,,`($4)/(L_unit)'),ifsvg(`&```amp''';',`\&'),Lg_body `$5',`$6')', `AND_gen(m4PN,ifelse(`$2',N,N)IBANONESEC,`$3',`$4',Lg_body `$5',`$6')')') `AND_gen(n,chars,[wid,[ht]],attributes) 0 <= n <= 16 B=base and straight sides; A=Arc; [N]NE,[N]SE,[N]I,[N]N,[N]S=inputs or circles, [N]O=output; C=center location or AND_gen(chars,chars,[wid,[ht]],attributes) arg1 is one or more of N|P eg PPPP defines 4 inputs, NPNP negates the first and third; arg2 as above except [N]I is ignored arg5 contains body attributes ' define(`AND_gen',`define(`m4arg1',ifelse(`$1',,2,`$1'))dnl ifelse(ifinstr(m4arg1,N,1)`'ifinstr(m4arg1,P,1),, `m4dupstr(ifinstr(`$2',N,N,P),m4arg1,`m4PN')',`define(`m4PN',m4arg1)')dnl define(`m4hor',`ifelse(`$3',,AND_wd,`($3)/(L_unit)')')dnl define(`m4ver',`ifelse(`$4',,ifelse(`$3',,AND_ht,AND_ht/(AND_wd)*m4hor), `($4)/(L_unit)')')define(`dna_',ifelse(`$2',,IBAONESEC,`$2'))dnl [ sc_draw(`dna_',B,`Bm: line from svec_(m4hor-m4ver/2,-m4ver/2) \ to svec_(0,-m4ver/2) \ then to svec_(0,m4ver/2) then \ to svec_(m4hor-m4ver/2,m4ver/2) Lg_body `$5' ')dnl sc_draw(`dna_',A,`Arc: arc cw rad m4ver/2 \ to rsvec_(0,-m4ver) \ with .c at rsvec_(0,-m4ver/2) Lg_body `$5' ')dnl sc_draw(`dna_',NNE, `NNE: svec_(m4hor-m4ver/2,0)+svec_(Rect_(m4ver/2+N_diam,45)) N_NNE: NOT_circle \ at svec_(m4hor-m4ver/2,0)+svec_(Rect_(m4ver/2+N_rad,45)) ')dnl sc_draw(`dna_',NSE, `NSE: svec_(m4hor-m4ver/2,0)+svec_(Rect_(m4ver/2+N_diam,-45)) N_NSE: NOT_circle \ at svec_(m4hor-m4ver/2,0)+svec_(Rect_(m4ver/2+N_rad,-45)) ')dnl sc_draw(`dna_',NE, `NE: svec_(m4hor-m4ver/2,0)+svec_(Rect_(m4ver/2,45)) ')dnl ifelse(ifinstr(`$1',N,1)`'ifinstr(`$1',P,1),, `sc_draw(`dna_',NI, `m4A_defs(ifelse(`$1',,2,`$1'),N)') ')dnl sc_draw(`dna_',SE, `SE: svec_(m4hor-m4ver/2,0)+svec_(Rect_(m4ver/2,-45)) ')dnl sc_draw(`dna_',NN, `N_NN: NOT_circle \ at svec_((m4hor-m4ver/2)/2,m4ver/2+N_rad) NN: svec_((m4hor-m4ver/2)/2,m4ver/2+N_diam) ')dnl sc_draw(`dna_',NS, `N_NS: NOT_circle \ at svec_((m4hor-m4ver/2)/2,-m4ver/2-N_rad) NS: svec_((m4hor-m4ver/2)/2,-m4ver/2-N_diam) ')dnl sc_draw(`dna_',NO, `N_Out: NOT_circle \ at svec_(m4hor+N_rad,0) Out: svec_(m4hor+N_diam,0) ')dnl sc_draw(`dna_',O, `Out: svec_(m4hor,0) ')dnl sc_draw(`dna_',N, `N: svec_(0,m4ver/2) ')dnl ifelse(ifinstr(`$1',N,1)`'ifinstr(`$1',P,1),, `sc_draw(`dna_',I, `m4A_defs(ifelse(`$1',,2,`$1'))')', `m4A_defs(`$1')') sc_draw(`dna_',S, `S: svec_(0,-m4ver/2) ')dnl sc_draw(`dna_',C, `C: svec_(m4hor/2,0)') `$6']') `Input locations, flat face m4A_defs(n,[N]) or m4A_defs(string) string = one or more N|P otherwise n inputs (negated if arg2 is N) Input points are defined as Inx and NOT circles as N_Inx where x is integer' `What about the IOdefs macro?' define(`m4A_defs',`define(`m4arg1',ifelse(`$1',,2,`$1'))dnl ifelse(ifinstr(m4arg1,N,1)`'ifinstr(m4arg1,P,1),, `m4dupstr(ifinstr(`$2',N,N,P),m4arg1,`m4AI')define(`m4AnI',m4arg1)', `define(`m4AI',m4arg1)define(`m4AnI',len(m4arg1))')dnl define(`m4m',`m4ver/2/min(m4AnI,Mx_pins-1)*min(m4AnI,3*(Mx_pins-1))')dnl ifelse(eval(m4AnI>Mx_pins),1, `line from svec_(0, m4m) \ to svec_(0,m4ver/2) line from svec_(0,-m4m) \ to svec_(0,-m4ver/2)') for_(1,m4AnI,1,`ifelse(substr(m4AI,0,1),N, `N_In`'m4x: NOT_circle at \ svec_(-N_rad,m4ver/min(m4AnI,Mx_pins-1)*((m4AnI+1)/2-m4x)) In`'m4x: svec_(-N_diam,m4ver/min(m4AnI,Mx_pins-1)*((m4AnI+1)/2-m4x)) ', `In`'m4x: svec_(0,m4ver/min(m4AnI,Mx_pins-1)*((m4AnI+1)/2-m4x))')dnl define(`m4AI',substr(m4AI,1)) ') ') `OR_gate or NOR_gate or XOR_gate or NXOR_gate drawn in the current direction args = (n,[N][B],wid,ht,attributes) 0 <= n <= 16; N=negated inputs, B=box shape or args = (chars,[B],wid,ht,attributes) arg1 is one or more of N|P ' define(`OR_gate',`m4_OR(OR,$@)') define(`NOR_gate',`m4_OR(NOR,$@)') define(`XOR_gate',`m4_OR(XOR,$@)') define(`NXOR_gate',`m4_OR(NXOR,$@)') define(`m4_OR',`define(`m4arg1',ifelse(`$2',,2,`$2'))dnl ifelse(ifinstr(m4arg1,N,1)`'ifinstr(m4arg1,P,1),, `m4dupstr(ifinstr(`$3',N,N,P),m4arg1,`m4PN')',`define(`m4PN',m4arg1)')dnl ifinstr(`$3',B, `BOX_gate(m4PN,ifinstr(`$3',N,N,P,`$6',`$7'), ifelse(`$4',,,`($4)/(L_unit)'),ifelse(`$5',,,`($5)/(L_unit)'), ifinstr(`$1',XOR,`=',`ifsvg(`>=1',`\geq 1')'),Lg_body `$6',`$7',`$8')', `OR_gen(m4PN, ifinstr(`$1',XOR,P)`'ifelse(`$3',N,N)IBA`'ifelse(substr(`$1',0,1),N,N)ONESEC, shift(shift(shift($@))))')') `OR_gen(n,chars,[wid,[ht]],attributes) 0 <= n <= 16 B=base and straight sides; A=Arc; [N]NE,[N]SE,[N]I,[N]N,[N]S=inputs or circles, [N]P=XOR arc; [N]O=output; C=center or OR_gen(chars,chars,[wid,[ht]],attributes) arg1 is one or more of N|P eg PPPP defines 4 inputs, NPNP negates the first and third; arg2 as above except [N]I is ignored If arg5 contains shaded rgbstring(...) the arguments of rgbstring may not contain parentheses' define(`OR_gen',`[define(`m4arg1',ifelse(`$1',,2,`$1'))dnl ifelse(ifinstr(m4arg1,N,1)`'ifinstr(m4arg1,P,1),, `m4dupstr(ifinstr(`$2',N,N,P),m4arg1,`m4PN')',`define(`m4PN',m4arg1)')dnl define(`m4hor',`ifelse(`$3',,AND_wd,`($3)/(L_unit)')')define(`m4o',0)dnl define(`m4ver',`ifelse(`$4',,ifelse(`$3',,AND_ht,AND_ht/(AND_wd)*m4hor), `($4)/(L_unit)')')define(`dna_',ifelse(`$2',,IBAONESEC,`$2'))dnl define(`m4gfill',`Lg_body `$5'')dnl dnl concave fill, painful code: ifelse(regexp(m4gfill,`^fill \|[^a-z]fill '),-1, `ifelse(regexp(m4gfill,`shaded *"'),-1, `define(`m4osh',`regexp(m4gfill,`.*shaded *\(sprintf([^)]*)\).*$',\1)') define(`m4att',`patsubst(m4gfill,`shaded *sprintf([^)]*)')')', `define(`m4osh',`regexp(m4gfill,`.*shaded *"\([^"]*\)".*$',"\1")') define(`m4att',`patsubst(m4gfill,`shaded *"[^"]*"')')')', `define(`m4osh',`regexp(m4gfill,`.*fill *\([^ ]*\) .*$',`graystring(\1)')') define(`m4att',`patsubst(m4gfill,`fill *[^ ]*[ $]')')') sc_draw(`dna_',P,`define(`m4o',XOR_off*m4ver/(AND_ht))dnl Parc: arc cw from svec_(0,m4ver/2) \ to svec_(0,-m4ver/2) \ with .c at svec_(-sqrt((OR_rad*m4ver/(AND_ht))^2-(m4ver/2)^2),0) ')dnl sc_draw(`dna_',B,dnl `r = OR_rad*m4ver/(AND_ht) rt = sqrt(r^2-(m4ver/2)^2) ifelse(m4osh,,,`for i=0 to int((r-rt)*L_unit/lthick+1) do { tx = i*lthick/L_unit {arc cw from svec_(tx+m4o-hlth,m4ver/2) to svec_(tx+m4o-hlth,-m4ver/2) \ with .c at svec_(tx+m4o-rt,0) outlined m4osh} } {rotbox(m4hor/3*L_unit,m4ver*L_unit,shaded m4osh invis) with .W \ at svec_(m4o+r-rt,0)} ') Bt: line from svec_(m4o+m4hor/3,m4ver/2) \ to svec_(m4o,m4ver/2) ifdpic(chop 0 chop -hlth) m4att ArcB: arc cw to svec_(m4o,-m4ver/2) with .c at svec_(m4o-rt,0) m4att Bb: line to svec_(m4o+m4hor/3,-m4ver/2) ifdpic(chop -hlth chop 0) m4att ')dnl sc_draw(`dna_',A,`define(`m4m',`((m4hor*2/3)^2-(m4ver/2)^2)/(m4ver)')dnl ifelse(m4osh,,,`{line invis from svec_(m4o+m4hor,0) \ to svec_(m4o+m4hor/3, m4ver/2) then to svec_(m4o+m4hor/3,-m4ver/2) \ then to svec_(m4o+m4hor,0) shaded m4osh}') ArcN: arc cw from svec_(m4o+m4hor/3, m4ver/2) \ to svec_(m4o+m4hor,0) with .c at svec_(m4o+m4hor/3,-m4m) m4gfill ArcS: arc ccw from svec_(m4o+m4hor/3,-m4ver/2) \ to svec_(m4o+m4hor,0) with .c at svec_(m4o+m4hor/3,m4m) m4gfill ')dnl sc_draw(`dna_',NNE, `N_NNE: NOT_circle \ at svec_(m4o+m4hor/3,-m4m)+svec_(Rect_(m4ver/2+m4m+N_rad,60)) NNE: svec_(m4o+m4hor/3,-m4m)+svec_(Rect_(m4ver/2+m4m+N_diam,60)) ')dnl sc_draw(`dna_',NSE, `N_NSE: NOT_circle \ at svec_(m4o+m4hor/3,m4m)+svec_(Rect_(m4ver/2+m4m+N_rad,-60)) NSE: svec_(m4o+m4hor/3,m4m)+svec_(Rect_(m4ver/2+m4m+N_diam,-60)) ')dnl sc_draw(`dna_',NE, `NE: svec_(m4o+m4hor/3,-m4m)+svec_(Rect_(m4ver/2+m4m,60)) ')dnl sc_draw(`dna_',SE, `SE: svec_(m4o+m4hor/3,m4m)+svec_(Rect_(m4ver/2+m4m,-60)) ')dnl ifelse(ifinstr(`$1',N,1)`'ifinstr(`$1',P,1),, `sc_draw(`dna_',NI, `m4O_defs(ifelse(`$1',,2,`$1'),N)') ')dnl sc_draw(`dna_',NN, `N_NN: NOT_circle \ at svec_(m4o+m4hor/6,m4ver/2+N_rad) NN: svec_(m4o+m4hor/6,m4ver/2+N_diam) ')dnl sc_draw(`dna_',NS, `N_NS: NOT_circle \ at svec_(m4o+m4hor/6,-m4ver/2-N_rad) NS: svec_(m4o+m4hor/6,-m4ver/2-N_diam) ')dnl sc_draw(`dna_',NO, `N_Out: NOT_circle \ at svec_(m4o+m4hor+N_rad,0) Out: svec_(m4o+m4hor+N_diam,0) ')dnl sc_draw(`dna_',O, `Out: svec_(m4o+m4hor,0) ')dnl sc_draw(`dna_',N, `N: svec_(m4o+m4hor/6,m4ver/2) ')dnl ifelse(ifinstr(`$1',N,1)`'ifinstr(`$1',P,1),, `sc_draw(`dna_',I, `m4O_defs(ifelse(`$1',,2,`$1'))')', `m4O_defs(`$1')') sc_draw(`dna_',S, `S: svec_(m4o+m4hor/6,-m4ver/2) ')dnl sc_draw(`dna_',C, `C: svec_(m4o+m4hor/2,0)') `$6']') `Input locations, curved face m4O_defs(n,[N]) or m4O_defs(string) string = one or more N|P otherwise n inputs (negated if arg2 is N) Input points are defined as Inx and NOT circles as N_Inx where x is integer' define(`m4O_defs',`define(`m4arg1',ifelse(`$1',,2,`$1'))dnl ifelse(ifinstr(m4arg1,N,1)`'ifinstr(m4arg1,P,1),, `m4dupstr(ifinstr(`$2',N,N,P),m4arg1,`m4OI')define(`m4OnI',m4arg1)', `define(`m4OI',m4arg1)define(`m4OnI',len(m4arg1))')dnl define(`m4om',`m4ver/2/min(m4OnI,Mx_pins-1)*min(m4OnI,3*(Mx_pins-1))')dnl ifelse(eval(m4OnI>Mx_pins),1, `arc ccw from svec_(0,m4ver/2) \ to svec_(0,m4ver) + M4O_pos(m4om-m4ver) \ with .c at svec_(-M4O_dst,m4ver) arc cw from svec_(0,-m4ver/2) \ to svec_(0,-m4ver) +M4O_pos(-(m4om-m4ver))\ with .c at svec_(-M4O_dst,-m4ver) ')dnl define(`m4on',`(eval((m4OnI-Mx_pins+1)/2))')dnl for_(1,m4OnI,1, `define(`m4oq',`m4ver/2/min(m4OnI,Mx_pins-1)*(m4OnI+1-2*m4x)')dnl In`'m4x: ifelse(eval(m4x<=m4on),1,`svec_(0, m4ver)+M4O_pos(m4oq-m4ver)', eval(m4x>(m4OnI-m4on)),1, `svec_(0,-m4ver)+M4O_pos(m4oq+m4ver)', `M4O_pos(m4oq)') ifelse(substr(m4OI,0,1),N, `N_In`'m4x: NOT_circle \ at In`'m4x+svec_(-N_rad,0) In`'m4x: In`'m4x+svec_(-N_diam,0) ')dnl define(`m4OI',substr(m4OI,1))dnl ') ') define(`M4O_dst',`sqrt((OR_rad*m4ver/(AND_ht))^2-(m4ver/2)^2)') define(`M4O_pos',`svec_(-M4O_dst+sqrt((OR_rad*m4ver/(AND_ht))^2-(`$1')^2), `$1')') `IOdefs(linespec,label,[P|N]*,L|R) Distribute named locations with optional NOT circles along a line eg IOdefs(up 1,Z,PNN,R) defines Z1 at 1/6 along the line, NOT circles N_Z2 and N_Z3 to the right at 1/2 and 5/6 along the line with Z2 and Z3 labeled at their right edges' define(`IOdefs',`define(`m4dm',`ifelse(`$3',,1,len(`$3'))') define(`m4da',`$3')define(`m4dv',`ifelse(`$2',,In,`$2')') M4TL: move `$1' M4PL: vperp(M4TL,L_unit) for_(1,m4dm,1, `m4dv`'m4x: ((m4x-1/2)/m4dm between M4TL.start and M4TL.end) dnl ifelse(substr(m4da,0,1),N,`\ ifelse(`$4',R,-,+)(M4PL.x*N_diam,M4PL.y*N_diam) {N_`'m4dv`'m4x: NOT_circle \ at m4dv`'m4x ifelse(`$4',R,+,-)(M4PL.x*N_rad,M4PL.y*N_rad)}') define(`m4da',substr(m4da,1))') ') `BUFFER_gen(chars, wd, ht, [N|P]*, [N|P]*, [N|P]*, attributes) chars: T=triangle (default), [N]O=output location Out (NO draws circle N_Out); [N]I,[N]N,[N]S,[N]NE,[N]SE input locations C=centre location. Args 4-6 define In, NE, and SE argument sequences' define(`BUFFER_gen', `define(`m4h',`ifelse(`$2',,BUF_wd,`($2)/(L_unit)')')define(`m4y',m4h)dnl define(`m4v',`ifelse(`$3',,BUF_ht,`($3)/(L_unit)')')dnl define(`dna_',ifelse(`$1',,ITOCNESE,`$1'))dnl define(`m4z',`N_rad/vlength(m4h,m4v/2)')dnl [sc_draw(`dna_',T,`Tm: line from svec_(m4h,0) \ to svec_(0,-m4v/2) \ then to svec_(0,m4v/2) \ then to svec_(m4h,0) Lg_body `$7' Tc: svec_(m4h/2,0) ')dnl sc_draw(`dna_',NNE,`N_NNE: NOT_circle \ at svec_(m4h/2,m4v/4)+svec_(m4v/2*m4z,m4h*m4z) NNE: svec_(m4h/2,m4v/4)+svec_(m4v*m4z,m4h*2*m4z) ')dnl sc_draw(`dna_',NSE,`N_NSE: NOT_circle \ at svec_(m4h/2,-m4v/4)-svec_(-m4v/2*m4z,m4h*m4z) NSE: svec_(m4h/2,-m4v/4)-svec_(-m4v*m4z,m4h*2*m4z) ')dnl sc_draw(`dna_',NI,`define(`m4y',m4y+N_diam) In1: svec_(-N_diam,0) N_In1: NOT_circle \ at svec_(-N_rad,0) ')dnl sc_draw(`dna_',NE, `NE: svec_(m4h/2,m4v/4) ')dnl sc_draw(`dna_',SE, `SE: svec_(m4h/2,-m4v/4) ')dnl sc_draw(`dna_',NN, `N_NN: NOT_circle \ at svec_(0,m4v/2+N_rad) NN: svec_(0,m4v/2+N_diam) ')dnl sc_draw(`dna_',NS, `N_NS: NOT_circle \ at svec_(0,-m4v/2-N_rad) NS: svec_(0,-m4v/2-N_diam) ')dnl sc_draw(`dna_',NO,`define(`m4y',m4y+N_diam) Out: svec_(m4h+N_diam,0) N_Out: NOT_circle \ at svec_(m4h+N_rad,0) ')dnl sc_draw(`dna_',N, `N: svec_(0,m4v/2) ')dnl sc_draw(`dna_',S, `S: svec_(0,-m4v/2) ')dnl sc_draw(`dna_',O, `Out: svec_(m4h,0) ')dnl sc_draw(`dna_',I, `In1: (0,0) ')dnl sc_draw(`dna_',C, `C: svec_(m4h/3,0) ')dnl ifelse(`$4',,,`IOdefs(from svec_(0,m4v/2) to svec_(0,-m4v/2),In,`$4',R)') ifelse(`$5',,,`IOdefs(from svec_(0,m4v/2) to svec_(m4h,0),NE,`$5')') ifelse(`$6',,,`IOdefs(from svec_(0,-m4v/2) to svec_(m4h,0),SE,`$6',R)') `$8']') `BUFFER_gate(linespec, [N|B], wid, ht, [N|P]*, [N|P]*, attributes) When linespec is blank then the element is composite and In1, Out, C, NE, and SE are defined; otherwise the element is drawn as two-terminal arg2: B=box gate' Args 5 and 6 define NE and SE argument sequences' define(`BUFFER_gate',`ifinstr(`$2',B, `BOX_gate(ifinstr(`$2',N,N,P),P,ifelse(`$3',,,`($3)/(L_unit)'), ifelse(`$4',,,`($4)/(L_unit)'),1,`$5',Lg_body `$7')', `ifelse(`$1',, `BUFFER_gen(ifelse(`$2',N,N)ITOCNESE,`$3',`$4',,`$5',`$6', Lg_body `$7',`$8')', `eleminit_(`$1') { BUFFER_gen(ifelse(`$2',N,N)ITOC,`$3',`$4',,`$5',`$6',Lg_body `$7',`$8') \ with .Tc at last line.c } { line to last [].In1; line from last [].Out to 2nd last line.end } line invis to rvec_(rp_len,0)')')') `NOT_gate(linespec, [B][N|n], wid, ht, attributes) When linespec is blank then the element is composite and In1, Out, C, NE, and SE are defined; otherwise the element is drawn as two-terminal arg2: B=box gate, N=negated input and output, n=negated input only' define(`NOT_gate',`ifinstr(`$2',B, `BOX_gate(ifinstr(`$2',N,N,`$2',n,N,P),ifinstr(`$2',n,P,N), ifelse(`$3',,,`($3)/(L_unit)'),ifelse(`$4',,,`($4)/(L_unit)'),1, Lg_body `$5',`$6')', `ifelse(`$1',, `BUFFER_gen(ifinstr(`$2',N,N,`$2',n,N)IT`'ifinstr(`$2',n,,N)OCNESE, `$3',`$4',,,,Lg_body `$5',`$6')', `eleminit_(`$1') { BUFFER_gen(ifinstr(`$2',N,N,`$2',n,N)IT`'ifinstr(`$2',n,,N)OC, `$3',`$4',,,,Lg_body `$5',`$6') \ with .C at last line.c } { line to last [].In1; line from last [].Out to 2nd last line.end } line invis to rvec_(rp_len,0)')')') ############################### `The comprehensive logic pin: lg_pin(location, label, Picname, n|e|s|w [L|M|I|O][N][E], pinno, optlen) label=text (indicating logical pin function, usually) Picname=pic label for referring to the pin n|e|s|w=orientation (north, east, south, west) L=active low out; M=active low in; I=inward arrow; O=outward arrow N=negated (NOT-circle); E=edge trigger' define(`lg_pin',`ifelse(`$1',,,`move to $1') define(`dna_',`substr(`$4',1)')define(`m4lE',)define(`m4lch',0)dnl define(`m4ld',`ifelse(`$4',,e,`substr(`$4',0,1)')')dnl define(`m4lph',`ifelse(m4ld,n,0,m4ld,w,-1,m4ld,s,0,1)')dnl define(`m4lpv',`ifelse(m4ld,n,1,m4ld,w,0,m4ld,s,-1,0)')dnl define(`m4lpl',`ifelse(`$6',,`lg_plen',(`$6')/L_unit)')dnl sc_draw(`dna_',E,`define(`m4lE',1)dnl { line from rsvec_(lp_xy(0,N_rad)) \ to rsvec_(lp_xy(-N_diam*sqrt(3)/2,0)) then to rsvec_(lp_xy(0,-N_rad)) }') ifelse(`$2',,, `{ lg_pintxt(`$2') ifelse(m4ld,w,`ljust_', m4ld,n,`below_', m4ld,s,`above_',`rjust_') at Here dnl ifxfig(`+(lp_xy(-0.72bp__,0))') dnl ifelse(m4lE,1,`+svec_(lp_xy(-N_diam*sqrt(3)/2,0))') }') sc_draw(`dna_',N,`define(`m4lch',N_diam*L_unit) { NOT_circle \ at rsvec_(lp_xy(N_rad,0)) }') sc_draw(`dna_',L,`define(`m4lch',N_rad*2.5*L_unit) {line from rsvec_(lp_xy(0, ifelse(m4ld,w,-,m4ld,s,-)N_rad*3/2)) to rsvec_(lp_xy(N_rad*2.5,0)) \ then to Here }') sc_draw(`dna_',M,`define(`m4lch',N_rad*2.5*L_unit) { line to rsvec_(lp_xy(N_rad*2.5, ifelse(m4ld,w,-,m4ld,s,-)N_rad*3/2)) then to rsvec_(lp_xy(N_rad*2.5,0)) \ then to Here}') {ifelse(`$3',,,`$3':) line to rsvec_(lp_xy(m4lpl,0)) chop m4lch chop 0 dnl ifinstr(dna_,I,` <- wid linethick*5.6bp__ ht linethick*7.2bp__ ')dnl ifinstr(dna_,O,` -> wid linethick*5.6bp__ ht linethick*7.2bp__ ') ifelse(`$5',,,`move to last line.c; lg_pintxt(`$5') dnl ifelse(m4ld,n,`rjust_', m4ld,w,`above_',m4ld,e,`above_',`rjust_')') } ') define(`lp_xy',`vrot_(`$1',`$2',m4lph,m4lpv)') `Mux(ni, label, [L][B|H|X][N[n]|S[n]][[N]OE], wid, ht, attributes) ni = number of inputs (default 2) The input pins are lines named In0, In1, ... In chars: L reverses pin numbering, B pin number labels in binary form, H pin number labels in hex form, X disable pin labels N[n] puts pin Sel or pins Sel0 .. Seln at the top (with respect to the drawing direction) S[n] puts the Sel inputs at the bottom (default) OE adds the OE input, NOE negated ' define(`Mux',`[ define(`m4Mwid',`ifelse(`$4',,Mux_wid,((`$4')/(L_unit)))')dnl define(`m4Mht',`ifelse(`$5',,Mux_ht,((`$5')/(L_unit)))')dnl W: (0,0) C: svec_(m4Mwid/2,0) E: svec_(m4Mwid,0) NW: svec_(0,m4Mht/2) SW: svec_(0,-m4Mht/2) NE: svec_(m4Mwid,m4Mht/2-2) SE: svec_(m4Mwid,-m4Mht/2+2) Line: line from W to NW then to NE then to SE then to SW then to W \ Lg_body`$6' ifelse(`$2',,,`"ifsvg(`svg_small($2,75)',`\scriptsize $2')" at C') lg_pin(E,,Out,e) define(`m4Mdna',`$3')define(`m4MOE')dnl sc_draw(`m4Mdna',L,`define(`m4ML',-)',`define(`m4ML')')dnl define(`m4MN')define(`m4MSel',S)dnl sc_draw(`m4Mdna',B,`define(`m4MN',B)')dnl sc_draw(`m4Mdna',H,`define(`m4MN',H)')dnl sc_draw(`m4Mdna',X,`define(`m4MN',X)')dnl sc_draw(`m4Mdna',NOE,`define(`m4MOE',N)')dnl sc_draw(`m4Mdna',OE,`define(`m4MOE',O)')dnl sc_draw(`m4Mdna',N,`define(`m4MSel',N)')dnl sc_draw(`m4Mdna',S,`define(`m4MSel',S)')dnl define(`m4Mn',`ifelse(`$1',,2,`ifelse(m4MOE,,`$1',incr(eval(`$1')))')')dnl ifinstr(m4MSel,N, dnl Draw Sel pins `ifelse(m4Mdna,,`lg_pin((0.5 between NW and NE),,Sel,n)',`for_(1,m4Mdna,1,` lg_pin((m4x-0.5)/m4Mdna between NW and NE,,Sel`'eval(m4x-1),n)')')', `ifelse(m4Mdna,,`lg_pin((0.5 between SW and SE),,Sel,s)',`for_(1,m4Mdna,1,` lg_pin((m4x-0.5)/m4Mdna between SW and SE,,Sel`'eval(m4x-1),s)')')') for_(1,m4Mn,1,`lg_pin( dnl Draw In pins svec_(0,m4ML`'m4Mht*(0.5+(0.5-m4x)/m4Mn)), define(`m4MNO',`ifelse(m4x,m4Mn,ifelse(m4MOE,,,N))')dnl ifelse(m4MNO,, `ifelse(m4MN,X,, m4MN,B,`define(`m4dg',`ifelse(eval(`$1'>8),1,4,eval(`$1'>4),1,3, eval(`$1'>2),1,2,1)') binary_(decr(m4x),m4dg)', m4MN,H,`define(`m4dg',`ifelse(eval(`$1'>16),1,2,1)')dnl hexadecimal_(decr(m4x),m4dg)', decr(m4x))', `ifelse(m4MN,X,,`ifelse(m4MOE,N,lg_bartxt(OE),OE)')'), ifelse(m4MNO,,In`'decr(m4x),ifelse(m4MOE,N,N)OE), w`'ifelse(m4MNO,,,ifelse(m4MOE,N,N)))') `$7']') `Demux(no, label, [L][B|H|X][N[n]|S[n]][[N]OE], wid, ht, attributes) no = number of outputs (default 2) The output pins are lines named Out0, Out1, ... Out chars: L reverses pin numbering, B displays binary pin numbers, H displays hexadecimal pin numbers, X disable pin labels N[n] puts Sel or Sel0 .. Seln at the top (with respect to the drawing direction) S[n] puts the Sel inputs at the bottom (default) OE adds the OE input, NOE negated ' define(`Demux',`[ define(`m4Dwid',`ifelse(`$4',,Mux_wid,((`$4')/(L_unit)))')dnl define(`m4Dht',`ifelse(`$5',,Mux_ht,((`$5')/(L_unit)))')dnl W: (0,0) C: svec_(m4Dwid/2,0) E: svec_(m4Dwid,0) NW: svec_(0,m4Dht/2-2) SW: svec_(0,-m4Dht/2+2) NE: svec_(m4Dwid,m4Dht/2) SE: svec_(m4Dwid,-m4Dht/2) Line: line from W to NW then to NE then to SE then to SW then to W \ Lg_body `$6' ifelse(`$2',,,`"ifsvg(`svg_small($2,75)',`\scriptsize $2')" at C') # lg pin(location, label, Picname, n|e|s|w [L|M|I|O][N][E], pinno, optlen) lg_pin(W,,In,w) define(`m4Ddna',`$3')define(`m4DOE')dnl sc_draw(`m4Ddna',L,`define(`m4DL',-)',`define(`m4DL')')dnl define(`m4DN')define(`m4DSel',S)dnl sc_draw(`m4Ddna',B,`define(`m4DN',B)')dnl sc_draw(`m4Ddna',H,`define(`m4DN',H)')dnl sc_draw(`m4Ddna',X,`define(`m4DN',X)')dnl sc_draw(`m4Ddna',NOE,`define(`m4DOE',N)')dnl sc_draw(`m4Ddna',OE,`define(`m4DOE',O)')dnl sc_draw(`m4Ddna',N,`define(`m4DSel',N)')dnl sc_draw(`m4Ddna',S,`define(`m4DSel',S)')dnl define(`m4Dn',`ifelse(`$1',,2,`$1')')dnl ifinstr(m4DSel,N, `ifelse(m4Ddna,,`lg_pin((0.5 between NW and NE),,Sel,n)',`for_(1,m4Ddna,1,` lg_pin((m4x-0.5)/m4Ddna between NW and NE,,Sel`'eval(m4x-1),n)')')', `ifelse(m4Ddna,,`lg_pin((0.5 between SW and SE),,Sel,s)',`for_(1,m4Ddna,1,` lg_pin((m4x-0.5)/m4Ddna between SW and SE,,Sel`'eval(m4x-1),s)')')') # lg pin(location, label, Picname, n|e|s|w [L|M|I|O][N][E], pinno, optlen) for_(1,m4Dn,1,`lg_pin( svec_(m4Dwid,m4DL`'m4Dht*(0.5+(0.5-m4x)/m4Dn)), ifelse(m4DN,X,, m4DN,B,`define(`m4dg',`ifelse(eval(`$1'>8),1,4,eval(`$1'>4),1,3, eval(`$1'>2),1,2,1)') binary_(decr(m4x),m4dg)', m4DN,H,`define(`m4dg',`ifelse(eval(`$1'>16),1,2,1)')dnl hexadecimal_(decr(m4x),m4dg)', decr(m4x)), Out`'decr(m4x), e)') ifelse(m4DOE,,,`ifelse(m4DOE,N, `lg_pin(svec_(0,m4Dht*(0.5+(1-m4Dn)/m4Dn)),lg_bartxt(OE),NOE,wN)', `lg_pin(svec_(0,m4Dht*(0.5+(1-m4Dn)/m4Dn)),OE,OE,w)')') `$7']') ########################################################################### AutoGate allowable functions (plus the ~ operator) define(`And',`_AutoGate(AND,$@)') define(`Or',`_AutoGate(OR,$@)') define(`Not',`_AutoGate(NOT,$@)') define(`NNot',`_AutoGate(NNOT,$@)') define(`Buffer',`_AutoGate(BUFFER,$@)') define(`Xor',`_AutoGate(XOR,$@)') define(`Nand',`_AutoGate(NAND,$@)') define(`Nor',`_AutoGate(NOR,$@)') define(`Nxor',`_AutoGate(NXOR,$@)') dnl Custom subcircuit template: dnl define(`Mygate',`_AutoGate(My,$@)') dnl define(`My_gate',`[ ... dnl Out: ... dnl In1: ... dnl In2: ... ]') dnl Not gate with negated input for NNOT above define(`NNOT_gate',`BUFFER_gate(,N)') Style parameters define(`gatelineth',linethick) #define(`gatelineth',1.0) use absolute pt define(`lineth',linethick) #define(`lineth',0.5) define(`autoinputsep',`(BUF_ht*L_unit*5/4)') # distance between inputs define(`autovsep',`L_unit') # vertical separation between input gates ########## Draw the gate with input sublayer Sg containing gates G1, G2, ... Example: `AutoGate(AND,$@)' define(`AutoGate',`[ lu = L_unit define(`m4dirt',m4_dir_) dnl Count the arguments (inputs) (could use $# ) pushdef(`m4nargs',0)dnl Loopover_(`arg',`define(`m4nargs',incr(m4nargs))',shift($@))dnl ifinstr(`$2',[,`define(`m4nargs',decr(m4nargs))dnl define(`m4PN',substr(`$2',1,eval(len(`$2')-2)))') `#' m4Delch(`$1') gate(m4nargs) # Draw the gate linethick = gatelineth ifelse(m4Delch(`$1'),NOT,`Q: NOT_gate()', m4Delch(`$1'),BUFFER,`Q: BUFFER_gate()', `Q: m4Delch(`$1')_gate(ifinstr(`$2',[,m4PN,m4nargs))') `#' End Q linethick = lineth ifelse(substr(m4Delch(`$1'),0,1),N, `Out: Q.Out', `line thick lineth from Q.Out m4dirt N_diam*L_unit; Out: Here') pushdef(`AutoOutNames',m4Path.Out) T: ifelse(m4dirt,right,`Q.w-(2*lu,0)',`Q.e+(2*lu,0)') # Sublayer Sg containing gates or vars G1, G2,... # with output vertical median at the height of Q.c Sg: [ pushdef(`m4_nct',0)dnl pushdef(`m4Path',m4Path.Sg)dnl Loopover_(`arg', `define(`m4_nct',incr(m4_nct))dnl dnl Variable or a sublayer gate dnl Inputs are labelled In_N or In_X dnl Remove initial white space; detect a name or gate pushdef(`m4Path',m4Path.G`'m4_nct)dnl m4ifboolvar_(arg, `G`'m4_nct: [dnl ifelse(substr(arg,0,1),~, `define(`m4xg',substr(arg,1))dnl pushdef(`m4InNames',m4Path.In`'m4xg`'_N)define(N_`'m4xg)dnl In`'m4xg`'_N: Here', `pushdef(`m4InNames',m4Path.In`'arg`'_X)define(X_`'arg)dnl In`'arg`'_X: Here ') Out: Here ] ht \ ifelse(m4nargs,1,`2*autovsep',`abs(Q.In1.y-Q.In2.y)-autovsep')', `pushdef(`AutoOutNames',m4Path.Out)dnl G`'m4_nct: m4Delch(arg)') ifelse(m4_nct,1,,`ifelse(m4dirt,right, \ `with .ne at last [].se',`with .nw at last [].sw')+(0,-autovsep)') popdef(`m4Path') ', ifinstr(`$2',[,`shift(shift($@))',`shift($@)')) MidOut: 0.5 between G`'eval((m4_nct+1)/2).Out and G`'eval((m4_nct+2)/2).Out popdef(`m4_nct')dnl popdef(`m4Path')dnl ] with .MidOut at T+(ifelse(m4dirt,right,-)m4nargs*lu,0) # end Sg # Draw the connecting lines define(`m4hhv',`(m4nargs-1)/2')dnl ifelse(m4dirt,right, `for_(1,m4nargs,1,`ifelse(m4x,1,,` ')line thick lineth from Q.In`'m4x \ left Q.In`'m4x.x-T.x+(m4hhv-abs(m4x-m4hhv-1))*lu \ then up Sg.G`'m4x.Out.y-Q.In`'m4x.y \ then to Sg.G`'m4x.Out')', `for_(1,m4nargs,1,`line thick lineth from Q.In`'eval(m4nargs+1-m4x) \ right T.x-Q.In`'eval(m4nargs+1-m4x).x+(m4hhv-abs(m4x-m4hhv-1))*lu \ then up Sg.G`'m4x.Out.y-Q.In`'eval(m4nargs+1-m4x).y \ then to Sg.G`'m4x.Out')') popdef(`m4nargs') m4xpand(m4dirt`'_) ]') define(`m4ifboolvar_', `ifelse(substr(`$1',0,1),_,`$3',`ifdef(`$1',`$3',`$2')')') Empty a stack while nonblank, executing arg2 define(`m4stackdump',`ifdef(`$1',`ifelse($1,,`popdef(`$1')', `$2`'popdef(`$1')m4stackdump(`$1',`$2')')')')dnl define(`DrawIn',` # Draw and label input $1 PrevInput: PrevInput-ifdef(`m4LI',`(0,autoinputsep)', `(ifelse(m4_dir_,left,-)autoinputsep,0)') In`'$1: PrevInput define(`m4dirt',m4_dir_) ifinstr(`$2',N, ` line thick lineth from PrevInput ifdef(`m4LI',m4_dir_`'_,down_) dimen_/2 linethick = gatelineth NOT_gate linethick = lineth InNt`'$1: Here', ` line thick lineth from PrevInput ifdef(`m4LI',m4_dir_,down) dimen_/4 Int`'$1: Here') m4xpand(m4dirt`'_) ') define(`DrawInNotIn',` # Draw and label input $1 inverted and uninverted. PrevInput: PrevInput-ifdef(`m4LI',`(0,autoinputsep)', `(ifelse(m4_dir_,left,-)autoinputsep*2,0)') In`'$1: PrevInput define(`m4dirt',m4_dir_) line thick lineth from PrevInput ifdef(`m4LI',m4_dir_,down) dimen_/4 ifdef(`m4LI',`PrevInput: PrevInput-(0,autoinputsep)') Int`'$1: dot line thick lineth ifdef(`m4LI',down,m4_dir_) autoinputsep \ then ifdef(`m4LI',m4_dir_`'_,down_) dimen_/4 linethick = gatelineth NOT_gate linethick = lineth InNt`'$1: Here; m4xpand(m4dirt`'_) ') ifelse(0,1,` dnl `Autologic(Boolean function, ****Obsolete dnl [N[oconnect]][L[eftinputs]][R][V][;offset=val])' ` Example: Autologic(`Nor(`And(x,y)',`And(~x,z)')') Second example, (the same function): Autologic(ZNor(ZAnd(x,y),ZAnd(~x,z))) ' define(`Autologic', `print `"*** Circuit_macros warning: Autologic is obsolete; use Autologix"' Autologix($@)') ') ############################ `Autologix(FunctionSpec;FunctionSpec;... , [M[irror]] [N[oconnect]] [L[eftinputs]] [R][V] [;offset=val], fill spec) FunctionSpec = Boolean-fn(args) [@ location attribute]' ` e.g. HalfAdder: Autologix(Xor(x,y);And(x,y),LVR) Drawing single gates is not enough; more general Boolean expressions in function notation should be drawn automatically. This macro draws one or more trees of gates with the output or outputs (roots) to the right (on the left if the M[irror] option is used). The predefined functions are And, Or, Not, Nand, Nor, Xor, Nxor, and Buffer, and may be nested; e.g., Or(And(x,~y),And(~x,y)). Function notation does not model internal connections such as feedback. However, internal nodes are labelled for later access. The exact appearance of a tree depends on the order in which terms and variables appear in the expressions. Gates can be placed relative to previously drawn objects using the @ location construct; e.g. @with .nw at last [].sw+(0,-dimen_). Autologix locates the distinct input variables (with variables preceded by ~ given NOT gates) in a row or column and connects them to the internal gates. If x1 (for example) appears as a variable, then the corresponding input location is Inx1. Inputs are placed in a row from right to left by default, in the order in which the variables first appear in the expressions. An R in the second argument reverses their order and a V reverses the order in which variables are scanned in the expresssions. An N in the second argument draws the expression tree or trees only, suppresses drawing of the inputs, and defines the gate inputs as In1, In2, ... Placing inputs in a column on the left using the L option introduces graphic complexity. Therefore, hand tuning may be required for complex expressions; putting offset=value in arg2 shifts the array of inputs. Time will tell whether this can be improved. Arbitrary subcircuits with inputs In1, In2, ... on the left and at least one output Out on the right can be included (see the example containing SRff). Internal blocks: Fx contains all the FunctionSpec trees; the input connections are drawn external to Fx Each Txi contains an input variable or a function tree Function tree i is drawn by the `AutoGate' macro. ' define(`Autologix', `[ `# Autologix' dnl Split arg1 into FunctionSpecs undefine(`m4BooleanR')stacksplit_(`m4BooleanR',`$1',;)dnl dnl ifinstr(`$2',L,`define(`m4LI')',`undefine(`m4LI')')dnl undefine(`m4InNames')undefine(`AutoOutNames')define(`m4Path',)dnl define(`m4nbf',0)undefine(`m4BooleanFn')undefine(`m4posattr')dnl dnl Separate functions from position attributes stackexec_(`m4BooleanR',, `define(`m4nbf',incr(m4nbf))define(`m4xi',regexp(m4BooleanR,[a-zA-Z_~]))dnl ifinstr(m4BooleanR,@,`define(`m4xc',index(m4BooleanR,@))dnl pushdef(`m4BooleanFn',substr(m4BooleanR,m4xi,eval(m4xc-m4xi)))dnl pushdef(`m4posattr',substr(m4BooleanR,incr(m4xc)))', `pushdef(`m4BooleanFn',substr(m4BooleanR,m4xi))dnl pushdef(`m4posattr')')dnl ')dnl pushdef(`Lg_body',`$3') dnl Sublayer of functions, outputs Out1, Out2,.. pushdef(`m4Path',ifelse(m4Path,,,m4Path.)Fx) Fx: [ define(`m4fno',0)dnl ifinstr(`$2',M,`left_ define(`m4corner',e)',`right_ define(`m4corner',w)') dnl stackexec_(`m4BooleanFn',, `define(`m4fno',incr(m4fno))dnl pushdef(`m4Path',m4Path.Tx`'m4fno)dnl dnl Simple variable or gate m4ifboolvar_(m4BooleanFn, `Tx`'m4fno: [ifelse(substr(m4BooleanFn,0,1),~, `define(`m4xg',substr(m4BooleanFn,1))dnl pushdef(`m4InNames',m4Path.In`'m4xg`'_N)define(N_`'m4xg) In`'m4xg`'_N: Here', `pushdef(`m4InNames',m4Path.In`'m4BooleanFn`'_X)dnl define(X_`'m4BooleanFn)dnl In`'m4BooleanFn`'_X: Here'); Out: Here ] ht 2*autovsep', `Tx`'m4fno: m4Delch(m4BooleanFn)') ifelse(m4posattr,, `ifelse(m4fno,1,,`with .n`'ifinstr(`$2',M,w,e) \ at Tx`'eval(m4fno-1).s`'ifinstr(`$2',M,w,e)+(0,-dimen_/4)')', m4posattr) dnl pushdef(`AutoOutNames',m4Path.Out)dnl popdef(`m4posattr') `#' End Tx`'m4fno popdef(`m4Path')dnl dnl Functions drawn dnl Out and Out1, Out2, ... pushdef(`AutoOutNames',m4Path.Out`'m4fno)dnl Out`'m4fno: Tx`'m4fno.Out ifelse(m4nbf,1,`; Out: Tx1.Out') ') ] `#' End Fx popdef(`m4Path')dnl stackreverse_(`m4InNames')dnl define(`M4TopInput',m4InNames)dnl dnl dnl Delete blank names; mark top and bottom inputs stackexec_(`m4InNames',,`ifelse(m4InNames,,, `pushdef(`m4R',m4InNames)define(`m4bn',basename_(m4R))dnl ifdef(Top`'m4bn,,`define(Top`'m4bn,m4R)')define(Bot`'m4bn,m4R)')') dnl dnl Check V option ifinstr(`$2',V, `stackexec_(`m4R',`m4TAL')stackexec_(`m4TAL',`m4InNames')', `stackexec_(`m4R',`m4InNames')') dnl dnl Count and extract bare input names undefine(`m4r')undefine(`m4f')define(`m4nargs',0)undefine(`m4N_')dnl stackexec_(`m4InNames',`m4R', `define(`m4nargs',incr(m4nargs))define(`m4bn',basename_(`m4InNames'))dnl ifelse(substr(m4bn,decr(len(m4bn))),N,`define(`m4N_')')dnl pushdef(`m4r',substr(m4bn,2,eval(len(m4bn)-4)))')dnl stackexec_(`m4R',`m4InNames')stackexec_(`m4r',`m4f') dnl dnl Unique bare input names in forward order stackexec_(`m4f',,`ifdef(D_`'m4f,,`pushdef(`m4r',m4f)define(D_`'m4f)')') stackexec_(`m4r',`m4f',`undefine(D_`'m4r)') dnl dnl Optional reverse of bare name order ifinstr(`$2',R,`stackreverse_(`m4f')') dnl Get the offset=value if any pushkey_($2,offset,0)dnl dnl Place reference for row or column of inputs PrevInput: ifdef(`m4LI',dnl `ifinstr(`$2',M, `(Fx.e.x+dimen_/2+m4nargs*2*L_unit`'ifdef(`m4N_',`+(BUF_wd+N_diam)*L_unit'),\ M4TopInput.y+autoinputsep+m4offset)', `(Fx.w.x-dimen_/2-m4nargs*2*L_unit`'ifdef(`m4N_',`-(BUF_wd+N_diam)*L_unit'),\ M4TopInput.y+autoinputsep+m4offset)')', `ifinstr(`$2',M, `Fx.ne+(-(autoinputsep/2+dimen_/4+m4offset),dimen_`'ifdef(`m4N_',,/4))', `Fx.nw+( autoinputsep/2+dimen_/4+m4offset, dimen_`'ifdef(`m4N_',,/4))')') popdef(`m4offset') dnl # Draw inputs right to left or top to bottom stackexec_(`m4f',`m4r',`ifinstr(`$2',N,,`ifdef(X_`'m4f, `ifdef(N_`'m4f,`DrawInNotIn(m4f)',`DrawIn(m4f)')',`DrawIn(m4f,N)')')') stackexec_(`m4r',,`undefine(N_`'m4r)undefine(X_`'m4r)') dnl dnl Show the internal inputs in comment lines dnl `# Internal input names' (m4nargs):dnl dnl stackexec_(`m4InNames',`m4R',` dnl `#' m4InNames') dnl stackexec_(`m4R',`m4InNames') dnl # Promote the In and Out locations to the top level `# Internal gate input labels' (m4nargs):dnl define(`m4ix',0)dnl stackexec_(`m4InNames',`m4R',`define(`m4ix',incr(m4ix)) In`'m4ix: m4InNames') stackexec_(`m4R',`m4InNames') `# Internal gate output labels (there may be duplicate locations)':dnl define(`m4ix',0)define(`m4lastout')dnl stackexec_(`AutoOutNames',, `ifelse(AutoOutNames,m4lastout,,`define(`m4ix',incr(m4ix)) Out`'m4ix: AutoOutNames define(`m4lastout',AutoOutNames)')') # Connect the gates to the inputs and clean up ifinstr(`$2',N,, `m4AutoConnex(Fx,`m4InNames',ifdef(`m4LI',`ifinstr(`$2',V,Top,Bot)',Bot))') stackexec_(`m4TAL',, `define(`m4bn',basename_(m4TAL))undefine(Bot`'m4bn)undefine(Top`'m4bn)') popdef(`Lg_body')dnl # Define the outputs ifelse(m4nbf,1,`Out: Fx.Out') for_(1,m4nbf,1,`Out`'m4x: Fx.Out`'m4x') `$4' `# End Autologix' ] ') define(`m4AutoConnex',`define(`m4cx',0)dnl define(`m4ltn',`substr(`$2',decr(len(`$2')))')dnl stackexec_(`$2',`m4TAL', `define(`m4ltn',substr($2,decr(len($2))))dnl ifelse(ifelse(m4ltn,N,T,m4ltn,X,T),T, `define(`m4bn',basename_($2))dnl define(`m4hn',substr(m4bn,2,eval(len(m4bn)-4)))dnl define(`m4cx',incr(m4cx))dnl ifdef(`m4LI', `Vx: ($1.w.x-(m4nargs-m4cx)*L_unit*2,$2.y) Tx: In`'ifelse(m4ltn,N,N)t`'m4hn line thick lineth from $2 to Vx \ then to (Vx,Tx) ifelse($2,m4xpand($3`'m4bn),` then to Tx',` dot') ', `Tx: In`'ifelse(m4ltn,N,N)t`'m4hn line thick lineth from $2 \ to (Tx,$2) ifelse($2,m4xpand($3`'m4bn),` then to Tx',` dot')') ')dnl ')') ############################ `FlipFlop( D|T|RS|JK, label, boxspec, pinlength )' define(`FlipFlop',`ifelse( `$1',D, `FlipFlopX(`$3',`$2', : D:PinD;E:CK:PinCK, , :Q:PinQ;:lg_bartxt(Q):PinNQ,,`$4',`$5')', `$1',T, `FlipFlopX(`$3',`$2', : T:PinT;E:CK:PinCK, , :Q:PinQ;:lg_bartxt(Q):PinNQ,,`$4',`$5')', `$1',RS, `FlipFlopX(`$3',`$2', : R:PinR;: S:PinS, , :Q:PinQ;:lg_bartxt(Q):PinNQ,,`$4',`$5')', `$1',SR, `FlipFlopX(`$3',`$2', : S:PinS;: R:PinR, , :Q:PinQ;:lg_bartxt(Q):PinNQ,,`$4',`$5')', `$1',JK, `FlipFlopX(`$3',`$2', : J:PinJ;NE:CK:PinCK;: K:PinK, N:CLR:PinCLR, :Q:PinQ;:lg_bartxt(Q):PinNQ, N:PR:PinPR,`$4',`$5')', `FlipFlopX(`$3',`$2', ::Pin;, , :Q:PinQ;:lg_bartxt(Q):PinNQ,,`$4',`$5')')') `FlipFlopX( boxspec, center label, leftpins, toppins, rightpins, bottompins, pinlength) General flipflop. Arg1 modifies the box (labelled Chip) default specification. Each of args 3 to 6 is null or a string of pinspecs separated by semicolons (;). A pinspec is either empty (null) or of the form [pinopts]:[label[:Picname]]. The first colon (:) draws the pin. Pins are placed top to bottom or left to right along the box edges with null pinspecs counted for placement. Pins are named by side and number by default; eg W1, W2, ..., N1, N2, ..., E1, ..., S1, ... ; however, if :Picname is present in a pinspec then Picname replaces the default name. A pinspec label is text placed at the pin base. Semicolons are not allowed in labels; use eg \char59{} instead. To put a bar over a label, use lg_bartxt(label). The pinopts are [L|M|I|O][N][E] L=active low out; M=active low in; I=inward arrow; O=outward arrow; N=pin with not circle; E=edge trigger Optional arg7 is the length of pins' define(`FlipFlopX',`[ Chip: box wid_ FF_wid*L_unit ht_ FF_ht*L_unit Lg_body `$1' dnl Center label ifelse(`$2',,,"ifsvg(`svg_small($2,75)',`\scriptsize $2')" at Chip.c) ifelse(`$7',,,`pushdef(`lg_plen',(`$7')/L_unit)')dnl ifelse(`$3',,, `m4_ffside(`(m4x-0.5)/m4_pc between Chip.nw_ and Chip.sw_',`$3',W,w)') ifelse(`$4',,, `m4_ffside(`(m4x-0.5)/m4_pc between Chip.nw_ and Chip.ne_',`$4',N,n)') ifelse(`$5',,, `m4_ffside(`(m4x-0.5)/m4_pc between Chip.ne_ and Chip.se_',`$5',E,e)') ifelse(`$6',,, `m4_ffside(`(m4x-0.5)/m4_pc between Chip.sw_ and Chip.se_',`$6',S,s)') ifelse(`$7',,,`popdef(`lg_plen')')dnl `$8' ]') dnl `Stack, count, and draw the pins on the given side' define(`m4_ffside', `stacksplit_(`m4pr',`$2',;)define(`m4_pc',0)dnl stackexec_(`m4pr',`m4pspec',`define(`m4_pc',incr(m4_pc))')dnl for_(1,m4_pc,1, `ifinstr(m4pspec,:, `m4drawpin(`$1',`$3'm4x,`$4'm4_pattocomma(m4pspec,:))')dnl popdef(`m4pspec')')')dnl define(`m4drawpin',`lg_pin(`$1',`$4',ifelse(`$5',,`$2',`$5'),`$3')') ########################################################################### divert(0)dnl