2023-03-23 21:02:28 +00:00
|
|
|
///////////////////////////////////////////
|
|
|
|
// bitmanipalu.sv
|
|
|
|
//
|
|
|
|
// Written: Kevin Kim <kekim@hmc.edu>
|
|
|
|
// Created: 23 March 2023
|
|
|
|
// Modified: 23 March 2023
|
|
|
|
//
|
|
|
|
// Purpose: RISC-V Arithmetic/Logic Unit Bit-Manipulation Extension
|
|
|
|
//
|
|
|
|
// Documentation: RISC-V System on Chip Design Chapter 15
|
|
|
|
//
|
|
|
|
// A component of the CORE-V-WALLY configurable RISC-V project.
|
|
|
|
//
|
|
|
|
// Copyright (C) 2021-23 Harvey Mudd College & Oklahoma State University
|
|
|
|
//
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH SHL-2.1
|
|
|
|
//
|
|
|
|
// Licensed under the Solderpad Hardware License v 2.1 (the “License”); you may not use this file
|
|
|
|
// except in compliance with the License, or, at your option, the Apache License version 2.0. You
|
|
|
|
// may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// https://solderpad.org/licenses/SHL-2.1/
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, any work distributed under the
|
|
|
|
// License is distributed on an “AS IS” BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
|
|
|
|
// either express or implied. See the License for the specific language governing permissions
|
|
|
|
// and limitations under the License.
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
`include "wally-config.vh"
|
|
|
|
|
2023-03-23 23:02:38 +00:00
|
|
|
module bitmanipalu #(parameter WIDTH=32) (
|
2023-03-24 04:56:03 +00:00
|
|
|
input logic [WIDTH-1:0] A, B, // Operands
|
|
|
|
input logic [2:0] ALUControl, // With Funct3, indicates operation to perform
|
|
|
|
input logic [2:0] ALUSelect, // ALU mux select signal
|
2023-03-24 05:20:37 +00:00
|
|
|
input logic [1:0] BSelect, // Binary encoding of if it's a ZBA_ZBB_ZBC_ZBS instruction
|
2023-03-24 04:56:03 +00:00
|
|
|
input logic [2:0] ZBBSelect, // ZBB mux select signal
|
|
|
|
input logic [2:0] Funct3, // With ALUControl, indicates operation to perform NOTE: Change signal name to ALUSelect
|
|
|
|
input logic [1:0] CompFlags, // Comparator flags
|
|
|
|
input logic [2:0] BALUControl, // ALU Control signals for B instructions in Execute Stage
|
|
|
|
input logic [WIDTH-1:0] CondExtA, // A Conditional Extend Intermediary Signal
|
|
|
|
input logic [WIDTH-1:0] ALUResult, FullResult, // ALUResult, FullResult signals
|
2023-03-24 05:20:37 +00:00
|
|
|
output logic [WIDTH-1:0] CondMaskB, // B is conditionally masked for ZBS instructions
|
|
|
|
output logic [WIDTH-1:0] CondShiftA, // A is conditionally shifted for ShAdd instructions
|
|
|
|
output logic [WIDTH-1:0] rotA, // Rotate source signal
|
2023-03-24 04:56:03 +00:00
|
|
|
output logic [WIDTH-1:0] Result); // Result
|
|
|
|
|
2023-03-24 05:20:37 +00:00
|
|
|
logic [WIDTH-1:0] ZBBResult, ZBCResult; // ZBB, ZBC Result
|
|
|
|
logic [WIDTH-1:0] MaskB; // BitMask of B
|
|
|
|
logic [WIDTH-1:0] RevA; // Bit-reversed A
|
|
|
|
logic W64; // RV64 W-type instruction
|
|
|
|
logic SubArith; // Performing subtraction or arithmetic right shift
|
|
|
|
logic ALUOp; // 0 for address generation addition or 1 for regular ALU ops
|
|
|
|
logic Rotate; // Indicates if it is Rotate instruction
|
|
|
|
logic Mask; // Indicates if it is ZBS instruction
|
|
|
|
logic PreShift; // Inidicates if it is sh1add, sh2add, sh3add instruction
|
|
|
|
logic [1:0] PreShiftAmt; // Amount to Pre-Shift A
|
2023-03-23 21:02:28 +00:00
|
|
|
|
|
|
|
// Extract control signals from ALUControl.
|
|
|
|
assign {W64, SubArith, ALUOp} = ALUControl;
|
|
|
|
|
|
|
|
// Extract control signals from bitmanip ALUControl.
|
2023-03-24 05:20:37 +00:00
|
|
|
assign {Mask, PreShift} = BALUControl[1:0];
|
2023-03-23 21:02:28 +00:00
|
|
|
|
2023-03-24 04:56:03 +00:00
|
|
|
// Mask Generation Mux
|
2023-03-23 21:02:28 +00:00
|
|
|
if (`ZBS_SUPPORTED) begin: zbsdec
|
|
|
|
decoder #($clog2(WIDTH)) maskgen (B[$clog2(WIDTH)-1:0], MaskB);
|
|
|
|
mux2 #(WIDTH) maskmux(B, MaskB, Mask, CondMaskB);
|
|
|
|
end else assign CondMaskB = B;
|
|
|
|
|
|
|
|
// shifter rotate source select mux
|
|
|
|
if (`ZBB_SUPPORTED & WIDTH == 64) begin
|
|
|
|
mux2 #(WIDTH) rotmux(A, {A[31:0], A[31:0]}, W64, rotA);
|
|
|
|
end else assign rotA = A;
|
|
|
|
|
2023-03-24 04:56:03 +00:00
|
|
|
// Pre-Shift Mux
|
2023-03-23 21:02:28 +00:00
|
|
|
if (`ZBA_SUPPORTED) begin: zbapreshift
|
2023-03-23 23:02:38 +00:00
|
|
|
assign PreShiftAmt = Funct3[2:1] & {2{PreShift}};
|
2023-03-23 21:02:28 +00:00
|
|
|
assign CondShiftA = CondExtA << (PreShiftAmt);
|
2023-03-23 23:02:38 +00:00
|
|
|
end else begin
|
|
|
|
assign PreShiftAmt = 2'b0;
|
|
|
|
assign CondShiftA = A;
|
2023-03-23 21:02:28 +00:00
|
|
|
end
|
|
|
|
|
2023-03-24 04:56:03 +00:00
|
|
|
// Bit reverse needed for some ZBB, ZBC instructions
|
2023-03-23 21:02:28 +00:00
|
|
|
if (`ZBC_SUPPORTED | `ZBB_SUPPORTED) begin: bitreverse
|
|
|
|
bitreverse #(WIDTH) brA(.A, .RevA);
|
|
|
|
end
|
|
|
|
|
2023-03-24 05:20:37 +00:00
|
|
|
// ZBC Unit
|
2023-03-23 21:02:28 +00:00
|
|
|
if (`ZBC_SUPPORTED) begin: zbc
|
|
|
|
zbc #(WIDTH) ZBC(.A, .RevA, .B, .Funct3, .ZBCResult);
|
|
|
|
end else assign ZBCResult = 0;
|
|
|
|
|
2023-03-24 05:20:37 +00:00
|
|
|
// ZBB Unit
|
2023-03-23 21:02:28 +00:00
|
|
|
if (`ZBB_SUPPORTED) begin: zbb
|
|
|
|
zbb #(WIDTH) ZBB(.A, .RevA, .B, .ALUResult, .W64, .lt(CompFlags[0]), .ZBBSelect, .ZBBResult);
|
|
|
|
end else assign ZBBResult = 0;
|
|
|
|
|
2023-03-24 05:20:37 +00:00
|
|
|
// Result Select Mux
|
2023-03-24 03:42:49 +00:00
|
|
|
always_comb
|
|
|
|
case (BSelect)
|
|
|
|
// 00: ALU, 01: ZBA/ZBS, 10: ZBB, 11: ZBC
|
|
|
|
2'b00: Result = ALUResult;
|
|
|
|
2'b01: Result = FullResult; // NOTE: We don't use ALUResult because ZBA/ZBS instructions don't sign extend the MSB of the right-hand word.
|
|
|
|
2'b10: Result = ZBBResult;
|
|
|
|
2'b11: Result = ZBCResult;
|
|
|
|
endcase
|
2023-03-23 23:02:38 +00:00
|
|
|
endmodule
|