## **Half Adder Component**



 ${[A/0, B/0], [A/1, B/1]} -> sum/0; {[A/1, B/0], [A/0, B/1]} -> sum/1;$ 



{[A/0, B/0], [A/1, B/0], [A/0, B/1]} -> carryout/0; [A/1, B/1] -> carryout/1;

#### **Symbolic specification**

```
// specify the half add component halfadd(A, B -> sum, carryout){
flow [A, B] -> [sum, carryout];
token {0:1} A, B, sum, carryout;
{[A/0, B/0], [A/1, B/1]} -> sum/0;
{[A/1, B/0], [A/0, B/1]} -> sum/1;
{[A/0, B/0], [A/1, B/0], [A/0, B/1]} -> carryout/0;
[A/1, B/1] -> carryout/1;
close A <- [sum/#, carryout/#];
close B <- [sum/#, carryout/#];
```







### input incompleteness for carry

Given two output flow paths of a single shared completeness behavior If the completeness of transition one output flow implies the completeness of transition of the input flow then the other output flow need not imply the completeness of input but only the completeness of propagation and the correcctness of the output. The AND of the completeness of the two output flows closes with the input and fulfills the completeness criterion

In the case of the half adder the sum implies the completeness of the input so the carry can be generated with partial input. Specifically, if A/0 then carry/0 and it is not necessary to wait on B to generate the carry. In the case of the counter B will be the carryin from the lower order digit. So to generate carryout half the time we will not have to wait on carryin.



#### Half Adder A

The carry orphan causes a wait for the null carry ripple



#### Half Adder B

The carry orphan does not cause a wait for the null carry ripple

### Composing the half adder ring



#### Symbolic specification



Red names indicate corespondences in the verilog code

twoD-counter\_ringA.v twoD-counter\_ringB.v twoD-counter\_ringC.v



nanada r component

halfadd(B, carryin -> count,carryout)

### halfadd ring counter component



### Canonical composition

direct mapping from symbolic specification

#### Symbolic specification

```
// sink the flow path
(count -> ){
flow count -> ;
token count[0:31]{0:1};
path count/i <
 (count/i -> );
// compose the flow path
counter( -> count){
flow -> count;
token count[0:31]{0:1};
token carry[0:32]{0:1};
path count/i <
 (1 -> carry/0);
 digit_count(carry/i -> count/i, carry/i+1)
(carry/max(i)+1 \rightarrow ));
close count <- ?/#;
                           sink
 constant 1
                       component
component
```



# Cost reduction of carry pipeline

digit\_counter32A.v digit\_counter32B.v

digit-counter\_ringA.v digit-counter\_ringB.v



# Imposition of full token completeness

fullword\_counter32A.v fullword\_counter32B.v fullword-counter\_ringA.v fullword-counter\_ringB.v

The flowing wavefronts will conform to whatever logical constraint is imposed

```
counter( -> count){
flow -> count;
token count[0:31]{0:1};
token carry[0:32]{0:1};
path count/i <
    (1 -> carry/0);
    digit_count(carry/i -> count/i, carry/i+1)
    (carry/max(i)+1 -> ));
>
close count <- ?/#;
}</pre>
```

Close count <- ?/#
specifies that the complete token
has to be closed each oscillation but
it does not specify how the closure
must occur.



# Fully integrated combinational logic

No gate in the circuit is performing solely combination service. Every gate in the circuit is performing flow coordination duty and a few are also performing combinational duty.



A long delay gate creates a long period oscillation which paces the rest of the circuit. Consequently it is slower than the non integrated circuit. But it is fewer gates.





# Fully integrated combinational logic

No gate in the circuit is performing solely combination service. Every gate in the circuit is performing flow coordination duty and a few are also performing combinational duty.



The combinational work is distributed over two oscillations with shorter periods. The circuit as a whole flows faster. but it has more gates and more transitions.



