这周有朋友问怎样在fpga中用数码管来显示一个十进制数,比如1000。每个数码管上显示一位十进制数。如果用高级语言来分离各位,只需要分别对该数做1000,100,10对应的取商和取余即可分离出千百十个位。但是FPGA做除法非常耗资源。有没有其它解决办法?因为用verilog写程序时虽然形式上可以写为比如256,但是实际存储对应的还是0100H,且一个数码管只能显示一个十进制数。因此这个问题相当于二进制如何转换为一个BCD(Binary Code Decimal)码数。

本文只考虑最常见的8421码的转换,而且是压缩BCD码。

一、算法原理

根据二进制与十进制转换的定义可知,将二进制数按位加权求和即可得到。用公式描述为n位二进制转换10进制公式

A(n-1)*2^(n-1)+A(n-2)*2^(n-2)+........+A(0)*2^0=对应十进制数    (1)

将公式(1)一直提取公因子2,可很容易将公式(1)转换为公式(2),如下所示:

(A(n-1)*2+A(n-2))*2+A(n-3))*2........+a(0) =对应十进制数          (2)

即对于n位二进制数的按权展开求和,由最高的2的n-1次幂,转换为乘以n-1次2,以此类推n-2次幂,n-3次幂……。因为左移一位相当于乘以2,这样乘法操作就转换为了左移操作。

1、BCD码有0~9共计10个数码,用四位二进制表示0000~1001即可表示,而四位二进制可以表示数的范围为0000~1001~1111,即0~9~15。总计多处了 6个数码A~F。我们知道十进制逢十进一,十六进制逢十六进一。从而产生了一个非常关键的问题,转换到大于等于10以后的数字后本应由低位向紧邻的高位进位,但是只能到大于16以后进位。如DH(12D)十进制本应进位产生十位1,和个位2,但是却变成十六进制的DH。所以必须进行修正,修正的方法是加6。

2、BCD码是二进制编码的十进制,逢十进一,10/2=5.因此得到判断条件,即判断每四位是否大于4,因为5-9进一位(左移)溢出。

3、对于左移操作,相当于每进一位就会丢掉6,那么就要加上6/2=3(3左移一位后相当于6)。每次调整在左移之前完成。

*因此二进制转BCD的方法是通过左移,然后每四位判断是否大于4,满足则加3.

二、基于Verilog的代码实现

为了简便,本代码是以十进制数1000为例进行测试。1000D = 3E8H = 0011 1110 1000B。

 /*******************************************************************
* 功能:将二进制数转换为BCD码数,在七段数码管上显示
* 原理:左移 + 3
* 作者:国静德远
* 时间:2017年4月16日
*******************************************************************/
module BinaryToBCD(CLOCK_50, HEX3, HEX2, HEX1, HEX0); input CLOCK_50;
output [:] HEX3, HEX2, HEX1, HEX0; wire [:] TestData;
wire [:] ResultData;
assign TestData = 'd1001;
//Binary to BCD
Binary2BCD ConvertData(CLOCK_50, TestData, ResultData);
//Display
Seg7 H1(ResultData[ : ], HEX0);
Seg7 H2(ResultData[ : ], HEX1);
Seg7 H3(ResultData[ : ], HEX2);
Seg7 H4(ResultData[ : ], HEX3); endmodule //Convert Data by left + 3
module Binary2BCD(inClk, inData, outData);
input inClk;
input [:] inData;
output [:]outData; reg [:]count10 = ;
reg [:]ShiftReg; always@(posedge inClk)
begin
if(count10 >= && count10 <= )
begin
count10 <= count10 - 'b1;
end
else
count10 <= 'd15;
end always @(posedge inClk)
begin
//for(i = 9; i >= 0; i = i - 1)
if(count10 >= && count10 <= )
begin
// shift left
ShiftReg = (ShiftReg << );
ShiftReg[] = inData[count10];
//adjust by add 3
if(ShiftReg[:] > )
ShiftReg[:] = ShiftReg[:] + 'd3;
else
ShiftReg[:] = ShiftReg[:]; if(ShiftReg[:] > )
ShiftReg[:] = ShiftReg[:] + 'd3;
else
ShiftReg[:] = ShiftReg[:]; if(ShiftReg[:] > )
ShiftReg[:] = ShiftReg[:] + 'd3;
else
ShiftReg[:] = ShiftReg[:]; if(ShiftReg[:] > )
ShiftReg[:] = ShiftReg[:] + 'd3;
else
ShiftReg[:] = ShiftReg[:]; end
else
ShiftReg = ShiftReg;
end assign outData = ShiftReg; endmodule //Display on 7seg
module Seg7(inData, outData);
input [:]inData;
output [:]outData;
//reg [3:0]inData;
reg [:]outData;
always@(inData)
begin
case(inData)
'd0: outData = 8'b1100_0000;
'd1: outData = 8'b1111_1001;
'd2: outData = 8'b1010_0100;
'd3: outData = 8'b1011_0000;
'd4: outData = 8'b1001_1001;
'd5: outData = 8'b1001_0010;
'd6: outData = 8'b1000_0010;
'd7: outData = 8'b1111_1000;
'd8: outData = 8'b1000_0000;
'd9: outData = 8'b1001_0000;
default: outData = 'b1111_1111;
endcase
end
endmodule

仿真结果如下图所示:

