我想要实现一种类似于'补充'计数器的功能,在VHDL中实现,其基本上会在每一步中反转/补充/否定计数器值,从而为测试提供稍微丰富一些的位模式。当然,我希望这个功能是可综合的(因此计数器值可能被分配到引脚),并且是可移植的代码(即仅实现IEEE库,没有
STD_LOGIC_ARITH
)。我也不想默认将所有内容都视为无符号(因此我想避免使用STD_LOGIC_UNSIGNED
)。简言之,这个计数器可以描述为:给定初始值C [0],然后在每个时钟周期的值将是:
C[i+1] = not(C[i]) + ( ( C[i]<(Cmax/2) ) ? 0 : 1 )
如果C是16位宽(这将导致无符号的Cmax = 65535和Cmax/2 = 32768),则可以写成以下形式:
C[i+1] = 65535 - C[i] + ( ( C[i]<32768 ) ? 0 : 1 )
这里的技巧在于计数器只应该增加一次 - 如果它同时增加补码和“正常”范围,那么就不会发生任何更改(方程将在两个值之间“振荡”)。
因此,既然检查 C[i] < (Cmax/2) 基本上与检查C的最高位(第15位)相同,我认为我可以轻松使用VHDL实现类似以下的内容:
Y <= not(Y) + Y(15);
哎呀,我对“轻松”这个词的理解完全错误了 :)
首先问题在于上述方程可能会得出65535+1的结果,这种情况下结果需要17位(即溢出);在我的情况下,我只想截断/忽略任何“进位位”。
这导致了使用什么的问题:
std_logic_vector
有补码not()
定义;但它没有加法+
定义natural
/integer
可能内部占用32位,因此它们的位宽并未必定;它们支持算术+
,但没有补码not()
- 我也尝试过
unsigned
,但遇到了一些问题(记不清是哪些问题了)
当Y是std_logic_vector
时才能提取第15位(MSB),在这种情况下,Y(15)是单个std_logic
- 但是,它需要转换为integer
类型,否则加法+
没有定义 :|
所以,我的当前解决方案(如下)首先有两个计数器寄存器的副本;一个是SIGNAL wCntReg : STD_LOGIC_VECTOR(15 DOWNTO 0)
;另一个是SIGNAL tmp_na : natural
。然后:
- 有两个时钟:一个“主”时钟@50 MHz,另一个是“计数器”时钟:主时钟的16倍频率分频(3.125 MHz)。
- “计数器”时钟应在下降沿激活计算计数器值
- 计算通过
natural
变量执行(它从STD_LOGIC_VECTOR
中复制) - 显然,只有将
std_logic
转换为std_logic_vector
后,才能将其转换为integer
(我很幸运在网上找到了vectorize
函数)。
最让人头疼的部分是如何将natural
变量值反馈到STD_LOGIC_VECTOR
中;我能构造的唯一有效命令是:
wCntReg <= std_logic_vector(to_unsigned(natural'pos(tmp_na), wCntReg'length));
然而,请注意,此命令基本上是“设置”值,该值将在下次运行相同命令时“生效”。因此,它无法在“计数器”时钟进程中运行 - 在下面的代码中,我将其放在更快的“主”时钟进程中。
最后,下面的代码确实有效(在ISE WebPack的行为模拟中通过),但我仍然想知道是否有更简单的方法来解决这个问题。
提前感谢任何答案, 干杯!
代码:
----------------------------------------------------------------------------------
library IEEE;
use IEEE.STD_LOGIC_1164.ALL;
-- use IEEE.STD_LOGIC_ARITH.ALL;
-- use IEEE.STD_LOGIC_UNSIGNED.ALL;
use IEEE.NUMERIC_STD.ALL;
ENTITY complement_count_test_tbw IS
END complement_count_test_tbw;
ARCHITECTURE testbench_arch OF complement_count_test_tbw IS
-- http://www.ingenieurbuero-eschemann.de/downloads/ipicregs/example/vhdl/test/timer_regs_tb.vhd
-- convert std_logic to std_logic_vector(0 downto 0)
function vectorize(s: std_logic) return std_logic_vector is
variable v: std_logic_vector(0 downto 0);
begin
v(0) := s;
return v;
end;
-- DECLARE REGISTERS ==========================
-- 'wires'
SIGNAL wtCLK : std_logic := '0';
-- counter register: 16 bit
SIGNAL wCntReg : STD_LOGIC_VECTOR(15 DOWNTO 0) := (others => 'Z');
-- temporary 'natural' copy of counter register
-- http://www.velocityreviews.com/forums/t21700-std_logic_vector-to-unsigned-type-casting.html
SIGNAL tmp_na : natural;
-- clock parameters
constant PERIODN : natural := 20; -- can be real := 20.0;
constant PERIOD : time := PERIODN * 1 ns;
constant DUTY_CYCLE : real := 0.5;
constant OFFSET : time := 100 ns;
-- freq divisor; with initial values
constant fdiv : natural := 16;
SIGNAL fdiv_cnt : natural := 1;
SIGNAL wfdiv_CLK : std_logic := '0';
BEGIN
-- initializations of connections:
-- instances of components, and their wiring (port maps)...
-- END instances of components, and their wiring (port maps)...
-- PROCESSES (STATE MACHINES) CODE =========
-- clock process for generating CLK
PROCESS
BEGIN
WAIT for OFFSET;
CLOCK_LOOP : LOOP
wtCLK <= '0';
-- MUST refresh counter reg here with value of tmp_na
wCntReg <= std_logic_vector(to_unsigned(natural'pos(tmp_na), wCntReg'length));
WAIT FOR (PERIOD - (PERIOD * DUTY_CYCLE));
wtCLK <= '1';
WAIT FOR (PERIOD * DUTY_CYCLE);
END LOOP CLOCK_LOOP;
END PROCESS;
-- freq divided clock
freq_divisor: PROCESS(wtCLK)
BEGIN
IF rising_edge(wtCLK) THEN -- posedge
IF fdiv_cnt = fdiv THEN
-- reset
fdiv_cnt <= 1 ;
wfdiv_CLK <= not(wfdiv_CLK);
ELSE
fdiv_cnt <= fdiv_cnt + 1;
END IF;
END IF;
END PROCESS freq_divisor;
-- sim: count
PROCESS
BEGIN
WAIT for 10 ns;
tmp_na <= 125;
WAIT for 10 ns;
TESTCOUNT_LOOP: LOOP
-- change counter on negedge of freq. divided clock
WAIT until falling_edge(wfdiv_CLK);
tmp_na <= to_integer(unsigned(not(wCntReg))) + to_integer(unsigned(vectorize(wCntReg(15))));
WAIT for 10 ns;
END LOOP TESTCOUNT_LOOP;
END PROCESS;
-- END PROCESSES (STATE MACHINES) CODE =====
-- END IMPLEMENT ENGINE of 'CORE' ===============
END testbench_arch;
-- END ARCHITECTURE -----------------------------