Verilog의 행동 모델링 및 타이밍

Verilog의 행동 모델에는 시뮬레이션을 제어하고 데이터 유형의 변수를 조작하는 절차 문이 포함되어 있습니다. 이러한 모든 진술은 절차에 포함되어 있습니다. 각 프로 시저에는 연관된 활동 흐름이 있습니다.

행동 모델을 시뮬레이션하는 동안 '항상'및 '초기'문으로 정의 된 모든 흐름은 시뮬레이션 시간 '0'에서 함께 시작됩니다. 초기 문은 한 번 실행되고 always 문은 반복적으로 실행됩니다. 이 모델에서 레지스터 변수 a와 b는 시뮬레이션 시간 '0'에서 각각 이진 1과 0으로 초기화됩니다. 그런 다음 초기 문이 완료되고 해당 시뮬레이션 실행 중에 다시 실행되지 않습니다. 이 초기 명령문은 명령문의 시작-끝 블록 (순차 블록이라고도 함)을 포함합니다. 이 시작-끝 유형 블록에서 a가 먼저 초기화 된 다음 b가 초기화됩니다.

행동 모델링의 예

module behave; 
reg [1:0]a,b; 

initial 
begin 
   a = ’b1; 
   b = ’b0; 
end 

always 
begin 
   #50 a = ~a; 
end 

always 
begin 
   #100 b = ~b; 
end 
End module

절차 적 할당

절차 지정은 reg, 정수, 시간 및 메모리 변수를 업데이트하기위한 것입니다. 아래에 설명 된대로 절차 적 할당과 연속 할당 사이에는 상당한 차이가 있습니다.

연속 할당은 순 변수를 구동하고 입력 피연산자가 값을 변경할 때마다 평가 및 업데이트됩니다.

절차 적 할당은 변수를 둘러싼 절차 적 흐름 구조의 제어하에 레지스터 변수의 값을 업데이트합니다.

프로 시저 할당의 오른쪽은 값으로 평가되는 모든 표현식이 될 수 있습니다. 그러나 오른쪽의 부분 선택에는 상수 인덱스가 있어야합니다. 왼쪽은 오른쪽에서 할당을받는 변수를 나타냅니다. 절차 할당의 왼쪽은 다음 형식 중 하나를 취할 수 있습니다.

  • 레지스터, 정수, 실수 또는 시간 변수-이러한 데이터 유형 중 하나의 이름 참조에 대한 할당.

  • 레지스터, 정수, 실수 또는 시간 변수의 비트 선택-다른 비트는 그대로 두는 단일 비트에 대한 할당.

  • 레지스터, 정수, 실수 또는 시간 변수의 부분 선택-나머지 비트는 그대로 두는 두 개 이상의 연속 비트의 부분 선택. 부품 선택 양식의 경우 상수 표현식 만 유효합니다.

  • 메모리 요소-메모리의 단일 단어. 비트 선택 및 부분 선택은 메모리 요소 참조에서 불법입니다.

  • 위의 모든 연결-이전 네 가지 형식 중 하나의 연결을 지정할 수 있습니다. 이는 오른쪽 표현식의 결과를 효과적으로 분할하고 분할 부분을 순서대로 연결의 여러 부분에 할당합니다.

할당 지연 (합성이 아님)

지연된 할당에서 Δt 시간 단위는 명령문이 실행되고 왼쪽 할당이 이루어지기 전에지나갑니다. 할당 내 지연을 사용하면 오른쪽이 즉시 평가되지만 결과가 왼쪽 할당에 배치되기 전에 Δt 지연이 있습니다. 다른 절차가 Δt 동안 오른쪽 신호를 변경하면 출력에 영향을주지 않습니다. 합성 도구는 지연을 지원하지 않습니다.

통사론

  • Procedural Assignment변수 = 표현식

  • Delayed assignment# Δt 변수 = 표현;

  • Intra-assignment delay변수 = # Δt 표현식;

reg [6:0] sum; reg h, ziltch; 
sum[7] = b[7] ^ c[7]; // execute now. 
ziltch = #15 ckz&h; /* ckz&a evaluated now; ziltch changed 
after 15 time units. */ 

#10 hat = b&c; /* 10 units after ziltch changes, b&c is
evaluated and hat changes. */

할당 차단

차단 절차 적 할당 문은 순차적 블록에서 그 뒤에 오는 문을 실행하기 전에 실행되어야합니다. 차단 절차 적 할당 문은 병렬 블록에서 그 뒤에 오는 문 실행을 막지 않습니다.

