§ Общие принципы
Взглянем правде в глаза: я так и не смог сделать модуль умножения для чисел float. Не смог. Но надо хотя бы попытаться, что ли!Для того, чтобы перемножить два числа, которые представлены в float форме, надо для начала, узнать, как они записываются:
Где S-это знак, P-порядок, M-мантисса.
Чтобы умножить 2 числа, нужно перемножить мантиссы между собой, сложить степени, и применить XOR к знаку. Если же мантисса превысила число 2, то сдвинуть итоговый порядок, прибавив +1.
§ Число в верилоге
Давайте теперь представим это число в верилоге:input [31:0] value; ... wire sign = value[31]; wire [7:0] exponent = value[30:23]; wire [23:0] mantissa = {exponent != 0, value[22:0]};Обращу внимание на скрытую единицу! Если экспонента не будет равна 0, то тогда туда записывается 1, иначе будет 0 — такое число называется денормализованным.
Я расскажу только про умножение нормализованных чисел.
При умножении двух чисел, может получится число, которое больше по количеству значащих знаков в 2 раза больше. Это надо учесть и потому взять итоговый результат только из более старших знаков.
§ Модуль вычисления
module fp32mul ( input [31:0] a, input [31:0] b, output [31:0] c ); wire sign = a[31] ^ b[31]; // Значение порядков и мантиссы wire [ 7:0] pa = a[30:23], pb = b[30:23]; wire [22:0] ma = a[22:0], mb = b[22:0]; // Умножение нормализованных чисел wire [47:0] t = {|pa, ma} * {|pb, mb}; wire [22:0] m = t[47] ? t[46:24] : t[45:23]; // Результирующая мантисса wire [ 8:0] p0 = pa + pb + t[47], // Переполнение вверх p1 = p0 - 8'h7F; // Переполнение вниз // Если порядок более 255, то +inf, если менее 0, то это будет 0 assign c = {sign, p0[8] ? {31{1'b1}} : p1[8] ? {31{1'b0}} : {p1, m}}; endmoduleНа вход подаются нормализованные float a и b, на выход получается c.