| /* | 
 |  *  PicoSoC - A simple example SoC using PicoRV32 | 
 |  * | 
 |  *  Copyright (C) 2017  Clifford Wolf <clifford@clifford.at> | 
 |  * | 
 |  *  Permission to use, copy, modify, and/or distribute this software for any | 
 |  *  purpose with or without fee is hereby granted, provided that the above | 
 |  *  copyright notice and this permission notice appear in all copies. | 
 |  * | 
 |  *  THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES | 
 |  *  WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF | 
 |  *  MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR | 
 |  *  ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES | 
 |  *  WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN | 
 |  *  ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF | 
 |  *  OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. | 
 |  * | 
 |  */ | 
 |  | 
 | `default_nettype wire | 
 |  | 
 | module simpleuart ( | 
 | 	input clk, | 
 | 	input resetn, | 
 |  | 
 | 	output ser_tx, | 
 | 	input  ser_rx, | 
 |  | 
 | 	input   [3:0] reg_div_we, | 
 | 	input  [31:0] reg_div_di, | 
 | 	output [31:0] reg_div_do, | 
 |  | 
 | 	input         reg_dat_we, | 
 | 	input         reg_dat_re, | 
 | 	input  [31:0] reg_dat_di, | 
 | 	output [31:0] reg_dat_do, | 
 | 	output        reg_dat_wait | 
 | ); | 
 | 	reg [31:0] cfg_divider; | 
 |  | 
 | 	reg [3:0] recv_state; | 
 | 	reg [31:0] recv_divcnt; | 
 | 	reg [7:0] recv_pattern; | 
 | 	reg [7:0] recv_buf_data; | 
 | 	reg recv_buf_valid; | 
 |  | 
 | 	reg [9:0] send_pattern; | 
 | 	reg [3:0] send_bitcnt; | 
 | 	reg [31:0] send_divcnt; | 
 | 	reg send_dummy; | 
 |  | 
 | 	assign reg_div_do = cfg_divider; | 
 |  | 
 | 	assign reg_dat_wait = reg_dat_we && (send_bitcnt || send_dummy); | 
 | 	assign reg_dat_do = recv_buf_valid ? recv_buf_data : ~0; | 
 |  | 
 | 	always @(posedge clk) begin | 
 | 		if (!resetn) begin | 
 | 			cfg_divider <= 1; | 
 | 		end else begin | 
 | 			if (reg_div_we[0]) cfg_divider[ 7: 0] <= reg_div_di[ 7: 0]; | 
 | 			if (reg_div_we[1]) cfg_divider[15: 8] <= reg_div_di[15: 8]; | 
 | 			if (reg_div_we[2]) cfg_divider[23:16] <= reg_div_di[23:16]; | 
 | 			if (reg_div_we[3]) cfg_divider[31:24] <= reg_div_di[31:24]; | 
 | 		end | 
 | 	end | 
 |  | 
 | 	always @(posedge clk) begin | 
 | 		if (!resetn) begin | 
 | 			recv_state <= 0; | 
 | 			recv_divcnt <= 0; | 
 | 			recv_pattern <= 0; | 
 | 			recv_buf_data <= 0; | 
 | 			recv_buf_valid <= 0; | 
 | 		end else begin | 
 | 			recv_divcnt <= recv_divcnt + 1; | 
 | 			if (reg_dat_re) | 
 | 				recv_buf_valid <= 0; | 
 | 			case (recv_state) | 
 | 				0: begin | 
 | 					if (!ser_rx) | 
 | 						recv_state <= 1; | 
 | 					recv_divcnt <= 0; | 
 | 				end | 
 | 				1: begin | 
 | 					if (2*recv_divcnt > cfg_divider) begin | 
 | 						recv_state <= 2; | 
 | 						recv_divcnt <= 0; | 
 | 					end | 
 | 				end | 
 | 				10: begin | 
 | 					if (recv_divcnt > cfg_divider) begin | 
 | 						recv_buf_data <= recv_pattern; | 
 | 						recv_buf_valid <= 1; | 
 | 						recv_state <= 0; | 
 | 					end | 
 | 				end | 
 | 				default: begin | 
 | 					if (recv_divcnt > cfg_divider) begin | 
 | 						recv_pattern <= {ser_rx, recv_pattern[7:1]}; | 
 | 						recv_state <= recv_state + 1; | 
 | 						recv_divcnt <= 0; | 
 | 					end | 
 | 				end | 
 | 			endcase | 
 | 		end | 
 | 	end | 
 |  | 
 | 	assign ser_tx = send_pattern[0]; | 
 |  | 
 | 	always @(posedge clk) begin | 
 | 		if (reg_div_we) | 
 | 			send_dummy <= 1; | 
 | 		send_divcnt <= send_divcnt + 1; | 
 | 		if (!resetn) begin | 
 | 			send_pattern <= ~0; | 
 | 			send_bitcnt <= 0; | 
 | 			send_divcnt <= 0; | 
 | 			send_dummy <= 1; | 
 | 		end else begin | 
 | 			if (send_dummy && !send_bitcnt) begin | 
 | 				send_pattern <= ~0; | 
 | 				send_bitcnt <= 15; | 
 | 				send_divcnt <= 0; | 
 | 				send_dummy <= 0; | 
 | 			end else | 
 | 			if (reg_dat_we && !send_bitcnt) begin | 
 | 				send_pattern <= {1'b1, reg_dat_di[7:0], 1'b0}; | 
 | 				send_bitcnt <= 10; | 
 | 				send_divcnt <= 0; | 
 | 			end else | 
 | 			if (send_divcnt > cfg_divider && send_bitcnt) begin | 
 | 				send_pattern <= {1'b1, send_pattern[9:1]}; | 
 | 				send_bitcnt <= send_bitcnt - 1; | 
 | 				send_divcnt <= 0; | 
 | 			end | 
 | 		end | 
 | 	end | 
 | endmodule |