LRM読解(Chap.11)
11. 名前つきblockとtaskの無効化 ~ Disabling of named blocks and tasks 'disable'文は,Verilog HDLの手続き上の記述の構造化された本質を維持している間,併発してアクティブな手続きの実行(activity)を終了することに関する能力を与えます.
disable文は,taskの命令を全て実行する前に終了させることや,loop文からのbreakすることや,loop文のほかの反復とともに連続するため命令をスキップすること,のメカニズムを与えます.ハードウェア割込みやglobal resetのような例外状態をハンドリングするのに有用です.
disable_statement ::= (From Annex A - A.6.5) disable hierarchical_task_identifier ; | disable hierarchical_block_identifier ;
それぞれの様式のdisable文は,taskまたは名前付けされたブロックを終了します.処理は,blockに続く命令から,または,後のtask enabling文により再開します.また,名前つきblock内や,task内で有効になった,全ての活動(activities)は,終えられるものとします.
task enable文がネストしていて(1つのtaskがほかのtaskをenableにするような場合),一方が他方をまだenbaleにしていないならば,連鎖の中でtaskをdisableにすることは,下位の連鎖全てのtaskをdisableにすることである.
If task enable statements are nested that is, one task enables another, and that one enables yet another then disabling a task within the chain shall disable all tasks downward on the chain.
taskが1回以上enableにされるならば,そのようなtaskを無効にすることは,そのtaskの全ての起動が無効にされるものとします.(disableのqueingされたものが全て消える,という意味かしら??)
If a task is enabled more than once, then disabling such a task shall disable all activations of the task.
taskがdisableにされるなら,taskによって開始されるかもしれない以下の活動(activities)の結果は定義されません:
- output引数とinout引数の結果
- スケジューリングされたけれども実行されていないノンブロッキング代入
- 手続き型連続代入(assign文とforce文)
disbale文は,disable文に含まれる,特殊なblockやtaskをdisableするのに,blockとtask内で使えます.disbale文は,function内の名前つきblockをdisableにすることができます.function内のdisable文があるところでは,blockかtaskをdisableにします.この挙動は未定義です.
automatic task中に,automatic taskまたはblockを無効化することは,タスクのすべての同時発生の実行のための通常のタスクのように続きます.(意味わからん)
Disabling an automatic task or a block inside an automatic task It proceeds as for regular tasks for all concurrent executions of the task.
例1) 本例は,blockが自身をdisableする方法を示します.
begin : block_name rega = regb; disable block_name; regc = rega; // この代入は実行されることはない. end
例2) 本例は,disable文を,名前つきblockで,前方goto文と似たように使うことを示します(意訳).disable文の後の,次の命令文は,名前つきblockに続くものとなります.
begin : block_name ... ... if (a == 0) disable block_name; ... end // 名前つきblockの終端 // 名前つきblockの後のコードから継続実行する ...
例3) この例は,taskからの早期returnのように使うdisable文を示します.しかし,diable文を使ってtask自身をdisableにすることは,プログラミング言語で見つけるようなreturn文のための速記(short-hand)ではありません.
task proc_a; begin ... ... if (a == 0) disable proc_a; // trueであれば返る. ... ... end endtask
例4) この例は,C言語のcontinue/break文の2つと同じように使うdisable文を示します.例は,名前つきblockが,ループカウンタがnに達するまで繰り返すか,変数aがbの値にセットされるまで実行される,制御コードを示します.名前つきブロック"break"は,a==bになるまで実行するようなコードを含んでいます.このポイントでは,"disable break;"文が,blockの実行を終了します.名前つきブロック"continue"は,for loopのそれぞれの繰り返しを実行するコードを含んでいます.毎回,このコードは"disable continue;"命令をh実行し,"continue"blockを終了します.そして,次のfor loopの反復処理へ移ります.
"continue"blockの反復制御には,命令文セットは,a!=0であれば実行する.他の命令セットは,a!=bであれば実行する.(訳注:要はこの文だけ.)
begin : break for (i = 0; i < n; i = i+1) begin : continue @clk if (a == 0) // "continue" loop disable continue; <statements> <statements> @clk if (a == b) // "break" from loop disable break; <statements> <statements> end end
例5) この例は,resetイベントが起きたときの,タイミング制御とtask"action"を同時実行を無効にするdisable文を示します.? being used to disable concurrently a sequence of timing controls and the task action, when the reset event occurs.
例は,"event_expr"と名づけられたシーケンシャルブロックと,イベント"reset"の発生を待つdisable文とから成るfork/join blockを示します.シーケンシャルブロックと,reset待ちとは,並行して実行します."event_expr"ブロックは,イベント"ev1"とイベント"trig"の三回の発生を待ちます.これら4つのイベントが生じて,さらに'd'時間単位経過したときに,task"action"が実行します.イベント"reset"がおきたとき,シーケンシャルブロック内のイベントにかかわらず,task"action"を含めて,fork/joinブロックは終了します.
fork begin : event_expr @ev1; repeat (3) @trig; #d action (areg, breg); end @reset disable event_expr; join
例6) 次の例は,再トリガ可能な単安定のビヘイビア記述の例です.名前がつけられた"retrig"イベントは,単安定時間周期で再開します."retrig"が,250時間単位以内で起こり続けていると,Qは1となります.
always begin : monostable #250 q = 0; end always @retrig begin disable monostable; q = 1; end
(個人的まとめ)
diable/continue文は,シミュレーションモデルで使うと効果的かもしれない.ハードウェア記述で使うのはナンセンスだろう.(合成できるかどうかも怪しいか. パラメータを使ってコンパイル時にのみインスタンスを複数生成することなんかには使えそう.)