FPGA中将十进制数在数码管中显示(verilog版)--二进制转换为BCD码的更多相关文章

  1. FPGA加三移位算法:硬件逻辑实现二进制转BCD码

    本文设计方式采用明德扬至简设计法.利用FPGA来完成显示功能不是个很理想的方式,当显示任务比较复杂,要通过各种算法显示波形或者特定图形时,当然要用单片机通过C语言完成这类流程控制复杂,又对时序要求不高 ...

  2. Delphi中对BCD码的直接支持 (转)

    最近在Delphi下写软件,需要将数据转换为BCD码和将BCD码转换为其它数据类型,从网上搜索了一下,没有发现好的函数,于是就想自定义函数来完成BCD与其它格式的数据转换功能.但最终没有动手写,先查查 ...

  3. PHP中将ip地址转成十进制数的两种实用方法

    As we all know that long2ip works as ip1.ip2.ip3.ip4 (123.131.231.212) long ip => (ip1 * 256 * 25 ...

  4. Java中实现十进制数转换为二进制的三种思路

    Java中实现十进制数转换为二进制 第一种:除基倒取余法 这是最符合我们平时的数学逻辑思维的,即输入一个十进制数n,每次用n除以2,把余数记下来,再用商去除以2...依次循环,直到商为0结束,把余数倒 ...

  5. 在Java中,为什么十六进制数0xFF取反之后对应的十进制数是-256呢?

    int number = 0xFF: 字面值是指在程序中无需变量保存,可直接表示为一个具体的数字或字符串的值. 0xFF是一个整数字面值,整数字面值的缺省类型是 int. 我们知道在Java中, in ...

  6. Java中实现十进制数转换为二进制的三种方法

    第一种:除基倒取余法 这是最符合我们平时的数学逻辑思维的,即输入一个十进制数n,每次用n除以2,把余数记下来,再用商去除以2...依次循环,直到商为0结束,把余数倒着依次排列,就构成了转换后的二进制数 ...

  7. verilog实验1:基于FPGA蜂鸣器演奏乐曲并数码管显示

    一.实验任务 利用FPGA进行代码开发,使蜂鸣器演奏出乐曲<生日快乐>,将音调显示在数码管.原理为蜂鸣器为交流源蜂鸣器,在引脚上加一定频率的方波就可以发声,而且发声的频率由所加方波决定.这 ...

  8. 将十进制数转为一个n位数的密码(每位都是个m进制数)

    例如一个6位数的10进制密码,共有106个密码,如果把每个6位数的密码编成号就是[0,106-1].这是十进制的情况,即6个位,每个位有10种选择.如果要遍历所有密码,需要6重for循环,每个循环10 ...

  9. 实验10.3_数值显示拓展_dword型数转变为表示十进制数的字符串

    assume cs:code data segment db 10 dup (0) data ends code segment start : mov ax,4240H;F4240H=1000000 ...

随机推荐

  1. 第九篇 C#实现螺旋矩阵

    C#语言,二维数组的用法和C++.java不同 其它两种也写了,差别不大这里不上传了 using System; namespace _NetRectangle { internal class Pr ...

  2. 【2017-02-21】分支语句if...else...、分支嵌套、变量的作用域

    语句是指程序命令,都是按照顺序执行的. 语句又分为: 顺序语句:从上到下按顺序执行,挨个执行一遍. 分支语句:选择性执行语句,有的可能会执行,有的可能不执行.满足条件执行. 循环语句: 一.分支语句 ...

  3. API抓屏

    需调用API函数 需在开头引入命名空间 using System.Runtime.InteropServices; 获取当前窗口句柄:GetForegroundWindow() [DllImport( ...

  4. Eclipse多个console的使用

    我们在使用Eclipse开发服务器端和客户端时,经常需要同时运行服务器端和客户端,以便联调运行情况.而我之前一般有两种做法: 使用Java的命令行模式运行,但这种做法通常比较麻烦,因为要使用大串的cl ...

  5. oracle expdp和impdp常用命令选项

    一.expdp导出数据库 1.按用户导出 expdp scott/tiger@orcl DIRECTORY=oracle_dmp dumpfile=bak.dmp schemas=scott vers ...

  6. 1965: [Ahoi2005]SHUFFLE 洗牌

    1965: [Ahoi2005]SHUFFLE 洗牌 Time Limit: 3 Sec  Memory Limit: 64 MBSubmit: 408  Solved: 240[Submit][St ...

  7. CoreAnimation 核心动画

    - (void)createBaseAnimation{ //基础动画 CABasicAnimation *animation = [CABasicAnimation animation]; anim ...

  8. Nginx uWSGI web.py 站点搭建

    一.安装nginx 在安装nginx前,需要先装nginx的依赖包. 1.如果没有yum则先安装yum   删除原有的yum  rpm -aq|grep yum|xargs rpm -e --node ...

  9. Array对象

    1.    Array定义 Array是可以保存任意变量的.也就是说,第一项可以是一个基本类型,第二项就可以是一个引用类型,长度也是根据内容动态增加的 var arr = new Array(); 当 ...

  10. js模块化加载器实现

    背景 自es6以前,JavaScript是天生模块化缺失的,即缺少类似后端语言的class, 作用域也只以函数作为区分.这与早期js的语言定位有关, 作为一个只需要在网页中嵌入几十上百行代码来实现一些 ...