0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內(nèi)不再提示

狀態(tài)機的一段式、二段式、三段式的區(qū)別

FPGA技術江湖 ? 來源:FPGA技術江湖 ? 2023-08-21 09:25 ? 次閱讀

本篇文章描述狀態(tài)機的一段式、二段式、三段式區(qū)別

一、狀態(tài)機

再次給出狀態(tài)機的示意圖:

1.1、摩爾型,輸出只與狀態(tài)寄存器的輸出狀態(tài)有關

1.2、米粒型,輸出不僅與狀態(tài)寄存器的輸出狀態(tài)有關,還與組合邏輯的輸入有關

e9fe45a2-3ee7-11ee-ac96-dac502259ad0.png

二、一段式、二段式、三段式區(qū)別

根據(jù)狀態(tài)機的結構,狀態(tài)機描述方式 可分為:一段式、二段式、三段式

1.1、一段式

整個狀態(tài)機寫到一個 always 模塊里面。在該模塊中既描述狀態(tài)轉(zhuǎn)移,又描述狀態(tài)的輸入和輸出。

1.2、二段式

用兩個 always 模塊來描述狀態(tài)機。

1.2.1、其中一個 always 模塊采用同步時序描述狀態(tài)轉(zhuǎn)移;

1.2.2、另一個 always模塊采用組合邏輯判斷狀態(tài)轉(zhuǎn)移條件,描述狀態(tài)轉(zhuǎn)移規(guī)律及其輸出,注意組合邏輯輸出要用阻塞賦值。

1.3、三段式

在兩個 always 模塊描述方法基礎上,使用三個 always 模塊。

1.3.1、 一個 always 模塊采用同步時序描述狀態(tài)轉(zhuǎn)移;

1.3.2、一個 always 采用組合邏輯判斷狀態(tài)轉(zhuǎn)移條件,描述狀態(tài)轉(zhuǎn)移規(guī)律,注意組合邏輯輸出要用阻塞賦值;

1.3.3、另一個 always 模塊描述狀態(tài)輸出(可以用組合電路輸出,也可以時序電路輸出),注意組合邏輯輸出要用阻塞賦值。

1.4、綜合

可以看出兩段式有限狀態(tài)機與一段式有限狀態(tài)機的區(qū)別是將時序部分(狀態(tài)轉(zhuǎn)移)和組合部分(判斷狀態(tài)轉(zhuǎn)移條件和產(chǎn)生輸出)分開,寫為兩個always語句,即為兩段式有限狀態(tài)機。將組合部分中的判斷狀態(tài)轉(zhuǎn)移條件和產(chǎn)生輸出再分開寫,則為三段式有限狀態(tài)機。

三、自動售貨機、一段式

module  auto_sell(
    input   clk,
    input       rst_n,
    input       coin_one,
    input       coin_half,
     
    output  reg     water,
    output  reg     coin_back
);
 
    parameter   ZERO        = 3'b000;
    parameter   HALF        = 3'b001;
    parameter   ONE         = 3'b010;
    parameter   ONE_HALF = 3'b011;
    parameter   TWO         = 3'b100;
     
    //一段式狀態(tài)機
    reg [2:0]       status;
     
    always@(posedge clk,negedge rst_n)begin
        if(!rst_n)
            begin
                status   <= ZERO;
                water    <= 0;
                coin_back <= 0;
            end
        else
            case(status)
                ZERO :
                        begin  
                            water    <= 0;
                            coin_back <= 0;
                            if(coin_half)
                                status <= HALF;
                            else if(coin_one)
                                status <= ONE;
                            else
                                status <= status;
                        end
                HALF :
                        begin  
                            water    <= 0;
                            coin_back <= 0; 
                            if(coin_half)
                                status <= ONE;
                            else if(coin_one)
                                status <= ONE_HALF;
                            else
                                status <= status;   
                        end
                ONE :  
                        begin  
                            water    <= 0;
                            coin_back <= 0;
                            if(coin_half)
                                status <= ONE_HALF;
                            else if(coin_one)
                                status <= TWO;
                            else
                                status <= status;               
                        end
                ONE_HALF :
                            begin  
                                if(coin_half)
                                    begin
                                        status <= TWO;
                                        water    <= 1'b0;
                                        coin_back <= 1'b0;
                                    end
                                else if(coin_one)
                                    begin
                                        status <= ZERO;
                                        water    <= 1'b1;
                                        coin_back <= 1'b0;
                                    end
                                else
                                    begin
                                        status <= status;   
                                        water    <= 1'b0;
                                        coin_back <= 1'b0;                                      
                                    end
                            end
                TWO :  
                        begin  
                            if(coin_half)
                                    begin
                                        status <= ZERO;
                                        water    <= 1'b1;
                                        coin_back <= 1'b0;
                                    end
                            else if(coin_one)
                                    begin
                                        status <= ZERO;
                                        water    <= 1'b1;
                                        coin_back <= 1'b1;
                                    end
                            else
                                    begin
                                        status <= status;   
                                        water    <= 1'b0;
                                        coin_back <= 1'b0;                                      
                                    end    
                        end
                default:
                            begin
                                status <= ZERO; 
                                water    <= 1'b0;
                                coin_back <= 1'b0;                                      
                            end
            endcase        
    end
 
