cam_simple.v 1.63 KB
Newer Older
Antoine Kaufmann's avatar
Antoine Kaufmann committed
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
module cam_simple #(
    parameter DATA_WIDTH = 64,
    parameter ADDR_WIDTH = 5,
    parameter SLICE_WIDTH = 9
)
(
    input  wire                     clk,
    input  wire                     rst,

    input  wire [ADDR_WIDTH-1:0]    write_addr,
    input  wire [DATA_WIDTH-1:0]    write_data,
    input  wire                     write_delete,
    input  wire                     write_enable,
    output wire                     write_busy,

    input  wire [DATA_WIDTH-1:0]    compare_data,
    output wire [2**ADDR_WIDTH-1:0] match_many,
    output wire [2**ADDR_WIDTH-1:0] match_single,
    output wire [ADDR_WIDTH-1:0]    match_addr,
    output wire                     match
);

localparam RAM_DEPTH = 2**ADDR_WIDTH;

reg [RAM_DEPTH - 1 : 0]match_many_raw;
reg valids[RAM_DEPTH];
reg [DATA_WIDTH-1:0]keys[RAM_DEPTH];

// reads
integer k;
always @(posedge clk) begin
    for (k = 0; k < RAM_DEPTH; k = k + 1) begin
        match_many_raw[k] <= valids[k] && (keys[k] == compare_data);
    end
end

// writes
integer i;
always @(posedge clk) begin
    if (rst) begin
        for (i = 0; i < RAM_DEPTH; i = i + 1) begin
            valids[i] <= 0;
            keys[i] <= 0;
        end
    end else if (write_enable) begin
        if (write_delete) begin
            valids[write_addr] <= 0;
        end else begin
            keys[write_addr] <= write_data;
            valids[write_addr] <= 1;
        end
    end
end

priority_encoder #(
    .WIDTH(RAM_DEPTH),
    .LSB_PRIORITY("HIGH")
)
priority_encoder_inst (
    .input_unencoded(match_many_raw),
    .output_valid(match),
    .output_encoded(match_addr),
    .output_unencoded(match_single)
);

endmodule