# TSTE12 Design of Digital Systems Lecture 6

Kent Palmkvist



TSTE12 Design of Digital Systems, Lecture 6

2024-09-13

2

# Agenda

- Practical issues
  - Handins
  - Audio codec function and interface
- High abstraction level modelling



## TSTE12 Deadlines Y,D,ED

- Final version of Requirement specification today 13 September
- Initial version design sketch and project plan tuesday 17 September
  - Design sketch (proper document) describe block structure and functions
  - Project plan describe activities (what, who and when)
- Weekly meetings should start
  - Internal weekly meeting with transcript sent to supervisor
- Lab 2 soft deadline moved to Thursday 24 September at 21.00
  - Lab 2 results will be checked after project end



TSTE12 Design of Digital Systems, Lecture 6

2024-09-13

1

## TSTE12 Deadlines MELE, erasmus

- First project meeting no later than Monday 16 September
- Tuesday 17 September: First version of requirement specification
- Wednesday 18 September 21.00: Lab 1 deadline
  - Pass required to be allowed continued project participation



#### 5

# Handin (homework), Individual!

- 1st handin deadline Monday 23 September
  - Available on web from Monday 16 September
- Submit answers using Lisam assignment function (individual work!!!)
- Theory question answers entered direct as text (see assignments on web)
- Use your own home directory for code answer testing (Do NOT use the lab group directory)
  - Use ~/TSTE12/
- Use a special terminal window when working with handins

module load TSTE12; TSTE12handin



TSTE12 Design of Digital Systems, Lecture 6

2024-09-13

6

## Individual handin task, cont.

- Create handin code answers using a plain text editor
  - emacs, vim, or the built-in editor in modelsim
  - See in the tutorial how start and use modelsim
- Upload the answers of the coding tasks onto Lisam (TSTE12 course room submission)
- Remember to compile and simulate the design
  - Will use source code for checking handin results
- Do not use handin directory for anything else but handin code and answers you make yourself!



## Individual handin, cont.

- Hand-ins are **individual** work!
  - Ask me if there are questions about the handin
- Hand-ins are checked automatically (using scripts)
  - Make sure all names and types are correct in code answers
    - Datatype bit is **not** the same as std\_logic!
  - Test your code! Do not assume that you have written correct code.
  - NOTE: Do NOT use hdl-designer (do not start the software using TSTE12lab or TSTE12proj)
  - See modelsim tutorial on exercise page
    - www.isy.liu.se/edu/kurs/TSTE12/kursmaterial



TSTE12 Design of Digital Systems, Lecture 6

2024-09-13

ρ

# DE2-115 board components

- · Audio codec used to input/output analog audio signal
- Codec function
  - Codec can be used in multiple configurations
  - Contains clock generators, A/D, D/A and filters
  - Loopback, volume control
- Codec configuration
  - Default configuration defined in documentation



#### 9

## Individual

- DE2-115 FPGA default design
  - DE2-115 loads a default design at power on
    - Microprocessor design running (NIOS II Soft process, i.e. written i VHDL)
    - Check switches SW3 downto SW0 to select what to do with the SRAM contents and Codec init
    - Infinite loop: read switches, update LEDs, updates 7-segment display.
    - · Help text shown on VGA screen
  - Default design is not the standard design described in the DE2-115 user manual
  - Do not depend on power-on defaults
  - Allows configuration of memory and codec for testing



TSTE12 Design of Digital Systems, Lecture 6

2024-09-13 10

# Codec programming (I2C)

- · Codec configured using an I2C bus
- General structure
  - Multichip bus (all chips connect to the same pins)
- Pullup
  - Only assign 'Z' or '0' on pins
  - Pullup will translate 'Z' to '1' if other chips does not sink that pin
  - Separate output and input signals to/from pin
  - Values do not change immediately (slow voltage changes)





STOP

## **I2C** protocol

- Bidirectional protocol
  - Wired-and using pull-up
  - Send byte by byte
  - MSB first
- FPGA work as master
- Slave (codec) responds with ack after each byte

SDIN

START

R ADDR R/W

- Pulls down SDA in ack cycle
- Simple solution: assume ack (do not check)



TSTE12 Design of Digital Systems, Lecture 6

# Sequential logic

- JK flipflop with asynchronous set/reset
- entity JKFF is
   generic(SRDEL,CLKDEL: TIME);
   port(S,R,J,K,CLK: in BIT;
   Q,QN: inout BIT);
  end JKFF;