endmodule

四、自動售貨機、二段式

1.1、二段式,寫法一

module  auto_sell(
    input   clk,
    input       rst_n,
    input       coin_one,
    input       coin_half,
     
    output  reg     water,
    output  reg     coin_back
);
 
    parameter   ZERO        = 3'b000;
    parameter   HALF        = 3'b001;
    parameter   ONE         = 3'b010;
    parameter   ONE_HALF = 3'b011;
    parameter   TWO         = 3'b100;
 
//--------------------二段式 1  ok--------------------------
    //二段式狀態(tài)機
    reg [2:0]   c_status;
    reg [2:0]   n_status;
     
    //狀態(tài)轉(zhuǎn)移
    always@(posedge clk,negedge rst_n)begin
        if(!rst_n)
            c_status <= ZERO;
        else
            c_status <= n_status;
    end
     
    //描述狀態(tài)轉(zhuǎn)移規(guī)律以及輸出
    always@(posedge clk,negedge rst_n)begin
        if(!rst_n)
            begin
                n_status <= ZERO;
                water <= 1'b0;
                coin_back <= 1'b0;  
            end
        else
            case(c_status)
                ZERO :
                        begin  
                            water <= 1'b0;
                            coin_back <= 1'b0;                          
                            if(coin_half)
                                n_status <= HALF;
                            else if(coin_one)
                                n_status <= ONE;
                            else
                                n_status <= ZERO;
                        end
                HALF :
                        begin  
                            water <= 1'b0;
                            coin_back <= 1'b0;                          
                            if(coin_half)
                                n_status <= ONE;
                            else if(coin_one)
                                n_status <= ONE_HALF;
                            else
                                n_status <= HALF;   
                        end
                ONE :  
                        begin  
                            water <= 1'b0;
                            coin_back <= 1'b0;                          
                            if(coin_half)
                                n_status <= ONE_HALF;
                            else if(coin_one)
                                n_status <= TWO;
                            else
                                n_status <= ONE;            
                        end
                ONE_HALF :
                            begin  
                                water <= 1'b0;
                                coin_back <= 1'b0;                              
                                if(coin_half)
                                        n_status <= TWO;
                                else if(coin_one)
                                    begin
                                        n_status <= ZERO;
                                        water <= 1'b1;
                                        coin_back <= 1'b0;  
                                    end
                                else
                                        n_status <= ONE_HALF;   
                            end
                TWO :  
                        begin  
                            water <= 1'b0;
                            coin_back <= 1'b0;                          
                            if(coin_half)
                                begin
                                    n_status <= ZERO;
                                    water <= 1'b1;
                                    coin_back <= 1'b0;                                      
                                end
                            else if(coin_one)
                                begin
                                    n_status <= ZERO;
                                    water <= 1'b1;
                                    coin_back <= 1'b1;                                          
                                end
                            else
                                n_status <= TWO;                                        
                        end
                default:
                            n_status <= ZERO;                                   
            endcase
    end
     
 
endmodule

1.2、二段式,寫法二

module  auto_sell(
    input   clk,
    input       rst_n,
    input       coin_one,
    input       coin_half,
     
    output  reg     water,
    output  reg     coin_back
);
 
    parameter   ZERO        = 3'b000;
    parameter   HALF        = 3'b001;
    parameter   ONE         = 3'b010;
    parameter   ONE_HALF = 3'b011;
    parameter   TWO         = 3'b100;
 
