blob: 5f014ce154346c55d3514c04f43535e4e687677e [file] [log] [blame]
shalanfd13eb52020-08-21 16:48:07 +02001module gpio_wb # (
2 parameter BASE_ADR = 32'h 2100_0000,
3 parameter GPIO_DATA = 8'h 00,
4 parameter GPIO_ENA = 8'h 04,
5 parameter GPIO_PU = 8'h 08,
6 parameter GPIO_PD = 8'h 0c
7) (
8 input wb_clk_i,
9 input wb_rst_i,
10
11 input [31:0] wb_dat_i,
12 input [31:0] wb_adr_i,
13 input [3:0] wb_sel_i,
14 input wb_cyc_i,
15 input wb_stb_i,
16 input wb_we_i,
17
18 output [31:0] wb_dat_o,
19 output wb_ack_o,
20
21 input [15:0] gpio_in_pad,
22 output [15:0] gpio,
23 output [15:0] gpio_oeb,
24 output [15:0] gpio_pu,
25 output [15:0] gpio_pd
26);
27
28 wire resetn;
29 wire valid;
30 wire ready;
31 wire [3:0] iomem_we;
32
33 assign resetn = ~wb_rst_i;
34 assign valid = wb_stb_i && wb_cyc_i;
35
36 assign iomem_we = wb_sel_i & {4{wb_we_i}};
37 assign wb_ack_o = ready;
38
39 gpio #(
40 .BASE_ADR(BASE_ADR),
41 .GPIO_DATA(GPIO_DATA),
42 .GPIO_ENA(GPIO_ENA),
43 .GPIO_PD(GPIO_PD),
44 .GPIO_PU(GPIO_PU)
45 ) gpio_ctrl (
46 .clk(wb_clk_i),
47 .resetn(resetn),
48
49 .gpio_in_pad(gpio_in_pad),
50
51 .iomem_addr(wb_adr_i),
52 .iomem_valid(valid),
53 .iomem_wstrb(iomem_we),
54 .iomem_wdata(wb_dat_i),
55 .iomem_rdata(wb_dat_o),
56 .iomem_ready(ready),
57
58 .gpio(gpio),
59 .gpio_oeb(gpio_oeb),
60 .gpio_pu(gpio_pu),
61 .gpio_pd(gpio_pd)
62 );
63
64endmodule
65
66module gpio #(
67 parameter BASE_ADR = 32'h 2100_0000,
68 parameter GPIO_DATA = 8'h 00,
69 parameter GPIO_ENA = 8'h 04,
70 parameter GPIO_PU = 8'h 08,
71 parameter GPIO_PD = 8'h 0c
72) (
73 input clk,
74 input resetn,
75
76 input [15:0] gpio_in_pad,
77
78 input [31:0] iomem_addr,
79 input iomem_valid,
80 input [3:0] iomem_wstrb,
81 input [31:0] iomem_wdata,
82 output reg [31:0] iomem_rdata,
83 output reg iomem_ready,
84
85 output [15:0] gpio,
86 output [15:0] gpio_oeb,
87 output [15:0] gpio_pu,
88 output [15:0] gpio_pd
89);
90
91 reg [15:0] gpio; // GPIO output data
92 reg [15:0] gpio_pu; // GPIO pull-up enable
93 reg [15:0] gpio_pd; // GPIO pull-down enable
94 reg [15:0] gpio_oeb; // GPIO output enable (sense negative)
95
96 wire gpio_sel;
97 wire gpio_oeb_sel;
98 wire gpio_pu_sel;
99 wire gpio_pd_sel;
100
101 assign gpio_sel = (iomem_addr[7:0] == GPIO_DATA);
102 assign gpio_oeb_sel = (iomem_addr[7:0] == GPIO_ENA);
103 assign gpio_pu_sel = (iomem_addr[7:0] == GPIO_PU);
104 assign gpio_pd_sel = (iomem_addr[7:0] == GPIO_PD);
105
106 always @(posedge clk) begin
107 if (!resetn) begin
108 gpio <= 0;
109 gpio_oeb <= 16'hffff;
110 gpio_pu <= 0;
111 gpio_pd <= 0;
112 end else begin
113 iomem_ready <= 0;
114 if (iomem_valid && !iomem_ready && iomem_addr[31:8] == BASE_ADR[31:8]) begin
115 iomem_ready <= 1'b 1;
116
117 if (gpio_sel) begin
118 iomem_rdata <= {gpio, gpio_in_pad};
119
120 if (iomem_wstrb[0]) gpio[ 7: 0] <= iomem_wdata[ 7: 0];
121 if (iomem_wstrb[1]) gpio[15: 8] <= iomem_wdata[15: 8];
122
123 end else if (gpio_oeb_sel) begin
124 iomem_rdata <= {16'd0, gpio_oeb};
125
126 if (iomem_wstrb[0]) gpio_oeb[ 7: 0] <= iomem_wdata[ 7: 0];
127 if (iomem_wstrb[1]) gpio_oeb[15: 8] <= iomem_wdata[15: 8];
128
129 end else if (gpio_pu_sel) begin
130 iomem_rdata <= {16'd0, gpio_pu};
131
132 if (iomem_wstrb[0]) gpio_pu[ 7: 0] <= iomem_wdata[ 7: 0];
133 if (iomem_wstrb[1]) gpio_pu[15: 8] <= iomem_wdata[15: 8];
134
135 end else if (gpio_pd_sel) begin
136 iomem_rdata <= {16'd0, gpio_pd};
137
138 if (iomem_wstrb[0]) gpio_pd[ 7: 0] <= iomem_wdata[ 7: 0];
139 if (iomem_wstrb[1]) gpio_pd[15: 8] <= iomem_wdata[15: 8];
140
141 end
142
143 end
144 end
145 end
146
147endmodule