Primitives Verilog définies par l'utilisateur
Primitives Verilog standard comme nand
et not
n'est pas toujours facile ou suffisant pour représenter une logique complexe. Nouveaux éléments primitifs appelés UDP ou primitives définies par l'utilisateur peut être défini pour modéliser une logique combinatoire ou séquentielle.
Tous les UDP ont exactement une sortie qui peut être 0, 1 ou X et jamais Z (non pris en charge). Toute entrée ayant la valeur Z sera traitée comme X.
Symboles Verilog UDP
Les primitives Verilog définies par l'utilisateur peuvent être écrites au même niveau que module définitions, mais jamais entre module
et endmodule
. Ils peuvent avoir plusieurs ports d'entrée mais toujours un port de sortie, et les ports bidirectionnels ne sont pas valides. Tous les signaux de port doivent être scalaires, ce qui signifie qu'ils doivent avoir une largeur de 1 bit.
Le comportement du matériel est décrit comme un primitif table d'état qui répertorie les différentes combinaisons possibles d'entrées et leur sortie correspondante dans table
et endtable
. Les valeurs des signaux d'entrée et de sortie sont indiquées à l'aide des symboles suivants.
Symbole | Commentaires |
---|---|
0 | 0 logique |
1 | Logique 1 |
x | Inconnu, peut être 0 ou 1 logique. Peut être utilisé comme entrée/sortie ou état actuel des UDP séquentiels |
? | Logique 0, 1 ou x. Ne peut être sorti d'aucun UDP |
- | Aucun changement, uniquement autorisé dans la sortie d'un UDP |
ab | Changement de valeur de a à b où a ou b est soit 0, 1 ou x |
* | Identique à ??, indique toute modification de la valeur d'entrée |
r | Identique à 01 -> front montant sur l'entrée |
f | Identique à 10 -> front descendant en entrée |
p | Front positif potentiel en entrée ; soit 0->1, 0->x ou x->1 |
n | Front descendant potentiel sur l'entrée ; soit 1->0, x->0, 1->x |
Exemple UDP combinatoire
// Output should always be the first signal in port list
primitive mux (out, sel, a, b);
output out;
input sel, a, b;
table
// sel a b out
0 1 ? : 1;
0 0 ? : 0;
1 ? 0 : 0;
1 ? 1 : 1;
x 0 0 : 0;
x 1 1 : 1;
endtable
endprimitive
Un ?
indique que le signal peut être 0, 1 ou x et n'a pas d'importance pour décider de la sortie finale.
Ci-dessous, un module de banc d'essai qui instancie l'UDP et lui applique des stimuli d'entrée.
module tb;
reg sel, a, b;
reg [2:0] dly;
wire out;
integer i;
// Instantiate the UDP - note that UDPs cannot
// be instantiated with port name connection
mux u_mux ( out, sel, a, b);
initial begin
a <= 0;
b <= 0;
$monitor("[T=%0t] a=%0b b=%0b sel=%0b out=%0b", $time, a, b, sel, out);
// Drive a, b, and sel after different random delays
for (i = 0; i < 10; i = i + 1) begin
dly = $random;
#(dly) a <= $random;
dly = $random;
#(dly) b <= $random;
dly = $random;
#(dly) sel <= $random;
end
end
endmodule
Journal de simulation xcelium> run [T=0] a=0 b=0 sel=x out=0 [T=4] a=1 b=0 sel=x out=x [T=5] a=1 b=1 sel=x out=1 [T=10] a=1 b=1 sel=1 out=1 [T=15] a=0 b=1 sel=1 out=1 [T=28] a=0 b=0 sel=1 out=0 [T=33] a=0 b=0 sel=0 out=0 [T=38] a=1 b=0 sel=0 out=1 [T=40] a=1 b=1 sel=0 out=1 [T=51] a=1 b=1 sel=1 out=1 [T=54] a=0 b=0 sel=1 out=0 [T=62] a=1 b=0 sel=1 out=0 [T=67] a=1 b=1 sel=1 out=1 [T=72] a=0 b=1 sel=1 out=1 [T=80] a=0 b=1 sel=0 out=0 [T=84] a=0 b=0 sel=0 out=0 [T=85] a=1 b=0 sel=0 out=1 xmsim: *W,RNQUIE: Simulation is complete.
Exemple UDP séquentiel
La logique séquentielle peut être sensible au niveau ou sensible aux bords et il existe donc deux types d'UDP séquentiels. Le port de sortie doit également être déclaré en tant que reg
type dans la définition UDP et peut éventuellement être initialisé dans un initial
déclaration.
Les UDP séquentiels ont un champ supplémentaire entre le champ d'entrée et de sortie qui est délimité par un :
qui représente l'état actuel.
UDP sensibles au niveau
primitive d_latch (q, clk, d);
output q;
input clk, d;
reg q;
table
// clk d q q+
1 1 : ? : 1;
1 0 : ? : 0;
0 ? : ? : -;
endtable
endprimitive
Dans le tableau ci-dessus, un trait d'union -
sur la dernière ligne du tableau indique aucun changement de valeur pour q+.
module tb;
reg clk, d;
reg [1:0] dly;
wire q;
integer i;
d_latch u_latch (q, clk, d);
always #10 clk = ~clk;
initial begin
clk = 0;
$monitor ("[T=%0t] clk=%0b d=%0b q=%0b", $time, clk, d, q);
#10; // To see the effect of X
for (i = 0; i < 50; i = i+1) begin
dly = $random;
#(dly) d <= $random;
end
#20 $finish;
end
endmodule
Journal de simulation xcelium> run [T=0] clk=0 d=x q=x [T=10] clk=1 d=1 q=1 [T=13] clk=1 d=0 q=0 [T=14] clk=1 d=1 q=1 [T=17] clk=1 d=0 q=0 [T=20] clk=0 d=1 q=0 [T=28] clk=0 d=0 q=0 [T=30] clk=1 d=1 q=1 [T=38] clk=1 d=0 q=0 [T=39] clk=1 d=1 q=1 [T=40] clk=0 d=1 q=1 [T=42] clk=0 d=0 q=1 [T=47] clk=0 d=1 q=1 [T=50] clk=1 d=0 q=0 [T=55] clk=1 d=1 q=1 [T=59] clk=1 d=0 q=0 [T=60] clk=0 d=0 q=0 [T=61] clk=0 d=1 q=0 [T=64] clk=0 d=0 q=0 [T=67] clk=0 d=1 q=0 [T=70] clk=1 d=0 q=0 [T=73] clk=1 d=1 q=1 [T=74] clk=1 d=0 q=0 [T=77] clk=1 d=1 q=1 [T=79] clk=1 d=0 q=0 [T=80] clk=0 d=0 q=0 [T=84] clk=0 d=1 q=0 [T=86] clk=0 d=0 q=0 [T=87] clk=0 d=1 q=0 [T=90] clk=1 d=1 q=1 [T=91] clk=1 d=0 q=0 [T=100] clk=0 d=0 q=0 [T=110] clk=1 d=0 q=0 Simulation complete via $finish(1) at time 111 NS + 0
UDP sensibles aux bords
Une bascule D est modélisée comme une primitive définie par l'utilisateur Verilog dans l'exemple ci-dessous. Notez que le front montant de l'horloge est spécifié par 01
ou 0?
primitive d_flop (q, clk, d);
output q;
input clk, d;
reg q;
table
// clk d q q+
// obtain output on rising edge of clk
(01) 0 : ? : 0;
(01) 1 : ? : 1;
(0?) 1 : 1 : 1;
(0?) 0 : 0 : 0;
// ignore negative edge of clk
(?0) ? : ? : -;
// ignore data changes on steady clk
? (??): ? : -;
endtable
endprimitive
Dans le testbench, l'UDP est instancié et piloté avec des valeurs d'entrée aléatoires d après un nombre aléatoire d'horloges.
module tb;
reg clk, d;
reg [1:0] dly;
wire q;
integer i;
d_flop u_flop (q, clk, d);
always #10 clk = ~clk;
initial begin
clk = 0;
$monitor ("[T=%0t] clk=%0b d=%0b q=%0b", $time, clk, d, q);
#10; // To see the effect of X
for (i = 0; i < 20; i = i+1) begin
dly = $random;
repeat(dly) @(posedge clk);
d <= $random;
end
#20 $finish;
end
endmodule
On peut voir sur l'image que la sortie q suit l'entrée d après 1 retard d'horloge, ce qui est le comportement souhaité pour une bascule D.
Journal de simulationxcelium> run [T=0] clk=0 d=x q=x [T=10] clk=1 d=1 q=x [T=20] clk=0 d=1 q=x [T=30] clk=1 d=1 q=1 [T=40] clk=0 d=1 q=1 [T=50] clk=1 d=1 q=1 [T=60] clk=0 d=1 q=1 [T=70] clk=1 d=0 q=1 [T=80] clk=0 d=0 q=1 [T=90] clk=1 d=1 q=0 [T=100] clk=0 d=1 q=0 [T=110] clk=1 d=1 q=1 [T=120] clk=0 d=1 q=1 [T=130] clk=1 d=1 q=1 [T=140] clk=0 d=1 q=1 [T=150] clk=1 d=0 q=1 [T=160] clk=0 d=0 q=1 [T=170] clk=1 d=0 q=0 [T=180] clk=0 d=0 q=0 [T=190] clk=1 d=0 q=0 [T=200] clk=0 d=0 q=0 [T=210] clk=1 d=1 q=0 [T=220] clk=0 d=1 q=0 [T=230] clk=1 d=1 q=1 [T=240] clk=0 d=1 q=1 [T=250] clk=1 d=1 q=1 [T=260] clk=0 d=1 q=1 [T=270] clk=1 d=1 q=1 [T=280] clk=0 d=1 q=1 [T=290] clk=1 d=1 q=1 [T=300] clk=0 d=1 q=1 [T=310] clk=1 d=1 q=1 [T=320] clk=0 d=1 q=1 [T=330] clk=1 d=1 q=1 [T=340] clk=0 d=1 q=1 [T=350] clk=1 d=1 q=1 [T=360] clk=0 d=1 q=1 [T=370] clk=1 d=0 q=1 [T=380] clk=0 d=0 q=1 [T=390] clk=1 d=0 q=0 [T=400] clk=0 d=0 q=0 [T=410] clk=1 d=1 q=0 [T=420] clk=0 d=1 q=0 [T=430] clk=1 d=1 q=1 [T=440] clk=0 d=1 q=1 [T=450] clk=1 d=1 q=1 [T=460] clk=0 d=1 q=1 [T=470] clk=1 d=1 q=1 [T=480] clk=0 d=1 q=1 Simulation complete via $finish(1) at time 490 NS + 0
Verilog