통사론

차단 절차 할당 구문은 다음과 같습니다-

<lvalue> = <timing_control> <expression>

여기서 lvalue는 절차 적 할당 문에 유효한 데이터 유형이고 =는 할당 연산자이며 타이밍 제어는 선택적 내부 할당 지연입니다. 타이밍 제어 지연은 지연 제어 (예 : # 6) 또는 이벤트 제어 (예 : @ (posedge clk)) 일 수 있습니다. 표현식은 시뮬레이터가 왼쪽에 할당하는 오른쪽 값입니다. 절차 적 할당을 차단하는 데 사용되는 = 할당 연산자는 절차 적 연속 할당 및 연속 할당에서도 사용됩니다.

rega = 0; 
rega[3] = 1;            // a bit-select 
rega[3:5] = 7;          // a part-select 
mema[address] = 8’hff;  // assignment to a memory element 
{carry, acc} = rega + regb;  // a concatenation

비 차단 (RTL) 할당

비 차단 절차 할당을 사용하면 절차 흐름을 차단하지 않고 할당을 예약 할 수 있습니다. 순서 나 종속성에 관계없이 동일한 시간 단계 내에서 여러 레지스터 할당을 원할 때마다 비 차단 절차 문을 사용할 수 있습니다.

통사론

비 차단 절차 할당의 구문은 다음과 같습니다.

<lvalue> <= <timing_control> <expression>

여기서 lvalue는 절차 적 할당 명령문에 유효한 데이터 유형이고, <=는 비 차단 할당 연산자이고, 타이밍 제어는 선택적 내부 할당 타이밍 제어입니다. 타이밍 제어 지연은 지연 제어 또는 이벤트 제어 일 수 있습니다 (예 : @ (posedge clk)). 표현식은 시뮬레이터가 왼쪽에 할당하는 오른쪽 값입니다. 비 차단 할당 연산자는 시뮬레이터가 같지 않은 관계 연산자에 사용하는 것과 동일한 연산자입니다. 시뮬레이터는 <= 연산자를 표현식에서 사용할 때 관계형 연산자로 해석하고 <= 연산자를 비 차단 절차 적 할당 구문에서 사용할 때 할당 연산자로 해석합니다.

시뮬레이터가 비 차단 절차 할당을 평가하는 방법 시뮬레이터가 비 차단 절차 할당을 만나면 시뮬레이터는 다음과 같이 두 단계로 비 차단 절차 할당을 평가하고 실행합니다.

  • 시뮬레이터는 오른쪽을 평가하고 절차 적 타이밍 제어에 지정된 시간에 수행되도록 새 값 할당을 예약합니다. 시뮬레이터는 오른쪽을 평가하고 절차 적 타이밍 제어에 지정된 시간에 수행되도록 새 값 할당을 예약합니다.

  • 주어진 지연이 만료되었거나 적절한 이벤트가 발생한 시간 단계의 끝에서 시뮬레이터는 값을 왼쪽에 할당하여 할당을 실행합니다.

module evaluates2(out); 
output out; 
reg a, b, c; 
initial 

begin 
   a = 0; 
   b = 1; 
   c = 0; 
end 
always c = #5 ~c; 
always @(posedge c) 

begin 
   a <= b; 
   b <= a; 
end 
endmodule

정황

조건문 (또는 if-else 문)은 명령문의 실행 여부를 결정하는 데 사용됩니다.

공식적으로 구문은 다음과 같습니다.

<statement> 
::= if ( <expression> ) <statement_or_null> 
||= if ( <expression> ) <statement_or_null> 
   else <statement_or_null> 
<statement_or_null> 

::= <statement> 
||= ;

<expression>이 평가됩니다. true이면 (즉, 알려진 값이 0이 아닌 경우) 첫 번째 문이 실행됩니다. false이면 (값이 0이거나 값이 x 또는 z 인 경우) 첫 번째 명령문이 실행되지 않습니다. else 문이 있고 <expression>이 false이면 else 문이 실행됩니다. if 표현식의 숫자 값이 0인지 테스트하기 때문에 특정 단축키가 가능합니다.

예를 들어, 다음 두 문장은 동일한 논리를 표현합니다.

if (expression) 
if (expression != 0)

if-else의 else 부분은 선택 사항이므로 중첩 된 if 시퀀스에서 else가 생략되면 혼동이 발생할 수 있습니다. 이것은 else가없는 경우 항상 else를 가장 가까운 이전 항목과 연관시킴으로써 해결됩니다.

if (index > 0) 
if (rega > regb) 
   result = rega; 
   else // else applies to preceding if 
   result = regb; 

If that association is not what you want, use a begin-end block statement 
to force the proper association 

if (index > 0) 
begin 

if (rega > regb) 
result = rega; 
end 
   else 
   result = regb;

구성 : if- else- if

다음 구성은 너무 자주 발생하므로 간단히 개별적으로 논의 할 가치가 있습니다.

Example

if (<expression>) 
   <statement> 
   else if (<expression>) 
   <statement> 
   else if (<expression>) 
   <statement> 
   else  
   <statement>

이 if의 시퀀스 (if-else-if 구문이라고 함)는 다 방향 결정을 작성하는 가장 일반적인 방법입니다. 식은 순서대로 평가됩니다. 표현식이 참이면 그와 관련된 명령문이 실행되고 전체 체인이 종료됩니다. 각 문은 단일 문 또는 문 블록입니다.

if-else-if 구문의 마지막 else 부분은 '위의 항목 없음'또는 다른 조건이 충족되지 않은 기본 케이스를 처리합니다. 때때로 기본값에 대한 명시적인 조치가 없습니다. 이 경우 후행 else를 생략하거나 불가능한 조건을 포착하기 위해 오류 검사에 사용할 수 있습니다.

사례 진술

case 문은식이 다른 여러 식 중 하나와 일치하는지 여부를 테스트하고 그에 따라 분기하는 특수 다 방향 결정 문입니다. case 문은 예를 들어 마이크로 프로세서 명령어의 디코딩을 설명하는 데 유용합니다. case 문에는 다음과 같은 구문이 있습니다.

Example

<statement> 
::= case ( <expression> ) <case_item>+ endcase 
||= casez ( <expression> ) <case_item>+ endcase 
||= casex ( <expression> ) <case_item>+ endcase 
<case_item> 
::= <expression> <,<expression>>* : <statement_or_null> 
||= default : <statement_or_null> 
||= default <statement_or_null>

케이스 표현식은 주어진 순서대로 평가되고 비교됩니다. 선형 검색 중에 케이스 항목 표현식 중 하나가 괄호 안의 표현식과 일치하면 해당 케이스 항목과 연관된 명령문이 실행됩니다. 모든 비교가 실패하고 기본 항목이 제공되면 기본 항목 문이 실행됩니다. 기본 문이 제공되지 않고 모든 비교가 실패하면 케이스 항목 문이 실행되지 않습니다.

구문과는 별도로 case 문은 두 가지 중요한 방식에서 다중 방식 if-else-if 구조와 다릅니다.

  • if-else-if 구문의 조건식은 case 문에서와 같이 한 식을 다른 식과 비교하는 것보다 더 일반적입니다.

  • case 문은 표현식에 x 및 z 값이있을 때 확실한 결과를 제공합니다.

루핑 문

네 가지 유형의 루핑 문이 있습니다. 명령문의 실행을 0 번, 1 번 또는 그 이상 제어하는 ​​수단을 제공합니다.

  • 영원히 계속해서 명령문을 실행합니다.

  • repeat은 명령문을 고정 된 횟수로 실행합니다.

  • while은 표현식이 거짓이 될 때까지 문을 실행합니다. 표현식이 거짓으로 시작하면 명령문이 전혀 실행되지 않습니다.

  • 다음과 같이 3 단계 프로세스에 의해 관련 명령문의 실행을 제어합니다.

    • 실행되는 루프 수를 제어하는 ​​변수를 초기화하는 데 일반적으로 사용되는 할당을 실행합니다.

    • 표현식을 평가합니다. 결과가 0이면 for 루프가 종료되고 0이 아니면 for 루프가 관련 명령문을 실행 한 다음 3 단계를 수행합니다.

    • loopcontrol 변수의 값을 수정하는 데 일반적으로 사용되는 할당을 실행 한 다음 2 단계를 반복합니다.

다음은 루핑 문에 대한 구문 규칙입니다-

Example

<statement> 
::= forever <statement> 
||=forever 
begin 
   <statement>+ 
end  

<Statement> 
::= repeat ( <expression> ) <statement> 
||=repeat ( <expression> ) 
begin
   <statement>+ 
end  

<statement> 
::= while ( <expression> ) <statement> 
||=while ( <expression> ) 
begin 
   <statement>+ 
end  
<statement> 
::= for ( <assignment> ; <expression> ; <assignment> ) 
<statement> 
||=for ( <assignment> ; <expression> ; <assignment> ) 
begin 
   <statement>+ 
end

지연 제어

지연 제어

절차 문의 실행은 다음 구문을 사용하여 지연 제어 할 수 있습니다.

<statement> 
::= <delay_control> <statement_or_null> 
<delay_control> 
::= # <NUMBER> 
||= # <identifier> 
||= # ( <mintypmax_expression> )

다음 예는 할당 실행을 10 시간 단위로 지연시킵니다.

# 10 rega = regb;

다음 세 가지 예는 숫자 기호 (#) 다음에 나오는 표현식을 제공합니다. 할당 실행은 표현식 값으로 지정된 시뮬레이션 시간만큼 지연됩니다.

이벤트 제어

절차 문의 실행은 다음 이벤트 제어 구문을 사용하여 net 또는 레지스터의 값 변경 또는 선언 된 이벤트 발생과 동기화 될 수 있습니다.

Example

<statement> 
::= <event_control> <statement_or_null> 

<event_control> 
::= @ <identifier> 
||= @ ( <event_expression> ) 

<event_expression> 
::= <expression> 
||= posedge <SCALAR_EVENT_EXPRESSION> 
||= negedge <SCALAR_EVENT_EXPRESSION> 
||= <event_expression> <or <event_expression>>

* <SCALAR_EVENT_EXPRESSION>은 1 비트 값으로 해석되는 표현식입니다.

네트 및 레지스터의 값 변경은 명령문 실행을 트리거하는 이벤트로 사용할 수 있습니다. 이를 암시 적 이벤트 감지라고합니다. Verilog 구문을 사용하면 변경 방향, 즉 값 1 (포지션) 또는 값 0 (부정) 방향에 따라 변경을 감지 할 수도 있습니다. 알 수없는 표현 값에 대한 posedge와 negedge의 동작은 다음과 같습니다.

  • 1에서 알 수 없음으로, 알 수 없음에서 0으로 전환시 부정이 감지됩니다.
  • 포즈가 0에서 알 수 없음으로, 알 수 없음에서 1로 전환 될 때 감지됩니다.

절차 : 항상 및 초기 블록

Verilog의 모든 절차는 다음 4 개의 블록 중 하나에 지정됩니다. 1) 초기 블록 2) 항상 블록 3) 태스크 4) 기능

