Contact me for VHDL projects or assignments

Monday, April 19, 2010

8 bit Binary to BCD converter - Double Dabble algorithm

    I have written a function for converting a 8-bit binary signal into a 12 bit BCD ( consisting of 3 BCD digits).An algorithm known as "double dabble" is used for this.The explanation for the algorithm can be found here.

The function code is given below:

function to_bcd ( bin : std_logic_vector(7 downto 0) ) return std_logic_vector is
variable i : integer:=0;
variable bcd : std_logic_vector(11 downto 0) := (others => '0');
variable bint : std_logic_vector(7 downto 0) := bin;

begin
for i in 0 to 7 loop  -- repeating 8 times.
bcd(11 downto 1) := bcd(10 downto 0);  --shifting the bits.
bcd(0) := bint(7);
bint(7 downto 1) := bint(6 downto 0);
bint(0) :='0';


if(i < 7 and bcd(3 downto 0) > "0100") then --add 3 if BCD digit is greater than 4.
bcd(3 downto 0) := bcd(3 downto 0) + "0011";
end if;

if(i < 7 and bcd(7 downto 4) > "0100") then --add 3 if BCD digit is greater than 4.
bcd(7 downto 4) := bcd(7 downto 4) + "0011";
end if;

if(i < 7 and bcd(11 downto 8) > "0100") then  --add 3 if BCD digit is greater than 4.
bcd(11 downto 8) := bcd(11 downto 8) + "0011";
end if;


end loop;
return bcd;
end to_bcd;

Some sample inputs and the corresponding outputs are shown below:
bin = "01100011"   ,     output = "0000 1001 1001"  (99).
bin = "11111110"   ,     output = "0010 0101 0100"  (254).
bin = "10111011"   ,     output = "0001 1000 0111"  (187).

The code is synthesisable, and the cell usage statistics for Virtex-5 FPGA is shown below:


# BELS                             : 24
#      GND                         : 1
#      LUT3                        : 1
#      LUT4                        : 2
#      LUT5                        : 12
#      LUT6                        : 7
#      MUXF7                       : 1
# IO Buffers                       : 20
#      IBUF                        : 8
#      OBUF                        : 12


Note :- The code can be modified to convert any length binary number to BCD digits.This require very little change in the code.