- Edge trigged using 'event
- Asynchronous update
  - Higher priority than clocked circuit function
- Synchronous update
  - Note use of elsif (must be used)
  - Edge trigged using 'event

```
architecture ALG of JKFF is
begin
  process(CLK,S,R)
  begin
     if S = '1' and R = '0' then
  Q <= '1' after SRDEL;
  QN <= '0' after SRDEL;</pre>
     elsif S = '0' and R = '1' then
        Q <= '0' after SRDEL;
        QN <= '1' after SRDEL;
     elsif CLK'EVENT and CLK = '1' and
             S='0' and R='0' then
        if J = '1' and K = '0' then Q <= '1' after CLKDEL; QN <= '0' after CLKDEL;
        elsif J = '0' and K = '1'
 Q \leftarrow '0' after CLKDEL;
                                             then
        QN <= '1' after CLKDEL;
elsif J= '1' and K= '1'
           Q <= not Q after CLKDEL;
           QN <= not QN after CLKDEL;
        end if;
     end if;
  end process;
end ALG;
```

2024-09-13



# Sequential logic, cont.

- Register with alternative design
  - Use a guarded statement
  - Use 'STABLE instead of 'EVENT

```
entity REG is
  generic(DEL: TIME);
  port(RESET, LOAD, CLK: in BIT;
        DATA_IN: in BIT_VECTOR(3 downto 0);
        Q: inout BIT_VECTOR(3 downto 0));
end REG;

architecture DF of REG is
begin
  REG: block(not CLK'STABLE and CLK ='1')
  begin
  Q <= guarded "0000" after DEL when RESET ='1' else
        DATA_IN after DEL when LOAD ='1' else
        Q;
  end block REG;
end DF;</pre>
```



TSTE12 Design of Digital Systems, Lecture 6

2024-09-13 14

# Output feedback problems

- Entity output can NOT be read in the architecture
- Three solutions
  - Use INOUT
    - Does not match OUT, enables output values to influence internal signal values
  - Use BUFFER
    - Does not match OUT, complicates building testbenches etc.
  - Use OUT with a temporary signal
    - use temporary signal everywhere needed (read and assign), assign entity out signal at the end of the architecture



# Sequential logic, oscillator

- Run signal indicate when to start generating clock pulses.
- Feedback example
  - Need extra variable to guarantee complete clock cycles
  - Simulation use only, will not synthesize

```
architecture ALG of CLOCK_GENERATOR is
  signal CLOCK: BIT;
begin
  process (RUN, CLOCK)
    variable CLKE: BIT := '0';
  beain
    if RUN='1' and not RUN'STABLE then
  CLKE := '1';
      CLOCK <= transport '0' after PER/2;
      CLOCK <= transport '1' after PER;
    end if;
    if RUN='0' and not RUN'STABLE then
    CLKE := '0';
end if;
    if CLOCK='1' and not CLOCK'STABLE
      and CLKE = '1'then
CLOCK <= transport '0' after PER/2;
      CLOCK <= transport '1' after PER;
    CLK <= CLOCK;
  end process;
end ALG;
```



TSTE12 Design of Digital Systems, Lecture 6

2024-09-13 16

# Sequential logic, oscillator

- Wait statement based
  - Can not have both wait and sensitivity list in process

```
entity COSC is
  generic(HI_TIME,LO_TIME: TIME);
  port(RUN: in BIT; CLOCK: out BIT := '0');
end COSC;
architecture ALG of COSC is
begin
  process
  begin
    wait until RUN ='1';
    while RUN = '1' loop
      CLOCK <= '1'
      wait for HI_TIME;
      CLOCK <= '0';
      wait for LO_TIME;
    end loop;
  end process;
end ALG;
```



## Numeric calculations

