# Chapter 3 working with combinational logic

## Working with combinational logic

- Simplification
  - two-level simplification
  - exploiting don't cares
  - algorithm for simplification
- Logic realization
  - two-level logic and canonical forms realized with NANDs and NORs
  - multi-level logic, converting between ANDs and ORs
- Time behavior
- Hardware description languages

The first important topic here is formalizing the process of boolean minimization. In the last chapter, we illustrated how logic functions (or its expressions) can be simplified by boolean cubes or K-maps. Here we will look at a systematic or algorithmic approach.



we'll need a 4-variable Karnaugh map for each of the 3 output functions

Before going into the formal minimization process, let's look at some examples. The first example compares two numbers, each of which is two bits long, N1 is AB and N2 is CD, where A and C are most significant bits (MSBs) while B and D are least significant bits (LSBs).



LT = A'B'D + A'C + B'CD  $EQ = A'B'C'D' + A'BC'D + ABCD + AB'CD' = (A xnor C) \cdot (B xnor D)$ GT = BC'D' + AC' + ABD'

LT and GT are similar (flip A/C and B/D) EQ = (AC + A'C')BD + (AC + A'C')B'D' = (AC + A'C')(BD + B'D')





This is a 2bit-by-2bit multiplier that generates 4 bit output (whose MSB is P8 and LSB is P1). Note that A2 and B2 are MSBs.



# Design example: BCD increment by 1



block diagram and truth table

| 18                                                                   | 14               | 12 | 11                                   | 08                                   | 04                    | 02                                             | 01           |
|----------------------------------------------------------------------|------------------|----|--------------------------------------|--------------------------------------|-----------------------|------------------------------------------------|--------------|
| 0                                                                    | 0                | 0  | 0                                    | 0                                    | 0                     | 0                                              | 1            |
| 0                                                                    | 0                | 0  | 1                                    | 0                                    | 0                     | 0<br>1                                         | 0            |
| 0                                                                    | 0                | 1  | 0                                    | 0                                    | 0                     | 1                                              | 1            |
| 0                                                                    | 0<br>0<br>0<br>0 | 1  | 1<br>0<br>1<br>0                     | 0                                    | 1                     | 0                                              | 0            |
| 0                                                                    | 1                | 0  |                                      | 0<br>0<br>0<br>0<br>0                | 1                     | 0                                              | 1            |
| 0                                                                    | 1                | 0  | 1                                    | 0                                    | 1                     | 1                                              | 0            |
| 0                                                                    | 1                | 1  | 0                                    |                                      | 1                     | 1                                              | 1            |
| 0                                                                    | 1                | 1  | 1                                    | 1                                    | 0                     | 0                                              | 0            |
| $\begin{array}{c} 0 \\ 0 \\ 0 \\ 0 \\ 0 \\ 0 \\ 1 \\ 1 \\ 1 \\ 1 \\$ | 0                | 0  | 1<br>0<br>1<br>0<br>1<br>0<br>1<br>0 | 1<br>0<br>X<br>X<br>X<br>X<br>X<br>X | 0<br>0                | 1<br>0<br>0<br>X<br>X<br>X<br>X<br>X<br>X<br>X | 0101010XXXXX |
| 1                                                                    | 0                | 0  | 1                                    | 0                                    | 0                     | 0                                              | 0            |
| 1                                                                    | 0                | 1  | 0                                    | X                                    | Х                     | Х                                              | Х            |
| 1                                                                    | 0<br>0<br>0      | 1  | 1                                    | X                                    | Х                     | X                                              | X            |
| 1                                                                    | 1                | Ó  | Ō                                    | X                                    | Х                     | X                                              | X            |
| 1                                                                    | 1                | Ō  | 1                                    | X                                    | O<br>X<br>X<br>X<br>X | X                                              | X            |
| 1                                                                    | 1                | 1  | Ò                                    | X                                    | X                     | X                                              | X            |
| 1                                                                    | 1                | 1  | 1                                    | X                                    | X                     | X                                              | X            |
|                                                                      |                  |    |                                      | 1                                    |                       |                                                |              |

4-variable K-map for each of the 4 output functions



In O8, we will interpret a don't care (DC) term as 1 if it helps to minimize the number of literals for the elements of the ON-set. The other DC terms will be treated as 0. To minimize the number of literals, we have to find out the maximum size subcube.

# Definition of terms for two-level simplification

- Implicant
  - single element of ON-set or DC-set or any group of these elements that can be combined to form a subcube (or an adjacent group)