시뮬레이션 시작시 initial 및 always 문이 활성화됩니다. 초기 블록은 한 번만 실행되고 명령문이 완료되면 활동이 종료됩니다. 반대로 always 블록은 반복적으로 실행됩니다. 해당 활동은 시뮬레이션이 종료 될 때만 종료됩니다. 모듈에서 정의 할 수있는 초기 및 항상 블록 수에는 제한이 없습니다. 작업 및 기능은 다른 프로 시저의 하나 이상의 위치에서 활성화되는 프로 시저입니다.

초기 블록

초기 명령문의 구문은 다음과 같습니다.

<initial_statement> 
::= initial <statement>

다음 예는 시뮬레이션 시작시 변수 초기화를위한 초기 문 사용을 보여줍니다.

Initial 
Begin 
   Areg = 0; // initialize a register 
   For (index = 0; index < size; index = index + 1) 
   Memory [index] = 0; //initialize a memory 
   Word 
End

초기 블록의 또 다른 일반적인 용도는 시뮬레이션되는 회로의 주요 부분에 자극을 제공하기 위해 한 번 실행되는 파형 설명의 사양입니다.

Initial 
Begin 
   Inputs = ’b000000; 
   // initialize at time zero 
   #10 inputs = ’b011001; // first pattern 
   #10 inputs = ’b011011; // second pattern 
   #10 inputs = ’b011000; // third pattern 
   #10 inputs = ’b001000; // last pattern 
End

항상 차단

'항상'문은 전체 시뮬레이션 실행 동안 지속적으로 반복됩니다. always 문의 구문은 다음과 같습니다.

<always_statement> 
::= always <statement>

'always'문은 루프 특성으로 인해 특정 형태의 타이밍 제어와 함께 사용할 때만 유용합니다. 'always'문이 시간을 진행할 수있는 수단을 제공하지 않는 경우 'always'문은 시뮬레이션 교착 상태를 만듭니다. 예를 들어 다음 코드는 무한 제로 지연 루프를 생성합니다.

Always areg = ~areg;

위의 코드에 타이밍 제어를 제공하면 다음 예제와 같이 잠재적으로 유용한 설명이 생성됩니다.

Always #half_period areg = ~areg;