1
0
mirror of https://github.com/corundum/corundum.git synced 2025-01-16 08:12:53 +08:00
corundum/fpga/common/rtl/mqnic_ptp_clock.v

249 lines
8.2 KiB
Coq
Raw Normal View History

2021-08-31 01:39:19 -07:00
/*
Copyright 2021, The Regents of the University of California.
All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:
1. Redistributions of source code must retain the above copyright notice,
this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright notice,
this list of conditions and the following disclaimer in the documentation
and/or other materials provided with the distribution.
THIS SOFTWARE IS PROVIDED BY THE REGENTS OF THE UNIVERSITY OF CALIFORNIA ''AS
IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
DISCLAIMED. IN NO EVENT SHALL THE REGENTS OF THE UNIVERSITY OF CALIFORNIA OR
CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY
OF SUCH DAMAGE.
The views and conclusions contained in the software and documentation are those
of the authors and should not be interpreted as representing official policies,
either expressed or implied, of The Regents of the University of California.
*/
// Language: Verilog 2001
`resetall
2021-08-31 01:39:19 -07:00
`timescale 1ns / 1ps
`default_nettype none
2021-08-31 01:39:19 -07:00
/*
* PTP hardware clock
*/
module mqnic_ptp_clock #
(
parameter PTP_PERIOD_NS_WIDTH = 4,
parameter PTP_OFFSET_NS_WIDTH = 32,
parameter PTP_FNS_WIDTH = 32,
parameter PTP_PERIOD_NS = 4'd4,
parameter PTP_PERIOD_FNS = 32'd0,
parameter PTP_PEROUT_ENABLE = 0,
parameter PTP_PEROUT_COUNT = 1
)
(
input wire clk,
input wire rst,
/*
* Register interface
*/
input wire [6:0] reg_wr_addr,
input wire [31:0] reg_wr_data,
input wire [3:0] reg_wr_strb,
input wire reg_wr_en,
output wire reg_wr_wait,
output wire reg_wr_ack,
input wire [6:0] reg_rd_addr,
input wire reg_rd_en,
output wire [31:0] reg_rd_data,
output wire reg_rd_wait,
output wire reg_rd_ack,
/*
* PTP clock
*/
output wire ptp_pps,
output wire [95:0] ptp_ts_96,
output wire ptp_ts_step
);
// control registers
reg reg_wr_ack_reg = 1'b0;
reg [31:0] reg_rd_data_reg = 0;
reg reg_rd_ack_reg = 1'b0;
reg [95:0] get_ptp_ts_96_reg = 0;
reg [95:0] set_ptp_ts_96_reg = 0;
reg set_ptp_ts_96_valid_reg = 0;
reg [PTP_PERIOD_NS_WIDTH-1:0] set_ptp_period_ns_reg = 0;
reg [PTP_FNS_WIDTH-1:0] set_ptp_period_fns_reg = 0;
reg set_ptp_period_valid_reg = 0;
reg [PTP_OFFSET_NS_WIDTH-1:0] set_ptp_offset_ns_reg = 0;
reg [PTP_FNS_WIDTH-1:0] set_ptp_offset_fns_reg = 0;
reg [15:0] set_ptp_offset_count_reg = 0;
reg set_ptp_offset_valid_reg = 0;
wire set_ptp_offset_active;
assign reg_wr_wait = 1'b0;
assign reg_wr_ack = reg_wr_ack_reg;
assign reg_rd_data = reg_rd_data_reg;
assign reg_rd_wait = 1'b0;
assign reg_rd_ack = reg_rd_ack_reg;
always @(posedge clk) begin
reg_wr_ack_reg <= 1'b0;
reg_rd_data_reg <= 0;
reg_rd_ack_reg <= 1'b0;
set_ptp_ts_96_valid_reg <= 1'b0;
set_ptp_period_valid_reg <= 1'b0;
set_ptp_offset_valid_reg <= 1'b0;
if (reg_wr_en && !reg_wr_ack_reg) begin
// write operation
reg_wr_ack_reg <= 1'b1;
case ({reg_wr_addr >> 2, 2'b00})
// PHC
7'h30: set_ptp_ts_96_reg[15:0] <= reg_wr_data; // PTP set fns
7'h34: set_ptp_ts_96_reg[45:16] <= reg_wr_data; // PTP set ns
7'h38: set_ptp_ts_96_reg[79:48] <= reg_wr_data; // PTP set sec l
7'h3C: begin
// PTP set sec h
set_ptp_ts_96_reg[95:80] <= reg_wr_data;
set_ptp_ts_96_valid_reg <= 1'b1;
end
7'h40: set_ptp_period_fns_reg <= reg_wr_data; // PTP period fns
7'h44: begin
// PTP period ns
set_ptp_period_ns_reg <= reg_wr_data;
set_ptp_period_valid_reg <= 1'b1;
end
7'h50: set_ptp_offset_fns_reg <= reg_wr_data; // PTP offset fns
7'h54: set_ptp_offset_ns_reg <= reg_wr_data; // PTP offset ns
7'h58: begin
// PTP offset count
set_ptp_offset_count_reg <= reg_wr_data;
set_ptp_offset_valid_reg <= 1'b1;
end
default: reg_wr_ack_reg <= 1'b0;
endcase
end
if (reg_rd_en && !reg_rd_ack_reg) begin
// read operation
reg_rd_ack_reg <= 1'b1;
case ({reg_rd_addr >> 2, 2'b00})
// PHC
7'h00: begin
// PHC features
reg_rd_data_reg[7:0] <= PTP_PEROUT_ENABLE ? PTP_PEROUT_COUNT : 0;
reg_rd_data_reg[15:8] <= 0;
reg_rd_data_reg[23:16] <= 0;
reg_rd_data_reg[31:24] <= 0;
end
7'h10: reg_rd_data_reg <= ptp_ts_96[15:0]; // PTP cur fns
7'h14: reg_rd_data_reg <= ptp_ts_96[45:16]; // PTP cur ns
7'h18: reg_rd_data_reg <= ptp_ts_96[79:48]; // PTP cur sec l
7'h1C: reg_rd_data_reg <= ptp_ts_96[95:80]; // PTP cur sec h
7'h20: begin
// PTP get fns
get_ptp_ts_96_reg <= ptp_ts_96;
reg_rd_data_reg <= ptp_ts_96[15:0];
end
7'h24: reg_rd_data_reg <= get_ptp_ts_96_reg[45:16]; // PTP get ns
7'h28: reg_rd_data_reg <= get_ptp_ts_96_reg[79:48]; // PTP get sec l
7'h2C: reg_rd_data_reg <= get_ptp_ts_96_reg[95:80]; // PTP get sec h
7'h30: reg_rd_data_reg <= set_ptp_ts_96_reg[15:0]; // PTP set fns
7'h34: reg_rd_data_reg <= set_ptp_ts_96_reg[45:16]; // PTP set ns
7'h38: reg_rd_data_reg <= set_ptp_ts_96_reg[79:48]; // PTP set sec l
7'h3C: reg_rd_data_reg <= set_ptp_ts_96_reg[95:80]; // PTP set sec h
7'h40: reg_rd_data_reg <= set_ptp_period_fns_reg; // PTP period fns
7'h44: reg_rd_data_reg <= set_ptp_period_ns_reg; // PTP period ns
7'h48: reg_rd_data_reg <= PTP_PERIOD_FNS; // PTP nom period fns
7'h4C: reg_rd_data_reg <= PTP_PERIOD_NS; // PTP nom period ns
7'h50: reg_rd_data_reg <= set_ptp_offset_fns_reg; // PTP offset fns
7'h54: reg_rd_data_reg <= set_ptp_offset_ns_reg; // PTP offset ns
7'h58: reg_rd_data_reg <= set_ptp_offset_count_reg; // PTP offset count
7'h5C: reg_rd_data_reg <= set_ptp_offset_active; // PTP offset status
default: reg_rd_ack_reg <= 1'b0;
endcase
end
if (rst) begin
reg_wr_ack_reg <= 1'b0;
reg_rd_ack_reg <= 1'b0;
end
end
// PTP clock
ptp_clock #(
.PERIOD_NS_WIDTH(PTP_PERIOD_NS_WIDTH),
.OFFSET_NS_WIDTH(PTP_OFFSET_NS_WIDTH),
.FNS_WIDTH(PTP_FNS_WIDTH),
.PERIOD_NS(PTP_PERIOD_NS),
.PERIOD_FNS(PTP_PERIOD_FNS),
.DRIFT_ENABLE(0)
)
ptp_clock_inst (
.clk(clk),
.rst(rst),
/*
* Timestamp inputs for synchronization
*/
.input_ts_96(set_ptp_ts_96_reg),
.input_ts_96_valid(set_ptp_ts_96_valid_reg),
.input_ts_64(0),
.input_ts_64_valid(1'b0),
/*
* Period adjustment
*/
.input_period_ns(set_ptp_period_ns_reg),
.input_period_fns(set_ptp_period_fns_reg),
.input_period_valid(set_ptp_period_valid_reg),
/*
* Offset adjustment
*/
.input_adj_ns(set_ptp_offset_ns_reg),
.input_adj_fns(set_ptp_offset_fns_reg),
.input_adj_count(set_ptp_offset_count_reg),
.input_adj_valid(set_ptp_offset_valid_reg),
.input_adj_active(set_ptp_offset_active),
/*
* Drift adjustment
*/
.input_drift_ns(0),
.input_drift_fns(0),
.input_drift_rate(0),
.input_drift_valid(0),
/*
* Timestamp outputs
*/
.output_ts_96(ptp_ts_96),
.output_ts_64(),
.output_ts_step(ptp_ts_step),
/*
* PPS output
*/
.output_pps(ptp_pps)
);
endmodule
`resetall