|
@@ -15,13 +15,13 @@
|
|
// [7] - ack bit
|
|
// [7] - ack bit
|
|
// [6] - SDA
|
|
// [6] - SDA
|
|
// [5] - SCL
|
|
// [5] - SCL
|
|
-// [4] - "started" (no S will be issued before next byte)
|
|
|
|
|
|
+// [4] - "started" (SDA low while idle, no S before next byte)
|
|
// [2:1] - bits [2:1] from last command
|
|
// [2:1] - bits [2:1] from last command
|
|
// [0] - busy
|
|
// [0] - busy
|
|
//
|
|
//
|
|
// 2 - baud rate divisor (f = clk/(4*(divisor+1)))
|
|
// 2 - baud rate divisor (f = clk/(4*(divisor+1)))
|
|
//
|
|
//
|
|
-// 3 - bit [0] - send dummy clocks on SCL when idle
|
|
|
|
|
|
+// 3 - bit [0] - reset (clears "started", returns to SCL = SDA = 1 idle)
|
|
//
|
|
//
|
|
// This unit handles S(r) and P conditions by considering two classes of
|
|
// This unit handles S(r) and P conditions by considering two classes of
|
|
// symbols: "normal", when SCL is asserted at the end of the
|
|
// symbols: "normal", when SCL is asserted at the end of the
|
|
@@ -55,7 +55,7 @@ module i2c (
|
|
reg started; // S sent, but not P
|
|
reg started; // S sent, but not P
|
|
|
|
|
|
reg [1:0] outsymb; // Output symbol [abnormal, data]
|
|
reg [1:0] outsymb; // Output symbol [abnormal, data]
|
|
-
|
|
|
|
|
|
+
|
|
reg scl_out = 1'b1;
|
|
reg scl_out = 1'b1;
|
|
reg sda_out = 1'b1;
|
|
reg sda_out = 1'b1;
|
|
|
|
|
|
@@ -65,18 +65,18 @@ module i2c (
|
|
always @(negedge rst_n or posedge clk)
|
|
always @(negedge rst_n or posedge clk)
|
|
if (~rst_n)
|
|
if (~rst_n)
|
|
begin
|
|
begin
|
|
- bitctr <= 4'd14; // Idle line - wait for start condition
|
|
|
|
- busy <= 1'b0;
|
|
|
|
- outsymb <= 2'b11; // A1
|
|
|
|
- baudctr <= 8'd0;
|
|
|
|
- divisor <= 8'd209; // 84 MHz -> 100 kHz
|
|
|
|
- scl_out <= 1'b1;
|
|
|
|
- sda_out <= 1'b1;
|
|
|
|
- phase <= 2'b00;
|
|
|
|
- do_read <= 1'b0;
|
|
|
|
- end_s <= 1'bx;
|
|
|
|
- end_p <= 1'bx;
|
|
|
|
- started <= 1'b0;
|
|
|
|
|
|
+ bitctr <= 4'd14; // Idle line - wait for start condition
|
|
|
|
+ busy <= 1'b0;
|
|
|
|
+ outsymb <= 2'b11; // A1
|
|
|
|
+ baudctr <= 8'd0;
|
|
|
|
+ divisor <= 8'd209; // 84 MHz -> 100 kHz
|
|
|
|
+ scl_out <= 1'b1;
|
|
|
|
+ sda_out <= 1'b1;
|
|
|
|
+ phase <= 2'b00;
|
|
|
|
+ do_read <= 1'b0;
|
|
|
|
+ end_s <= 1'b0;
|
|
|
|
+ end_p <= 1'b0;
|
|
|
|
+ started <= 1'b0;
|
|
end
|
|
end
|
|
else
|
|
else
|
|
begin
|
|
begin
|
|
@@ -120,7 +120,9 @@ module i2c (
|
|
// started or not.
|
|
// started or not.
|
|
if (~busy)
|
|
if (~busy)
|
|
begin
|
|
begin
|
|
- outsymb <= { 1'b1, ~started };
|
|
|
|
|
|
+ started <= started & ~end_p;
|
|
|
|
+ bitctr <= started & ~end_p ? 4'd0 : 4'd14;
|
|
|
|
+ outsymb <= { 1'b1, ~(started & ~end_p) };
|
|
end
|
|
end
|
|
else
|
|
else
|
|
begin
|
|
begin
|
|
@@ -155,7 +157,7 @@ module i2c (
|
|
4'd13: begin
|
|
4'd13: begin
|
|
started <= 1'b0;
|
|
started <= 1'b0;
|
|
outsymb <= 2'b11; // A1
|
|
outsymb <= 2'b11; // A1
|
|
- busy <= 1'b0;
|
|
|
|
|
|
+ busy <= 1'b0;
|
|
end
|
|
end
|
|
|
|
|
|
// Start condition
|
|
// Start condition
|
|
@@ -199,6 +201,16 @@ module i2c (
|
|
if (wstrb[0])
|
|
if (wstrb[0])
|
|
divisor <= wdata[7:0];
|
|
divisor <= wdata[7:0];
|
|
end
|
|
end
|
|
|
|
+ 2'b11: begin
|
|
|
|
+ if (wstrb[0])
|
|
|
|
+ begin
|
|
|
|
+ if (wdata[0])
|
|
|
|
+ begin
|
|
|
|
+ end_p <= 1'b1;
|
|
|
|
+ end_s <= 1'b0;
|
|
|
|
+ end
|
|
|
|
+ end
|
|
|
|
+ end
|
|
default: begin
|
|
default: begin
|
|
// Do nothing
|
|
// Do nothing
|
|
end
|
|
end
|
|
@@ -215,12 +227,12 @@ module i2c (
|
|
2'b01: rdata = { 16'b0, rreg, i2c_sda, i2c_scl, started, 1'b0,
|
|
2'b01: rdata = { 16'b0, rreg, i2c_sda, i2c_scl, started, 1'b0,
|
|
end_p, end_s, busy | do_read };
|
|
end_p, end_s, busy | do_read };
|
|
2'b10: rdata = { 24'b0, divisor };
|
|
2'b10: rdata = { 24'b0, divisor };
|
|
- default: rdata = 32'bx;
|
|
|
|
|
|
+ default: rdata = 32'b0;
|
|
endcase // casez (addr)
|
|
endcase // casez (addr)
|
|
|
|
|
|
//
|
|
//
|
|
// IRQ (level) when unit idle
|
|
// IRQ (level) when unit idle
|
|
//
|
|
//
|
|
- assign irq = ~(busy | do_read);
|
|
|
|
|
|
+ assign irq = ~(busy | do_read | unstart);
|
|
|
|
|
|
endmodule // i2c
|
|
endmodule // i2c
|