- Prime implicant (PI)
  - implicant that can't be combined with another to form a larger subcube
- Essential prime implicant
  - prime implicant is essential if it alone covers an element of ON-set
  - will participate in ALL possible covers of the ON-set
  - DC-set used to form prime implicants but not to make implicant essential
- Objective:
  - grow implicant into prime implicants (to minimize literals per term)
  - cover the ON-set with as few prime implicants as possible (to minimize number of product terms)

So far we have examined a few examples of logic design simplification

From now on, we will try to perform simplification in a systematic or algorithmic way. To do so, we first have to define some terminologies.



First of all, we have to find out all the possible prime implicants. Then we first transform the essential prime implicants to boolean expressions. Then we will try to find the minimum set of prime implicants to cover the entire on-set, called minimum cover.

### Algorithm for two-level simplification

- Algorithm: minimum sum-of-products expression from a Karnaugh map
  - Step 1: choose an element of the ON-set
  - Step 2: find "maximal" groupings of 1s and Xs adjacent to that element
    - consider top/bottom row, left/right column, and corner adjacencies
    - this forms prime implicants (number of elements always a power of 2)
  - Repeat Steps 1 and 2 to find all prime implicants
  - Step 3: revisit the 1s in the K-map
    - if covered by single prime implicant, it is essential, and participates in final cover
    - 1s covered by essential prime implicant do not need to be revisited
  - □ Step 4: if there remain 1s not covered by essential prime implicants
    - select the smallest number of prime implicants that cover the remaining 1s



For all 1s, check the PIs that include the 1. The PIs should be considered for all the 1s and DCs around that can be united. Then we first choose essential PIs and then find out the minimum cover, the minimum number of PIs that cover the remaining 1s.

# Activity

List all prime implicants for the following K-map:



- Which are essential prime implicants?
- What is the minimum cover?

Let's start with prime implicants. Among PIs, check which are essential. Finally, we should find out the minimum cover, which is the minimum number of PIs that cover all the elements of the ON-set (including essential PIs)

# Implementations of two-level logic ABC

- Sum-of-products
  - AND gates to form product terms (minterms)
  - OR gate to form sum

Product-of-sums

- OR gates to form sum terms (maxterms)
- AND gates to form product

In this section, we will focus on how to implement logic networks with NAND or NOR gates. Again there are two kinds of canonical forms: S-O-P and P-O-S. A small circle is an inverter.



NAND/NOR gates requires less CMOS TRs than AND/OR gates. So we want to change AND/OR gates into NAND gates only (or NOR gates only)

The simplest way is to insert double inverters between AND and OR gates. Then what happens is that AND becomes NAND just by placing bubbles. How about the OR gate?

# Two-level logic using NAND gates (cont'd)

- OR gate with inverted inputs is a NAND gate
  - de Morgan's:  $A' + B' = (A \bullet B)'$
- Two-level NAND-NAND network
  - inverted inputs are not counted
  - in a typical circuit, inversion is done once and signal distributed



Recall de Morgan's Law. When inverters are passing through a gate, the gate should be changed from OR to AND and vice versa.

# Two-level logic using NOR gates

- Replace maxterm OR gates with NOR gates
- Place compensating inversion at inputs of AND gate



In the case of P-O-S forms, the same technique is used; however, this time, the NOR gate is the results of conversion.

Again, we insert two bubbles between OR and AND gates and push those bubbles in the opposite directions.

# Two-level logic using NOR gates (cont'd)

- AND gate with inverted inputs is a NOR gate
  - de Morgan's:  $A' \bullet B' = (A + B)'$
- Two-level NOR-NOR network
  - inverted inputs are not counted
  - □ in a typical circuit, inversion is done once and signal distributed



Using de Morgan's theorem again, the AND gate with inverted inputs is transformed into the NOR gate as shown in the above.

# Two-level logic using NAND and NOR gates

- NAND-NAND and NOR-NOR networks
  - □ de Morgan's law:  $(A + B)' = A' \cdot B'$   $(A \cdot B)' = A' + B'$
  - written differently:  $A + B = (A' \bullet B')'$   $(A \bullet B) = (A' + B')'$
- In other words
  - OR is the same as NAND with complemented inputs
  - AND is the same as NOR with complemented inputs
  - NAND is the same as OR with complemented inputs
  - NOR is the same as AND with complemented inputs



This slide summarizes what I explained about conversion from AND-OR combination to either NAND or NOR gates. All the conversions are just variations of de morgan's theorem.

