VHDL driving signal from different processes

前端 未结 5 1038
醉梦人生
醉梦人生 2021-01-04 12:22

I have a little problem with following VHDL code:

process (zbroji)
begin
    if rising_edge(zbroji) then
        oduzima <= \'0\';
        ucitanPrvi <         


        
相关标签:
5条回答
  • 2021-01-04 12:30

    If you drive a std_logic signal from more than one process (and remember that a continuous assignment outside of a process also creates an implied process!) then all but one of them must be driving Z onto the signal. To a first approximation, the resolution function (that decides what the final value should be) will produce Xs unless this happens.

    I'm not sure how best to change your code - you need to decide when a particular process should not drive the signal and have it drive a Z at that point.


    The full definition of how the multiple drivers are resolved is defined in the ieee.std_logic_1164 package and covers all possibilities, such as a 1 and an L driving etc. The IEEE get shirty about copyright, so I'm not going to post even an excerpt here, but you'll be able to find it in the source libraries of your simulator.

    0 讨论(0)
  • 2021-01-04 12:36

    Unless zbroji and oduzmi are seperate clocks this is my recommended implementation

    This registers the zbroji and oduzmi and checks if the value in the register is the opposite of the original signal. This should only occur when zbroji/oduzmi go from 0 to 1 and the register has not yet updated the change in signal.

    process (clk)
    begin
        if rising_edge(clk) then
            if zbroji = '1' and zbroji_old = '0' then
                oduzima <= '0';
                ucitanPrvi <= '1';
                broj1 <= ulaz_broj;
            elif oduzmi = '1' and oduzmi_old = '0' then
                oduzima <= '1';
                ucitanPrvi <= '1';
                broj1 <= ulaz_broj;
            end if;
            zbroji_old <= zbroji;
            oduzmi_old <= oduzmi;
        end if;
    end process;
    

    Also it appears that ucitanPrvi and broj1 are always the same thing. Either the signals are useless, this was orignally a typo or you are creating "update" pulses in which case you need the statement

        ucitanPrvi <= '0'
        broj1 <= (others=>'0') -- assumed reset?
    

    following the if(rising_edge(clk) statement

    0 讨论(0)
  • 2021-01-04 12:50

    If you want to synthesize your design for a real FPGA or ASIC, you are going to have to think of VHDL in terms of real hardware (wires, flip flops, gates, etc.). Also, if you want to perform a real rising edge detect in hardware, you will need a system clock that drives a flip flop. Given your original code sample, it doesn't seem that zbroji or oduzmi are system clocks, but just std_logic signals. I wrote this code example assuming basic functionality from your example, hopefully, you can take my code and comments and accomplish what you need.

    library ieee;
    use ieee.std_logic_1164.all;
    use ieee.numeric_std.all;
    
    entity example is
      port (Reset      : in  std_logic;
            SysClk     : in  std_logic;
            zbroji     : in  std_logic;
            oduzmi     : in  std_logic;
            ulaz_broj  : in  std_logic;
            oduzima    : out std_logic;
            ucitanPrvi : out std_logic;
            broj1      : out std_logic
            );
    
    end example;
    
    architecture Behavioral of example is
    
      -- Delayed version of input signals (1 clock cycle delay)
      signal zbroji_d : std_logic;
      signal oduzmi_d : std_logic;
    
      signal zbrojiRE : std_logic;
      signal oduzmiRE : std_logic;
    
    begin
    
      -- Generate 1 clock cycle delayed version of
      -- signals we want to detect the rising edge
      -- Assumes active high reset
      -- Note: You should only use the rising_edge macro
      -- on an actual global or regional clock signal. FPGA's and
      -- ASICs place timing constraints on defined clock signals
      -- that make it possible to use rising_edge, otherwise, we have
      -- to generate our own rising edge signals by comparing delayed
      -- versions of a signal with the current signal.
      -- Also, with any respectable synthesizer / simulator using
      -- rising_edge is almos exactly the same as (clk'event and clk='1')
      -- except rising_edge only returns a '1' when the clock makes a
      -- valid '0' to '1' transition. (see link below)
      EdgeDetectProc : process (Reset, SysClk)
      begin
        if Reset = '1' then
          zbroji_d <= '0';
          oduzmi_d <= '0';
        elsif rising_edge(SysClk) then
          zbroji_d <= zbroji;
          oduzmi_d <= oduzmi;
        end if;
      end process EdgeDetectProc;
    
      -- Assert risinge edge signals for one clock cycle 
      zbrojiRE <= '1' when zbroji = '1' and zbroji_d = '0' else '0';
      oduzmiRE <= '1' when oduzmi = '1' and oduzmi_d = '0' else '0';
    
      -- Assumes that you want a single cycle pulse on ucitanPrvi on the
      -- rising edege of zbroji or oduzmi;
      ucitanPrvi <= zbrojiRE or oduzmiRE;
    
      -- Based on your example, I can't tell what you want to do with the
      -- broj1 signal, but this logic will drive broj1 with ulaz_broj on
      -- either the zbroji or oduzmi rising edge, otherwise '0'.
      broj1 <= ulaz_broj when zbrojiRE = '1' else
               ulaz_broj when oduzmiRE = '1' else
               '0';
    
      -- Finally, it looks like you want to clear oduzima on the rising
      -- edge of zbroji and assert oduzima on the rising edge of
      -- oduzmi
      LatchProc : process (Reset, SysClk)
      begin
        if Reset = '1' then
          oduzima <= '0';
        elsif rising_edge(SysClk) then
          if zbrojiRE = '1' then
            oduzima <= '0';
          elsif oduzmiRE = '1' then
            oduzima <= '1';
          end if;
        end if;
      end process LatchProc;
    
    end Behavioral;
    

    The previous code assumes you have a system clock. In a simulator like ModelSim (free student edition), you can generate a 100 MHz clock with non-synthesizable testbench code like this...

    ClockProc : process
    begin 
       SysClk <= '0';
       wait for 5 ns;
       SysClk <= '1';
       wait for 5 ns;
    end process ClockProc;
    

    In an actual FPGA/ASIC implementation, you will probably want to use an external oscillator that you run into your chip, drive the signal into a DCM (Digital clock manager), which will output a very clean clock signal to all of your VHDL logic, so you can have a glitch free design.

    And finally, here is a great explanation on the differences between rising_edge and (clk'event and clk='1')

    http://vhdlguru.blogspot.com/2010/04/difference-between-risingedgeclk-and.html

    Hope that helps.

    0 讨论(0)
  • 2021-01-04 12:54

    Driving signals from multiple processes is a bad idea unless you really know what you're doing. You can re-write this code in a single process like this.

    process (zbroji, oduzmi)
    begin
        if rising_edge(zbroji) then
            oduzima <= '0';
            ucitanPrvi <= '1';
            broj1 <= ulaz_broj;
        end if;
        if rising_edge(oduzmi) then
            oduzima <= '1';
            ucitanPrvi <= '1';
            broj1 <= ulaz_broj;
        end if;
    end process;
    

    Note that if you do this, and you get a rising edge on both zbroji & oduzmi then oduzima will get the value 1 as it happens last in the process. Before you'd have been trying to set it to 0 and 1 at the same time. That would simulate to X, and probably wouldn't synthesize. If it did synthesize you'd be connecting power and ground together in a CMOS design.


    An alternative method is to have each process drive it's own version of the signal, and then resolve them externally with what ever function you like (or another process). In this case I used or:

    process (zbroji)
    begin
        if rising_edge(zbroji) then
            ucitanPrvi_1 <= '1';
        end if;
    end process;
    
    process (oduzmi)
    begin
        if rising_edge(oduzmi) then
            ucitanPrvi_2 <= '1';
        end if;
    
    end process;
    
    ucitanPrvi <= ucitanPrvi_1 or ucitanPrvi_2;
    
    0 讨论(0)
  • 2021-01-04 12:56

    When you're changing same signal value from multiple process, the simulator will be creating multiple signal drivers for this. The output of them will essentially will be unresolved. Think of it as the output of multiple gates connected together, what do you expect?

    To overcome this, what you need to implement is, a resolution function, that drivers the output to signal.

    http://www.csee.umbc.edu/portal/help/VHDL/misc.html#resf

    If you have any doubts, let me know.

    0 讨论(0)
提交回复
热议问题