gb_gates.w
上传用户:rrhhcc
上传日期:2015-12-11
资源大小:54129k
文件大小:75k
源码类别:

通讯编程

开发平台:

Visual C++

  1. % This file is part of the Stanford GraphBase (c) Stanford University 1993
  2. @i boilerplate.w %<< legal stuff: PLEASE READ IT BEFORE MAKING ANY CHANGES!
  3. @i gb_types.w
  4. deftitle{GB_,GATES}
  5. prerequisite{GB_,GRAPH}
  6. @* Introduction. This GraphBase module provides six external subroutines:
  7. $$vbox{hsize=.8hsize everypar{hangindent3em}
  8. noindent|risc|, a routine that creates a directed acyclic graph based on the
  9.   logic of a simple RISC computer;par
  10. noindent|prod|, a routine that creates a directed acyclic graph based on the
  11.   logic of parallel multiplication circuits;par
  12. noindent|print_gates|, a routine that outputs a symbolic representation of
  13.   such directed acyclic graphs;par
  14. noindent|gate_eval|, a routine that evaluates such directed acyclic graphs by
  15.   assigning boolean values to each gate;par
  16. noindent|partial_gates|, a routine that extracts a subgraph by assigning
  17.   random values to some of the input gates;par
  18. noindent|run_risc|, a routine that can be used to play with the output
  19.   of |risc|.}$$
  20. Examples of the use of these routines can be found in the demo programs
  21. {sc TAKE_,RISC} and {sc MULTIPLY}.
  22. @(gb_gates.h@>=
  23. #define print_gates p_gates /* abbreviation for Procrustean linkers */
  24. extern Graph *risc(); /* make a network for a microprocessor */
  25. extern Graph *prod(); /* make a network for high-speed multiplication */
  26. extern void print_gates(); /* write a network to standard output file */
  27. extern long gate_eval(); /* evaluate a network */
  28. extern Graph *partial_gates(); /* reduce network size */
  29. extern long run_risc(); /* simulate the microprocessor */
  30. extern unsigned long risc_state[]; /* the output of |run_risc| */
  31. @ The directed acyclic graphs produced by {sc GB_,GATES} are GraphBase
  32. graphs with special conventions related to logical networks. Each vertex
  33. represents a gate of a network, and utility field |val| is a boolean
  34. value associated with that gate. Utility field |typ| is an ASCII code
  35. that tells what kind of gate is present:
  36. {advanceparindent 2em
  37. smallskip
  38. item{|'I'|} denotes an input gate, whose value is specified externally.
  39. smallskip
  40. item{|'&'|} denotes an .{AND} gate, whose value is the logical {sc AND} of
  41. two or more previous gates (namely, 1 if all those gates are~1, otherwise~0).
  42. smallskip
  43. item{|'|'|} denotes an .{OR} gate, whose value is the logical {sc OR} of
  44. two or more previous gates (namely, 0 if all those gates are~0, otherwise~1).
  45. smallskip
  46. item{|'^'|} denotes an .{XOR} gate, whose value is the logical {sc
  47. EXCLUSIVE-OR} of two or more previous gates (namely, their sum modulo~2).
  48. smallskip
  49. item{|'~'|} denotes an inverter, whose value is the logical complement of
  50. the value of a single previous gate.
  51. smallskip
  52. item{|'L'|} denotes a latch, whose value depends on past history; it is
  53. the value that was assigned to a subsequent gate when the network was most
  54. recently evaluated. Utility field |alt| points to that subsequent gate.
  55. smallskip}noindent
  56. Latches can be used to include ``state'' information in a circuit; for example,
  57. they correspond to registers of the RISC machine constructed by |risc|.
  58. The |prod| procedure does not use latches.
  59. The vertices of the directed acyclic graph appear in a special ``topological''
  60. order convenient for evaluation: All the input gates come first, followed
  61. by all the latches; then come the other types of gates, whose values are
  62. computed from their predecessors. The arcs of the graph run from each gate
  63. to its arguments, and all arguments to a gate precede that gate.
  64. If |g| points to such a graph of gates, the utility field |g->outs| points to
  65. a list of |Arc| records, denoting ``outputs'' that might be used in
  66. certain applications. For example, the outputs of the graphs
  67. created by |prod| correspond to the bits of the product of the numbers
  68. represented in the input gates.
  69. A special convention is used so that the routines will support partial
  70. evaluation: The |tip| fields in the output list either point to a
  71. vertex or hold one of the constant values 0 or~1 when regarded as an
  72. unsigned long integer.
  73. @d val x.I /* the field containing a boolean value */
  74. @d typ y.I /* the field containing the gate type */
  75. @d alt z.V /* the field pointing to another related gate */
  76. @d outs zz.A /* the field pointing to the list of output gates */
  77. @d is_boolean(v) ((unsigned long)(v)<=1) /* is a |tip| field constant? */
  78. @d the_boolean(v) ((long)(v)) /* if so, this is its value */
  79. @d tip_value(v) (is_boolean(v)? the_boolean(v): (v)->val)
  80. @d AND '&'
  81. @d OR '|'
  82. @d NOT '~'
  83. @d XOR '^'
  84. @(gb_gates.h@>=
  85. #define val @tquad@> x.I /* the definitions are repeated in the header file */
  86. #define typ @tquad@> y.I
  87. #define alt @tquad@> z.V
  88. #define outs @tquad@> zz.A
  89. #define is_boolean(v) @tquad@> ((unsigned long)(v)<=1)
  90. #define the_boolean(v) @tquad@> ((long)(v))
  91. #define tip_value(v) @tquad@> (is_boolean(v)? the_boolean(v): (v)->val)
  92. #define AND @tquad@> '&'
  93. #define OR @tquad@> '|'
  94. #define NOT @tquad@> '~'
  95. #define XOR @tquad@> '^'
  96. @ Let's begin with the |gate_eval| procedure, because it is quite simple
  97. and because it illustrates the conventions just explained. Given a gate
  98. graph |g| and optional pointers |in_vec| and |out_vec|, the procedure
  99. |gate_eval| will assign values to each gate of~|g|. If |in_vec| is
  100. non-null, it should point to a string of characters, each |'0'| or~|'1'|,
  101. that will be assigned to the first gates of the network, in order;
  102. otherwise |gate_eval| assumes that all input gates have already received
  103. appropriate values and it will not change them. New values are computed for
  104. each gate after the bits of |in_vec| have been consumed.
  105. If |out_vec| is non-null, it should point to a memory area capable of
  106. receiving |m+1| characters, where |m| is the number of outputs of~|g|;
  107. a string containing the respective output values will be deposited there.
  108. If |gate_eval| encounters an unknown gate type, it terminates execution
  109. prematurely and returns the value |-1|. Otherwise it returns~0.
  110. @<The |gate_eval| routine@>=
  111. long gate_eval(g,in_vec,out_vec)
  112.   Graph *g; /* graph with gates as vertices */
  113.   char *in_vec; /* string for input values, or |NULL| */
  114.   char *out_vec; /* string for output values, or |NULL| */
  115. {@+register Vertex *v; /* the current vertex of interest */
  116.   register Arc *a; /* the current arc of interest */
  117.   register char t; /* boolean value being computed */
  118.   if (!g) return -2; /* no graph supplied! */
  119.   v=g->vertices;
  120.   if (in_vec) @<Read a sequence of input values from |in_vec|@>;
  121.   for (; v<g->vertices+g->n; v++) {
  122.     switch (v->typ) { /* branch on type of gate */
  123.     case 'I': continue; /* this input gate's value should be externally set */
  124.     case 'L': t=v->alt->val;@+break;
  125.     @t44@>@<Compute the value |t| of a classical logic gate@>;
  126.     default: return -1; /* unknown gate type! */
  127.     }
  128.     v->val=t; /* assign the computed value */
  129.   }
  130.   if (out_vec) @<Store the sequence of output values in |out_vec|@>;
  131.   return 0;
  132. }
  133. @ @<Read a sequence...@>=
  134. while (*in_vec && v<g->vertices+g->n)
  135.   (v++)->val = *in_vec++ - '0';
  136. @ @<Store the sequence of output values in |out_vec|@>=
  137. {
  138.   for (a=g->outs; a; a=a->next)
  139.     *out_vec++='0'+tip_value(a->tip);
  140.   *out_vec=0; /* terminate the string */
  141. }
  142. @ @<Compute the value |t| of a classical logic gate@>=
  143. case AND: t=1;
  144.   for (a=v->arcs; a; a=a->next)
  145.     t &= a->tip->val;
  146.   break;
  147. case OR: t=0;
  148.   for (a=v->arcs; a; a=a->next)
  149.     t |= a->tip->val;
  150.   break;
  151. case XOR: t=0;
  152.   for (a=v->arcs; a; a=a->next)
  153.     t ^= a->tip->val;
  154.   break;
  155. case NOT: t=1-v->arcs->tip->val;
  156.   break;
  157. @ Here now is an outline of the entire {sc GB_,GATES} module, as seen by
  158. the CEE/ compiler:
  159. @p
  160. #include "gb_flip.h"
  161.  /* we will use the {sc GB_,FLIP} routines for random numbers */
  162. #include "gb_graph.h"
  163.  /* and we will use the {sc GB_,GRAPH} data structures */
  164. @h@#
  165. @<Private variables@>@;
  166. @<Global variables@>@;
  167. @<Internal subroutines@>@;
  168. @<The |gate_eval| routine@>@;
  169. @<The |print_gates| routine@>@;
  170. @<The |risc| routine@>@;
  171. @<The |run_risc| routine@>@;
  172. @<The |prod| routine@>@;
  173. @<The |partial_gates| routine@>@;
  174. @* The RISC netlist. The subroutine call |risc(regs)| creates a
  175. gate graph having |regs| registers; the value of |regs| must be
  176. between 2 and~16, inclusive, otherwise |regs| is set to~16.
  177. This gate graph describes the circuitry for a small RISC computer, defined
  178. below. The total number of gates turns out to be |1400+115*regs|;
  179. thus it lies between 1630 (when |regs=2|) and 3240 (when |regs=16|).
  180. {sc EXCLUSIVE-OR} gates are not used; the effect of xoring is obtained where
  181. needed by means of {sc AND}s, {sc OR}s, and inverters.
  182. If |risc| cannot do its thing, it returns |NULL| (.{NULL})
  183.  and sets |panic_code|
  184. to indicate the problem. Otherwise |risc| returns a pointer to the graph.
  185. @d panic(c) @+{@+panic_code=c;@+gb_trouble_code=0;@+return NULL;@+}
  186. @<The |risc| routine@>=
  187. Graph *risc(regs)
  188.   unsigned long regs; /* number of registers supported */
  189. {@+@<Local variables for |risc|@>@;
  190.   @#
  191.   @<Initialize |new_graph| to an empty graph of the appropriate size@>;
  192.   @<Add the RISC data to |new_graph|@>;
  193.   if (gb_trouble_code) {
  194.     gb_recycle(new_graph);
  195.     panic(alloc_fault); /* oops, we ran out of memory somewhere back there */
  196.   }
  197.   return new_graph;
  198. }
  199. @ @<Local variables for |risc|@>=
  200. Graph *new_graph; /* the graph constructed by |risc| */
  201. register long k,r; /* all-purpose indices */
  202. @ This RISC machine works with 16-bit registers and 16-bit data words.
  203. It cannot write into memory, but it assumes the existence of an
  204. external read-only memory. The circuit has 16 outputs, representing
  205. the 16 bits of a memory address register. It also has 17 inputs, the
  206. last 16 of which are supposed to be set to the contents of the memory
  207. address computed on the previous cycle. Thus we can run the machine
  208. by accessing memory between calls of |gate_eval|.  The first input
  209. bit, called .{RUN}, is normally set to~1; if it is~0, the other
  210. inputs are effectively ignored and all registers and outputs will be
  211. cleared to~0. Input bits for the memory appear in ``little-endian
  212. order,'' that is, least significant bit first; but the output bits for
  213. the memory address register appear in ``big-endian order,'' most
  214. significant bit first.
  215. Words read from memory are interpreted as instructions having the following
  216. format:
  217. $$vbox{offinterlineskip
  218.  def\#1&{omit&#1&}
  219.  hrule
  220.  halign{&vrule#&strutsevenrmhbox to 1.7em{hfil#hfil}cr
  221.  height 5pt&multispan7hfill&&multispan7hfill&&multispan3hfill
  222.   &&multispan3hfill&&multispan7hfill&cr
  223.  &multispan7hfill.{DST}hfill&&multispan7hfill.{MOD}hfill
  224.   &&multispan3hfill.{OP}hfill&&multispan3hfill.{A}hfill
  225.   &&multispan7hfill.{SRC}hfill&cr
  226.  height 5pt&multispan7hfill&&multispan7hfill&&multispan3hfill
  227.   &&multispan3hfill&&multispan7hfill&cr
  228.  noalign{hrule}
  229.  \15&\14&\13&\12&\11&\10&\9&\8&\7&\6&\5&\4&\3&\2&\1&%
  230.   \0&omitcr}}$$
  231. The .{SRC} and .A fields specify a ``source'' value.
  232. If $.A=0$, the source is .{SRC}, treated as a 16-bit signed
  233. number between $-8$ and $+7$ inclusive.
  234. If $.A=1$, the source is the contents of register .{DST} plus the
  235. (signed) value of .{SRC}. If $.A=2$, the source is the contents of register
  236. .{SRC}. And if $.A=3$, the source is the contents of the memory location
  237. whose address is the contents of register .{SRC}. Thus, for example,
  238. if $.{DST}=3$ and $.{SRC}=10$, and if .{r3} contains 17 while .{r10}
  239. contains 1009, the source value will be $-6$ if $.A=0$,
  240. or $17-6=11$ if $.A=1$, or 1009 if $.A=2$, or the contents of memory location
  241. 1009 if $.A=3$.
  242. The .{DST} field specifies the number of the destination register. This
  243. register receives a new value based on its previous value and the source
  244. value, as prescribed by the operation defined in the .{OP} and .{MOD}
  245. fields. For example, when $.{OP}=0$, a general logical operation is
  246. performed, as follows:
  247. Suppose the bits of .{MOD} are called $mu_{11}mu_{10}mu_{01}
  248. mu_{00}$ from left to right. Then if the $k$th bit of the destination register
  249. currently is equal to~$i$ and the $k$th bit of the source value is
  250. equal to~$j$, the general logical operator changes the $k$th bit of
  251. the destination register to~$mu_{ij}$. If the .{MOD} bits are,
  252. for example, $1010$, the source value is simply copied to the
  253. destination register; if $.{MOD}=0110$, an exclusive-or is done;
  254. if $.{MOD}=0011$, the destination register is complemented and the
  255. source value is effectively ignored.
  256. The machine contains four status bits called .S (sign), .N (nonzero),
  257. .K (carry), and .V (overflow). Every general logical operation sets
  258. .S equal to the sign of the new result transferred to the destination
  259. register; this is bit~15, the most significant bit. A general logical
  260. operation also sets .N to~1 if any of the other 15 bits are~1, to~0
  261. if all of the other bits are~0. Thus .S and .N both become zero if and
  262. only if the new result is entirely zero. Logical operations do not change
  263. the values of .K and~.V; the latter are affected only by the arithmetic
  264. operations described below.
  265. The status of the .S and .N bits can be tested by using the
  266. conditional load operator, $.{OP}=2$: This operation loads the source
  267. value into the destination register if and only if .{MOD} bit
  268. $mu_{ij}=1$, where $i$ and~$j$ are the current values of .S and~.N,
  269. respectively. For example, if $.{MOD}=0011$, the source value is
  270. loaded if and only if $.S=0$, which means that the last value
  271. affecting .S and~.N was greater than or equal to zero. If
  272. $.{MOD}=1111$, loading is always done; this option provides a way
  273. to move source to destination without affecting .S or~.N.
  274. A second conditional load operator, $.{OP}=3$, is similar, but
  275. it is used for testing the status of .K and~.V instead of
  276. .S and~.N. For example, a command having $.{MOD}=1010$,
  277. $.{OP}=3$, $.A=1$, and $.{SRC}=1$ adds the current overflow bit to the
  278. destination register. (Please take a moment to understand why
  279. this is true.)
  280. We have now described all the operations except those that
  281. are performed when $.{OP}=1$.
  282. As you might expect, our machine is able to do rudimentary arithmetic.
  283. The general addition and subtraction operators belong to this final case,
  284. together with various shift operators, depending on the value of .{MOD}.
  285. Eight of the $.{OP}=1$ operations set the destination register to a shifted
  286. version of the source value: $.{MOD}=0$ means ``shift left~1,''
  287. which is equivalent to multiplying the source by~2; $.{MOD}=1$ means
  288. ``cyclic shift left~1,'' which is the same except that it also adds the
  289. previous sign bit to the result; $.{MOD}=2$ means ``shift left~4,''
  290. which is equivalent to multiplying by~16; $.{MOD}=3$ means ``cyclic
  291. shift left~4''; $.{MOD}=4$ means ``shift right~1,'' which is
  292. equivalent to dividing the source by~2 and rounding down to the
  293. next lower integer if there was a remainder; $.{MOD}=5$ means
  294. ``unsigned shift right~1,'' which is the same except that the
  295. most significant bit is always set to zero instead of retaining the
  296. previous sign; $.{MOD}=6$ means ``shift right~4,'' which is equivalent
  297. to dividing the source by~16 and rounding down; $.{MOD}=7$ means
  298. ``unsigned shift right~4.'' Each of these shift operations affects
  299. .S and~.N, as in the case of logical operations. They also affect
  300. .K and~.V, as follows: Shifting left sets .K to~1 if and
  301. only if at least one of the bits shifted off the left was nonzero,
  302. and sets .V to~1 if and only if the corresponding multiplication
  303. would cause overflow.
  304. Shifting right~1 sets .K to the value of the bit
  305. shifted out, and sets .V to~0;
  306. shifting right~4 sets .K to the value of the last
  307. bit shifted out, and sets .V to the logical {sc OR} of the other three
  308. lost bits. The same values of .K and .V arise from cyclic or unsigned
  309. shifts as from ordinary shifts.
  310. When $.{OP}=1$ and $.{MOD}=8$, the source value is added to the
  311. destination register. This sets .S, .N, and .V as you would expect;
  312. and it sets .K to the carry you would get if you were treating the operands
  313. as 16-bit unsigned integers. Another addition operation, having
  314. $.{MOD}=9$, is similar, but the current value of .K is also added to
  315. the result; in this case, the new value of .N will be zero if and only if
  316. the 15 non-sign bits of the result are zero and the previous values of
  317. .S and~.N were also zero. This means
  318. that you can use the first addition operation on the lower
  319. halves of a 32-bit number and the second operation on the upper halves,
  320. thereby obtaining a correct 32-bit result, with appropriate sign,
  321. nonzero, carry, and overflow bits set.
  322. Higher precision (48 bits, 64 bits, etc.)~can be obtained in a similar way.
  323. When $.{OP}=1$ and $.{MOD}=10$, the source value is subtracted
  324. from the destination register. Again, .S, .N, .K, and .V are set;
  325. the .K value in this case represents the ``borrow'' bit.
  326. An auxiliary subtraction operation, having $.{MOD}=11$, subtracts
  327. also the current value of .K, thereby allowing for correct 32-bit subtraction.
  328. The operations for $.{OP}=1$ and $.{MOD}=12$, 13, and~14 are
  329. ``reserved for future expansion.'' Actually they will never change, however,
  330. since this RISC chip is purely academic. If you check out the logic
  331. below, you will find that they simply set the destination register and
  332. the four status bits all to zero.
  333. A final operation, called .{JUMP}, will be explained momentarily.
  334. It has $.{OP}=1$ and $.{MOD}=15$. It does not affect .S, .N, .K, or~.V.
  335. If the RISC is made with fewer than 16 registers, the higher-numbered ones
  336. will effectively contain zero whenever their values are fetched.
  337. But if you use them as destination registers, you will set
  338. .S, .N, .K, and~.V as if actual numbers were being stored.
  339. Register 0 is different from the other 15 registers: It is the location
  340. of the current instruction. Therefore if you change the contents of
  341. register~0, you are changing the control flow of the program. If you
  342. do not change register~0, it automatically increases by~1.
  343. Special treatment occurs when $.A=3$ and $.{SRC}=0$.
  344. In such a case, the normal rules given above say that the source value
  345. should be the contents of the memory location specified by register~0. But
  346. that memory location holds the current instruction; so the machine
  347. uses the {sl following/} location instead, as a 16-bit source
  348. operand. If the contents of register~0 are not changed by such a
  349. two-word instruction, register~0 will increase by~2 instead of~1.
  350. We have now discussed everything about the machine except the operation
  351. of the .{JUMP} command. This command moves the source value to
  352. register~0, thereby changing the flow of control. Furthermore, if
  353. $.{DST}ne0$, it also sets register .{DST} to the location of the
  354. instruction following the .{JUMP}. Assembly language programmers will
  355. recognize this as a convenient way to jump to a subroutine.
  356. Example programs can be found in the {sc TAKE_,RISC} module, which includes
  357. a simple subroutine for multiplication and division.
  358. @ A few auxiliary functions will ameliorate the task of constructing
  359. the RISC logic. First comes a routine that ``christens'' a new gate,
  360. assigning it a name and a type. The name is constructed from a prefix
  361. and a serial number, where the prefix indicates the current portion of
  362. logic being created.
  363. @<Internal...@>=
  364. static Vertex* new_vert(t)
  365.   char t; /* the type of the new gate */
  366. {@+register Vertex *v;
  367.   v=next_vert++;
  368.   if (count<0) v->name=gb_save_string(prefix);
  369.   else {
  370.     sprintf(name_buf,"%s%ld",prefix,count);
  371.     v->name=gb_save_string(name_buf);
  372.     count++;
  373.   }
  374.   v->typ=t;
  375.   return v;
  376. }
  377. @ @d start_prefix(s) strcpy(prefix,s);@+count=0
  378. @d numeric_prefix(a,b) sprintf(prefix,"%c%ld:",a,b);@+count=0;
  379. @<Private...@>=
  380. static Vertex* next_vert; /* the first vertex not yet assigned a name */
  381. static char prefix[5]; /* prefix string for vertex names */
  382. static long count; /* serial number for vertex names */
  383. static char name_buf[100]; /* place to form vertex names */
  384. @ Here are some trivial routines to create gates with 2, 3, or more
  385. arguments. The arcs from such a gate to its inputs are assigned length 100.
  386. Other routines, defined below,
  387. assign length~1 to the arc between an inverter and its unique
  388. input. This convention makes the lengths of shortest paths in the resulting
  389. network a bit more interesting than they would otherwise be.
  390. @d DELAY 100L
  391. @<Internal...@>=
  392. static Vertex* make2(t,v1,v2)
  393.   char t; /* the type of the new gate */
  394.   Vertex *v1,*v2;
  395. {@+register Vertex *v=new_vert(t);
  396.   gb_new_arc(v,v1,DELAY);
  397.   gb_new_arc(v,v2,DELAY);
  398.   return v;
  399. }
  400. @#
  401. static Vertex* make3(t,v1,v2,v3)
  402.   char t; /* the type of the new gate */
  403.   Vertex *v1,*v2,*v3;
  404. {@+register Vertex *v=new_vert(t);
  405.   gb_new_arc(v,v1,DELAY);
  406.   gb_new_arc(v,v2,DELAY);
  407.   gb_new_arc(v,v3,DELAY);
  408.   return v;
  409. }
  410. @#
  411. static Vertex* make4(t,v1,v2,v3,v4)
  412.   char t; /* the type of the new gate */
  413.   Vertex *v1,*v2,*v3,*v4;
  414. {@+register Vertex *v=new_vert(t);
  415.   gb_new_arc(v,v1,DELAY);
  416.   gb_new_arc(v,v2,DELAY);
  417.   gb_new_arc(v,v3,DELAY);
  418.   gb_new_arc(v,v4,DELAY);
  419.   return v;
  420. }
  421. @#
  422. static Vertex* make5(t,v1,v2,v3,v4,v5)
  423.   char t; /* the type of the new gate */
  424.   Vertex *v1,*v2,*v3,*v4,*v5;
  425. {@+register Vertex *v=new_vert(t);
  426.   gb_new_arc(v,v1,DELAY);
  427.   gb_new_arc(v,v2,DELAY);
  428.   gb_new_arc(v,v3,DELAY);
  429.   gb_new_arc(v,v4,DELAY);
  430.   gb_new_arc(v,v5,DELAY);
  431.   return v;
  432. }
  433. @ We use utility field |w.V| to store a pointer to the complement
  434. of a gate, if that complement has been formed. This trick prevents the creation
  435. of excessive gates that are equivalent to each other. The following subroutine
  436. returns a pointer to the complement of a given gate.
  437. @d bar w.V /* field pointing to complement, if known to exist */
  438. @d even_comp(s,v) ((s)&1? v: comp(v))
  439. @<Internal...@>=
  440. static Vertex* comp(v)
  441.   Vertex *v;
  442. {@+register Vertex *u;
  443.   if (v->bar) return v->bar;
  444.   u=next_vert++;
  445.   u->bar=v;@+v->bar=u;
  446.   sprintf(name_buf,"%s~",v->name);
  447.   u->name=gb_save_string(name_buf);
  448.   u->typ=NOT;
  449.   gb_new_arc(u,v,1L);
  450.   return u;
  451. }
  452. @ To create a gate for the {sc EXCLUSIVE-OR} of two arguments, we can
  453. either construct the {sc OR} of two {sc AND}s or the {sc AND} of two
  454. {sc OR}s. We choose the former alternative:
  455. @<Internal...@>=
  456. static Vertex* make_xor(u,v)
  457.   Vertex *u,*v;
  458. {@+register Vertex *t1,*t2;
  459.   t1=make2(AND,u,comp(v));
  460.   t2=make2(AND,comp(u),v);
  461.   return make2(OR,t1,t2);
  462. }
  463. @ OK, let's get going.
  464. @<Initialize |new_graph|...@>=
  465. if (regs<2 || regs>16) regs=16;
  466. new_graph=gb_new_graph(1400+115*regs);
  467. if (new_graph==NULL)
  468.   panic(no_room); /* out of memory before we're even started */
  469. sprintf(new_graph->id,"risc(%lu)",regs);
  470. strcpy(new_graph->util_types,"ZZZIIVZZZZZZZA");
  471. next_vert=new_graph->vertices;
  472. @ @<Add the RISC data to |new_graph|@>=
  473. @<Create the inputs and latches@>;
  474. @<Create gates for instruction decoding@>;
  475. @<Create gates for fetching the source value@>;
  476. @<Create gates for the general logic operation@>;
  477. @<Create gates for the conditional load operations@>;
  478. @<Create gates for the arithmetic operations@>;
  479. @<Create gates that bring everything together properly@>;
  480. if (next_vert!=new_graph->vertices+new_graph->n)
  481.   panic(impossible); /* oops, we miscounted; this should be impossible */
  482. @ Internal names will make it convenient to refer to the most important
  483. gates. Here are the names of inputs and latches.
  484. @<Local variables for |risc|@>=
  485. Vertex *run_bit; /* the .{RUN} input */
  486. Vertex *mem[16]; /* 16 bits of input from read-only memory */
  487. Vertex *prog; /* first of 10 bits in the program register */
  488. Vertex *sign; /* the latched value of .S */
  489. Vertex *nonzero; /* the latched value of .N */
  490. Vertex *carry; /* the latched value of .K */
  491. Vertex *overflow; /* the latched value of .V */
  492. Vertex *extra; /* latched status bit: are we doing an extra memory cycle? */
  493. Vertex *reg[16]; /* the least-significant bit of a given register */
  494. @ @d first_of(n,t) new_vert(t);@+for (k=1;k<n;k++)@+new_vert(t);
  495. @<Create the inputs and latches@>=
  496. strcpy(prefix,"RUN");@+count=-1;@+run_bit=new_vert('I');
  497. start_prefix("M");@+for (k=0;k<16;k++)@+mem[k]=new_vert('I');
  498. start_prefix("P");@+prog=first_of(10,'L');
  499. strcpy(prefix,"S");@+count=-1;@+sign=new_vert('L');
  500. strcpy(prefix,"N");@+nonzero=new_vert('L');
  501. strcpy(prefix,"K");@+carry=new_vert('L');
  502. strcpy(prefix,"V");@+overflow=new_vert('L');
  503. strcpy(prefix,"X");@+extra=new_vert('L');
  504. for (r=0;r<regs;r++) {
  505.   numeric_prefix('R',r);
  506.   reg[r]=first_of(16,'L');
  507. }
  508. @ The order of evaluation of function arguments is not defined in CEE/,
  509. so we introduce a few macros that force left-to-right order.
  510. @d do2(result,t,v1,v2)
  511.      {@+t1=v1;@+t2=v2;
  512.      result=make2(t,t1,t2);@+}
  513. @d do3(result,t,v1,v2,v3)
  514.      {@+t1=v1;@+t2=v2;@+t3=v3;
  515.      result=make3(t,t1,t2,t3);@+}
  516. @d do4(result,t,v1,v2,v3,v4)
  517.      {@+t1=v1;@+t2=v2;@+t3=v3;@+t4=v4;
  518.      result=make4(t,t1,t2,t3,t4);@+}
  519. @d do5(result,t,v1,v2,v3,v4,v5)
  520.      {@+t1=v1;@+t2=v2;@+t3=v3;@+t4=v4;@+t5=v5;
  521.      result=make5(t,t1,t2,t3,t4,t5);@+}
  522. @<Local variables for |risc|@>=
  523. Vertex *t1,*t2,*t3,*t4,*t5; /* temporary holds to force evaluation order */
  524. Vertex *tmp[16]; /* additional holding places for partial results */
  525. Vertex *imm; /* is the source value immediate (a given constant)? */
  526. Vertex *rel; /* is the source value relative to the
  527.                   current destination register? */
  528. Vertex *dir; /* should the source value be fetched directly
  529.                   from a source register? */
  530. Vertex *ind; /* should the source value be fetched indirectly from memory? */
  531. Vertex *op; /* least significant bit of .{OP} */
  532. Vertex *cond; /* most significant bit of .{OP} */
  533. Vertex *mod[4]; /* the .{MOD} bits */
  534. Vertex *dest[4]; /* the .{DEST} bits */
  535. @ The sixth line of the program here can be translated into the logic
  536. equation
  537. $$ |op|=(|extra|land|prog|)lor(mskip1muoverline{|extra|}land|mem[6]|),.$$
  538. Once you see why, you'll be able to read the rest of this curious code.
  539. @<Create gates for instruction decoding@>=
  540. start_prefix("D");
  541. do3(imm,AND,comp(extra),comp(mem[4]),comp(mem[5])); /* $.A=0$ */
  542. do3(rel,AND,comp(extra),mem[4],comp(mem[5])); /* $.A=1$ */
  543. do3(dir,AND,comp(extra),comp(mem[4]),mem[5]); /* $.A=2$ */
  544. do3(ind,AND,comp(extra),mem[4],mem[5]); /* $.A=3$ */
  545. do2(op,OR,make2(AND,extra,prog),make2(AND,comp(extra),mem[6]));
  546. do2(cond,OR,make2(AND,extra,prog+1),make2(AND,comp(extra),mem[7]));
  547. for (k=0;k<4;k++) {
  548.   do2(mod[k],OR,make2(AND,extra,prog+2+k),make2(AND,comp(extra),mem[8+k]));
  549.   do2(dest[k],OR,make2(AND,extra,prog+6+k),make2(AND,comp(extra),mem[12+k]));
  550. }
  551. @ @<Create gates for fetching the source value@>=
  552. start_prefix("F");
  553. @<Set |old_dest| to the present value of the destination register@>;
  554. @<Set |old_src| to the present value of the source register@>;
  555. @<Set |inc_dest| to |old_dest| plus .{SRC}@>;
  556. for (k=0;k<16;k++)@/
  557.   do4(source[k],OR,
  558.     make2(AND,imm,mem[k<4?k:3]),
  559.     make2(AND,rel,inc_dest[k]),@|
  560.     make2(AND,dir,old_src[k]),
  561.     make2(AND,extra,mem[k]));
  562. @ Here and in the immediately following section we create {sc OR}
  563. gates |old_dest[k]| and |old_src[k]| that might have as many as
  564. 16~inputs. (The actual number of inputs is |regs|.) All other
  565. gates in the network will have at most five inputs.
  566. @<Set |old_dest| to the present value of the destination register@>=
  567. for (r=0;r<regs;r++) @/
  568.   do4(dest_match[r],AND,even_comp(r,dest[0]),even_comp(r>>1,dest[1]),@|
  569.                           even_comp(r>>2,dest[2]),even_comp(r>>3,dest[3]));
  570. for (k=0;k<16;k++) {
  571.   for (r=0;r<regs;r++)@/
  572.     tmp[r]=make2(AND,dest_match[r],reg[r]+k);
  573.   old_dest[k]=new_vert(OR);
  574.   for (r=0;r<regs;r++) gb_new_arc(old_dest[k],tmp[r],DELAY);
  575. }
  576. @ @<Set |old_src| to the present value of the source register@>=
  577. for (k=0;k<16;k++) {
  578.   for (r=0;r<regs;r++)@/
  579.     do5(tmp[r],AND,reg[r]+k,even_comp(r,mem[0]),even_comp(r>>1,mem[1]),
  580.                             even_comp(r>>2,mem[2]),even_comp(r>>3,mem[3]));
  581.   old_src[k]=new_vert(OR);
  582.   for (r=0;r<regs;r++) gb_new_arc(old_src[k],tmp[r],DELAY);
  583. }
  584. @ @<Local variables for |risc|@>=
  585. Vertex *dest_match[16]; /* |dest_match[r]==1| iff $.{DST}=r$ */
  586. Vertex *old_dest[16]; /* contents of destination register before operation */
  587. Vertex *old_src[16]; /* contents of source register before operation */
  588. Vertex *inc_dest[16]; /* |old_dest| plus the .{SRC} field */
  589. Vertex *source[16]; /* source value for the operation */
  590. Vertex *log[16]; /* result of general logic operation */
  591. Vertex *shift[18]; /* result of shift operation, with carry and overflow */
  592. Vertex *sum[18]; /* |old_dest| plus |source| plus optional carry */
  593. Vertex *diff[18]; /* |old_dest| minus |source| minus optional borrow */
  594. Vertex *next_loc[16]; /* contents of register 0, plus 1 */
  595. Vertex *next_next_loc[16]; /* contents of register 0, plus 2 */
  596. Vertex *result[18]; /* result of operating on |old_dest| and |source| */
  597. @ @<Create gates for the general logic operation@>=
  598. start_prefix("L");
  599. for (k=0;k<16;k++)@/
  600.   do4(log[k],OR,@|
  601.     make3(AND,mod[0],comp(old_dest[k]),comp(source[k])),@|
  602.     make3(AND,mod[1],comp(old_dest[k]),source[k]),@|
  603.     make3(AND,mod[2],old_dest[k],comp(source[k])),@|
  604.     make3(AND,mod[3],old_dest[k],source[k]));
  605. @ @<Create gates for the conditional load operations@>=
  606. start_prefix("C");
  607. do4(tmp[0],OR,@|
  608.   make3(AND,mod[0],comp(sign),comp(nonzero)),@|
  609.   make3(AND,mod[1],comp(sign),nonzero),@|
  610.   make3(AND,mod[2],sign,comp(nonzero)),@|
  611.   make3(AND,mod[3],sign,nonzero));
  612. do4(tmp[1],OR,@|
  613.   make3(AND,mod[0],comp(carry),comp(overflow)),@|
  614.   make3(AND,mod[1],comp(carry),overflow),@|
  615.   make3(AND,mod[2],carry,comp(overflow)),@|
  616.   make3(AND,mod[3],carry,overflow));
  617. do3(change,OR,comp(cond),make2(AND,tmp[0],comp(op)),make2(AND,tmp[1],op));
  618. @ @<Local variables for |risc|@>=
  619. Vertex *change; /* is the destination register supposed to change? */
  620. @ Hardware is like software except that it performs all the operations
  621. all the time and then selects only the results it needs. (If you think about
  622. it, this is a profound observation about economics, society, and nature.
  623. Gosh.)
  624. @<Create gates that bring everything together properly@>=
  625. start_prefix("Z");
  626. @<Create gates for the |next_loc| and |next_next_loc| bits@>;
  627. @<Create gates for the |result| bits@>;
  628. @<Create gates for the new values of registers 1 to |regs|@>;
  629. @<Create gates for the new values of .S, .N, .K, and .V@>;
  630. @<Create gates for the new values of the program register and |extra|@>;
  631. @<Create gates for the new values of register 0
  632.      and the memory address register@>;
  633. @ @<Create gates for the |next_loc|...@>=
  634. next_loc[0]=comp(reg[0]);@+next_next_loc[0]=reg[0];
  635. next_loc[1]=make_xor(reg[0]+1,reg[0]);@+next_next_loc[1]=comp(reg[0]+1);
  636. for (t5=reg[0]+1,k=2;k<16;t5=make2(AND,t5,reg[0]+k++)) {
  637.   next_loc[k]=make_xor(reg[0]+k,make2(AND,reg[0],t5));
  638.   next_next_loc[k]=make_xor(reg[0]+k,t5);
  639. }
  640. @ @<Create gates for the |result| bits@>=
  641. jump=make5(AND,op,mod[0],mod[1],mod[2],mod[3]); /* assume |cond=0| */
  642. for (k=0;k<16;k++) {
  643.   do5(result[k],OR,@|
  644.     make2(AND,comp(op),log[k]),@|
  645.     make2(AND,jump,next_loc[k]),@|
  646.     make3(AND,op,comp(mod[3]),shift[k]),@|
  647.     make5(AND,op,mod[3],comp(mod[2]),comp(mod[1]),sum[k]),@|
  648.     make5(AND,op,mod[3],comp(mod[2]),mod[1],diff[k]));
  649.   do2(result[k],OR,@|
  650.     make3(AND,cond,change,source[k]),@|
  651.     make2(AND,comp(cond),result[k]));
  652. }
  653. for (k=16;k<18;k++) /* carry and overflow bits of the result */
  654.   do3(result[k],OR,@|
  655.     make3(AND,op,comp(mod[3]),shift[k]),@|
  656.     make5(AND,op,mod[3],comp(mod[2]),comp(mod[1]),sum[k]),@|
  657.     make5(AND,op,mod[3],comp(mod[2]),mod[1],diff[k]));
  658. @ The program register |prog| and the |extra| bit are needed for
  659. the case when we must spend an extra cycle to fetch a word from memory.
  660. On the first cycle, |ind| is true, so a ``result'' is calculated but not
  661. actually used. On the second cycle, |extra| is true.
  662. A slight optimization has been introduced in order to make the circuit
  663. a bit more interesting: If a conditional load instruction occurs with
  664. indirect addressing and a false condition, the extra cycle is not taken.
  665. (The |next_next_loc| values were computed for this reason.)
  666. @d latchit(u,@!latch)
  667.   (latch)->alt=make2(AND,u,run_bit) /* |u&run_bit| is new value for |latch| */
  668. @<Create gates for the new values of the program reg...@>=
  669. for (k=0;k<10;k++)
  670.   latchit(mem[k+6],prog+k);
  671. do2(nextra,OR,make2(AND,ind,comp(cond)),make2(AND,ind,change));
  672. latchit(nextra,extra);
  673. nzs=make4(OR,mem[0],mem[1],mem[2],mem[3]);
  674. nzd=make4(OR,dest[0],dest[1],dest[2],dest[3]);
  675. @ @<Local variables for |risc|@>=
  676. Vertex *jump; /* is this command a .{JUMP}, assuming |cond| is false? */
  677. Vertex *nextra; /* must we take an extra cycle? */
  678. Vertex *nzs; /* is the .{SRC} field nonzero? */
  679. Vertex *nzd; /* is the .{DST} field nonzero? */
  680. @ @<Create gates for the new values of registers 1 to |regs|@>=
  681. t5=make2(AND,change,comp(ind)); /* should destination register change? */
  682. for (r=1;r<regs;r++) {
  683.   t4=make2(AND,t5,dest_match[r]); /* should register |r| change? */
  684.   for (k=0;k<16;k++) {
  685.     do2(t3,OR,make2(AND,t4,result[k]),make2(AND,comp(t4),reg[r]+k));
  686.     latchit(t3,reg[r]+k);
  687.   }
  688. }
  689. @ @<Create gates for the new values of .S, .N, .K, and .V@>=
  690. do4(t5,OR,@|
  691.   make2(AND,sign,cond),@|
  692.   make2(AND,sign,jump),@|
  693.   make2(AND,sign,ind),@|
  694.   make4(AND,result[15],comp(cond),comp(jump),comp(ind)));
  695. latchit(t5,sign);
  696. do4(t5,OR,@|
  697.   make4(OR,result[0],result[1],result[2],result[3]),@|
  698.   make4(OR,result[4],result[5],result[6],result[7]),@|
  699.   make4(OR,result[8],result[9],result[10],result[11]),@|
  700.   make4(OR,result[12],result[13],result[14],@|
  701. @thskip5em@>make5(AND,make2(OR,nonzero,sign),op,mod[0],comp(mod[2]),mod[3])));
  702. do4(t5,OR,@|
  703.   make2(AND,nonzero,cond),@|
  704.   make2(AND,nonzero,jump),@|
  705.   make2(AND,nonzero,ind),@|
  706.   make4(AND,t5,comp(cond),comp(jump),comp(ind)));
  707. latchit(t5,nonzero);
  708. do5(t5,OR,@|
  709.   make2(AND,overflow,cond),@|
  710.   make2(AND,overflow,jump),@|
  711.   make2(AND,overflow,comp(op)),@|
  712.   make2(AND,overflow,ind),@|
  713.   make5(AND,result[17],comp(cond),comp(jump),comp(ind),op));
  714. latchit(t5,overflow);
  715. do5(t5,OR,@|
  716.   make2(AND,carry,cond),@|
  717.   make2(AND,carry,jump),@|
  718.   make2(AND,carry,comp(op)),@|
  719.   make2(AND,carry,ind),@|
  720.   make5(AND,result[16],comp(cond),comp(jump),comp(ind),op));
  721. latchit(t5,carry);
  722. @ As usual, we have left the hardest case for last, hoping that we will
  723. have learned enough tricks to handle it when the time of reckoning
  724. finally arrives.
  725. The most subtle part of the logic here is perhaps the case of a
  726. .{JUMP} command with $.A=3$. We want to increase register~0 by~1
  727. during the first cycle of
  728. such a command, if $.{SRC}=0$, so that the |result| will be
  729. correct on the next cycle.
  730. @<Create gates for the new values of register 0...@>=
  731. skip=make2(AND,cond,comp(change)); /* false conditional? */
  732. hop=make2(AND,comp(cond),jump); /* .{JUMP} command? */
  733. do4(normal,OR,@|
  734.   make2(AND,skip,comp(ind)),@|
  735.   make2(AND,skip,nzs),@|
  736.   make3(AND,comp(skip),ind,comp(nzs)),@|
  737.   make3(AND,comp(skip),comp(hop),nzd));
  738. special=make3(AND,comp(skip),ind,nzs);
  739. for (k=0;k<16;k++) {
  740.   do4(t5,OR,@|
  741.     make2(AND,normal,next_loc[k]),@|
  742.     make4(AND,skip,ind,comp(nzs),next_next_loc[k]),@|
  743.     make3(AND,hop,comp(ind),source[k]),@|
  744.     make5(AND,comp(skip),comp(hop),comp(ind),comp(nzd),result[k]));
  745.   do2(t4,OR,@|
  746.     make2(AND,special,reg[0]+k),@|
  747.     make2(AND,comp(special),t5));
  748.   latchit(t4,reg[0]+k);
  749.   do2(t4,OR,@|
  750.     make2(AND,special,old_src[k]),@|
  751.     make2(AND,comp(special),t5));
  752.   {@+register Arc *a=gb_virgin_arc();
  753.     a->tip=make2(AND,t4,run_bit);
  754.     a->next=new_graph->outs;
  755.     new_graph->outs=a; /* pointer to memory address bit */
  756.   }
  757. }  /* arcs for output bits will appear in big-endian order */
  758. @ @<Local variables for |risc|@>=
  759. Vertex *skip; /* are we skipping a conditional load operation? */
  760. Vertex *hop; /* are we doing a .{JUMP}? */
  761. Vertex *normal; /* is this a case where register 0 is simply incremented? */
  762. Vertex *special; /* is this a case where register 0 and the memory address
  763.   register will not coincide? */
  764. @* Serial addition. We haven't yet specified the parts of |risc| that
  765. deal with addition and subtraction; somehow, those parts wanted to
  766. be separate from the rest. To complete our mission, we will use
  767. subroutine calls of the form `|make_adder(n,x,y,z,carry,add)|',
  768. where |x| and |y| are |n|-bit arrays of input gates and
  769. |z|~is an |(n+1)|-bit array of output gates. If |add!=0|, the subroutine
  770. computes |x+y|, otherwise it computes |x-y|. If |carry!=0|, the |carry| gate
  771. is effectively added to~|y| before the operation.
  772. A simple |n|-stage serial scheme, which reduces the problem of |n|-bit
  773. addition to |(n-1)|-bit addition, is adequate for our purposes here.
  774. (A parallel adder, which gains efficiency by reducing the problem size
  775. from |n| to~$n/phi$, can be found in the |prod| routine below.)
  776. The handy identity $x-y=overline{overline x+y}$ is used to reduce
  777. subtraction to addition.
  778. @<Internal...@>=
  779. static void make_adder(n,x,y,z,carry,add)
  780.   unsigned long n; /* number of bits */
  781.   Vertex *x[],*y[]; /* input gates */
  782.   Vertex *z[]; /* output gates */
  783.   Vertex *carry; /* add this to |y|, unless it's null */
  784.   char add; /* should we add or subtract? */
  785. {@+register long k;
  786.   Vertex *t1,*t2,*t3,*t4; /* temporary storage used by |do4| */
  787.   if (!carry) {
  788.     z[0]=make_xor(x[0],y[0]);
  789.     carry=make2(AND,even_comp(add,x[0]),y[0]);
  790.     k=1;
  791.   }@+else k=0;
  792.   for (;k<n;k++) {
  793.     comp(x[k]);@+comp(y[k]);@+comp(carry); /* generate inverse gates */
  794.     do4(z[k],OR,@|
  795.       make3(AND,x[k],comp(y[k]),comp(carry)),@|
  796.       make3(AND,comp(x[k]),y[k],comp(carry)),@|
  797.       make3(AND,comp(x[k]),comp(y[k]),carry),@|
  798.       make3(AND,x[k],y[k],carry));
  799.     do3(carry,OR,@|
  800.       make2(AND,even_comp(add,x[k]),y[k]),@|
  801.       make2(AND,even_comp(add,x[k]),carry),@|
  802.       make2(AND,y[k],carry));
  803.   }
  804.   z[n]=carry;
  805. }
  806. @ OK, now we can add. What good does that do us?
  807. In the first place, we need a 4-bit adder to compute the least
  808. significant bits of $|old_dest|+.{SRC}$. The other 12 bits of that
  809. sum are simpler.
  810. @<Set |inc_dest| to |old_dest| plus .{SRC}@>=
  811. make_adder(4L,old_dest,mem,inc_dest,NULL,1);
  812. up=make2(AND,inc_dest[4],comp(mem[3])); /* remaining bits must increase */
  813. down=make2(AND,comp(inc_dest[4]),mem[3]); /* remaining bits must decrease */
  814. for (k=4;;k++) {
  815.   comp(up);@+comp(down);
  816.   do3(inc_dest[k],OR,@|
  817.     make2(AND,comp(old_dest[k]),up),@|
  818.     make2(AND,comp(old_dest[k]),down),@|
  819.     make3(AND,old_dest[k],comp(up),comp(down)));
  820.   if (k<15) {
  821.     up=make2(AND,up,old_dest[k]);
  822.     down=make2(AND,down,comp(old_dest[k]));
  823.   }@+else break;
  824. }
  825. @ @<Local variables for |risc|@>=
  826. Vertex *up,*down; /* gates used when computing |inc_dest| */
  827. @ In the second place, we need a 16-bit adder and a 16-bit subtracter
  828. for the four addition/subtraction commands.
  829. @<Create gates for the arithmetic operations@>=
  830. start_prefix("A");
  831. @<Create gates for the shift operations@>;
  832. make_adder(16L,old_dest,source,sum,make2(AND,carry,mod[0]),1); /* adder */
  833. make_adder(16L,old_dest,source,diff,make2(AND,carry,mod[0]),0); /* subtracter */
  834. do2(sum[17],OR,@|
  835.   make3(AND,old_dest[15],source[15],comp(sum[15])),@|
  836.   make3(AND,comp(old_dest[15]),comp(source[15]),sum[15])); /* overflow */
  837. do2(diff[17],OR,@|
  838.   make3(AND,old_dest[15],comp(source[15]),comp(diff[15])),@|
  839.   make3(AND,comp(old_dest[15]),source[15],diff[15])); /* overflow */
  840. @ @<Create gates for the shift operations@>=
  841. for (k=0;k<16;k++)@/
  842.   do4(shift[k],OR,@|
  843.     (k==0? make4(AND,source[15],mod[0],comp(mod[1]),comp(mod[2])):@|
  844.      @thskip5em@>make3(AND,source[k-1],comp(mod[1]),comp(mod[2]))),@|
  845.     (k<4? make4(AND,source[k+12],mod[0],mod[1],comp(mod[2])):@|
  846.      @thskip5em@>make3(AND,source[k-4],mod[1],comp(mod[2]))),@|
  847.     (k==15? make4(AND,source[15],comp(mod[0]),comp(mod[1]),mod[2]):@|
  848.      @thskip5em@>make3(AND,source[k+1],comp(mod[1]),mod[2])),@|
  849.     (k>11? make4(AND,source[15],comp(mod[0]),mod[1],mod[2]):@|
  850.      @thskip5em@>make3(AND,source[k+4],mod[1],mod[2])));
  851. do4(shift[16],OR,@|
  852.   make2(AND,comp(mod[2]),source[15]),@|
  853.   make3(AND,comp(mod[2]),mod[1],
  854.     make3(OR,source[14],source[13],source[12])),@|
  855.   make3(AND,mod[2],comp(mod[1]),source[0]),@|
  856.   make3(AND,mod[2],mod[1],source[3])); /* ``carry'' */
  857. do3(shift[17],OR,@|
  858.   make3(AND,comp(mod[2]),comp(mod[1]),
  859.    make_xor(source[15],source[14])),@|
  860.   make4(AND,comp(mod[2]),mod[1],@|
  861.    @thskip5em@>make5(OR,source[15],source[14],
  862.       source[13],source[12],source[11]),@|
  863.    @thskip5em@>make5(OR,comp(source[15]),comp(source[14]),
  864.       comp(source[13]),@|
  865.     @thskip10em@>comp(source[12]),comp(source[11]))),@|
  866.   make3(AND,mod[2],mod[1],
  867.       make3(OR,source[0],source[1],source[2]))); /* ``overflow'' */
  868. @* RISC management. The |run_risc| procedure takes a gate graph output by
  869. |risc| and simulates its behavior, given the contents of its read-only memory.
  870. (See the demonstration program {sc TAKE_,RISC}, which appears in a module
  871. by itself, for a typical illustration of how |run_risc| might be used.)
  872. This procedure clears the simulated machine and begins executing the program
  873. that starts at address~0. It stops when it gets to an address greater
  874. than the size of read-only memory supplied. One way to stop it
  875. is therefore to execute a command such as |0x0f00|, which will transfer
  876. control to location |0xffff|; even better is |0x0f8f|, which transfers
  877. to location |0xffff| without changing the status of .S and .N.
  878. However, if the given read-only memory
  879. contains a full set of $2^{16}$ words, |run_risc| will never stop.
  880. When |run_risc| does stop, it returns 0 and puts the final contents of the
  881. simulated registers into the global array |risc_state|.
  882. Or, if |g| was not a decent graph, |run_risc| returns a negative value and
  883. leaves |risc_state| untouched.
  884. @<The |run_risc|...@>=
  885. long run_risc(g,rom,size,trace_regs)
  886.   Graph *g; /* graph output by |risc| */
  887.   unsigned long rom[]; /* contents of read-only memory */
  888.   unsigned long size; /* length of |rom| vector */
  889.   unsigned long trace_regs; /* if nonzero, this many registers will be traced */
  890. {@+register unsigned long l; /* memory address */
  891.   register unsigned long m; /* memory or register contents */
  892.   register Vertex *v; /* the current gate of interest */
  893.   register Arc *a; /* the current output list element of interest */
  894.   register long k,r; /* general-purpose indices */
  895.   long x,s,n,c,o; /* status bits */
  896.   if (trace_regs) @<Print a headline@>;
  897.   r=gate_eval(g,"0",NULL); /* reset the RISC by turning off the .{RUN} bit */
  898.   if (r<0) return r; /* not a valid gate graph! */
  899.   g->vertices->val=1; /* turn the .{RUN} bit on */
  900.   while (1) {
  901.     for (a=g->outs,l=0;a;a=a->next) l=2*l+a->tip->val;
  902.         /* set $l=null$memory address */
  903.     if (trace_regs) @<Print register contents@>;
  904.     if (l>=size) break; /* stop if memory check occurs */
  905.     for (v=g->vertices+1,m=rom[l];v<=g->vertices+16;v++,m>>=1)
  906.       v->val=m&1; /* store bits of memory word in the input gates */
  907.     gate_eval(g,NULL,NULL); /* do another RISC cycle */
  908.   }
  909.   if (trace_regs) @<Print a footline@>;
  910.   @<Dump the register contents into |risc_state|@>;
  911.   return 0;
  912. }
  913. @ If tracing is requested, we write on the standard output file.
  914. @<Print a headline@>=
  915. {
  916.   for (r=0;r<trace_regs;r++) printf(" r%-2ld ",r); /* register names */
  917.   printf(" P XSNKV MEMn"); /* |prog|, |extra|, status bits, memory */
  918. }
  919. @ @<Print a footline@>=
  920. printf("Execution terminated with memory address %04lx.n",l);
  921. @ Here we peek inside the circuit to see what values are about to
  922. be latched.
  923. @<Print register contents@>=
  924. { for (r=0;r<trace_regs;r++) {
  925.     v=g->vertices+(16*r+47); /* most significant bit of register |r| */
  926.     m=0;
  927.     if (v->typ=='L')
  928.       for (k=0,m=0;k<16;k++,v--) m=2*m+v->alt->val;
  929.     printf("%04lx ",m);
  930.   }
  931.   for (k=0,m=0,v=g->vertices+26;k<10;k++,v--) m=2*m+v->alt->val; /* |prog| */
  932.   x=(g->vertices+31)->alt->val; /* |extra| */
  933.   s=(g->vertices+27)->alt->val; /* |sign| */
  934.   n=(g->vertices+28)->alt->val; /* |nonzero| */
  935.   c=(g->vertices+29)->alt->val; /* |carry| */
  936.   o=(g->vertices+30)->alt->val; /* |overflow| */
  937.   printf("%03lx%c%c%c%c%c ",m<<2,
  938.      x?'X':'.', s?'S':'.', n?'N':'.', c?'K':'.', o?'V':'.');
  939.   if (l>=size) printf("????n");
  940.   else printf("%04lxn",rom[l]);
  941. }
  942. @ @<Dump...@>=
  943. for (r=0;r<16;r++) {
  944.   v=g->vertices+(16*r+47); /* most significant bit of register |r| */
  945.   m=0;
  946.   if (v->typ=='L')
  947.     for (k=0,m=0;k<16;k++,v--) m=2*m+v->alt->val;
  948.   risc_state[r]=m;
  949. }
  950. for (k=0,m=0,v=g->vertices+26;k<10;k++,v--) m=2*m+v->alt->val; /* |prog| */
  951. m=4*m+(g->vertices+31)->alt->val; /* |extra| */
  952. m=2*m+(g->vertices+27)->alt->val; /* |sign| */
  953. m=2*m+(g->vertices+28)->alt->val; /* |nonzero| */
  954. m=2*m+(g->vertices+29)->alt->val; /* |carry| */
  955. m=2*m+(g->vertices+30)->alt->val; /* |overflow| */
  956. risc_state[16]=m; /* program register and status bits go here */
  957. risc_state[17]=l;
  958.  /* this is the out-of-range address that caused termination */
  959. @ @<Global variables@>=
  960. unsigned long risc_state[18];
  961. @*Generalized gate graphs. For intermediate computations, it is
  962. convenient to allow two additional types of gates:
  963. {advanceparindent 2em
  964. smallskip
  965. item{|'C'|} denotes a constant gate of value |z.I|.
  966. smallskip
  967. item{|'='|} denotes a copy of a previous gate; utility field |alt|
  968. points to that previous gate.
  969. smallskip}noindent
  970. Such gates might appear anywhere in the graph, possibly interspersed with
  971. the inputs and latches.
  972. Here is a simple subroutine that prints a symbolic representation of
  973. a generalized gate graph on the standard output file:
  974. @d bit z.I /* field containing the constant value of a |'C'| gate */
  975. @d print_gates p_gates /* abbreviation makes chopped-off name unique */
  976. @<The |print_gates| routine@>=
  977. static void pr_gate(v)
  978.   Vertex *v;
  979. {@+register Arc *a;
  980.   printf("%s = ",v->name);
  981.   switch(v->typ) {
  982.   case 'I':printf("input");@+break;
  983.   case 'L':printf("latch");
  984.     if (v->alt) printf("ed %s",v->alt->name);
  985.     break;
  986.   case '~':printf("~ ");@+break;
  987.   case 'C':printf("constant %ld",v->bit); break;
  988.   case '=':printf("copy of %s",v->alt->name);
  989.   }
  990.   for (a=v->arcs;a;a=a->next) {
  991.     if (a!=v->arcs) printf(" %c ",(char)v->typ);
  992.     printf(a->tip->name);
  993.   }
  994.   printf("n");
  995. }
  996. @#
  997. void print_gates(g)
  998.   Graph *g;
  999. {@+register Vertex *v;
  1000.   register Arc *a;
  1001.   for (v=g->vertices;v<g->vertices+g->n;v++) pr_gate(v);
  1002.   for (a=g->outs;a;a=a->next)
  1003.     if (is_boolean(a->tip)) printf("Output %ldn",the_boolean(a->tip));
  1004.     else printf("Output %sn",a->tip->name);
  1005. }
  1006. @ @(gb_gates.h@>=
  1007. #define bit @tquad@> z.I
  1008. @ The |reduce| routine takes a generalized graph |g| and uses the identities
  1009. $overline{overline x}=x$ and
  1010. $$openup1jot
  1011. vbox{halign{hfil$x#0=null$&$#$,hfilquad
  1012.              &hfil$x#1=null$&$#$,hfilquad
  1013.              &hfil$x#x=null$&$#$,hfilquad
  1014.              &hfil$x#overline x=null$&$#$,hfilcr
  1015. land&0&land&x&land&x&land&0cr
  1016. lor&x&lor&1&lor&x&lor&1cr
  1017. oplus&x&oplus&overline x&oplus&0&oplus&1cr}}$$
  1018. to create an equivalent graph having no
  1019. |'C'| or |'='| or obviously redundant gates. The reduced graph also excludes
  1020. any gates that are not used directly or indirectly in the computation of
  1021. the output values.
  1022. @<Internal...@>=
  1023. static Graph* reduce(g)
  1024.   Graph *g;
  1025. {@+register Vertex *u, *v; /* the current vertices of interest */
  1026.   register Arc *a, *b; /* the current arcs of interest */
  1027.   Arc *aa, *bb; /* their predecessors */
  1028.   Vertex *latch_ptr; /* top of the latch list */
  1029.   long n=0; /* the number of marked gates */
  1030.   Graph *new_graph; /* the reduced gate graph */
  1031.   Vertex *next_vert=NULL, *max_next_vert=NULL; /* allocation of new vertices */
  1032.   Arc *avail_arc=NULL; /* list of recycled arcs */
  1033.   Vertex *sentinel; /* end of the vertices */
  1034.   if (g==NULL) panic(missing_operand); /* where is |g|? */
  1035.   sentinel=g->vertices+g->n;
  1036.   while (1) {
  1037.     latch_ptr=NULL;
  1038.     for (v=g->vertices;v<sentinel;v++)
  1039.       @<Reduce gate |v|, if possible, or put it on the latch list@>;
  1040.     @<Check to see if any latch has become constant; if not, |break|@>;
  1041.   }
  1042.   @<Mark all gates that are used in some output@>;
  1043.   @<Copy all marked gates to a new graph@>;
  1044.   gb_recycle(g);
  1045.   return new_graph;
  1046. }
  1047. @ We will link latches together via their |v.V| fields.
  1048. @<Check to see if any latch has become constant; if not, |break|@>=
  1049. {@+char no_constants_yet=1;
  1050.   for (v=latch_ptr;v;v=v->v.V) {
  1051.     u=v->alt; /* the gate whose value will be latched */
  1052.     if (u->typ=='=')
  1053.       v->alt=u->alt;
  1054.     else if (u->typ=='C') {
  1055.       v->typ='C';@+v->bit=u->bit;@+no_constants_yet=0;
  1056.     }
  1057.   }
  1058.   if (no_constants_yet) break;
  1059. }
  1060. @ @d foo x.V /* link field used to find all the gates later */
  1061. @<Reduce gate |v|, if possible, or put it on the latch list@>=
  1062. {
  1063.   switch(v->typ) {
  1064.     case 'L': v->v.V=latch_ptr;@+latch_ptr=v;@+break;
  1065.     case 'I': case 'C': break;
  1066.     case '=': u=v->alt;
  1067.       if (u->typ=='=')
  1068.         v->alt=u->alt;
  1069.       else if (u->typ=='C') {
  1070.         v->bit=u->bit;@+goto make_v_constant;
  1071.       }
  1072.       break;
  1073.     case NOT:@<Try to reduce an inverter, then |goto done|@>;
  1074.     case AND:@<Try to reduce an {sc AND} gate@>;@+goto test_single_arg;
  1075.     case OR:@<Try to reduce an {sc OR} gate@>;@+goto test_single_arg;
  1076.     case XOR:@<Try to reduce an {sc EXCLUSIVE-OR} gate@>;
  1077.   test_single_arg: if (v->arcs->next) break;
  1078.     v->alt=v->arcs->tip;
  1079.   make_v_eq: v->typ='=';@+goto make_v_arcless;
  1080.   make_v_1: v->bit=1;@+goto make_v_constant;
  1081.   make_v_0: v->bit=0;
  1082.   make_v_constant: v->typ='C';
  1083.   make_v_arcless: v->arcs=NULL;
  1084.   }
  1085. v->bar=NULL; /* this field will point to the complement, if computed later */
  1086. done: v->foo=v+1; /* this field will link all the vertices together */
  1087. }
  1088. @ @<Try to reduce an inverter...@>=
  1089. u=v->arcs->tip;
  1090. if (u->typ=='=')
  1091.   u=v->arcs->tip=u->alt;
  1092. if (u->typ=='C') {
  1093.   v->bit=1-u->bit;@+goto make_v_constant;
  1094. }@+else if (u->bar) { /* this inverse already computed */
  1095.   v->alt=u->bar;@+goto make_v_eq;
  1096. }@+else {
  1097.   u->bar=v;@+v->bar=u;@+goto done;
  1098. }
  1099. @ @<Try to reduce an {sc AND} gate@>=
  1100. for (a=v->arcs,aa=NULL;a;a=a->next) {
  1101.   u=a->tip;
  1102.   if (u->typ=='=')
  1103.     u=a->tip=u->alt;
  1104.   if (u->typ=='C') {
  1105.     if (u->bit==0) goto make_v_0;
  1106.     goto bypass_and;
  1107.   }@+else@+for (b=v->arcs;b!=a;b=b->next) {
  1108.     if (b->tip==u) goto bypass_and;
  1109.     if (b->tip==u->bar) goto make_v_0;
  1110.   }
  1111.   aa=a;@+continue;
  1112. bypass_and: if (aa) aa->next=a->next;
  1113.   else v->arcs=a->next;
  1114. }
  1115. if (v->arcs==NULL) goto make_v_1;
  1116. @ @<Try to reduce an {sc OR} gate@>=
  1117. for (a=v->arcs,aa=NULL;a;a=a->next) {
  1118.   u=a->tip;
  1119.   if (u->typ=='=')
  1120.     u=a->tip=u->alt;
  1121.   if (u->typ=='C') {
  1122.     if (u->bit) goto make_v_1;
  1123.     goto bypass_or;
  1124.   }@+else@+for (b=v->arcs;b!=a;b=b->next) {
  1125.     if (b->tip==u) goto bypass_or;
  1126.     if (b->tip==u->bar) goto make_v_1;
  1127.   }
  1128.   aa=a;@+continue;
  1129. bypass_or: if (aa) aa->next=a->next;
  1130.   else v->arcs=a->next;
  1131. }
  1132. if (v->arcs==NULL) goto make_v_0;
  1133. @ @<Try to reduce an {sc EXCLUSIVE-OR} gate@>=
  1134. {@+long cmp=0;
  1135.   for (a=v->arcs,aa=NULL;a;a=a->next) {
  1136.     u=a->tip;
  1137.     if (u->typ=='=')
  1138.       u=a->tip=u->alt;
  1139.     if (u->typ=='C') {
  1140.       if (u->bit) cmp=1-cmp;
  1141.       goto bypass_xor;
  1142.     }@+else@+for (bb=NULL,b=v->arcs;b!=a;b=b->next) {
  1143.       if (b->tip==u) goto double_bypass;
  1144.       if (b->tip==u->bar) {
  1145.         cmp=1-cmp;
  1146.         goto double_bypass;
  1147.       }
  1148.       bb=b;@+ continue;
  1149.     double_bypass: if (bb) bb->next=b->next;
  1150.       else v->arcs=b->next;
  1151.       goto bypass_xor;
  1152.     }
  1153.     aa=a;@+ continue;
  1154.   bypass_xor: if (aa) aa->next=a->next;
  1155.     else v->arcs=a->next;
  1156.     a->a.A=avail_arc;
  1157.     avail_arc=a;
  1158.   }
  1159.   if (v->arcs==NULL) {
  1160.     v->bit=cmp;
  1161.     goto make_v_constant;
  1162.   }
  1163.   if (cmp) @<Complement one argument of |v|@>;
  1164. }
  1165. @ @<Complement one argument of |v|@>=
  1166. {
  1167.   for (a=v->arcs;;a=a->next) {
  1168.     u=a->tip;
  1169.     if (u->bar) break; /* good, the complement is already known */
  1170.     if (a->next==NULL) { /* oops, this is our last chance */
  1171.       @<Create a new vertex for complement of |u|@>;
  1172.       break;
  1173.     }
  1174.   }
  1175.   a->tip=u->bar;
  1176. }
  1177. @ Here we've come to a subtle point: If a lot of |XOR| gates involve
  1178. an input that is set to the constant value~1, the ``reduced'' graph
  1179. might actually be larger than the original, in the sense of having
  1180. more vertices (although fewer arcs).  Therefore we must have the
  1181. ability to allocate new vertices during the reduction phase of
  1182. |reduce|. At least one arc has been added to the |avail_arc| list
  1183. whenever we reach this portion of the program.
  1184. @<Create a new vertex for complement of |u|@>=
  1185. if (next_vert==max_next_vert) {
  1186.   next_vert=gb_typed_alloc(7,Vertex,g->aux_data);
  1187.   if (next_vert==NULL) {
  1188.     gb_recycle(g);
  1189.     panic(no_room+1); /* can't get auxiliary storage! */
  1190.   }
  1191.   max_next_vert=next_vert+7;
  1192. }
  1193. next_vert->typ=NOT;
  1194. sprintf(name_buf,"%s~",u->name);
  1195. next_vert->name=gb_save_string(name_buf);
  1196. next_vert->arcs=avail_arc; /* this is known to be non-|NULL| */
  1197. avail_arc->tip=u;
  1198. avail_arc=avail_arc->a.A;
  1199. next_vert->arcs->next=NULL;
  1200. next_vert->bar=u;
  1201. next_vert->foo=u->foo;
  1202. u->foo=u->bar=next_vert++;
  1203. @ During the marking phase, we will use the |w.V| field to link the
  1204. list of nodes-to-be-marked. That field will turn out to be non-|NULL|
  1205. only in the marked nodes. (We no longer use its former meaning related
  1206. to complementation, so we call it |lnk| instead of |bar|.)
  1207. @d lnk w.V /* stack link for marking */
  1208. @<Mark all gates that are used in some output@>=
  1209. {
  1210.   for (v=g->vertices;v!=sentinel;v=v->foo) v->lnk=NULL;
  1211.   for (a=g->outs;a;a=a->next) {
  1212.     v=a->tip;
  1213.     if (is_boolean(v)) continue;
  1214.     if (v->typ=='=')
  1215.       v=a->tip=v->alt;
  1216.     if (v->typ=='C') { /* this output is constant, so make it boolean */
  1217.       a->tip=(Vertex*)v->bit;
  1218.       continue;
  1219.     }
  1220.     @<Mark all gates that are used to compute |v|@>;
  1221.   }
  1222. }
  1223. @ @<Mark all gates that are used to compute |v|@>=
  1224. if (v->lnk==NULL) {
  1225.   v->lnk=sentinel;
  1226.    /* |v| now represents the top of the stack of nodes to be marked */
  1227.   do@+{
  1228.     n++;
  1229.     b=v->arcs;
  1230.     if (v->typ=='L') {
  1231.       u=v->alt; /* latch vertices have a ``hidden'' dependency */
  1232.       if (u<v) n++; /* latched input value will get a special gate */
  1233.       if (u->lnk==NULL) {
  1234.         u->lnk=v->lnk;
  1235.         v=u;
  1236.       }@+else v=v->lnk;
  1237.     }@+else v=v->lnk;
  1238.     for (;b;b=b->next) {
  1239.       u=b->tip;
  1240.       if (u->lnk==NULL) {
  1241.         u->lnk=v;
  1242.         v=u;
  1243.       }
  1244.     }
  1245.   }@+while (v!=sentinel);
  1246. }
  1247. @ It is easier to copy a directed acyclic graph than to copy a general graph,
  1248. but we do have to contend with the feedback in latches.
  1249. @d reverse_arc_list(@!alist)
  1250.   {@+for (aa=alist,b=NULL;aa;b=aa,aa=a) {
  1251.       a=aa->next;
  1252.       aa->next=b;
  1253.      }
  1254.    alist=b;@+}
  1255. @<Copy all marked gates to a new graph@>=
  1256. new_graph=gb_new_graph(n);
  1257. if (new_graph==NULL) {
  1258.   gb_recycle(g);
  1259.   panic(no_room+2); /* out of memory */
  1260. }
  1261. strcpy(new_graph->id,g->id);
  1262. strcpy(new_graph->util_types,"ZZZIIVZZZZZZZA");
  1263. next_vert=new_graph->vertices;
  1264. for (v=g->vertices,latch_ptr=NULL;v!=sentinel;v=v->foo) {
  1265.   if (v->lnk) { /* yes, |v| is marked */
  1266.     u=v->lnk=next_vert++; /* make note of where we've copied it */
  1267.     @<Make |u| a copy of |v|; put it on the latch list if it's a latch@>;
  1268.   }
  1269. }
  1270. @<Fix up the |alt| fields of the newly copied latches@>;
  1271. reverse_arc_list(g->outs);
  1272. for (a=g->outs;a;a=a->next) {
  1273.   b=gb_virgin_arc();
  1274.   b->tip=is_boolean(a->tip)? a->tip: a->tip->lnk;
  1275.   b->next=new_graph->outs;
  1276.   new_graph->outs=b;
  1277. }
  1278. @ @<Make |u| a copy of |v|; put it on the latch list if it's a latch@>=
  1279. u->name=gb_save_string(v->name);
  1280. u->typ=v->typ;
  1281. if (v->typ=='L') {
  1282.   u->alt=latch_ptr;@+latch_ptr=v;
  1283. }
  1284. reverse_arc_list(v->arcs);
  1285. for (a=v->arcs;a;a=a->next)
  1286.   gb_new_arc(u,a->tip->lnk,a->len);
  1287. @ @<Fix up the |alt| fields of the newly copied latches@>=
  1288. while (latch_ptr) {
  1289.   u=latch_ptr->lnk; /* the copy of a latch */
  1290.   v=u->alt;
  1291.   u->alt=latch_ptr->alt->lnk;
  1292.   latch_ptr=v;
  1293.   if (u->alt<u) @<Replace |u->alt| by a new gate that copies an input@>;
  1294. }
  1295. @ Suppose we had a latch whose value was originally the {sc AND} of
  1296. two inputs, where one of those inputs has now been set to~1. Then the
  1297. latch should still refer to a subsequent gate, equal to the value of the
  1298. other input on the previous cycle. We create such a gate here, making
  1299. it an {sc OR} of two identical inputs. We do this because we're not supposed
  1300. to leave any |'='| in the result of |reduce|, and because every {sc OR}
  1301. is supposed to have at least two inputs.
  1302. @<Replace |u->alt| by a new gate that copies an input@>=
  1303. {
  1304.   v=u->alt; /* the input gate that should be copied for latching */
  1305.   u->alt=next_vert++;
  1306.   sprintf(name_buf,"%s>%s",v->name,u->name);
  1307.   u=u->alt;
  1308.   u->name=gb_save_string(name_buf);
  1309.   u->typ=OR;
  1310.   gb_new_arc(u,v,DELAY);@+gb_new_arc(u,v,DELAY);
  1311. }
  1312. @* Parallel multiplication. Now comes the |prod| routine,
  1313. which constructs a rather different network of gates, based this time
  1314. on a divide-and-conquer paradigm. Let's take a breather before we tackle it.
  1315. (Deep breath.)
  1316. The subroutine call |prod(m,n)| creates
  1317. a network for the binary multiplication of unsigned
  1318. |m|-bit numbers by |n|-bit numbers, assuming that |m>=2| and |n>=2|.
  1319. There is no upper limit on the sizes of |m| and~|n|, except of course
  1320. the limits imposed by the size of memory in which this routine is run.
  1321. The overall strategy used by |prod| is to start with a generalized
  1322. gate graph for multiplication in which many of the gates are
  1323. identically zero or copies of other gates.  Then the |reduce| routine
  1324. will perform local optimizations leading to the desired result. Since
  1325. there are no latches, some of the complexities of the general |reduce|
  1326. routine are avoided.
  1327. All of the |AND|, |OR|, and |XOR| gates of the network returned by
  1328. |prod| have exactly two inputs. The depth of the circuit (i.e., the
  1329. length of its longest path) is $3log m/!log 1.5 + log(m+n)/!logphi
  1330. +O(1)$, where $phi=(1+sqrt5,)/2$ is the golden ratio. The grand total
  1331. number of gates is $6mn+5m^2+Obigl((m+n)log(m+n)bigr)$.
  1332. There is a demonstration program called {sc MULTIPLY} that uses |prod| to
  1333. compute products of large integers.
  1334. @<The |prod| routine@>=
  1335. Graph* prod(m,n)
  1336.   unsigned long m,n; /* lengths of the binary numbers to be multiplied */
  1337. {@+@<Local variables for |prod|@>@;
  1338. @#
  1339.   if (m<2) m=2;
  1340.   if (n<2) n=2;
  1341.   @<Allocate space for a temporary graph |g| and for auxiliary tables@>;
  1342.   @<Fill |g| with generalized gates that do parallel multiplication@>;
  1343.   if (gb_trouble_code) {
  1344.     gb_recycle(g);@+panic(alloc_fault); /* too big */
  1345.   }
  1346.   g=reduce(g);
  1347.   return g; /* if |g==NULL|, the |panic_code| was set by |reduce| */
  1348. }
  1349. @ The divide-and-conquer recurrences used in this network lead to interesting
  1350. patterns. First we use a method for parallel column addition that reduces
  1351. the sum of three numbers to the sum of two numbers. Repeated use of this
  1352. reduction makes it possible to reduce the sum of |m| numbers to a sum of
  1353. just two numbers, with a total circuit depth that satisfies the
  1354. recurrence $T(3N)=T(2N)+O(1)$. Then when the result has been reduced
  1355. to a sum of two numbers, we use a parallel addition scheme based on
  1356. recursively ``golden sectioning the data''; in other words, the recursion
  1357. partitions the data into two parts such that the ratio of the larger part
  1358. to the smaller part is approximately $phi$. This technique proves to be
  1359. slightly better than a binary partition would be, both asymptotically and
  1360. for small values of~$m+n$.
  1361. defflog{mathop{rm flog}nolimits}
  1362. We define $flog N$, the Fibonacci logarithm of~$N$, to be the smallest
  1363. @^Fibonacci, Leonardo, numbers@>
  1364. nonnegative integer~$k$ such that $Nle F_{k+1}$. Let $N=m+n$. Our parallel
  1365. adder for two numbers of $N$ bits will turn out to have depth at most
  1366. $2+flog N$. The unreduced graph~|g| in our circuit for multiplication
  1367. will have fewer than $(6m+3flog N)N$ gates.
  1368. @<Allocate space for a temporary graph |g| and for auxiliary tables@>=
  1369. m_plus_n=m+n;@+@<Compute $f=flog(m+n)$@>;
  1370. g=gb_new_graph((6*m-7+3*f)*m_plus_n);
  1371. if (g==NULL) panic(no_room); /* out of memory before we're even started */
  1372. sprintf(g->id,"prod(%lu,%lu)",m,n);
  1373. strcpy(g->util_types,"ZZZIIVZZZZZZZA");
  1374. long_tables=gb_typed_alloc(2*m_plus_n+f,long,g->aux_data);
  1375. vert_tables=gb_typed_alloc(f*m_plus_n,Vertex*,g->aux_data);
  1376. if (gb_trouble_code) {
  1377.   gb_recycle(g);
  1378.   panic(no_room+1); /* out of memory trying to create auxiliary tables */
  1379. }
  1380. @ @<Local variables for |prod|@>=
  1381. unsigned long m_plus_n; /* guess what this variable holds */
  1382. long f; /* initially $flog(m+n)$, later flog of other things */
  1383. Graph *g; /* graph of generalized gates, to be reduced eventually */
  1384. long *long_tables; /* beginning of auxiliary array of |long| numbers */
  1385. Vertex **vert_tables; /* beginning of auxiliary array of gate pointers */
  1386. @ @<Compute $f=flog(m+n)$@>=
  1387. f=4;@+j=3;@+k=5; /* $j=F_f$, $k=F_{f+1}$ */
  1388. while (k<m_plus_n) {
  1389.   k=k+j;
  1390.   j=k-j;
  1391.   f++;
  1392. }
  1393. @ The well-known formulas for a ``full adder,''
  1394. $$ x+y+z=s+2c,qquad
  1395.    hbox{where $s=xoplus yoplus z$ and $c=xylor yzlor zx$},$$
  1396. can be applied to each bit of an $N$-bit number, thereby providing us
  1397. with a way to reduce the sum of three numbers to the sum of two.
  1398. The input gates of our network will be called $x_0$, $x_1$, dots,~$x_{m-1}$,
  1399. $y_0$,~$y_1$, dots,~$y_{n-1}$, and the outputs will be called
  1400. $z_0$, $z_1$, dots,~$z_{m+n-1}$. The logic of the |prod| network will compute
  1401. $$(z_{m+n-1}ldots z_1z_0)_2=(x_{m-1}ldots x_1x_0)_2cdot
  1402.                              (y_{n-1}ldots y_1y_0)_2,,$$
  1403. by first considering the product to be the $m$-fold sum
  1404. $A_0+A_1+cdots+A_{m-1}$, where
  1405. $$A_j=2^jx_jcdot(y_{n-1}ldots y_1y_0)_2,,qquad 0le j<m.$$
  1406. Then the three-to-two rule for addition is used to define further
  1407. numbers $A_m$, $A_{m+1}$, dots,~$A_{3m-5}$ by the scheme
  1408. $$A_{m+2j}+A_{m+2j+1}=A_{3j}+A_{3j+1}+A_{3j+2},,qquad 0le jle m-3.$$
  1409. [A similar but slightly less efficient scheme was used by Pratt and
  1410. Stockmeyer in {sl Journal of Computer and System Sciences bf12} (1976),
  1411. @^Pratt, Vaughan Ronald@>
  1412. @^Stockmeyer, Larry Joseph@>
  1413. Proposition~5.3. The recurrence used here is related to the Josephus
  1414. @^Josephus, Flavius, problem@>
  1415. problem with step-size~3; see {sl Concrete Mathematics},
  1416. {mathhexbox278}3.3.]
  1417. For this purpose, we compute intermediate results $P_j$, $Q_j$, and~$R_j$
  1418. by the rules
  1419. $$eqalign{P_j&=A_{3j}oplus A_{3j+1},;cr
  1420.            Q_j&=A_{3j}land A_{3j+1},;cr
  1421.       A_{m+2j}&=P_joplus A_{3j+2},;cr
  1422.            R_j&=P_jland A_{3j+2},;cr
  1423.     A_{m+2j+1}&=2(Q_jlor R_j),.cr}$$
  1424. Finally we let
  1425. $$eqalign{U&=A_{3m-6}oplus A_{3m-5},,cr
  1426.            V&=A_{3m-6}land A_{3m-5},;cr}$$
  1427. these are the values that would be $P_{m-2}$ and $Q_{m-2}$ if the previous
  1428. formulas were allowed to run past $j=m-3$. The final result
  1429. $Z=(z_{m+n-1}ldots z_1z_0)_2$ can now be expressed as
  1430. $$Z=U+2V,.$$
  1431. The gates of the first part of the network are conveniently obtained
  1432. in groups of $N=m+n$, representing the bits of the quantities $A_j$,
  1433. $P_j$, $Q_j$, $R_j$, $U$, and~$V$. We will put the least significant bit
  1434. of $A_j$ in gate position |g->vertices+a(j)*N|, where $a(j)=j+1$ for
  1435. $0le j<m$ and $a(m+2j+t)=m+5j+3+2t$ for $0le jle m-3$, $0le tle1$.
  1436. @<Fill |g| with generalized gates that do parallel multiplication@>=
  1437. next_vert=g->vertices;
  1438. start_prefix("X");@+x=first_of(m,'I');
  1439. start_prefix("Y");@+y=first_of(n,'I');
  1440. @<Define $A_j$ for $0le j<m$@>;
  1441. @<Define $P_j$, $Q_j$, $A_{m+2j}$, $R_j$, and $A_{m+2j+1}$
  1442.   for $0le jle m-3$@>;
  1443. @<Define $U$ and $V$@>;
  1444. @<Compute the final result $Z$ by parallel addition@>;
  1445. @ @<Local variables for |prod|@>=
  1446. register long i,j,k,l; /* all-purpose indices */
  1447. register Vertex *v; /* current vertex of interest */
  1448. Vertex *x,*y; /* least-significant bits of the input gates */
  1449. Vertex *alpha,*beta; /* least-significant bits of arguments */
  1450. @ @<Define $A_j$ for $0le j<m$@>=
  1451. for (j=0; j<m; j++) {
  1452.   numeric_prefix('A',j);
  1453.   for (k=0; k<j; k++) {
  1454.     v=new_vert('C');@+v->bit=0; /* this gate is the constant 0 */
  1455.   }
  1456.   for (k=0; k<n; k++)
  1457.     make2(AND,x+j,y+k);
  1458.   for (k=j+n; k<m_plus_n; k++) {
  1459.     v=new_vert('C');@+v->bit=0; /* this gate is the constant 0 */
  1460.   }
  1461. }
  1462. @ Since |m| is |unsigned|, it is necessary to say `|j<m-2|' here instead
  1463. of `|j<=m-3|'.
  1464. @d a_pos(j) (j<m? j+1: m+5*((j-m)>>1)+3+(((j-m)&1)<<1))
  1465. @<Define $P_j$, $Q_j$, $A_{m+2j}$, $R_j$, and $A_{m+2j+1}$...@>=
  1466. for (j=0; j<m-2; j++) {
  1467.   alpha=g->vertices+(a_pos(3*j)*m_plus_n);
  1468.   beta=g->vertices+(a_pos(3*j+1)*m_plus_n);
  1469.   numeric_prefix('P',j);
  1470.   for (k=0; k<m_plus_n; k++)
  1471.     make2(XOR,alpha+k,beta+k);
  1472.   numeric_prefix('Q',j);
  1473.   for (k=0; k<m_plus_n; k++)
  1474.     make2(AND,alpha+k,beta+k);
  1475.   alpha=next_vert-2*m_plus_n;
  1476.   beta=g->vertices+(a_pos(3*j+2)*m_plus_n);
  1477.   numeric_prefix('A',(long)m+2*j);
  1478.   for (k=0; k<m_plus_n; k++)
  1479.     make2(XOR,alpha+k,beta+k);
  1480.   numeric_prefix('R',j);
  1481.   for (k=0; k<m_plus_n; k++)
  1482.     make2(AND,alpha+k,beta+k);
  1483.   alpha=next_vert-3*m_plus_n;
  1484.   beta=next_vert-m_plus_n;
  1485.   numeric_prefix('A',(long)m+2*j+1);
  1486.   v=new_vert('C');@+v->bit=0; /* another 0, it multiplies $Qlor R$ by 2 */
  1487.   for (k=0; k<m_plus_n-1; k++)
  1488.     make2(OR,alpha+k,beta+k);
  1489. }
  1490. @ Actually $v_{m+n-1}$ will never be used (it has to be zero); but we
  1491. compute it anyway. We don't have to worry about such nitty gritty details
  1492. because |reduce| will get rid of all the obvious redundancy.
  1493. @<Define $U$ and $V$@>=
  1494. alpha=g->vertices+(a_pos(3*m-6)*m_plus_n);
  1495. beta=g->vertices+(a_pos(3*m-5)*m_plus_n);
  1496. start_prefix("U");
  1497. for (k=0; k<m_plus_n; k++)
  1498.   make2(XOR,alpha+k,beta+k);
  1499. start_prefix("V");
  1500. for (k=0; k<m_plus_n; k++)
  1501.   make2(AND,alpha+k,beta+k);
  1502. @* Parallel addition. It's time now to take another deep breath. We
  1503. have finished the parallel multiplier except for one last step, the
  1504. design of a parallel adder.
  1505. The adder is based on the following theory:
  1506. We want to perform the binary addition
  1507. $$vbox{halign{hfil$#$&& hfil$#$cr
  1508.   u_{N-1}&ldots&u_2&u_1&u_0cr
  1509.   v_{N-2}&ldots&v_1&v_0cr
  1510. noalign{kern2pthrulekern4pt}
  1511.   z_{N-1}&ldots&z_2&z_1&z_0cr}}$$
  1512. where we know that $u_k+v_kle1$ for all~$k$. It follows that $z_k=u_koplus
  1513. w_k$, where $w_0=0$ and
  1514. $$ w_k;=;v_{k-1};lor;u_{k-1}v_{k-2};lor;u_{k-1}u_{k-2}v_{k-3};lor
  1515.   ;cdots;lor;u_{k-1}ldots u_1v_0$$
  1516. for $k>0$. The problem has therefore been reduced to the evaluation
  1517. of $w_1$, $w_2$, dots, $w_{N-1}$.
  1518. Let $c_k^{,j}$ denote the {sc OR} of the first $j$ terms in the formula
  1519. that defines $w_k$, and let $d_k^{,j}$ denote the $j$-fold product
  1520. $u_{k-1}u_{k-2}ldots u_{k-j}$.
  1521. Then $w_k=c_k^k$, and we can use a recursive scheme of the form
  1522. $$c_k^{,j}=c_k^{,i}lor d_k^{,i}c_{k-i}^{,j-i},,qquad
  1523.   d_k^{,j}=d_k^{,i}d_{k-i}^{,j-i},,qquad jge2,$$
  1524. to do the evaluation.
  1525. defdown{mathop{rm down}}
  1526. It turns out that this recursion behaves very nicely if we choose
  1527. $i=down[j]$, where $down[j]$ is defined for $j>1$ by the formula
  1528. $$down[j];=;j-F_{(flog j)-1},.$$
  1529. For example, $flog18=7$ because $F_7=13<18le21=F_8$,
  1530. hence $down[18]=18-F_6=10$.
  1531. Let us write $jtodown[j]$, and consider the oriented tree on the set
  1532. of all positive integers that is defined by this relation. One of the
  1533. paths in this tree, for example, is $18to10to5to3to2to1$. Our
  1534. recurrence for $w_{18}=c_{18}^{18}$ involves $c_{18}^{10}$, which
  1535. involves $c_{18}^5$, which involves $c_{18}^3$, and so on. In general,
  1536. we will compute $c_k^{,j}$ for all $j$ with $kto^*j$, and we will
  1537. compute $d_k^{,j}$ for all $j$ with $kto^+j$. It is not difficult to
  1538. prove that $$k;to^*;j;to;iqquadhbox{implies}qquad
  1539. k-i;to^*;j-i,;$$ therefore the auxiliary factors $c_{k-i}^{,j-i}$
  1540. and $d_{k-i}^{,j-i}$ needed in the recurrence scheme will already
  1541. have been evaluated. (Indeed, one can prove more: Let $l=flog k$. If
  1542. the complete path from $k$ to~$1$ in the tree is $k=k_0to
  1543. k_1tocdotsto k_t=1$, then the differences $k_0-k_1$, $k_1-k_2$,
  1544. dots, $k_{t-2}-k_{t-1}$ will consist of precisely the Fibonacci
  1545. numbers $F_{l-1}$, $F_{l-2}$, dots,~$F_2$, except for the numbers that
  1546. appear when $F_{l+1}-k$ is written as a sum of non-consecutive
  1547. Fibonacci numbers.)
  1548. It can also be shown that, when $k>1$, we have
  1549. $$flog k=min_{0<j<n},maxbigl(1+flog j,,2+flog(k-j)bigr),,$$
  1550. and that $down[k]$ is the smallest~$j$ such that the minimum is
  1551. achieved in this equation. Therefore the depth of the circuit for
  1552. computing $w_k$ from the $u$'s and~$v$'s is exactly $flog k$.
  1553. In particular, we can be sure that at most $3flog N$ gates will be
  1554. created when computing $z_k$, and that there will be at most $3Nflog N$
  1555. gates in the parallel addition portion of the circuit.
  1556. @<Compute the final result $Z$ by parallel addition@>=
  1557. @<Set up auxiliary tables to handle Fibonacci-based recurrences@>;
  1558. @<Create the gates for $W$, remembering intermediate results that
  1559.   might be reused later@>;
  1560. @<Compute the last gates $Z=Uoplus W$, and record their locations
  1561.   as outputs of the network@>;
  1562. g->n=next_vert-g->vertices; /* reduce to the actual number of gates used */
  1563. @ After we have created a gate for $w_k$, we will store its address as
  1564. the value of $w[k]$ in an auxiliary table. After we've created a gate
  1565. for $c_k^{,i}$ where $i<k$ is a Fibonacci number~$F_{l+1}$ and
  1566. $l=flog ige2$, we will store its address as the value of
  1567. $c[k+(l-2)N]$; the gate $d_k^{,i}$ will immediately follow this one.
  1568. Tables of $flog j$ and $down[j]$ will facilitate all these
  1569. manipulations.
  1570. @<Set up auxiliary tables to handle Fibonacci-based recurrences@>=
  1571. w=vert_tables;
  1572. c=w+m_plus_n;
  1573. flog=long_tables;
  1574. down=flog+m_plus_n+1;
  1575. anc=down+m_plus_n;
  1576. flog[1]=0;@+flog[2]=2;
  1577. down[1]=0;@+down[2]=1;
  1578. for (i=3,j=2,k=3,l=3; l<=m_plus_n; l++) {
  1579.   if (l>k) {
  1580.     k=k+j;
  1581.     j=k-j;
  1582.     i++; /* $F_i=j<lle k=F_{i+1}$ */
  1583.   }
  1584.   flog[l]=i;
  1585.   down[l]=l-k+j;
  1586. }
  1587. @ @<Local variables for |prod|@>=
  1588. Vertex *uu, *vv; /* pointer to $u_0$ and $v_0$ */
  1589. Vertex **w; /* table of pointers to $w_k$ */
  1590. Vertex **c; /* table of pointers to potentially
  1591.                  important intermediate values $c_k^{,i}$ */
  1592. Vertex *cc,*dd; /* pointers to $c_k^{,i}$ and $d_k^{,i}$ */
  1593. long *flog; /* table of flog values */
  1594. long *down; /* table of down values */
  1595. long *anc; /* table of ancestors of the current $k$ */
  1596. @ @<Create the gates for $W$, remembering intermediate results that
  1597.   might be reused later@>=
  1598. vv=next_vert-m_plus_n;@+uu=vv-m_plus_n;
  1599. start_prefix("W");
  1600. v=new_vert('C');@+v->bit=0;@+w[0]=v; /* $w_0=0$ */
  1601. v=new_vert('=');@+v->alt=vv;@+w[1]=v; /* $w_1=v_0$ */
  1602. for (k=2;k<m_plus_n;k++) {
  1603.   @<Set the |anc| table to a list of the ancestors of |k| in decreasing order,
  1604.       stopping with |anc[l]=2|@>;
  1605.   i=1;@+cc=vv+k-1;@+dd=uu+k-1;
  1606.   while (1) {
  1607.     j=anc[l]; /* now $i=down[j]$ */
  1608.     @#
  1609.     @<Compute the gate $b_k^{,j}=d_k^{,i}land c_{k-i}^{,j-i}$@>;
  1610.     @<Compute the gate $c_k^{,j}=c_k^{,i}lor b_k^{,j}$@>;
  1611.     if (flog[j]<flog[j+1]) /* $j$ is a Fibonacci number */
  1612.     c[k+(flog[j]-2)*m_plus_n]=v;
  1613.     if (l==0) break;
  1614.     cc=v;
  1615.     @<Compute the gate $d_k^{,j}=d_k^{,i}land d_{k-i}^{,j-i}$@>;
  1616.     dd=v;
  1617.     i=j;
  1618.     l--;
  1619.   }
  1620.   w[k]=v;
  1621. }
  1622. @ If $kto j$, we call $j$ an ``ancestor'' of $k$ because we are thinking
  1623. of the tree defined by `$to$'; this tree is rooted at $2to1$.
  1624. @<Set the |anc| table to a list of the ancestors of |k| in decreasing order,
  1625.       stopping with |anc[l]=2|@>=
  1626. for (l=0,j=k;;l++,j=down[j]) {
  1627.   anc[l]=j;
  1628.   if (j==2) break;
  1629. }
  1630. @ @d spec_gate(v,a,k,j,t)
  1631.     v=next_vert++;
  1632.     sprintf(name_buf,"%c%ld:%ld",a,k,j);
  1633.     v->name=gb_save_string(name_buf);
  1634.     v->typ=t;
  1635. @<Compute the gate $b_k^{,j}=d_k^{,i}land c_{k-i}^{,j-i}$@>=
  1636. spec_gate(v,'B',k,j,AND);
  1637. gb_new_arc(v,dd,DELAY); /* first argument is $d_k^{,i}$ */
  1638. f=flog[j-i]; /* get ready to compute the second argument, $c_{k-i}^{,j-i}$ */
  1639. gb_new_arc(v,f>0? c[k-i+(f-2)*m_plus_n]:vv+k-i-1,DELAY);
  1640. @ @<Compute the gate $c_k^{,j}=c_k^{,i}lor b_k^{,j}$@>=
  1641. if (l) {
  1642.   spec_gate(v,'C',k,j,OR);
  1643. }@+else v=new_vert(OR); /* if $l$ is zero, this gate is $c_k^k=w_k$ */
  1644. gb_new_arc(v,cc,DELAY); /* first argument is $c_k^{,i}$ */
  1645. gb_new_arc(v,next_vert-2,DELAY); /* second argument is $b_k^{,j}$ */
  1646. @ Here we reuse the value $f=flog(j-i)$ computed a minute ago.
  1647. @<Compute the gate $d_k^{,j}=d_k^{,i}land d_{k-i}^{,j-i}$@>=
  1648. spec_gate(v,'D',k,j,AND);
  1649. gb_new_arc(v,dd,DELAY); /* first argument is $d_k^{,i}$ */
  1650. gb_new_arc(v,f>0? c[k-i+(f-2)*m_plus_n]+1:uu+k-i-1,DELAY);
  1651.  /* $d_{k-i}^{,j-i}$ */
  1652. @ The output list will contain the gates in ``big-endian order''
  1653. $z_{m+n-1}$, dots, $z_1$, $z_0$, because we insert them into the
  1654. |outs| list in little-endian order.
  1655. @<Compute the last gates $Z=Uoplus W$...@>=
  1656. start_prefix("Z");
  1657. for (k=0;k<m_plus_n;k++) {@+register Arc *a=gb_virgin_arc();
  1658.   a->tip=make2(XOR,uu+k,w[k]);
  1659.   a->next=g->outs;
  1660.   g->outs=a;
  1661. }
  1662. @* Partial evaluation. The subroutine call |partial_gates(g,r,prob,seed,buf)|
  1663. creates a new gate graph from a given gate graph~|g| by ``partial evaluation,''
  1664. i.e., by setting some of the inputs to constant values and simplifying the
  1665. result. The new graph is usually smaller than |g|; it might, in fact, be
  1666. a great deal smaller. Graph~|g| is destroyed in the process.
  1667. The first |r| inputs of |g| are retained unconditionally. Each
  1668. remaining input is retained with probability |prob/65536|, and if not
  1669. retained it is assigned a random constant value. For example, about
  1670. half of the inputs will become constant if |prob=32768|.  The |seed|
  1671. parameter defines a machine-independent source of random numbers, and
  1672. it may be given any value between $0$ and $2^{31}-1$.
  1673. If the |buf| parameter is non-null, it should be the address of a string.
  1674. In such a case, |partial_gates| will put a record of its partial evaluation
  1675. into that string; |buf| will contain one character for each input gate
  1676. after the first |r|, namely |'*'| if the input was
  1677. retained, |'0'| if it was set to~$0$, or |'1'| if it was set to~$1$.
  1678. The new graph will contain only gates that contribute to the computation
  1679. of at least one output value. Therefore some input gates might disappear
  1680. even though they were supposedly ``retained,'' i.e., even though their
  1681. value has not been set constant. The |name| field of a vertex can be
  1682. used to determine exactly which input gates have survived.
  1683. If graph |g| was created by |risc|, users will probably want to make
  1684. |r>=1|, since the whole RISC circuit collapses to zero whenever its
  1685. first input `.{RUN}' is set to 0.
  1686. An interesting class of graphs is produced by
  1687. the function call |partial_gates(prod(m,n),m,0,seed,NULL)|, which
  1688. creates a graph corresponding to a circuit that multiplies a given |m|-bit
  1689. number by a fixed (but randomly selected) |n|-bit constant. If the constant
  1690. is not zero, all |m| of the ``retained'' input gates necessarily survive.
  1691. The demo program called {sc MULTIPLY} illustrates such circuits.
  1692. The graph |g| might be a generalized network; that is, it might
  1693. involve the |'C'| or |'='| gates described earlier. Notice that if |r| is
  1694. sufficiently large, |partial_gates| becomes equivalent to the |reduce|
  1695. routine. Therefore we need not make that private routine public.
  1696. As usual, the result will be |NULL|, and |panic_code| will be set,
  1697. if |partial_gates| is unable to complete its task.
  1698. @<The |partial_gates| routine@>=
  1699. Graph *partial_gates(g,r,prob,seed,buf)
  1700.   Graph *g; /* generalized gate graph */
  1701.   unsigned long r; /* the number of initial gates to leave untouched */
  1702.   unsigned long prob;
  1703.    /* scaled probability of not touching subsequent input gates */
  1704.   long seed; /* seed value for random number generation */
  1705.   char *buf; /* optional parameter for information about partial assignment */
  1706. {@+register Vertex *v; /* the current gate of interest */
  1707.   if (g==NULL) panic(missing_operand); /* where is |g|? */
  1708.   gb_init_rand(seed); /* get them random numbers rolling */
  1709.   for (v=g->vertices+r;v<g->vertices+g->n;v++)
  1710.     switch (v->typ) {
  1711.     case 'C': case '=': continue; /* input gates might still follow */
  1712.     case 'I': if ((gb_next_rand()>>15)>=prob) {
  1713.         v->typ='C';@+v->bit=gb_next_rand()>>30;
  1714.         if (buf) *buf++=v->bit+'0';
  1715.       }@+else if (buf) *buf++='*';
  1716.       break;
  1717.     default: goto done; /* no more input gates can follow */
  1718.     }
  1719. done:if (buf) *buf=0; /* terminate the string */
  1720.   g=reduce(g);
  1721.   @<Give the reduced graph a suitable |id|@>;
  1722.   return g; /* if |(g==NULL)|, a |panic_code| has been set by |reduce| */
  1723. }
  1724. @ The |buf| parameter is not recorded in the graph's |id| field, since it
  1725. has no effect on the graph itself.
  1726. @<Give the reduced graph a suitable |id|@>=
  1727. if (g) {
  1728.   strcpy(name_buf,g->id);
  1729.   if (strlen(name_buf)>54) strcpy(name_buf+51,"...");
  1730.   sprintf(g->id,"partial_gates(%s,%lu,%lu,%ld)",name_buf,r,prob,seed);
  1731. }
  1732. @* Index. Here is a list that shows where the identifiers of this program are
  1733. defined and used.