## Conversion between forms

- Convert from networks of ANDs and ORs to networks of NANDs and NORs
  - introduce appropriate inversions ("bubbles")
- Each introduced "bubble" must be matched by a corresponding "bubble"
  - conservation of inversions
  - o do not alter logic function
- Example: AND/OR to NAND/NAND



Again, inverters inserted between gates are called bubbles. In order to make no changes in the logic function, the bubbles are always paired.

#### Conversion between forms (cont'd)

Example: verify equivalence of two forms



$$Z = [(A \cdot B)' \cdot (C \cdot D)']'$$
  
= [(A' + B') \cdot (C' + D')]'  
= [(A' + B')' + (C' + D')']  
= (A \cdot B) + (C \cdot D) \checkmark

Let's verify the conversion rule by boolean expressions and boolean theorems



When an input variable, say A, is complemented, it is denoted by A

When a S-o-P canonical form is converted to NOR networks, we have to insert two bubbles at the input stage. And the same thing happens at the output stage.

#### Conversion between forms (cont'd)

Example: verify equivalence of two forms





$$Z = \{ [(A' + B')' + (C' + D')']' \}'$$
  
= 
$$\{ (A' + B') \cdot (C' + D') \}'$$
  
= 
$$(A' + B')' + (C' + D')'$$
  
= 
$$(A \cdot B) + (C \cdot D) \checkmark$$

This is the boolean logic proof of the conversion in the previous slide.

# Multi-level logic

#### x = A D F + A E F + B D F + B E F + C D F + C E F + G

- reduced sum-of-products form already simplified
- 6 x 3-input AND gates + 1 x 7-input OR gate (that may not even exist!)
- 25 wires (19 literals plus 6 internal wires)
- x = (A + B + C) (D + E) F + G
  - factored form not written as two-level S-o-P
  - □ 1 x 3-input OR gate, 2 x 2-input OR gates, 1 x 3-input AND gate
  - 10 wires (7 literals plus 3 internal wires)



If there are common parts in a canonical form, it may be better to use multi-level logic to reduce the number of literals and gates at the cost of delay. Again tradeoff between delay and gate count



Normally when we add two bubbles in a wire, two levels are converted to NAND gates. Here we add two bubbles between AND and OR gates.

#### Conversion between forms

Example





В

add double bubbles to invert all inputs of OR gate



add double bubbles to invert output of AND gate

original circuit



insert inverters to eliminate double bubbles on a wire

This slide illustrates how we can convert a combination of AND and OR gates into NAND and NOT gates. As mentioned before, NOT gates can be replaced by NAND gates by splitting the input.



Here we add bubbles in different position to use NOR gates. Note that the final inverter is implemented by NOR!

# AND-OR-Invert (AOI) gates

- AOI function: three stages of logic AND, OR, Invert
  - multiple gates "packaged" as a single circuit block



Here is a special case of AND-OR-Inverter gates, which is a popular combination in a logic package. The reason it becomes a popular combination is that it can be implemented compactly with CMOS TRs.

# AOI example

• Why AOI is more compact than NAND or NOR?



30

#### Conversion to AOI forms

- General procedure to place in AOI form
  - compute the complement of the function in sum-of-products form
  - □ by grouping the 0s in the Karnaugh map
- Example: XOR implementation
  - $\Box A \text{ xor } B = A' B + A B'$
  - □ AOI form:
    - F = (A' B' + A B)'



Let's take an example to use AOI to implement a logic function. Suppose we have to implement a XOR function.

F = AB'+A'B. first of all, we consider F' (note that there is an inverter at the end of AOI).

F' = AB + A'B'. So we implement F' in the SOP form.

# Summary for multi-level logic

- Advantages
  - circuits may be smaller
  - gates have smaller fan-in
- Disadvantages
  - circuits will be slower
  - more difficult to design
  - tools for optimization are not as good as for two-level
  - analysis is more complex

Multi-level logic design can reduce the number of gates or at least the number of fan-ins of gates. However, optimization is more complex.

#### Time behavior of combinational networks

- Waveforms
  - visualization of values carried on signal wires over time
  - useful in explaining sequences of events (changes in value)
- Simulation tools are used to create these waveforms
  - input to the simulator includes gates and their connections
  - input stimulus, that is, input signal waveforms
- Some terms
  - □ gate delay time for change at input to cause change at output
    - min delay typical/nominal delay max delay
    - careful designers design for the worst case
  - □ rise time time for output to transition from low to high voltage
  - □ fall time time for output to transition from high to low voltage
  - pulse width time that an output stays high or stays low between changes

The next topic of this chapter is the behavior of combinational logic as time goes by.

The waveform of a system can be simulated by a tool considering gates and their connections. The output of the system is triggered by the input stimulus.

#### Momentary changes in outputs

- Can be useful pulse shaping circuits
- Can be a problem incorrect circuit operation (glitches/hazards)
- Example: pulse shaping circuit

$$\Box A' \bullet A = 0$$

delays matter



Let's see how a waveform changes over time in this case. Here, each gate is assumed to incur 10 time unit delay. This time-varying behavior is utilized to make a periodic pulse.



Assume that each gate delay is 10 time units. Here, the output of NAND is feedback to its input with a couple of inverters. Let's look at the waveform of B. what does it look like?

# Hardware description languages (HDLs)

- Describe hardware at varying levels of abstraction
- Structural description
  - textual replacement for schematic
  - hierarchical composition of modules from primitives
- Data-flow style description
  - textual replacement of truth table
- Behavioral/functional description
  - describe what module does, not how
  - synthesis generates circuit for module
- Simulation semantics

This is the last topic of chapter 3.

So far, we rely on boolean expressions and schematic drawings to describe logic functions. However, as a logic function gets complicated, it will become extremely hard to write and understand the logic system. Hierarchy can help to mitigate this problem; but it is not enough. HDLs are proposed to deal with this problem.

Using HDLs, we can describe any complicated logic system. Moreover, the languages can be executed, they run like s/w. A program emulates the behavior of the designed system as faithfully as possible. It radically reduces the time to design a system  $\frac{36}{36}$ 

## HDLs

- Abel (circa 1983) developed by Data-I/O
  - targeted to programmable logic devices
  - not good for much more than state machines
- ISP (circa 1977) research project at CMU
  - simulation, but no synthesis
- Verilog (circa 1985) developed by Gateway (absorbed by Cadence)
  - similar to Pascal and C
  - delay is only interaction with simulator
  - fairly efficient and easy to write
  - IEEE standard
- VHDL (circa 1987) DoD sponsored standard V: very high speed IC
  - similar to Ada (emphasis on re-use and maintainability)
  - simulation semantics visible
  - very general but verbose
  - IEEE standard

Verilog and VHDL are the most popular HDLs.

This course does not aim to cover HDLs in-depth.

### VHDL power

- Alternative to schematics (interconnection of components)
- Much more powerful than schematics
  - Boolean eqs.
  - Truth table
  - □ Complex operations (addition, ...)
- Represent entire system designed as a hierarchy
- Many tools available

### VHDL design example

#### Design half of a 74x139 (Dual 2-to-4 decoder)

- Three different types of programming (description)
  - Structural description
  - Data-flow style description
  - Behavioral description

#### 74x139





(b) Copyright © 2000 by Prentice Hall, Inc. Digital Design Principles and Practices, 3/e



```
library IEEE;
use IEEE.STD LOGIC 1164.ALL;
entity v74x139h is
    Port ( G L, B, A: in std logic;
           YO L, Y1 L, Y2 L, Y3 L: out std logic );
end v74x139h;
architecture v74x139h a of v74x139h is
   signal B L, A L, G, B i, A i: STD LOGIC;
   component inv port(o: out STD LOGIC; i: in STD LOGIC);
   end component;
   component nand3 port(o: out STD LOGIC; i2, i1, i0: in STD LOGIC );
   end component:
begin
   Ul: inv port map (G, G L);
   U2: inv port map (B L, B);
   U3: inv port map (A_L, A);
   U4: inv port map (B_i, B_L);
   U5: inv port map (A i, A L);
   U6: NAND3 port map (Y0 L, B L, A L, G);
   U7: NAND3 port map (Y1 L, B L, A i, G);
   U8: NAND3 port map (Y2 L, B i, A L, G);
   U9: NAND3 port map (Y3 L, B i, A i, G);
```

end v74x139h\_a;

```
library IEEE;
use IEEE.STD LOGIC 1164.ALL;
entity v74x139h is
    Port ( G L, B, A: in std logic;
           Y L: out std logic vector(3 downto 0) );
end v74x139h;
architecture v74x139h b of v74x139h is
   signal G: STD LOGIC;
   signal Y i: STD LOGIC VECTOR (3 downto 0);
   signal S: std logic vector (1 downto 0);
begin
   G <= not G L;
   S \leq B \& A:
   with S select Y i <=
      "0001" when "00",
      "0010" when "01",
      "0100" when "10",
      "1000" when "11",
      "0000" when others:
   y l \leq not y i when G='l' else "llll";
end v74x139h b;
```

## Behavioral Description

- To this point, structural and data-flow description → alternatives of schematic and truth table
- Behavioral description: richest set of language element → algorithmic description of hardware
- Main element: process



```
library IEEE;
use IEEE.STD_LOGIC_1164.ALL;
entity v74x139h is
    Port ( G L: in std logic;
              S: in STD LOGIC VECTOR(1 downto 0);
           Y L: out std logic vector(3 downto 0) );
end v74x139h;
architecture v74x139h c of v74x139h is
   signal G: STD LOGIC;
   signal Y: STD LOGIC VECTOR (3 downto 0);
begin
   G \ll not G L;
   Y L \leq not Y;
   -- S <= B \& A; this is automatically taken care of
                   in the entity definition
   ___
   process(S, G)
   begin
      if G='1' then
          case S is
             when "00" \Rightarrow Y <= "0001";
             when "01" \Rightarrow Y <= "0010";
             when "10" => Y <= "0100";
             when "11" \Rightarrow Y <= "1000";
             when others \Rightarrow Y <= "0000";
          end case;
      else Y <= "0000";
      end if:
   end process;
end v74x139h c;
```





# <sup>1</sup>/<sub>2</sub> 74x139 timing diagram

|       | 1    | 1    | 1             |               |               |              |  |              |                |       |
|-------|------|------|---------------|---------------|---------------|--------------|--|--------------|----------------|-------|
| ν/Ь   | 0    |      |               |               |               |              |  |              |                |       |
| ı/a   | 0    |      |               |               |               |              |  |              |                |       |
| 1/g_l | 1    |      |               |               |               |              |  |              |                |       |
| v/y_l | 1111 | 1110 | <u>)(1101</u> | <u>)(1011</u> | <u>)(0111</u> | <u>)1111</u> |  | <u>)0111</u> | <u>)1011 (</u> | (1111 |
|       |      |      |               |               |               |              |  |              |                |       |

# 1/2 74x139 fitting result



## Magnification of a MC (Macrocell)



### Hierarchical Design $(2 \times \frac{1}{2} 74 \times 139)$

```
library IEEE;
use IEEE.STD LOGIC 1164.ALL;
use IEEE.STD LOGIC ARITH.ALL;
use IEEE.STD LOGIC UNSIGNED.ALL;
entity v72x139 is
    Port ( G L1, G L2: in std logic;
           S1, S2: in std logic vector(1 downto 0);
          Y L1, Y L2: out std logic vector(3 downto 0) );
end v72x139;
architecture v74x139 arch of v72x139 is
   component V74x139h port (G L: in STD LOGIC;
         S: in STD LOGIC VECTOR (1 downto 0);
         Y L: out STD LOGIC VECTOR (3 downto 0));
   end component:
begin
   U1: v74x139h port map (G L1, S1, Y L1);
   U2: v74x139h port map (G_L2, S2, Y_L2);
end v74x139 arch;
```

## HDLs vs. programming languages (PLs)

- Program structure
  - instantiation of multiple components of the same type
  - specify interconnections between modules via schematic
  - hierarchy of modules
- Assignment
  - continuous assignment (logic always computes)
  - propagation delay (computation takes time)
  - timing of signals is important (when does computation have its effect)
- Data structures
  - size explicitly spelled out no dynamic structures
  - no pointers
- Parallelism
  - hardware is naturally parallel (must support multiple threads)
  - assignments can occur in parallel (not just sequentially)

Even though a hierarchical structure is common on HDLs and PLs, there are some fundamental differences between HDLs and PLs.

For example, continuous assignment and propagation delay are not typically supported in PLs.

### HDLs and combinational logic

- Modules specification of inputs, outputs, bidirectional, and internal signals
- Continuous assignment a gate's output is a function of its inputs at all times (doesn't need to wait to be "called")
- Propagation delay- concept of time and delay in input affecting gate output
- Composition connecting modules together with wires
- Hierarchy modules encapsulate functional blocks

HDLs can describe every aspect of combinational logic systems.

### Working with combinational logic summary

- Design problems
  - filling in truth tables
  - incompletely specified functions
  - simplifying two-level logic
- Realizing two-level logic
  - NAND and NOR networks
  - networks of Boolean functions and their time behavior
- Time behavior
- Hardware description languages
- Later
  - combinational logic technologies
  - more design case studies