

module i2c-master (

input logic clk,reset,

↳ could have used wire, but  
it's preferred System verilog.  
standard for signals.

we can use continuous (assign)  
or procedural assignment.

— [7:0] din,

↳ writing : 8-bit data to slave  
Reading : din[0] for acknowledge  
that we have read.

— [15:0] dver,

— [2:0] cmd,

↳ to tell FSM what operation  
to perform (start stop - write)  
5 states in total)

— wr\_i2c,

↳ when cmd is received, then  
wr\_i2c goes high for one  
cycle.

Tells FSM to execute that  
command.

— tri scl,

— tri sda,

bidirectional —  
↳ tri-state buffer.



when  $0 \rightarrow$  line 1'b0  
 $1 \rightarrow$  line 1'b2(open)

letting one external pull-up resistor to pull line high.

— ready, done-tick, ack,  
 when 1, module  
 is in idle or hold state.  
 goes high for a moment  
 a byte (read or write) is  
 completed.  
 - used to know when done is  
 valid

$0 \rightarrow$  slave acknow-  
 ledges the byte.  
 $1 \rightarrow$  slave did not ackn—  
 or is busy.

— [7:0] dout . );

localparam

|         |   |   |   |
|---------|---|---|---|
| Start   | = | 0 | : |
| Write   | = | 1 | : |
| read.   | = | 2 | : |
| Stop    | = | 3 | , |
| Restart | = | 4 | ; |

```
typedef enum { ... } static_type;
```

static\_type state\_reg, state\_next;

c-reg. c-rect → counter

quarter, half → 2 clk pulses

1 clk pulse

— [8:0] tx\_rey, th\_nent

→ 8 for date, 1 for ask

— [B:0] em, ey, en, nest.

— — cmd reg —

— [3.0] bit reg → to count the 9 bits

sda\_out, sd\_out, sda-reg, sd-reg

to determine if  
motor wants to  
drive or not

when in data  
phase.

— into → when master should be bettering.

→ treated as input.

- mark → din [o]

// output control logic

## the idle state

in  $\Sigma 2C$  is

pulled high

① always @ ( )

$$\frac{4}{3} (\text{resct}) = 1' b 1 ;$$

else

$$-\text{reg} = -\text{out};$$

② assign scl = (scl - reg) ? 1'b2 : 1'b0

open-drain (0, z)  
not tri-state buffer (0, 1, z)

↳ here we used open drain logic because it allows slave to stretch the clock.

i.e. • If master want  $\text{scl} = 1$ ,  $\text{scl\_reg} = 1$ .

driven goes z, then pull up resistor makes the line 1.

• If slave is not ready, it can pull the line low (0).

• If used tri-state driver, the master would actively drive '1' & the slave pulling '0' would cause a short circuit.

③ assign int0 = ( ————— || ————— )

↳ to declare when master should release the sda line to listen.

↳ (data-phase & cmd-reg == RD cmd & bit-reg < 8) : Reading data.

↳ during a read command, for the first 8 bits

↳ (data-phase & & cmd-reg == WR-cmd  
& & bit-reg == 8) ; Receiving ack.

↳ during write command, on the 9<sup>th</sup> bit, the master should listen for the slave ack.

④ assign sda = (into || sda-reg) ? 1'b2:1'bo;

when listening ↗  
release the line. ↗ we want to send a line

⑤ — dout = rx-reg [8:1];  
ack = rx-reg [0];  
nack = dim(0);

// fsmd (with datapath)

① always @ ( — )  
→ FSM to idle state &  
clear all counters.  
→ cmd clk pulse.

stage ++

cmd reg <= next

c-reg ++

th

bit-reg ++

rx

② assign qtr = dwsr;  
assign half = (qtr \* 2) to get  
half period

## # next-state Logic

case (state\_reg)

idle: ready<sup>i</sup> = 1'b1;  
if (wr\_i2 & cmd == start)  
next-state

start1: sda\_out = 1'b0,  
waits for half a period.  
next-state.

start2: scl\_out = 1'b0  
waits for quarter wave  
next-state.

hold: ready<sup>i</sup> = 1'b1.  
scl\_out = 1'b0  
sda\_out = 1'b0  
waits for wr\_i2C  
when it gets a command.

stop CMD  $\rightarrow$  goes to stop2.

Restart CMD  $\rightarrow$  goes to restart.

WR CMD or RD-CMD  $\rightarrow$

→ reset bit counter.

Load the transmit register  
next state = data1.

data1: scl\_out = 1'b0

sda\_out = tx\_reg[8]

wait qtr.

then to data2.

data2: scl\_out = 1'b1.

sm-next = {rx\_reg[7:0], sda1};

wait qtr.

next state

data3: scl\_out = 1'b1.

wait qtr

next state

data4: scl\_out = 1'b0;

wait qtr.

if (bit\_reg == 8).

↳ ✓ don'tick = 1 & go to  
data end

↳ ✗ bit-next ++, shift

tx-reg.left & goes to  
data2.

data.end : to keep lines low &  
wait quit  
goes to hold to wait for  
next command.

restart : wait half  
 $c_{\text{rest}} = 0$   
goes to start 1.

stop 1 : sdu-out = 1'b0.  
wait half.  
goes to stop 2.