- Bit-vectors (and std\_logic\_vectors) does not correspond to a numeric value
  - "1011" could mean 11 in decimal (unsigned), or -5 in decimal (2's complement), or even -3 if sign-magnitude would be used
- Additional definitions are included in supporting packages to enable arithmetic on bit-vectors and std\_logic\_vectors
  - ieee.numeric\_bit.all
  - ieee.numeric\_std.all
- Must use defined types signed or unsigned to allow calculations
  - Same definitions as bit\_vector and std\_logic\_vector
  - Can copy values between types due to same element type



TSTE12 Design of Digital Systems, Lecture 6

2024-09-13

Ω

Type conversion (vectors, signed, unsigned, integer)

- Casting between vector datatypes does not change element pattern
  - unsigned(A)
  - signed(B)
  - std logic vector(C)
  - bit vector(D)
- Conversion to/from integer require separate translation function





# Numeric calculations example

- Counter incrementing 3-bit count value each clock cycle
  - Asynchronous reset example
  - Clock edge detection using 'event

```
library ieee;
use ieee.numeric_bit.all;
```

```
entity INL3_KB is
port (
C: in bit;
R: in bit;
Q: out bit_vector(1 to 3));
end entity;
```

```
architecture KB of INL3_KB is begin
```

```
process(C,R)
  variable count : unsigned(1 to 3);
begin
  if R = '1' then
    count := (others => '0');
  elsif C'event and (C='1') then
    count := count + 1;
  end if;
  Q <= bit_vector(count);
end process;</pre>
```

end architecture;



TSTE12 Design of Digital Systems, Lecture 6

2024-09-13 20

## Numeric calculations details

- - Must signextend to detect carry
- Adding different length vectors will sign extend the shortest one
  - May still get overflow
- Multiplication always generates an output number of bits equal to the total number of input bits
  - Multiplying a 3-bit input with a 4-bit input generates a 7-bit output result



## Avoid old packages

- Before the introduction of numeric\_std and numeric\_bit there where other libraries
  - std\_logic\_unsigned, std\_logic\_signed
  - std\_logic\_arith
- Do NOT use these, they are obsolete
  - Made it difficult/impossible to mix signed and unsigned



TSTE12 Design of Digital Systems, Lecture 6

2024-09-13 22

# Including integers

- Integers can be used for synthesis
  - If synthesis tool cannot figure out the limits, the result is 32-bit arithmetic
  - Subtypes (limiting range) help to reduce hardware and catch unexpected use
- Integers will be implemented as bitvectors
  - Either unsigned or signed (2's complement)
  - Translation between integer and bitvectors exist
    - x\_signed := to\_signed(y\_int,x\_signed'size);
  - Translation other way around (unsigned to integer value)
    - y\_int = to\_integer(x\_signed);



## Another aspect of signal assignment

- One signal can be assigned from different parts of the code
  - Support multiple entities driving the same wire
  - Example: Databus in a computer connecting multiple memories and CPU
- Modelling must be strict and clear
  - Same result independant of simulator tool
  - Should not be able to detect the order the processes where calculated
- Not all data types support multiple sources for the value



TSTE12 Design of Digital Systems, Lecture 6

2024-09-13 24

# Multiple assignment on one signal

- Each process containing a signal assignment will have a driver in the simulator generating a contribution to the final signal value
  - Concurrent signal assignments will have one driver each
  - Processes only have one driver for each signal (even with multiple assignment)
  - The signal update seen before is done individually on each driver
  - One driver does not know anything about other drivers
- When the value of a signal is fetched, the contributions from the different drivers current values are collected.
  - The resulting signal value depends on the definition of how to combine the values from the different drivers, using a resolution function



# Example of data types supporting multiple drivers

- Signals driven by multiple drivers must be resolved
  - Use a special function that resolves multiple drivers
- Resolution function
  - Example: Wired-OR
    - signal X1 : WIRED\_OR Bit;
    - subtype STD\_LOGIC is RESOLVED STD\_ULOGIC;
    - signal Y2 : STD\_LOGIC;
  - RESOLVED is the resolution function name
    - Called every time the value of the signal is calculated
    - Gets all driver values as input



### 26

## Multivalued logic

- Not enough with 0 and 1 to model "real" logic
- Example: Bus
  - Requires bus release
  - Signal assignment driver can not drop its value
  - Use a value to indicate not driven, and indicate non-driven signals (Z)
  - Need to indicate conflicting driver (X)

## Multivalued logic in VHDL

- Alternative to data type BIT
   Type MVL4 is ('X', '0', '1', 'Z');
   Type MVL4\_VECTOR is array(NATURAL range <>) of MVL4;
- X leftmost to make it the initial value unless explicitly initialized in the code

28

## Multidriver signals

- Requires a resolution signal
- Different combinations possible
  - X always overrides others
  - 0 and 1 at the same time gives X
  - Z and Z gives Z

### Resolution function definition

Subtype DotX is wiredX MVL4;

- WiredX is the name of the resolution function
   Function WiredX (V:MVL4\_VECTOR)
   return MVL4;
- Where V is a vector containing all values of all drivers of a signal

30

## Resolution function implementation

Implement as a loop and lookup table

```
Function wiredX (V: MVL4_VECTOR) return MVL4 is
   Variable result: MVL4:= 'Z';
Begin
   For i in V'RANGE loop -- range not known in advance
    Result = table_WIREDX(result, V(i));
    Exit when result = 'X';
   End loop;
   Return result;
End wiredX;
```

## Resolution function impl., cont.

- Check of X in loop is not necessary, but speed up simulation
- Table should then look like:

32

## Resolution function impl. Cont.

- Table lookup may be used for most functions
  - Not possible to know the order of the value in V, may therefore require a more complex algorithm

## Bus data type

Type busX is array (Natural range <>) of DotX;

- However, a new data type requires all logic operations to be specified
  - Complicated
- Better approach: conversion function
  - Only read bus using a call to a Sense function

Function Sense (value : busX) return bit vector;

- Only assign value to the bus using the Drive function

Function Drive (value : bit vector) return busX



TSTE12 Design of Digital Systems, Lecture 6

2024-09-13 34

# Algorithmic level development

- Specification in many cases in natural language
  - Ambigous description in many cases
- Want an executable specification
  - Allows testing of the behavior the description describes
- Use VHDL to capture the specification
  - Use the full language capabilities
  - Description not intended for synthesis



## Process Model Graph (PMG)

- · Typical example
- Arcs describes signals with names and delays
  - example process 4 to 1
- VHDL Code example: S <= xxx after DEL S;
- · Physical or functional partitioning
  - Single process may map to multiple hardware units
  - Multiple process may map to single hardware unit



TSTE12 Design of Digital Systems, Lecture 6

2024-09-13 36

# **Graph Elements**

- Two types of signals
  - Triggering signal, put in sensitivity list
  - ─⊳ Sampled signal
- Signals without delay information has a delay of one delta-t
- Signals may be driven by multiple processes. Requires a resolution function
- Signals may be bidirectional. Requires also a resolution function







## Example approach

- Map groups of sentences onto VHDL processes
- Assign each process an activity list
- Develop VHDL code that implements each activity



TSTE12 Design of Digital Systems, Lecture 6

2024-09-13 38

## Example: Serial to Paralell converter

- English text description
  - The 8-bit parallel word (PARIN) is loaded into the converter when the control signal LD makes a zero to one transition At this time the status signal BUSY is set high. The data is shifted out serially at a rate controlled by the input shift clock SHCLK. Shifting occurs at the rise of the clock. BUSY remains high until shifting is complete. While BUSY is high, no further loads will be accepted.
- Note some sentences are shared between functions
- Two processes: LOAD and SHIFT



## Serial to Paralell converter, cont.

- LOAD: (a) 8-bit parallel word (PARIN) load when LD makes a zero to one transition. Set BUSY high. (b) BUSY remains high until shift complete. No new loads while BUSY high
- SHIFT: (a) Data shifted out controlled by rising edge of SHCLK.
   (b) BUSY remain high until shift complete

```
PARIN LOAD PREG SHIFT SO SH_COMP SHCLK
```

```
entity PAR_TO_SER is
  port(LD,SHCLK: in BIT;
     PARIN: in BIT_VECTOR(0 to 7);
     BUSY: inout BIT := '0';
     S0: out BIT);
end PAR_TO_SER;
```



TSTE12 Design of Digital Systems, Lecture 6

## PMG version

- Corresponding code based on processes
- PMG defines interface of each process + signals between the processes
- Code start by defining processes and comments about activities

```
2024-09-13 40
architecture TWO_PROC of PAR_TO_SER is
  signal SH_COMP: BIT :='0';
  signal PREG: BIT_VECTOR(0 to 7);
  LOAD:process(LD,SH COMP)
  begin
    ---- Activities:
      ----1)Register Load
      ----2)Busy Set
      ----3)Busy Reset
  end process LOAD;
  SHIFT: process(BUSY, SHCLK)
    variable COUNT: INTEGER;
    variable OREG: BIT VECTOR(0 to 7);
    ----Activities:
     ----1)Shift Initialize
      ----2)Shift
      ----3)Shift Complete
  end process SHIFT;
end TWO_PROC;
```



## PMG -> Code

 Each process has a check for an event, and then a part that execute the data operations

```
SHIFT: process (BUSY, SHCLK)
  variable COUNT: INTEGER;
  variable OREG: BIT_VECTOR(0 to 7);
begin
   ---Activities:
  if BUSY'EVENT and
    BUSY = '1' then
    ----1)Shift Initialize
                               LOAD:process(LD, SH_COMP)
   COUNT := 7;
                               begin
     OREG := PREG;
                                 ---- Activities:
   SH_COMP <= '0';
                                 if LD'EVENT and LD='1'
 end if;
                                   and BUSY='0' then
  if SHCLK'EVENT and
                                   ----1)Register Load
     SHCLK= '1'and
    BUSY='1' then
                                  PREG <= PARIN;</pre>
                                   ----2)Busy Set
    ----2)Shift
                                  BUSY <= '1';
    SO<=OREG(COUNT);
                                 end if;
    COUNT := COUNT - 1;
                                   if SH_COMP'EVENT
    ----3)Shift Complete
                                     and SH_COMP='1' then
    if COUNT < 0 then
                                     ----3)Busy Reset
      SH_COMP <= '1';
                                     BUSY <= '0';
    end if;
                                   end if;
  end if;
                               end process LOAD;
end process SHIFT;
```



TSTE12 Design of Digital Systems, Lecture 6

2024-09-13 42

# Timing example

- New model: Buffered register
  - Loaded on rise of the strobe (STRB)
- English description:
  - The register is loaded on the rise of the strobe (STRB), and assuming that the output buffers are enabled, the output of the buffers will change  $t_{\rm SD}$  nanoseconds later. The enable condition for the register buffer is the AND of the DS1 and invers of DS2 inputs. Any change in the enable condition will cause the outputs to change  $t_{\rm FD}$  nanoseconds later.



# Timing example, cont.

- Three processes: PREG, ENABLE, OUTPUT
- Add delay on wires

```
t_{SD} = STRB_DEL + ODEL

t_{ED} = EN_DEL + ODEL
```





TSTE12 Design of Digital Systems, Lecture 6

2024-09-13 44

Timing example, cont.

PREG: process(STRB)

```
entity BUFF_REG is
  generic(
    STRB_DEL,EN_DEL,ODEL: TIME);
port(
    DI: in BIT_VECTOR(1 to 8);
    STRB: in BIT;
    DS1: in BIT;
    NDS2: in BIT;
    D0: out BIT_VECTOR(1 to 8));
end BUFF_REG;

architecture THREE_PROC of BUFF_REG is
    signal REG: BIT_VECTOR(1 to 8);
    signal ENBLD: BIT;
begin
```

```
begin
      if (STRB = '1') then
        RÈG <=DI after STRB_DEL;
      end if;
 end process PREG;
 ENABLE: process(DS1, NDS2)
      ENBLD <= DS1 and not NDS2 after EN_DEL;
 end process ENABLE;
 OUTPUT: process(REG, ENBLD)
      if (ENBLD = '1') then
        DO <= REG after ODEL;
      else
        DO <= "11111111" after ODEL;
      end if;
 end process OUTPUT;
end THREE_PROC;
```



# Process complexity trade-off

- Number of signals
  - Many signals => slow simulation
- · Large processes
  - Complex behavior may not match specification
- Ease of mapping to hardware
  - More processes may simplify mapping



TSTE12 Design of Digital Systems, Lecture 6

2024-09-13 46

# Checking timing

- · Additional requirements
  - DI stable SUT ns before STRB rise
  - DI stable HT ns after STRB rise
  - STRB minimum high duration MPW ns
- · Implement checks using assert statements



## Timing Check placement

- Tests in architecture must be copied between architectures
  - May introduce errors
  - If changed, many architectures must be changed
- · Solution: Place checks in the entity
  - Check always executed, independent of selected architecture



TSTE12 Design of Digital Systems, Lecture 6

2024-09-13 48

# Timing check example

```
Entity BUFF_REG is
   Generic (STRB_DEL, EN_DEL, ODEL, SUT, HT, MPW: TIME);
   Port (DI: in bit_vector(1 to 8);
        STRB : in bit ; DS1 : in bit;
        NDS2 : in bit;
        D0 : out bit_vector(1 to 8));

Begin
   Assert STRB'stable or (STRB = '0') or DI'stable(SUT)
        Report "Setup time Failure";

Assert STRB'delayed(HUT)'stable or
        (STRB'delayed(HT) = '0') or DI'STABLE(HT)
        Report "Hold Time Failure";

Assert STRB'stable or (STRB = '1') or
        STRB'delayed'stable(MPW)
        Report "Minimum pulse width failure";
```

End BUFF\_REG;