18 comments:

  1. Hi,

    is it possible to create these netlist-graphics from the command-line using xilinx tools?

    ReplyDelete
  2. @marvin2k : See this link from xilinx.
    http://www.xilinx.com/itp/xilinx4/data/docs/xst/command_line9.html

    ReplyDelete
  3. @vipin: this is to synthesize a design? I meant something like the "RTL Viewer" mentioned in other posts (sorry, in fact I wanted to post in another blogentry of your blog...) to view the generated netlist.

    ReplyDelete
  4. There is a command like this in that link :
    run -ifn watchvhd.vhd -ifmt VHDL -ofn watchvhd.ngc -ofmt NGC -p xcv50-bg256-6 -opt_mode Speed
    -opt_level 1

    This will generate the ngc file.You have to open this file with a xilinx ngc file viewer.

    ReplyDelete
  5. Yes, and my question was if one can invoke such a tool from the commandline... sorry for this much text...

    ReplyDelete
  6. I havent explored command line options till.I guess you should try whatever is given in that link.Also share your experience here.

    ReplyDelete
  7. @rourab : yes. you just have to understand the concept.you can make it for n bit. But the code will be more complicated.

    ReplyDelete
  8. function to_bcd ( bin : std_logic_vector((n-1) downto 0) ) return std_logic_vector is
    variable i : integer:=0;
    variable j : integer:=1;
    variable bcd : std_logic_vector(((4*q)-1) downto 0) := (others => '0');
    variable bint : std_logic_vector((n-1) downto 0) := bin;

    begin
    for i in 0 to n-1 loop -- repeating 8 times.
    bcd(((4*q)-1) downto 1) := bcd(((4*q)-2) downto 0); --shifting the bits.
    bcd(0) := bint(n-1);
    bint((n-1) downto 1) := bint((n-2) downto 0);
    bint(0) :='0';

    l1: for j in 1 to q loop
    if(i < n-1 and bcd(((4*q)-1) downto ((4*q)-4)) > "0100") then --add 3 if BCD digit is greater than 4.
    bcd(((4*q)-1) downto ((4*q)-4)) := bcd(((4*q)-1) downto ((4*q)-4)) + "0011";

    end if;

    end loop l1;
    end loop;
    return bcd;
    end to_bcd;

    ReplyDelete
  9. the previous code i have written in generic form ,where q is the number bcd digit,in that case i got the desire result up to 9 but when it exceed over 9 it gives the A,B,C,D,E,F. i cant get my mistake,

    ReplyDelete
  10. vipin i have solved my problem just replacing the q by j in the inner loop

    ReplyDelete
  11. rourab, please, can you tell me which q you replaced by j in the inner loop?

    I wrote similar code, but without generic parameter, and I had same problem (A,B,C..F).

    ReplyDelete
  12. function to_bcd ( bin : std_logic_vector((n-1) downto 0) ) return std_logic_vector is
    variable i : integer:=0;
    variable j : integer:=1;
    variable bcd : std_logic_vector(((4*q)-1) downto 0) := (others => '0');
    variable bint : std_logic_vector((n-1) downto 0) := bin;

    begin
    for i in 0 to n-1 loop -- repeating 8 times.
    bcd(((4*q)-1) downto 1) := bcd(((4*q)-2) downto 0); --shifting the bits.
    bcd(0) := bint(n-1);
    bint((n-1) downto 1) := bint((n-2) downto 0);
    bint(0) :='0';

    l1: for j in 1 to q loop
    if(i < n-1 and bcd(((4*j)-1) downto ((4*j)-4)) > "0100") then --add 3 if BCD digit is greater than 4.
    bcd(((4*j)-1) downto ((4*j)-4)) := bcd(((4*j)-1) downto ((4*j)-4)) + "0011";

    end if;

    end loop l1;
    end loop;
    return bcd;
    end to_bcd;



    this is my code

    ReplyDelete
  13. rourab,

    thank you very much, but I still have problem with this code, even when its generic.

    I wanted to convert 24-bit binary to 32-bit bcd and I inserted your function into my code, where I defined q := 8, because (4*q)-1) would be 32 bits.

    When I wanna do 8-bit bin to 12-bit bcd, I need to define q := 3.

    My question is:

    What is relation between n and q, what is relation between number of bits for input (bin) and number of bits for output (bcd)?

    Please, answer again...I am beginner and your little help is very great for me.

    Sory for my bad english, i hope that you can understand me.

    ReplyDelete
  14. I found answer for my question, but I need help for implementing that solution.

    Relation between n and q is :

    **********************************************
    q = round(n*(log(2)))

    where q must be rounded to greater integer !!!,
    **********************************************

    for example:

    for n = 24 and q = 8 it would be:

    q = round (32*(log (2)))= round (24*0.3010)=round (7.224)= 8


    I need help for implement this calculus for generic parameter (I want to make automatic calculus q = f(n)).

    Is this possible to be done with "real" data type,could we use not-synthesizable data type for generic parameter , to make synthesizable entity ?

    ReplyDelete
  15. how to implement for D-ALGORITHM IN TESTING OF VLSI CIRCUITS" IN VHDL OR VERILOG?"

    ReplyDelete
  16. Thanks, Men! That function helped me a lot. I made a physical implementation in a Spartan 3E using 8 switches as input and 3 displays as output. I wrote a post (Spanish) about it. I linked yours, of course!

    ReplyDelete
  17. Hi, I have the same question/problem which Aleksander mentioned above.

    ------------------------
    Is this possible to be done with "real" data type,could we use not-synthesizable data type for generic parameter , to make synthesizable entity ?
    ------------------------
    but, with a simpler implementation(from wiki): q = Roundup(n/3)

    Could anyone help me with their views please ??
    Thanks in advance.

    ReplyDelete

Related Posts with Thumbnails

Download this article as PDF