aboutsummaryrefslogtreecommitdiffstats
path: root/fpga/usrp3/lib/packet_proc/vita_eth_framer.v
diff options
context:
space:
mode:
Diffstat (limited to 'fpga/usrp3/lib/packet_proc/vita_eth_framer.v')
-rw-r--r--fpga/usrp3/lib/packet_proc/vita_eth_framer.v137
1 files changed, 137 insertions, 0 deletions
diff --git a/fpga/usrp3/lib/packet_proc/vita_eth_framer.v b/fpga/usrp3/lib/packet_proc/vita_eth_framer.v
new file mode 100644
index 000000000..a5e02e899
--- /dev/null
+++ b/fpga/usrp3/lib/packet_proc/vita_eth_framer.v
@@ -0,0 +1,137 @@
+
+// vita_eth_framer
+// Takes a vita stream in and adds udp, ip, and ethernet framing
+// Uses 8 setting reg addresses. First 4 are simple registers:
+// BASE+0 : Upper 16 bits of ethernet src mac
+// BASE+1 : Lower 32 bits of ethernet src mac
+// BASE+2 : IP src address
+// BASE+3 : UDP src port
+//
+// Next 4 control write ports on a RAM indexed by destination field of stream ID
+// BASE+4 : Dest SID for next 3 regs
+// BASE+5 : Dest IP
+// BASE+6 : Dest UDP port, upper 16 bits of dest mac
+// BASE+7 : Lower 32 bits of dest mac
+
+module vita_eth_framer
+ #(parameter BASE=0)
+ (input clk, input reset, input clear,
+ input set_stb, input [7:0] set_addr, input [31:0] set_data,
+ input [63:0] in_tdata, input in_tlast, input in_tvalid, output in_tready,
+ output [63:0] out_tdata, output [3:0] out_tuser, output out_tlast, output out_tvalid, input out_tready,
+ output [31:0] debug );
+
+ localparam SR_AWIDTH = 8;
+
+ reg [31:0] sid;
+ reg [15:0] vita_len;
+
+ reg [2:0] vef_state;
+ localparam VEF_IDLE = 3'd0;
+ localparam VEF_PAYLOAD = 3'd7;
+
+ reg [63:0] tdata;
+
+ always @(posedge clk)
+ if(reset | clear)
+ begin
+ vef_state <= VEF_IDLE;
+ sid <= 32'd0;
+ vita_len <= 16'd0;
+ end
+ else
+ case(vef_state)
+ VEF_IDLE :
+ if(in_tvalid)
+ begin
+ vef_state <= 1;
+ sid <= in_tdata[31:0];
+ vita_len <= in_tdata[47:32];
+ end
+ VEF_PAYLOAD :
+ if(in_tvalid & out_tready)
+ if(in_tlast)
+ vef_state <= VEF_IDLE;
+ default :
+ if(out_tready)
+ vef_state <= vef_state + 3'd1;
+ endcase // case (vef_state)
+
+ assign in_tready = (vef_state == VEF_PAYLOAD) ? out_tready : 1'b0;
+ assign out_tvalid = (vef_state == VEF_PAYLOAD) ? in_tvalid : (vef_state == VEF_IDLE) ? 1'b0 : 1'b1;
+ assign out_tlast = (vef_state == VEF_PAYLOAD) ? in_tlast : 1'b0;
+ assign out_tuser = ((vef_state == VEF_PAYLOAD) & in_tlast) ? {1'b0,vita_len[0],2'b00} : 4'b0000;
+ assign out_tdata = tdata;
+
+ wire [15:0] vita_len_in_bytes = {vita_len[13:0],2'b00}; // Vita length is in 32-bit words
+
+ wire [47:0] pad = 48'h0;
+ wire [47:0] mac_src, mac_dst;
+ wire [15:0] eth_type = 16'h0800;
+ wire [15:0] misc_ip = { 4'd4 /* IPv4 */, 4'd5 /* IP HDR Len */, 8'h00 /* DSCP and ECN */};
+ wire [15:0] ip_len = (16'd28 + vita_len_in_bytes); // 20 for IP, 8 for UDP
+ wire [15:0] ident = 16'h0;
+ wire [15:0] flag_frag = { 3'b010 /* don't fragment */, 13'h0 };
+ wire [15:0] ttl_prot = { 8'h10 /* TTL */, 8'h11 /* UDP */ };
+ wire [15:0] iphdr_checksum;
+ wire [31:0] ip_src, ip_dst;
+ wire [15:0] udp_src, udp_dst;
+ wire [15:0] udp_len = (16'd8 + vita_len_in_bytes);
+ wire [15:0] udp_checksum = 16'h0;
+
+ setting_reg #(.my_addr(BASE), .awidth(SR_AWIDTH), .width(16)) set_mac_upper
+ (.clk(clk), .rst(reset),
+ .strobe(set_stb), .addr(set_addr), .in(set_data),
+ .out(mac_src[47:32]), .changed());
+
+ setting_reg #(.my_addr(BASE+1), .awidth(SR_AWIDTH), .width(32)) set_mac_lower
+ (.clk(clk), .rst(reset),
+ .strobe(set_stb), .addr(set_addr), .in(set_data),
+ .out(mac_src[31:0]), .changed());
+
+ setting_reg #(.my_addr(BASE+2), .awidth(SR_AWIDTH), .width(32)) set_ip
+ (.clk(clk), .rst(reset),
+ .strobe(set_stb), .addr(set_addr), .in(set_data),
+ .out(ip_src), .changed());
+
+ setting_reg #(.my_addr(BASE+3), .awidth(SR_AWIDTH), .width(16)) set_udp
+ (.clk(clk), .rst(reset),
+ .strobe(set_stb), .addr(set_addr), .in(set_data),
+ .out(udp_src), .changed());
+
+ // Tables of MAC/IP/UDP addresses
+ wire [8:0] ram_addr; // FIXME we could skip this part if we had wider SR addresses
+
+ setting_reg #(.my_addr(BASE+4), .awidth(SR_AWIDTH), .width(9)) set_ram_addr
+ (.clk(clk), .rst(reset),
+ .strobe(set_stb), .addr(set_addr), .in(set_data),
+ .out(ram_addr), .changed());
+
+ ram_2port #(.DWIDTH(32), .AWIDTH(9)) ram_ip
+ (.clka(clk), .ena(1'b1), .wea(set_stb & (set_addr == BASE+5)), .addra(ram_addr), .dia(set_data), .doa(),
+ .clkb(clk), .enb(1'b1), .web(1'b0), .addrb(sid[8:0]), .dib(32'hFFFF_FFFF), .dob(ip_dst));
+
+ ram_2port #(.DWIDTH(32), .AWIDTH(9)) ram_udpmac
+ (.clka(clk), .ena(1'b1), .wea(set_stb & (set_addr == BASE+6)), .addra(ram_addr), .dia(set_data), .doa(),
+ .clkb(clk), .enb(1'b1), .web(1'b0), .addrb(sid[8:0]), .dib(32'hFFFF_FFFF), .dob({udp_dst,mac_dst[47:32]}));
+
+ ram_2port #(.DWIDTH(32), .AWIDTH(9)) ram_maclower
+ (.clka(clk), .ena(1'b1), .wea(set_stb & (set_addr == BASE+7)), .addra(ram_addr), .dia(set_data), .doa(),
+ .clkb(clk), .enb(1'b1), .web(1'b0), .addrb(sid[8:0]), .dib(32'hFFFF_FFFF), .dob(mac_dst[31:0]));
+
+ ip_hdr_checksum ip_hdr_checksum
+ (.clk(clk), .in({misc_ip,ip_len,ident,flag_frag,ttl_prot,16'd0,ip_src,ip_dst}),
+ .out(iphdr_checksum));
+
+ always @*
+ case(vef_state)
+ 1 : tdata <= { pad[47:0], mac_dst[47:32]};
+ 2 : tdata <= { mac_dst[31:0], mac_src[47:16]};
+ 3 : tdata <= { mac_src[15:0], eth_type[15:0], misc_ip[15:0], ip_len[15:0] };
+ 4 : tdata <= { ident[15:0], flag_frag[15:0], ttl_prot[15:0], iphdr_checksum[15:0]};
+ 5 : tdata <= { ip_src, ip_dst};
+ 6 : tdata <= { udp_src, udp_dst, udp_len, udp_checksum};
+ default : tdata <= in_tdata;
+ endcase // case (vef_state)
+
+endmodule // vita_eth_framer