//---------------------二段式  2 ok--------------------------------------
 
    //二段式狀態(tài)機
    reg [2:0]   status;
     
    //狀態(tài)轉(zhuǎn)移
    always@(posedge clk,negedge rst_n)begin
        if(!rst_n)
            status <= ZERO;
        else
            begin
                case(status)
                    ZERO :
                            begin                          
                                if(coin_half)
                                    status <= HALF;
                                else if(coin_one)
                                    status <= ONE;
                                else
                                    status <= ZERO;
                            end
                    HALF :
                            begin                          
                                if(coin_half)
                                    status <= ONE;
                                else if(coin_one)
                                    status <= ONE_HALF;
                                else
                                    status <= HALF; 
                            end
                    ONE :  
                            begin                      
                                if(coin_half)
                                    status <= ONE_HALF;
                                else if(coin_one)
                                    status <= TWO;
                                else
                                    status <= ONE;              
                            end
                    ONE_HALF :
                                begin                              
                                    if(coin_half)
                                            status <= TWO;
                                    else if(coin_one)
                                        begin
                                            status <= ZERO;
                                        end
                                    else
                                            status <= ONE_HALF; 
                                end
                    TWO :  
                            begin                          
                                if(coin_half)
                                    begin
                                        status <= ZERO;                                 
                                    end
                                else if(coin_one)
                                    begin
                                        status <= ZERO;                                     
                                    end
                                else
                                    status <= TWO;                                          
                            end
                    default:
                                status <= ZERO;                                 
                endcase        
            end
    end
 
 
    //輸出 時序邏輯
    always@(posedge clk,negedge rst_n)begin
        if(!rst_n)
            begin
                water <= 1'b1;
                coin_back <= 1'b0;  
            end
        else
            case(status)
                ONE_HALF:
                            begin
                                if(coin_one)
                                    begin
                                        water <= 1'b1;
                                        coin_back <= 1'b0;
                                    end
                                else
                                    begin
                                        water <= 1'b0;
                                        coin_back <= 1'b0;                              
                                    end
                            end
                TWO:
                            begin
                                if(coin_half)
                                    begin
                                        water <= 1'b1;
                                        coin_back <= 1'b0;
                                    end
                                else if(coin_one)
                                    begin
                                        water <= 1'b1;
                                        coin_back <= 1'b1;                              
                                    end
                                else
                                    begin
                                        water <= 1'b0;
                                        coin_back <= 1'b0;                                  
                                    end
                            end
                default:
                            begin
                                water <= 1'b0;
                                coin_back <= 1'b0;                                  
                            end        
            endcase
    end
 
endmodule

五、自動售貨機、三段式

module  auto_sell(
    input   clk,
    input       rst_n,
    input       coin_one,
    input       coin_half,
     
    output  reg     water,
    output  reg     coin_back
);
 
    parameter   ZERO        = 3'b000;
    parameter   HALF        = 3'b001;
    parameter   ONE         = 3'b010;
    parameter   ONE_HALF = 3'b011;
    parameter   TWO         = 3'b100;
 
    //三段式狀態(tài)機
    reg [2:0]   c_status;
    reg [2:0]   n_status;
     
    //狀態(tài)轉(zhuǎn)移
    always@(posedge clk,negedge rst_n)begin
        if(!rst_n)
            c_status <= ZERO;
        else
            c_status <= n_status;
    end
     
    //狀態(tài)轉(zhuǎn)移規(guī)律及狀態(tài)輸出,組合邏輯輸出只與輸入有關
    //如果有n_status = n_status,電路會出錯;
    always@(*)begin
        case(c_status)
            ZERO :
                    begin  
                        if(coin_half)
                            n_status = HALF;
                        else if(coin_one)
                            n_status = ONE;
                        else
                            n_status = ZERO;
                    end
            HALF :
                    begin  
                        if(coin_half)
                            n_status = ONE;
                        else if(coin_one)
                            n_status = ONE_HALF;
                        else
                            n_status = HALF;   
                    end
            ONE :  
                    begin  
                        if(coin_half)
                            n_status = ONE_HALF;
                        else if(coin_one)
                            n_status = TWO;
                        else
                            n_status = ONE;            
                    end
            ONE_HALF :
                        begin  
                            if(coin_half)
                                n_status = TWO;
                            else if(coin_one)
                                n_status = ZERO;
                            else
                                n_status = ONE_HALF;                                       
                        end
            TWO :  
                    begin  
                        if(coin_half)
                            n_status = ZERO;
                        else if(coin_one)
                            n_status = ZERO;
                        else
                            n_status = TWO;        
                    end
            default:
                        n_status = ZERO;                                   
        endcase        
    end
 
    always@(posedge clk,negedge rst_n)begin
        if(!rst_n)
            begin
                water = 1'b1;
                coin_back = 1'b0;  
            end
        else
            case(c_status)
                ONE_HALF:
                            begin
                                if(coin_one)
                                    begin
                                        water = 1'b1;
                                        coin_back = 1'b0;
                                    end
                                else
                                    begin
                                        water = 1'b0;
                                        coin_back = 1'b0;                              
                                    end
                            end
                TWO:
                            begin
                                if(coin_half)
                                    begin
                                        water = 1'b1;
                                        coin_back = 1'b0;
                                    end
                                else if(coin_one)
                                    begin
                                        water = 1'b1;
                                        coin_back = 1'b1;                              
                                    end
                                else
                                    begin
                                        water = 1'b0;
                                        coin_back = 1'b0;                                  
                                    end
                            end
                default:
                            begin
                                water = 1'b0;
                                coin_back = 1'b0;                                  
                            end        
            endcase
    end
     
endmodule

六、仿真腳本

`timescale 1ns/1ps
 
module auto_sell_tb;
 
    reg     clk;
    reg rst_n;
    reg coin_one;
    reg coin_half;
     
    wire    water;
    wire    coin_back;
     
     
     
    initial begin
        clk = 0;
        rst_n = 0;
        coin_one = 0;
        coin_half = 0;
        #20;
        rst_n = 1;
        //延時200us
        #10000
         
        //投2.5元
        coin_half = 1;
        #20;
        coin_half = 0;
        #20;
         
        coin_one = 1;
        #20;   
        coin_one = 0;  
        #20;
                 
        coin_half = 1;
        #20;
        coin_half = 0;
        #20;
                 
        coin_half = 1;
        #20;
        coin_half = 0; 
        #20;
                 
        //延時200us
        #10000
         
        //投3元
        coin_half = 1;
        #20;
        coin_half = 0;
        #20;
                 
        coin_one = 1;
        #20;   
        coin_one = 0;  
        #20;
                 
        coin_half = 1;
        #20;
        coin_half = 0;
        #20;
                 
        coin_one = 1;
        #20;   
        coin_one = 0;
        #20;
                 
        //延時200us
        #10000 
        $stop;
    end
     
    auto_sell auto_sell_inst(
        .clk            (clk),
        .rst_n      (rst_n),
        .coin_one   (coin_one),
        .coin_half  (coin_half),
     
        .water      (water),
        .coin_back  (coin_back)
    );
     
    always #10 clk = ~clk;
 
endmodule

七、仿真結果

eaa5221e-3ee7-11ee-ac96-dac502259ad0.png

eac8262e-3ee7-11ee-ac96-dac502259ad0.png

eaea7bc0-3ee7-11ee-ac96-dac502259ad0.png

審核編輯:湯梓紅
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 寄存器
    +關注

    關注

    31

    文章

    5359

    瀏覽量

    120790
  • 仿真
    +關注

    關注

    50

    文章

    4111

    瀏覽量

    133782
  • 時序電路
    +關注

    關注

    1

    文章

    114

    瀏覽量

    21723
  • 狀態(tài)機
    +關注

    關注

    2

    文章

    492

    瀏覽量

    27593

原文標題:筆記連載精選 |【狀態(tài)機:一段式、二段式、三段式】 【原理及verilog仿真】篇

文章出處:【微信號:HXSLH1010101010,微信公眾號:FPGA技術江湖】歡迎添加關注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關推薦

    普通三段式充電原理

    普通三段式充電原理   普通三段式鉛酸蓄電池充電器,充電過程如下: ① 恒流充電階段,充電器充電電流保持恒定,充入
    發(fā)表于 11-16 14:24 ?2.8w次閱讀

    三段式”過電流保護含意

    過流即過電流保護。三段式電流保護指的是電流速斷保護(第一段)、限時電流速斷保護(第二段)、定時限過電流保護(第三段),相互配合構成的套過電
    的頭像 發(fā)表于 02-27 09:17 ?6668次閱讀

    徹底搞懂狀態(tài)機一段式、兩段式、三段式)!個實例,種方法對比看!?。。ǔ绦颍?/a>

    該模塊中既描述狀態(tài)轉(zhuǎn)移,又描述狀態(tài)的輸入和輸出;(2)二段式:用兩個always模塊來描述狀態(tài)機,其中個always模塊采用同步時序描述
    發(fā)表于 06-27 22:13

    Verilog三段式狀態(tài)機描述及模版

    個always完成。三段式建模描述FSM的狀態(tài)機輸出時,只需指定case敏感表為次態(tài)寄存器, 然后直接在每個次態(tài)的case分支中描述該狀態(tài)
    發(fā)表于 07-03 10:13

    Verilog三段式狀態(tài)機描述及模版

    個always完成。三段式建模描述FSM的狀態(tài)機輸出時,只需指定case敏感表為次態(tài)寄存器, 然后直接在每個次態(tài)的case分支中描述該狀態(tài)
    發(fā)表于 07-09 01:55

    三段式和四段式耳機的引腳定義

      耳機插座在我們?nèi)粘I钪惺潜容^常見的種電子元件,其耳機插座的類型規(guī)格也區(qū)分有四段式耳機插座、三段式耳機插座等。三段式和四段式耳機的引腳
    發(fā)表于 12-25 15:26

    FPGA狀態(tài)機一段式簡介

    (41)FPGA狀態(tài)機一段式1.1 目錄1)目錄2)FPGA簡介3)Verilog HDL簡介4)FPGA狀態(tài)機一段式5)結語1.2 FPGA簡介FPGA(Field Programm
    發(fā)表于 02-23 06:45

    Verilog三段式狀態(tài)機描述(轉(zhuǎn)載)

    時序電路的狀態(tài)狀態(tài)變量集合,這些狀態(tài)變量在任意時刻的值都包含了為確定電路的未來行為而必需考慮的所有歷史信息。 狀態(tài)機采用Verilog
    發(fā)表于 02-09 09:42 ?1156次閱讀

    雙電源自動轉(zhuǎn)換開關基礎知識(工作原理、工作步驟、選型、二段式三段式區(qū)別

    本文主要介紹了雙電源自動轉(zhuǎn)換開關基礎知識(工作原理、工作步驟、選型、二段式三段式區(qū)別)。它是種可以在常用電源和備用電源之間進行自動轉(zhuǎn)換的開關。它是由
    發(fā)表于 03-23 11:54 ?5.3w次閱讀
    雙電源自動轉(zhuǎn)換開關基礎知識(工作原理、工作步驟、選型、<b class='flag-5'>二段式</b>和<b class='flag-5'>三段式</b>的<b class='flag-5'>區(qū)別</b>)

    淺談FPGA 四段式狀態(tài)機

    段式不是指個always代碼,而是四程序。使用四段式的寫法,可參照明德?lián)PGVIM特色指令Ztj產(chǎn)生的狀態(tài)機模板。
    發(fā)表于 05-28 10:50 ?2840次閱讀

    關于使用FPGA三段式狀態(tài)機點好處,你有什么看法?

    三段式描述狀態(tài)機的好處,國內(nèi)外各位大牛都已經(jīng)說的很多了,大致可歸為以下點:
    發(fā)表于 08-17 11:43 ?1.6w次閱讀
    關于使用FPGA<b class='flag-5'>三段式</b><b class='flag-5'>狀態(tài)機</b>的<b class='flag-5'>三</b>點好處,你有什么看法?

    FPGA三段式描述狀態(tài)機的好處

    先談談第點關于思維習慣。我發(fā)現(xiàn)有些人會有這樣種習慣,先用一段式狀態(tài)機實現(xiàn)功能,仿真ok后,再將其轉(zhuǎn)成三段式,他們對這種開發(fā)方式的解釋是
    發(fā)表于 07-14 14:59 ?1513次閱讀

    三段式狀態(tài)機編寫問題及三段式狀態(tài)機各部分功能分析

    在 Verilog的江湖里,流傳著一段,兩三段式狀態(tài)機的傳說。它們各有優(yōu)劣,本文就書寫三段式狀態(tài)機
    的頭像 發(fā)表于 06-20 10:35 ?4376次閱讀
    <b class='flag-5'>三段式</b><b class='flag-5'>狀態(tài)機</b>編寫問題及<b class='flag-5'>三段式</b><b class='flag-5'>狀態(tài)機</b>各部分功能分析

    三段式距離保護和三段式電流保護原理是相同的,它們有哪些區(qū)別?

    上也有著相似之處。但是它們在實際應用中存在些差異,下面我們將逐分析它們的區(qū)別、基本原理: 1. 三段式距離保護
    的頭像 發(fā)表于 09-18 09:57 ?5522次閱讀

    隨著電流的增大,三段式過流保護的一段二段保護動作時限變???

    為什么隨著電流的增大,三段式過流保護的一段、二段保護動作時限變?。?隨著電流的增大,三段式過流保護的一段
    的頭像 發(fā)表于 09-18 09:57 ?1174次閱讀