Conception VLSI - Guide rapide
L'intégration à très grande échelle (VLSI) est le processus de création d'un integrated circuit (IC) en combinant des milliers de transistorsen une seule puce. VLSI a débuté dans les années 1970 lorsque le complexesemiconductor et communicationdes technologies étaient en cours de développement. lemicroprocessor est un périphérique VLSI.
Avant l'introduction de la technologie VLSI, la plupart des circuits intégrés disposaient d'un ensemble limité de fonctions qu'ils pouvaient exécuter. Unelectronic circuit pourrait consister en un CPU, ROM, RAM et autre glue logic. VLSI permet aux concepteurs de circuits intégrés d'ajouter tout cela dans une seule puce.
L'industrie électronique a connu une croissance phénoménale au cours des dernières décennies, principalement en raison des progrès rapides des technologies d'intégration à grande échelle et des applications de conception de systèmes. Avec l'avènement des conceptions d'intégration à très grande échelle (VLSI), le nombre d'applications des circuits intégrés (CI) dans le calcul haute performance, les commandes, les télécommunications, le traitement d'image et vidéo et l'électronique grand public a augmenté à un rythme très rapide.
Les technologies de pointe actuelles, telles que les communications vidéo et cellulaires à haute résolution et à faible débit binaire, offrent aux utilisateurs finaux une merveilleuse quantité d'applications, de puissance de traitement et de portabilité. Cette tendance devrait se développer rapidement, avec des implications très importantes sur la conception VLSI et la conception des systèmes.
Flux de conception VLSI
Le flux de conception des circuits IC VLSI est illustré dans la figure ci-dessous. Les différents niveaux de conception sont numérotés et les blocs montrent les processus dans le flux de conception.
Les spécifications viennent en premier, elles décrivent de manière abstraite, la fonctionnalité, l'interface et l'architecture du circuit IC numérique à concevoir.
Une description comportementale est ensuite créée pour analyser la conception en termes de fonctionnalité, de performance, de conformité à des normes données et d'autres spécifications.
La description RTL est effectuée à l'aide des HDL. Cette description RTL est simulée pour tester la fonctionnalité. À partir de maintenant, nous avons besoin de l'aide des outils EDA.
La description RTL est ensuite convertie en une netlist au niveau de la porte à l'aide d'outils de synthèse logique. Une netlist gatelevel est une description du circuit en termes de portes et de connexions entre elles, qui sont faites de telle manière qu'elles répondent aux spécifications de synchronisation, de puissance et de zone.
Enfin, une mise en page physique est réalisée, qui sera vérifiée puis envoyée à la fabrication.
Graphique Y
Le diagramme en Y de Gajski-Kuhn est un modèle qui capture les considérations lors de la conception de dispositifs à semi-conducteurs.
Les trois domaines de la carte Y de Gajski-Kuhn sont sur des axes radiaux. Chacun des domaines peut être divisé en niveaux d'abstraction, en utilisant des anneaux concentriques.
Au niveau supérieur (anneau extérieur), on considère l'architecture de la puce; aux niveaux inférieurs (anneaux intérieurs), nous affinons successivement la conception en une mise en œuvre plus fine et détaillée -
La création d'une description structurelle à partir d'une description comportementale est réalisée grâce aux processus de synthèse de haut niveau ou de synthèse logique.
La création d'une description physique à partir d'une description structurelle est réalisée par la synthèse de mise en page.
Hiérarchie de conception - Structurelle
La hiérarchie de conception implique le principe de «Divide and Conquer». Il ne s'agit que de diviser la tâche en tâches plus petites jusqu'à ce qu'elle atteigne son niveau le plus simple. Ce procédé est le plus adapté car la dernière évolution de la conception est devenue si simple que sa fabrication devient plus facile.
Nous pouvons concevoir la tâche donnée dans le domaine du processus de flux de conception (comportemental, structurel et géométrique). Pour comprendre cela, prenons un exemple de conception d'un additionneur 16 bits, comme le montre la figure ci-dessous.
Ici, toute la puce de l'additionneur 16 bits est divisée en quatre modules d'additionneurs 4 bits. En outre, diviser l'additionneur 4 bits en additionneur 1 bit ou demi additionneur. L'addition de 1 bit est le processus de conception le plus simple et son circuit interne est également facile à fabriquer sur la puce. Maintenant, en connectant les quatre derniers additionneurs, nous pouvons concevoir un additionneur 4 bits et passer à autre chose, nous pouvons concevoir un additionneur 16 bits.
FPGA - Présentation
La forme complète de FPGA est "Field Programmable Gate Array». Il contient dix mille à plus d'un million de portes logiques avec interconnexion programmable. Des interconnexions programmables sont disponibles pour les utilisateurs ou les concepteurs pour exécuter facilement des fonctions données. Un modèle de puce FPGA typique est montré dans la figure donnée. Il existe des blocs d'E / S, qui sont conçus et numérotés en fonction de leur fonction. Pour chaque module de composition de niveau logique, il y aCLB’s (Configurable Logic Blocks).
CLB exécute l'opération logique donnée au module. L'interconnexion entre les blocs CLB et E / S est réalisée à l'aide de canaux de routage horizontaux, de canaux de routage verticaux et de PSM (multiplexeurs programmables).
Le nombre de CLB qu'il contient décide uniquement de la complexité du FPGA. Les fonctionnalités des CLB et PSM sont conçues par VHDL ou tout autre langage descriptif matériel. Après la programmation, CLB et PSM sont placés sur puce et connectés les uns aux autres avec des canaux de routage.
Avantages
- Cela prend très peu de temps; du processus de conception à la puce fonctionnelle.
- Aucune étape de fabrication physique n'est impliquée.
- Le seul inconvénient est qu'il est coûteux que les autres styles.
Conception de réseau de portes
le gate array (GA)se classe deuxième après le FPGA, en termes de capacité de prototypage rapide. Alors que la programmation utilisateur est importante pour la mise en œuvre de la conception de la puce FPGA, la conception et le traitement des masques métalliques sont utilisés pour GA. La mise en œuvre d'un réseau de portes nécessite un processus de fabrication en deux étapes.
La première phase se traduit par un réseau de transistors non validés sur chaque puce GA. Ces puces non validées peuvent être stockées pour une personnalisation ultérieure, qui est complétée par la définition des interconnexions métalliques entre les transistors du réseau. La structuration des interconnexions métalliques se fait à la fin du processus de fabrication de la puce, de sorte que le temps de rotation peut encore être court, de quelques jours à quelques semaines. La figure ci-dessous montre les étapes de traitement de base pour l'implémentation d'un tableau de portes.
Les plates-formes de réseau de portes typiques utilisent des zones dédiées appelées canaux, pour le routage intercellulaire entre des lignes ou des colonnes de transistors MOS. Ils simplifient les interconnexions. Les modèles d'interconnexion qui exécutent des portes logiques de base sont stockés dans une bibliothèque, qui peut ensuite être utilisée pour personnaliser des lignes de transistors non validés en fonction de la netlist.
Dans la plupart des GA modernes, plusieurs couches métalliques sont utilisées pour le routage des canaux. Avec l'utilisation de plusieurs couches interconnectées, le routage peut être réalisé sur les zones de cellules actives; afin que les canaux de routage puissent être supprimés comme dans les puces Sea-of-Gates (SOG). Ici, toute la surface de la puce est recouverte de transistors nMOS et pMOS non engagés. Les transistors voisins peuvent être personnalisés à l'aide d'un masque métallique pour former des portes logiques de base.
Pour le routage entre cellules, certains des transistors non validés doivent être sacrifiés. Ce style de conception se traduit par plus de flexibilité pour les interconnexions et généralement par une densité plus élevée. Le facteur d'utilisation de la puce GA est mesuré par la surface de la puce utilisée divisée par la surface totale de la puce. Il est supérieur à celui du FPGA, tout comme la vitesse de la puce.
Conception à base de cellules standard
Une conception à base de cellules standard nécessite le développement d'un jeu de masques personnalisé complet. La cellule standard est également connue sous le nom de polycell. Dans cette approche, toutes les cellules logiques couramment utilisées sont développées, caractérisées et stockées dans une bibliothèque de cellules standard.
Une bibliothèque peut contenir quelques centaines de cellules, y compris des onduleurs, des portes NAND, des portes NOR, des portes AOI complexes, des portes OAI, des verrous en D et des bascules. Chaque type de porte peut être implémenté en plusieurs versions pour fournir une capacité d'entraînement adéquate pour différentes épanouissements. La porte de l'onduleur peut avoir une taille standard, une taille double et une taille quadruple afin que le concepteur de puce puisse sélectionner la taille appropriée pour obtenir une vitesse de circuit et une densité de disposition élevées.
Chaque cellule est caractérisée selon plusieurs catégories de caractérisation différentes, telles que,
- Temps de retard en fonction de la capacité de charge
- Modèle de simulation de circuit
- Modèle de simulation de synchronisation
- Modèle de simulation de défaut
- Données de cellule pour le lieu et l'itinéraire
- Masquer les données
Pour le placement et le routage automatisés des cellules, chaque disposition de cellule est conçue avec une hauteur fixe, de sorte qu'un certain nombre de cellules peuvent être délimitées côte à côte pour former des lignes. Les rails d'alimentation et de masse sont parallèles aux limites supérieure et inférieure de la cellule. Ainsi, les cellules voisines partagent un bus d'alimentation et un bus de masse communs. La figure ci-dessous est un plan d'étage pour la conception à base de cellules standard.
Conception entièrement personnalisée
Dans une conception entièrement personnalisée, l'ensemble de la conception du masque est renouvelée, sans l'utilisation d'aucune bibliothèque. Le coût de développement de ce style de conception augmente. Ainsi, le concept de réutilisation de la conception devient célèbre pour réduire le temps de cycle de conception et les coûts de développement.
La conception personnalisée complète la plus difficile peut être la conception d'une cellule mémoire, qu'elle soit statique ou dynamique. Pour la conception de puces logiques, une bonne négociation peut être obtenue en utilisant une combinaison de différents styles de conception sur la même puce, c'est-à-dire des cellules standard, des cellules de chemin de données etprogrammable logic arrays (PLAs).
En pratique, le concepteur fait la mise en page personnalisée complète, c'est-à-dire la géométrie, l'orientation et le placement de chaque transistor. La productivité de la conception est généralement très faible; typiquement quelques dizaines de transistors par jour, par concepteur. Dans le CMOS VLSI numérique, la conception entièrement personnalisée est à peine utilisée en raison du coût élevé de la main-d'œuvre. Ces styles de conception incluent la conception de produits à volume élevé tels que des puces de mémoire, des microprocesseurs haute performance et des FPGA.
La technologie MOSFET complémentaire (CMOS) est aujourd'hui largement utilisée pour former des circuits dans des applications nombreuses et variées. Les ordinateurs, processeurs et téléphones portables actuels utilisent le CMOS en raison de plusieurs avantages clés. Le CMOS offre une faible dissipation de puissance, une vitesse relativement élevée, des marges de bruit élevées dans les deux états et fonctionnera sur une large gamme de tensions de source et d'entrée (à condition que la tension de la source soit fixe)
Pour les processus que nous allons discuter, le type de transistor disponible est le transistor à effet de champ métal-oxyde-semiconducteur (MOSFET). Ces transistors sont formésas a ‘sandwich’consistant en une couche semi-conductrice, généralement une tranche, ou plaquette, à partir d'un monocristal de silicium; une couche de dioxyde de silicium (l'oxyde) et une couche de métal.
Structure d'un MOSFET
Comme le montre la figure, la structure MOS contient trois couches -
The Metal Gate Electrode
The Insulating Oxide Layer (SiO2)
P – type Semiconductor (Substrate)
La structure MOS forme un condensateur, la grille et le substrat étant en tant que deux plaques et une couche d'oxyde en tant que matériau diélectrique. L'épaisseur du matériau diélectrique (SiO 2 ) est généralement comprise entre 10 nm et 50 nm. La concentration et la distribution des porteurs dans le substrat peuvent être manipulées par une tension externe appliquée à la grille et à la borne du substrat. Maintenant, pour comprendre la structure du MOS, considérons d'abord les propriétés électriques de base du substrat semi-conducteur de type P.
La concentration du support dans le matériau semi-conducteur suit toujours la Mass Action Law. La loi de l'action de masse est donnée par -
$$n.p=n_{i}^{2}$$
Où,
n est la concentration de porteurs d'électrons
p est la concentration de porteurs de trous
ni est la concentration intrinsèque des porteurs de silicium
Supposons maintenant que le substrat est également dopée avec l' accepteur (bore) concentration N A . Ainsi, la concentration d'électrons et de trous dans un substrat de type p est
$$n_{po}=\frac{n_{i}^{2}}{N_{A}}$$
$$p_{po}=N_{A}$$
Ici, concentration de dopage NAest (10 15 à 10 16 cm -3 ) supérieure à la concentration intrinsèque ni. Maintenant, pour comprendre la structure MOS, considérons le diagramme de niveau d'énergie du substrat de silicium de type p.
Comme le montre la figure, la bande interdite entre la bande de conduction et la bande de cantonnière est de 1,1 eV. Ici, le potentiel de Fermi Φ F est la différence entre le niveau de Fermi intrinsèque (E i ) et le niveau de Fermi (E FP ).
Où le niveau de Fermi E F dépend de la concentration de dopage. Le potentiel de Fermi Φ F est la différence entre le niveau de Fermi intrinsèque (E i ) et le niveau de Fermi (E FP ).
Mathématiquement,
$$\Phi_{Fp}=\frac{E_{F}-E_{i}}{q}$$
La différence de potentiel entre la bande de conduction et l'espace libre est appelée affinité électronique et est notée qx.
Ainsi, l'énergie requise pour qu'un électron passe du niveau de Fermi à l'espace libre est appelée fonction de travail (qΦ S ) et est donnée par
$$q\Phi _{s}=(E_{c}-E_{F})+qx$$
La figure suivante montre le diagramme de bande d'énergie des composants qui composent le MOS.
Comme le montre la figure ci-dessus, la couche isolante de SiO 2 a une large bande interdite d'énergie de 8 eV et la fonction de travail est de 0,95 eV. La porte métallique a une fonction de travail de 4,1 eV. Ici, les fonctions de travail sont différentes, donc cela créera une chute de tension dans le système MOS. La figure ci-dessous montre le diagramme de bande d'énergie combinée du système MOS.
Comme le montre cette figure, le niveau de potentiel fermi de la grille métallique et du semi-conducteur (Si) sont au même potentiel. Le potentiel de Fermi à la surface est appelé potentiel de surface Φ S et il est plus petit que le potentiel de Fermi Φ F en magnitude.
Fonctionnement d'un MOSFET
Le MOSFET se compose d'un condensateur MOS avec deux jonctions pn placées à proximité de la région de canal et cette région est contrôlée par la tension de grille. Pour rendre la jonction pn polarisée en inverse, le potentiel du substrat est maintenu inférieur au potentiel des trois autres bornes.
Si la tension de grille est augmentée au-delà de la tension de seuil (V GS > V TO ), une couche d'inversion sera établie en surface et un canal de type n se formera entre la source et le drain. Ce canal de type n transportera le courant de drain en fonction de la valeur V DS .
Pour différentes valeurs de V DS , le MOSFET peut fonctionner dans différentes régions comme expliqué ci-dessous.
Région linéaire
À V DS = 0, l'équilibre thermique existe dans la région du canal inversé et le courant de drain I D = 0. Maintenant, si une petite tension de drain, V DS > 0 est appliquée, un courant de drain proportionnel au V DS commencera à circuler de la source égoutter par le canal.
Le canal donne un chemin continu pour le flux de courant de la source au drain. Ce mode de fonctionnement est appelélinear region. La vue en coupe transversale d'un MOSFET à canal n, fonctionnant dans une région linéaire, est représentée sur la figure ci-dessous.
Au bord de la région de saturation
Maintenant, si le V DS est augmenté, les charges dans le canal et la profondeur du canal diminuent à la fin du drain. Pour V DS = V DSAT , les charges dans le canal sont réduites à zéro, ce qui est appelépinch – off point. La vue en coupe transversale du MOSFET à canal n fonctionnant au bord de la région de saturation est représentée sur la figure ci-dessous.
Région de saturation
Pour V DS > V DSAT , une surface appauvrie se forme à proximité du drain, et en augmentant la tension de drain, cette région appauvrie s'étend jusqu'à la source.
Ce mode de fonctionnement est appelé Saturation region. Les électrons venant de la source à l'extrémité du canal, entrent dans la région d'appauvrissement du drain et sont accélérés vers le drain dans un champ électrique élevé.
Courant MOSFET - Caractéristiques de tension
Pour comprendre la caractéristique courant-tension du MOSFET, une approximation du canal est effectuée. Sans cette approximation, l'analyse en trois dimensions du système MOS devient complexe. leGradual Channel Approximation (GCA) pour la caractéristique courant-tension réduira le problème d'analyse.
Approximation graduelle des canaux (GCA)
Considérons la vue en coupe transversale du MOSFET à canal n fonctionnant en mode linéaire. Ici, la source et le substrat sont connectés à la terre. V S = V B = 0. La porte - à - source (V GS ) et de drain - à - tension de source (V DS ) de tension sont les paramètres externes qui contrôlent le courant de drain I D .
La tension, V GS est fixée à une tension supérieure à la tension de seuil V TO , pour créer un canal entre la source et le drain. Comme le montre la figure, la direction x est perpendiculaire à la surface et la direction y est parallèle à la surface.
Ici, y = 0 à l'extrémité source comme indiqué sur la figure. La tension du canal, par rapport à la source, est représentée parVC(Y). Supposons que la tension de seuil VTO soit constante le long de la région de canal, entre y = 0 et y = L. Les conditions aux limites pour la tension de canal V C sont -
$$V_{c}\left ( y = 0 \right ) = V_{s} = 0 \,and\,V_{c}\left ( y = L \right ) = V_{DS}$$
On peut aussi supposer que
$$V_{GS}\geq V_{TO}$$ et
$$V_{GD} = V_{GS}-V_{DS}\geq V_{TO}$$
Soit Q1 (y) la charge totale d'électrons mobiles dans la couche d'inversion de surface. Cette charge électronique peut être exprimée par -
$$Q1(y)=-C_{ox}.[V_{GS}-V_{C(Y)}-V_{TO}]$$
La figure ci-dessous montre la géométrie spatiale de la couche d'inversion de surface et indique ses dimensions. La couche d'inversion diminue à mesure que nous passons du drain à la source. Maintenant, si nous considérons la petite région dy de la longueur de canal L, alors la résistance incrémentale dR offerte par cette région peut être exprimée comme -
$$dR=-\frac{dy}{w.\mu _{n}.Q1(y)}$$
Ici, le signe moins est dû à la polarité négative de la charge de couche d'inversion Q1 et μ n est la mobilité de surface, qui est constante. Maintenant, remplacez la valeur de Q1 (y) dans l'équation dR -
$$dR=-\frac{dy}{w.\mu _{n}.\left \{ -C_{ox}\left [ V_{GS}-V_{C\left ( Y \right )} \right ]-V_{TO} \right \}}$$
$$dR=\frac{dy}{w.\mu _{n}.C_{ox}\left [ V_{GS}-V_{C\left ( Y \right )} \right ]-V_{TO}}$$
Maintenant, la chute de tension dans la petite région de dy peut être donnée par
$$dV_{c}=I_{D}.dR$$
Mettez la valeur de dR dans l'équation ci-dessus
$$dV_{C}=I_{D}.\frac{dy}{w.\mu_{n}.C_{ox}\left [ V_{GS}-V_{C(Y)} \right ]-V_{TO}}$$
$$w.\mu _{n}.C_{ox}\left [ V_{GS}-V_{C(Y)}-V_{TO} \right ].dV_{C}=I_{D}.dy$$
Pour obtenir le courant de drain ID sur toute la région du canal, l'équation ci-dessus peut être intégrée le long du canal de y = 0 à y = L et des tensions V C (y) = 0 à V C (y) = V DS ,
$$C_{ox}.w.\mu _{n}.\int_{V_{c}=0}^{V_{DS}} \left [ V_{GS}-V_{C\left ( Y \right )}-V_{TO} \right ].dV_{C} = \int_{Y=0}^{L}I_{D}.dy$$
$$\frac{C_{ox}.w.\mu _{n}}{2}\left ( 2\left [ V_{GS}-V_{TO} \right ] V_{DS}-V_{DS}^{2}\right ) = I_{D}\left [ L-0 \right ]$$
$$I_{D} = \frac{C_{ox}.\mu _{n}}{2}.\frac{w}{L}\left ( 2\left [ V_{GS}-V_{TO} \right ]V_{DS}-V_{DS}^{2} \right )$$
Pour la région linéaire V DS <V GS - V TO . Pour la région de saturation, la valeur de V DS est supérieure à (V GS - V TO ). Par conséquent, pour la région de saturation V DS = (V GS - V TO ).
$$I_{D} = C_{ox}.\mu _{n}.\frac{w}{2}\left ( \frac{\left [ 2V_{DS} \right ]V_{DS}-V_{DS}^{2}}{L} \right )$$
$$I_{D} = C_{ox}.\mu _{n}.\frac{w}{2}\left ( \frac{2V_{DS}^{2}-V_{DS}^{2}}{L} \right )$$
$$I_{D} = C_{ox}.\mu _{n}.\frac{w}{2}\left ( \frac{V_{DS}^{2}}{L} \right )$$
$$I_{D} = C_{ox}.\mu _{n}.\frac{w}{2}\left ( \frac{\left [ V_{GS}-V_{TO} \right ]^{2}}{L} \right )$$
L'onduleur est véritablement le noyau de toutes les conceptions numériques. Une fois que son fonctionnement et ses propriétés sont clairement compris, la conception de structures plus complexes telles que des portes NAND, des additionneurs, des multiplicateurs et des microprocesseurs est grandement simplifiée. Le comportement électrique de ces circuits complexes peut être presque entièrement dérivé en extrapolant les résultats obtenus pour les onduleurs.
L'analyse des onduleurs peut être étendue pour expliquer le comportement de portes plus complexes telles que NAND, NOR ou XOR, qui à leur tour forment les blocs de construction de modules tels que les multiplicateurs et les processeurs. Dans ce chapitre, nous nous concentrons sur une seule incarnation de la porte de l'onduleur, à savoir l'onduleur CMOS statique - ou l'onduleur CMOS, en bref. C'est certainement le plus populaire à l'heure actuelle et mérite donc notre attention particulière.
Principe d'opération
Le symbole logique et la table de vérité de l'onduleur idéal sont indiqués dans la figure ci-dessous. Ici, A est l'entrée et B est la sortie inversée représentée par leurs tensions de nœud. En utilisant la logique positive, la valeur booléenne de la logique 1 est représentée par V dd et la logique 0 est représentée par 0. V th est la tension de seuil de l'onduleur, qui est V dd / 2, où V dd est la tension de sortie.
La sortie est commutée de 0 à V dd lorsque l'entrée est inférieure à V th . Ainsi, pour 0 <V in <V e sortie est égal à l'entrée logique 0 et V th <V in <V dd est égal à l'entrée logique 1 pour le variateur.
Les caractéristiques présentées sur la figure sont idéales. La structure de circuit généralisée d'un onduleur nMOS est illustrée dans la figure ci-dessous.
À partir de la figure donnée, nous pouvons voir que la tension d'entrée de l'onduleur est égale à la tension grille-source du transistor nMOS et la tension de sortie de l'onduleur est égale à la tension drain-source du transistor nMOS. La tension source-substrat de nMOS est également appelée pilote pour transistor qui est mis à la terre; donc V SS = 0. Le nœud de sortie est connecté avec une capacité localisée utilisée pour VTC.
Inverseur de charge résistive
La structure de base d'un onduleur à charge résistive est illustrée dans la figure ci-dessous. Ici, le type d'amélioration nMOS agit comme le transistor pilote. La charge est constituée d'une résistance linéaire simple , R L . L'alimentation du circuit est V DD et le courant de drain I D est égal au courant de charge R .
Fonctionnement du circuit
Lorsque l'entrée du transistor de commande est inférieure à la tension de seuil V TH (V en <V TH ), le transistor de commande est dans la région de coupure et ne conduit aucun courant. Ainsi, la chute de tension aux bornes de la résistance de charge est nulle et la tension de sortie est égale à V DD . Maintenant, lorsque la tension d'entrée augmente encore, le transistor pilote commencera à conduire le courant non nul et le nMOS passera dans la région de saturation.
Mathématiquement,
$$I_{D} = \frac{K_{n}}{2}\left [ V_{GS}-V_{TO} \right ]^{2}$$
En augmentant davantage la tension d'entrée, le transistor d'attaque entrera dans la région linéaire et la sortie du transistor d'attaque diminuera.
$$I_{D} = \frac{K_{n}}{2}2\left [ V_{GS}-V_{TO} \right ]V_{DS}-V_{DS}^{2}$$
VTC de l'onduleur de charge résistive, illustré ci-dessous, indique le mode de fonctionnement du transistor pilote et les points de tension.
Onduleur avec charge MOSFET de type N
Le principal avantage de l'utilisation du MOSFET comme dispositif de charge est que la zone de silicium occupée par le transistor est plus petite que la zone occupée par la charge résistive. Ici, le MOSFET est une charge active et l'onduleur à charge active donne de meilleures performances que l'onduleur à charge résistive.
Amélioration de la charge NMOS
Deux onduleurs avec dispositif de charge de type amélioré sont représentés sur la figure. Le transistor de charge peut fonctionner soit, dans la région de saturation, soit dans la région linéaire, en fonction de la tension de polarisation appliquée à sa borne de grille. L'onduleur de charge d'amélioration saturée est illustré sur la fig. (une). Elle nécessite une seule alimentation en tension et le processus de fabrication simple et donc V OH est limitée au V DD - V T .
L'onduleur de charge à amélioration linéaire est illustré à la fig. (b). Il fonctionne toujours en région linéaire; donc le niveau de V OH est égal à V DD .
L'onduleur de charge linéaire a une marge de bruit plus élevée par rapport à l'onduleur d'amélioration saturé. Mais, l'inconvénient de l'onduleur à amélioration linéaire est qu'il nécessite deux blocs d'alimentation séparés et que les deux circuits souffrent d'une dissipation de puissance élevée. Par conséquent, les onduleurs améliorés ne sont utilisés dans aucune application numérique à grande échelle.
Épuisement de la charge NMOS
Les inconvénients de l'inverseur de charge d'amélioration peuvent être surmontés en utilisant un inverseur de charge d'épuisement. Par rapport à l'onduleur de charge d'amélioration, l'onduleur de charge d'épuisement nécessite quelques étapes de fabrication supplémentaires pour que l'implant de canal ajuste la tension de seuil de charge.
Les avantages de l'onduleur de charge d'épuisement sont: une transition VTC nette, une meilleure marge de bruit, une alimentation électrique unique et une zone de disposition globale plus petite.
Comme le montre la figure, la porte et la borne source de la charge sont connectées; Donc, V GS = 0. Ainsi, la tension de seuil de la charge est négative. Par conséquent,
$$V_{GS,load}> V_{T,load}$$ est satisfait
Par conséquent, le dispositif de charge a toujours un canal de conduction quel que soit le niveau de tension d'entrée et de sortie.
Lorsque le transistor de charge est dans la région de saturation, le courant de charge est donné par
$$I_{D,load} = \frac{K_{n,load}}{2}\left [ -V_{T,load}\left ( V_{out} \right ) \right ]^{2}$$
Lorsque le transistor de charge est dans la région linéaire, le courant de charge est donné par
$$I_{D,load} = \frac{K_{n,load}}{2}\left [ 2\left | V_{T,load}\left ( V_{out} \right ) \right |.\left ( V_{DD}-V_{out} \right )-\left ( V_{DD}-V_{out} \right )^{2} \right ]$$
Les caractéristiques de transfert de tension de l'onduleur de charge d'appauvrissement sont indiquées dans la figure ci-dessous -
Inverseur CMOS - Circuit, fonctionnement et description
Le circuit de l'onduleur CMOS est illustré sur la figure. Ici, les transistors nMOS et pMOS fonctionnent comme des transistors pilotes; lorsqu'un transistor est activé, l'autre est désactivé.
Cette configuration s'appelle complementary MOS (CMOS). L'entrée est connectée à la borne de grille des deux transistors de sorte que les deux peuvent être commandés directement avec des tensions d'entrée. Le substrat du nMOS est connecté à la terre et le substrat du pMOS est connecté à l'alimentation, V DD .
Donc V SB = 0 pour les deux transistors.
$$V_{GS,n}=V_{in}$$
$$V_{DS,n}=V_{out}$$
Et,
$$V_{GS,p}=V_{in}-V_{DD}$$
$$V_{DS,p}=V_{out}-V_{DD}$$
Lorsque l 'entrée de nMOS est inférieure à la tension de seuil (V dans <V TO, n ), le nMOS est coupé et pMOS est dans la région linéaire. Ainsi, le courant de drain des deux transistors est nul.
$$I_{D,n}=I_{D,p}=0$$
Par conséquent, la tension de sortie V OH est égale à la tension d'alimentation.
$$V_{out}=V_{OH}=V_{DD}$$
Lorsque la tension d'entrée est supérieure à V DD + V TO, p , le transistor pMOS est dans la région de coupure et le nMOS est dans la région linéaire, de sorte que le courant de drain des deux transistors est nul.
$$I_{D,n}=I_{D,p}=0$$
Par conséquent, la tension de sortie V OL est égale à zéro.
$$V_{out}=V_{OL}=0$$
Le nMOS opère dans la région de saturation si V en > V TO et si les conditions suivantes sont satisfaites.
$$V_{DS,n}\geq V_{GS,n}-V_{TO,n} $$
$$V_{out}\geq V_{in}-V_{TO,n} $$
Le pMOS fonctionne dans la région de saturation si V dans <V DD + V TO, p et si les conditions suivantes sont satisfaites.
$$V_{DS,p}\leq V_{GS,p}-V_{TO,p} $$
$$V_{out}\leq V_{in}-V_{TO,p} $$
Pour différentes valeurs de tensions d'entrée, les régions de fonctionnement sont répertoriées ci-dessous pour les deux transistors.
Région | V dans | V sur | nMOS | pMOS |
---|---|---|---|---|
UNE | <V TO, n | V OH | Couper | Linéaire |
B | V IL | Haut ≈ V OH | Saturation | Linéaire |
C | V ème | V ème | Saturation | Saturation |
ré | V IH | Faible ≈ V OL | Linéaire | Saturation |
E | > (V DD + V TO, p ) | V OL | Linéaire | Couper |
Le VTC du CMOS est illustré dans la figure ci-dessous -
Les circuits ou portes logiques combinatoires, qui effectuent des opérations booléennes sur plusieurs variables d'entrée et déterminent les sorties en tant que fonctions booléennes des entrées, sont les éléments de base de tous les systèmes numériques. Nous examinerons les configurations de circuits simples telles que les portes NAND et NOR à deux entrées, puis étendrons notre analyse à des cas plus généraux de structures de circuits à entrées multiples.
Ensuite, les circuits logiques CMOS seront présentés de manière similaire. Nous soulignerons les similitudes et les différences entre la logique d'appauvrissement-charge nMOS et les circuits logiques CMOS et soulignerons les avantages des portes CMOS avec des exemples. Dans sa forme la plus générale, un circuit logique combinatoire, ou porte, exécutant une fonction booléenne peut être représenté comme un système à entrées multiples et à sortie unique, comme illustré sur la figure.
Les tensions de nœud, référencées au potentiel de terre, représentent toutes les variables d'entrée. En utilisant la convention de logique positive, la valeur booléenne (ou logique) de «1» peut être représentée par une haute tension de VDD, et la valeur booléenne (ou logique) de «0» peut être représentée par une basse tension de 0. La sortie Le nœud est chargé d'une capacité C L , qui représente les capacités combinées du dispositif parasite dans le circuit.
Circuits logiques CMOS
Porte NOR CMOS à deux entrées
Le circuit se compose d'un réseau n connecté en parallèle et d'un réseau p complémentaire connecté en série. Les tensions d'entrée V X et V Y sont appliquées aux grilles d'un transistor nMOS et d'un transistor pMOS.
Lorsque l'une ou les deux entrées sont hautes, c'est-à-dire lorsque le n-net crée un chemin conducteur entre le noeud de sortie et la masse, le p-net est coupé. Si les deux tensions d'entrée sont faibles, c'est-à-dire que le n-net est coupé, alors le p-net crée un chemin conducteur entre le noeud de sortie et la tension d'alimentation.
Pour toute combinaison d'entrée donnée, la structure du circuit complémentaire est telle que la sortie est connectée soit à V DD soit à la terre via un chemin à faible résistance et un chemin de courant continu entre le V DD et la terre n'est établi pour aucune combinaison d'entrée. La tension de sortie du CMOS, la porte NOR à deux entrées obtiendra une tension logique basse de V OL = 0 et une tension logique haute de V OH = V DD . L'équation de la tension de seuil de commutation V th est donnée par
$$V_{th}\left ( NOR2 \right ) = \frac{V_{T,n}+\frac{1}{2}\sqrt{\frac{k_{p}}{k_{n}}\left ( V_{DD}-\left | V_{T,p} \right | \right )}}{1+\frac{1}{2}\sqrt{\frac{k_{p}}{k_{n}}}}$$
Disposition de la porte NOR CMOS à 2 entrées
La figure montre un exemple de disposition de la porte NOR CMOS à 2 entrées, utilisant un métal monocouche et du silicium polycristallin monocouche. Les caractéristiques de cette mise en page sont -
- Polylignes verticales uniques pour chaque entrée
- Formes actives uniques pour les appareils N et P, respectivement
- Bus métalliques fonctionnant à l'horizontale
Le diagramme de bâton pour la porte CMOS N0R2 est illustré dans la figure ci-dessous; qui correspond directement à la mise en page, mais ne contient pas d'informations W et L. Les zones de diffusion sont représentées par des rectangles, les connexions métalliques et les lignes pleines et les cercles représentent respectivement les contacts, et les bandes hachurées représentent les colonnes en polysilicium. Le diagramme en bâtons est utile pour planifier une topologie de mise en page optimale.
Porte NAND CMOS à deux entrées
Le schéma de circuit de la porte CMOS NAND à deux entrées est donné dans la figure ci-dessous.
Le principe de fonctionnement du circuit est exactement le double du fonctionnement NOR CMOS à deux entrées. Le n-net constitué de deux transistors nMOS connectés en série crée un chemin conducteur entre le noeud de sortie et la terre, si les deux tensions d'entrée sont logiquement élevées. Les deux transistors pMOS connectés en parallèle dans p-net seront désactivés.
Pour toutes les autres combinaisons d 'entrées, l' un ou les deux transistors pMOS seront activés, tandis que p - net est coupé, créant ainsi un chemin de courant entre le noeud de sortie et la tension d 'alimentation. Le seuil de commutation pour cette porte est obtenu comme -
$$V_{th}\left ( NAND2 \right ) = \frac{V_{T,n}+2\sqrt{\frac{k_{p}}{k_{n}}\left ( V_{DD}-\left | V_{T,p} \right | \right )}}{1+2\sqrt{\frac{k_{p}}{k_{n}}}}$$
Les caractéristiques de cette mise en page sont les suivantes -
- Les lignes de silicium polycristallin simples pour les entrées s'étendent verticalement sur les régions actives N et P.
- Des formes actives uniques sont utilisées pour construire à la fois des périphériques nMOS et les deux périphériques pMOS.
- Le bus d'alimentation fonctionne horizontalement en haut et en bas de la mise en page.
- Les fils de sortie sont horizontaux pour une connexion facile au circuit voisin.
Circuits logiques complexes
Porte logique complexe de charge de déplétion NMOS
Pour réaliser des fonctions complexes de plusieurs variables d'entrée, les structures de circuit de base et les principes de conception développés pour NOR et NAND peuvent être étendus à des portes logiques complexes. La possibilité de réaliser des fonctions logiques complexes, en utilisant un petit nombre de transistors, est l'une des caractéristiques les plus intéressantes des circuits logiques nMOS et CMOS. Considérez la fonction booléenne suivante comme exemple.
$$\overline{Z=P\left ( S+T \right )+QR}$$
La porte logique complexe d'épuisement-charge nMOS utilisée pour réaliser cette fonction est représentée sur la figure. Sur cette figure, la branche de commande nMOS gauche de trois transistors de commande est utilisée pour exécuter la fonction logique P (S + T), tandis que la branche de droite exécute la fonction QR. En connectant les deux branches en parallèle et en plaçant le transistor de charge entre le noeud de sortie et la tension d'alimentationVDD,on obtient la fonction complexe donnée. Chaque variable d'entrée est affectée à un seul pilote.
L'inspection de la topologie du circuit donne des principes de conception simples du réseau pull-down -
- Les opérations OR sont effectuées par des pilotes connectés en parallèle.
- Les opérations ET sont effectuées par des pilotes connectés en série.
- L'inversion est fournie par la nature du fonctionnement du circuit MOS.
Si toutes les variables d'entrée sont logiquement hautes dans le circuit réalisant la fonction, le pilote équivalent (W/L) le rapport du réseau pull-down composé de cinq transistors nMOS est
$$\frac{W}{L}=\frac{1}{\frac{1}{\left ( W/L \right )Q}+\frac{1}{\left ( W/L \right )R}}+\frac{1}{\frac{1}{\left ( W/L \right )P}+\frac{1}{\left ( W/L \right )S+\left ( W/L \right )Q}}$$
Portes logiques CMOS complexes
La réalisation du réseau n-net, ou réseau pull-down, est basée sur les mêmes principes de conception de base examinés pour la porte logique complexe d'épuisement-charge nMOS. Le réseau pull-up pMOS doit être le double réseau du n-net.
Cela signifie que toutes les connexions parallèles dans le réseau nMOS correspondront à une connexion en série dans le réseau pMOS, et que toutes les connexions en série dans le réseau nMOS correspondront à une connexion parallèle dans le réseau pMOS. La figure montre une construction simple du graphique dual p-net (pull-up) à partir du graphique n-net (pull-down).
Chaque transistor pilote dans le réseau pull-down est représenté par ai et chaque nœud est représenté par un sommet dans le graphique pull-down. Ensuite, un nouveau sommet est créé dans chaque zone confinée du graphe d'extraction, et les sommets voisins sont reliés par des arêtes qui ne traversent chaque arête du graphe déroulant qu'une seule fois. Ce nouveau graphique montre le réseau pull-up.
Technique de mise en page utilisant la méthode du graphe d'Euler
La figure montre l'implémentation CMOS d'une fonction complexe et son diagramme en bâton fait avec un ordre de porte arbitraire qui donne une disposition très non optimale pour la porte CMOS.
Dans ce cas, la séparation entre les colonnes en polysilicium doit permettre une séparation diffusion-diffusion entre les deux. Cela consomme certainement une quantité considérable de surface de silicium supplémentaire.
En utilisant le chemin Euler, nous pouvons obtenir une mise en page optimale. Le chemin d'Euler est défini comme un chemin ininterrompu qui traverse chaque arête (branche) du graphe exactement une fois. Trouvez le chemin d'Euler à la fois dans l'arborescence déroulante et dans l'arborescence déroulante avec un ordre identique des entrées.
Les circuits logiques sont divisés en deux catégories - (a) les circuits combinés et (b) les circuits séquentiels.
Dans les circuits combinés, la sortie dépend uniquement de l'état des dernières entrées.
Dans les circuits séquentiels, la sortie dépend non seulement des dernières entrées, mais également de l'état des entrées précédentes. Les circuits séquentiels contiennent des éléments de mémoire.
Les circuits séquentiels sont de trois types -
Bistable- Les circuits bistables ont deux points de fonctionnement stables et seront dans l'un ou l'autre des états. Exemple - Cellules de mémoire, verrous, bascules et registres.
Monostable- Les circuits monostables n'ont qu'un seul point de fonctionnement stable et même s'ils sont temporairement perturbés à l'état opposé, ils reviendront dans le temps à leur point de fonctionnement stable. Exemple: minuteries, générateurs d'impulsions.
Astable- les circuits n'ont pas de point de fonctionnement stable et oscillent entre plusieurs états. Exemple - Oscillateur en anneau.
Circuits logiques CMOS
SR Latch basé sur NOR Gate
Si l'entrée réglée (S) est égale à la logique "1" et l'entrée de réinitialisation est égale à la logique "0." alors la sortie Q sera forcée à la logique "1". Tandis que$\overline{Q}$ est obligé de logique "0". Cela signifie que le verrou SR sera activé, quel que soit son état précédent.
De même, si S est égal à "0" et R est égal à "1" alors la sortie Q sera forcée à "0" tandis que $\overline{Q}$ est obligé de "1". Cela signifie que le verrou est réinitialisé, quel que soit son état précédemment maintenu. Enfin, si les deux entrées S et R sont égales à la logique"1" alors les deux sorties seront forcées à la logique "0" qui entre en conflit avec la complémentarité de Q et $\overline{Q}$.
Par conséquent, cette combinaison d'entrées n'est pas autorisée pendant le fonctionnement normal. Le tableau de vérité de SR Latch basé sur NOR est donné dans le tableau.
S | R | Q | $\overline{Q}$ | Opération |
---|---|---|---|---|
0 | 0 | Q | $\overline{Q}$ | Tenir |
1 | 0 | 1 | 0 | Ensemble |
0 | 1 | 0 | 1 | Réinitialiser |
1 | 1 | 0 | 0 | Interdit |
Le verrou CMOS SR basé sur la porte NOR est illustré dans la figure ci-dessous.
Si le S est égal à V OH et le R est égal à V OL , les deux transistors connectés en parallèle M1 et M2 seront passant. La tension sur le nœud$\overline{Q}$supposera un niveau logique bas de V OL = 0.
Dans le même temps, M3 et M4 sont désactivés, ce qui se traduit par une tension logique haute V OH au nœud Q. Si le R est égal à V OH et le S est égal à V OL , M1 et M2 sont désactivés et M3 et M4 sont allumés.
SR Latch basé sur NAND Gate
Le schéma de principe et le schéma de niveau de porte du verrou SR basé sur NAND sont illustrés sur la figure. Les petits cercles aux bornes d'entrée S et R indiquent que le circuit répond aux signaux d'entrée bas actifs. La table de vérité du verrou SR basé sur NAND est donnée dans le tableau
S | R | Q | Q ′ | |
0 | 0 | NC | NC | Pas de changement. Latch est resté dans l'état actuel. |
1 | 0 | 1 | 0 | Latch SET. |
0 | 1 | 0 | 1 | Verrouiller RESET. |
1 | 1 | 0 | 0 | Condition non valide. |
Si S va à 0 (tandis que R = 1), Q va haut, tirant $\overline{Q}$ bas et le verrou entre dans l'état Set
S = 0 alors Q = 1 (si R = 1)
Si R va à 0 (tandis que S = 1), Q va haut, tirant $\overline{Q}$ bas et le loquet est réinitialisé
R = 0 alors Q = 1 (si S = 1)
L'état de maintien nécessite que S et R soient tous deux élevés. Si S = R = 0, la sortie n'est pas autorisée, car elle entraînerait un état indéterminé. Le verrou CMOS SR basé sur la porte NAND est illustré dans la figure.
Le verrou nMOS SR à charge d'épuisement basé sur la porte NAND est illustré sur la figure. Le fonctionnement est similaire à celui du verrou CMOS NAND SR. L'implémentation du circuit CMOS a une faible dissipation de puissance statique et une marge de bruit élevée.
Circuits logiques CMOS
Loquet SR cadencé
La figure montre un verrou SR basé sur NOR avec une horloge ajoutée. Le verrou ne répond aux entrées S et R que lorsque CLK est à l'état haut.
Lorsque CLK est bas, le verrou conserve son état actuel. Observez que Q change d'état -
- Lorsque S devient élevé pendant CLK positif.
- Sur le bord avant CLK après des changements de S & R pendant le temps bas CLK.
- Un pépin positif dans S alors que CLK est élevé
- Lorsque R devient élevé pendant CLK positif.
L'implémentation CMOS AOI du verrou SR cadencé basé sur NOR est illustrée sur la figure. Notez que seuls 12 transistors sont nécessaires.
Lorsque CLK est bas, deux bornes série dans N arbre N sont ouvertes et deux transistors parallèles dans l'arbre P sont ON, conservant ainsi l'état dans la cellule mémoire.
Lorsque l'horloge est haute, le circuit devient simplement un verrou CMOS basé sur NOR qui répondra aux entrées S et R.
Clocked SR Latch based on NAND Gate
Le circuit est implémenté avec quatre portes NAND. Si ce circuit est implémenté avec CMOS, il nécessite 16 transistors.
- Le verrou ne répond à S ou R que si CLK est élevé.
- Si les signaux d'entrée et les signaux CLK sont tous deux actifs à l'état haut: c'est-à-dire que la sortie de verrouillage Q sera définie lorsque CLK = "1" S = "1" et R = "0"
- De même, le verrou sera réinitialisé lorsque CLK = "1", "S =" 0, "et
Lorsque CLK est bas, le verrou conserve son état actuel.
Loquet JK cadencé
La figure ci-dessus montre un verrou JK cadencé, basé sur des portes NAND. L'inconvénient d'un verrou SR est que lorsque S et R sont tous deux hauts, son état de sortie devient indéterminé. Le verrou JK élimine ce problème en utilisant la rétroaction de la sortie vers l'entrée, de sorte que tous les états d'entrée de la table de vérité sont autorisés. Si J = K = 0, le verrou conservera son état actuel.
Si J = 1 et K = 0, le verrou sera positionné sur le prochain front d'horloge positif, c'est-à-dire Q = 1, $\overline{Q}$ = 0
Si J = 0 et K = 1, le verrou se réinitialise sur le prochain front d'horloge positif, c'est-à-dire Q = 1 et $\overline{Q}$ = 0.
Si J = K = 1, le verrou bascule sur le prochain front d'horloge positif
Le fonctionnement du verrou JK cadencé est résumé dans la table de vérité donnée dans le tableau.
J |
K |
Q |
$\overline{Q}$ | S |
R |
Q |
$\overline{Q}$ | Operation |
0 | 0 | 0 | 1 | 1 | 1 | 0 | 1 | Tenir |
1 | 0 | 1 | 1 | 1 | 0 | |||
0 | 1 | 0 | 1 | 1 | 1 | 0 | 1 | Réinitialiser |
1 | 0 | 1 | 0 | 0 | 1 | |||
1 | 0 | 0 | 1 | 0 | 1 | 1 | 0 | Ensemble |
1 | 0 | 1 | 1 | 1 | 0 | |||
1 | 1 | 0 | 1 | 0 | 1 | 1 | 0 | basculer |
1 | 0 | 1 | 0 | 0 | 1 |
Implémentation CMOS D Latch
Le verrou D est normalement implémenté avec des commutateurs de porte de transmission (TG) comme indiqué sur la figure. L'entrée TG est activée avec CLK tandis que la boucle de retour de verrouillage TG est activée avec CLK. L'entrée D est acceptée lorsque CLK est haut. Lorsque CLK passe au niveau bas, l'entrée est ouverte et le verrou est réglé avec les données précédentes D.
VHDL est synonyme de langage de description de matériel de circuit intégré à très haute vitesse. C'est un langage de programmation utilisé pour modéliser un système numérique par flux de données, style comportemental et structurel de modélisation. Ce langage a été introduit pour la première fois en 1981 pour le département de la Défense (DoD) dans le cadre du programme VHSIC.
Décrire un design
En VHDL, une entité est utilisée pour décrire un module matériel. Une entité peut être décrite en utilisant,
- Déclaration d'entité
- Architecture
- Configuration
- Déclaration de colis
- Corps du paquet
Voyons ce que c'est?
Déclaration d'entité
Il définit les noms, les signaux de sortie d'entrée et les modes d'un module matériel.
Syntax -
entity entity_name is
Port declaration;
end entity_name;
Une déclaration d'entité doit commencer par «entité» et se terminer par des mots-clés «fin». La direction sera entrée, sortie ou inout.
Dans | Le port peut être lu |
En dehors | Le port peut être écrit |
Inout | Le port peut être lu et écrit |
Tampon | Le port peut être lu et écrit, il ne peut avoir qu'une seule source. |
Architecture -
L'architecture peut être décrite en utilisant un style structurel, de flux de données, comportemental ou mixte.
Syntax -
architecture architecture_name of entity_name
architecture_declarative_part;
begin
Statements;
end architecture_name;
Ici, nous devons spécifier le nom de l'entité pour laquelle nous écrivons le corps de l'architecture. Les déclarations d'architecture doivent être à l'intérieur des mots-clés «begin» et «énd». La partie déclarative de l'architecture peut contenir des variables, des constantes ou une déclaration de composant.
Modélisation des flux de données
Dans ce style de modélisation, le flux de données à travers l'entité est exprimé en utilisant un signal simultané (parallèle). Les instructions simultanées dans VHDL sont WHEN et GENERATE.
Outre eux, les affectations utilisant uniquement des opérateurs (AND, NOT, +, *, sll, etc.) peuvent également être utilisées pour construire du code.
Enfin, un type spécial d'affectation, appelé BLOCK, peut également être utilisé dans ce type de code.
Dans le code simultané, les éléments suivants peuvent être utilisés -
- Operators
- L'instruction WHEN (WHEN / ELSE ou WITH / SELECT / WHEN);
- L'instruction GENERATE;
- L'instruction BLOCK
Modélisation comportementale
Dans ce style de modélisation, le comportement d'une entité en tant qu'ensemble d'instructions est exécuté séquentiellement dans l'ordre spécifié. Seules les instructions placées dans un PROCESS, FUNCTION ou PROCEDURE sont séquentielles.
PROCESSES, FONCTIONS et PROCÉDURES sont les seules sections de code qui sont exécutées séquentiellement.
Cependant, dans l'ensemble, n'importe lequel de ces blocs est toujours en même temps que toute autre instruction placée en dehors de celui-ci.
Un aspect important du code de comportement est qu'il n'est pas limité à la logique séquentielle. En effet, avec lui, nous pouvons construire des circuits séquentiels ainsi que des circuits combinatoires.
Les instructions de comportement sont IF, WAIT, CASE et LOOP. Les VARIABLES sont également restreintes et sont censées être utilisées uniquement dans le code séquentiel. VARIABLE ne peut jamais être globale, donc sa valeur ne peut pas être transmise directement.
Modélisation structurelle
Dans cette modélisation, une entité est décrite comme un ensemble de composants interconnectés. Une instruction d'instanciation de composant est une instruction simultanée. Par conséquent, l'ordre de ces déclarations n'est pas important. Le style structurel de la modélisation ne décrit qu'une interconnexion de composants (considérés comme des boîtes noires), sans impliquer aucun comportement des composants eux-mêmes ni de l'entité qu'ils représentent collectivement.
Dans la modélisation structurelle, le corps de l'architecture est composé de deux parties: la partie déclarative (avant le mot-clé begin) et la partie instruction (après le mot-clé begin).
Fonctionnement logique - AND GATE
X | Oui | Z |
---|---|---|
0 | 0 | 0 |
0 | 1 | 0 |
1 | 0 | 0 |
1 | 1 | 1 |
VHDL Code:
Library ieee;
use ieee.std_logic_1164.all;
entity and1 is
port(x,y:in bit ; z:out bit);
end and1;
architecture virat of and1 is
begin
z<=x and y;
end virat;
Formes d'onde
Fonctionnement logique - OU Gate
X | Oui | Z |
---|---|---|
0 | 0 | 0 |
0 | 1 | 1 |
1 | 0 | 1 |
1 | 1 | 1 |
VHDL Code:
Library ieee;
use ieee.std_logic_1164.all;
entity or1 is
port(x,y:in bit ; z:out bit);
end or1;
architecture virat of or1 is
begin
z<=x or y;
end virat;
Formes d'onde
Opération logique - PAS Gate
X | Oui |
---|---|
0 | 1 |
1 | 0 |
VHDL Code:
Library ieee;
use ieee.std_logic_1164.all;
entity not1 is
port(x:in bit ; y:out bit);
end not1;
architecture virat of not1 is
begin
y<=not x;
end virat;
Formes d'onde
Fonctionnement logique - NAND Gate
X | Oui | z |
---|---|---|
0 | 0 | 1 |
0 | 1 | 1 |
1 | 0 | 1 |
1 | 1 | 0 |
VHDL Code:
Library ieee;
use ieee.std_logic_1164.all;
entity nand1 is
port(a,b:in bit ; c:out bit);
end nand1;
architecture virat of nand1 is
begin
c<=a nand b;
end virat;
Formes d'onde
Fonctionnement logique - Porte NOR
X | Oui | z |
---|---|---|
0 | 0 | 1 |
0 | 1 | 0 |
1 | 0 | 0 |
1 | 1 | 0 |
VHDL Code:
Library ieee;
use ieee.std_logic_1164.all;
entity nor1 is
port(a,b:in bit ; c:out bit);
end nor1;
architecture virat of nor1 is
begin
c<=a nor b;
end virat;
Formes d'onde
Fonctionnement logique - XOR Gate
X | Oui | Z |
---|---|---|
0 | 0 | 1 |
0 | 1 | 1 |
1 | 0 | 1 |
1 | 1 | 0 |
VHDL Code:
Library ieee;
use ieee.std_logic_1164.all;
entity xor1 is
port(a,b:in bit ; c:out bit);
end xor1;
architecture virat of xor1 is
begin
c<=a xor b;
end virat;
Formes d'onde
Fonctionnement logique - X-NOR Gate
X | Oui | Z |
---|---|---|
0 | 0 | 1 |
0 | 1 | 1 |
1 | 0 | 1 |
1 | 1 | 0 |
VHDL Code:
Library ieee;
use ieee.std_logic_1164.all;
entity xnor1 is
port(a,b:in bit ; c:out bit);
end xnor1;
architecture virat of xnor1 is
begin
c<=not(a xor b);
end virat;
Formes d'onde
Ce chapitre explique la programmation VHDL pour les circuits combinés.
Code VHDL pour un demi-additionneur
VHDL Code:
Library ieee;
use ieee.std_logic_1164.all;
entity half_adder is
port(a,b:in bit; sum,carry:out bit);
end half_adder;
architecture data of half_adder is
begin
sum<= a xor b;
carry <= a and b;
end data;
Formes d'onde
Code VHDL pour un additionneur complet
Library ieee;
use ieee.std_logic_1164.all;
entity full_adder is port(a,b,c:in bit; sum,carry:out bit);
end full_adder;
architecture data of full_adder is
begin
sum<= a xor b xor c;
carry <= ((a and b) or (b and c) or (a and c));
end data;
Formes d'onde
Code VHDL pour un demi-soustracteur
Library ieee;
use ieee.std_logic_1164.all;
entity half_sub is
port(a,c:in bit; d,b:out bit);
end half_sub;
architecture data of half_sub is
begin
d<= a xor c;
b<= (a and (not c));
end data;
Formes d'onde
Code VHDL pour un soustracteur complet
Library ieee;
use ieee.std_logic_1164.all;
entity full_sub is
port(a,b,c:in bit; sub,borrow:out bit);
end full_sub;
architecture data of full_sub is
begin
sub<= a xor b xor c;
borrow <= ((b xor c) and (not a)) or (b and c);
end data;
Formes d'onde
Code VHDL pour un multiplexeur
Library ieee;
use ieee.std_logic_1164.all;
entity mux is
port(S1,S0,D0,D1,D2,D3:in bit; Y:out bit);
end mux;
architecture data of mux is
begin
Y<= (not S0 and not S1 and D0) or
(S0 and not S1 and D1) or
(not S0 and S1 and D2) or
(S0 and S1 and D3);
end data;
Formes d'onde
Code VHDL pour un démultiplexeur
Library ieee;
use ieee.std_logic_1164.all;
entity demux is
port(S1,S0,D:in bit; Y0,Y1,Y2,Y3:out bit);
end demux;
architecture data of demux is
begin
Y0<= ((Not S0) and (Not S1) and D);
Y1<= ((Not S0) and S1 and D);
Y2<= (S0 and (Not S1) and D);
Y3<= (S0 and S1 and D);
end data;
Formes d'onde
Code VHDL pour un encodeur 8 x 3
library ieee;
use ieee.std_logic_1164.all;
entity enc is
port(i0,i1,i2,i3,i4,i5,i6,i7:in bit; o0,o1,o2: out bit);
end enc;
architecture vcgandhi of enc is
begin
o0<=i4 or i5 or i6 or i7;
o1<=i2 or i3 or i6 or i7;
o2<=i1 or i3 or i5 or i7;
end vcgandhi;
Formes d'onde
Code VHDL pour un décodeur 3 x 8
library ieee;
use ieee.std_logic_1164.all;
entity dec is
port(i0,i1,i2:in bit; o0,o1,o2,o3,o4,o5,o6,o7: out bit);
end dec;
architecture vcgandhi of dec is
begin
o0<=(not i0) and (not i1) and (not i2);
o1<=(not i0) and (not i1) and i2;
o2<=(not i0) and i1 and (not i2);
o3<=(not i0) and i1 and i2;
o4<=i0 and (not i1) and (not i2);
o5<=i0 and (not i1) and i2;
o6<=i0 and i1 and (not i2);
o7<=i0 and i1 and i2;
end vcgandhi;
Formes d'onde
Code VHDL - additionneur parallèle 4 bits
library IEEE;
use IEEE.STD_LOGIC_1164.all;
entity pa is
port(a : in STD_LOGIC_VECTOR(3 downto 0);
b : in STD_LOGIC_VECTOR(3 downto 0);
ca : out STD_LOGIC;
sum : out STD_LOGIC_VECTOR(3 downto 0)
);
end pa;
architecture vcgandhi of pa is
Component fa is
port (a : in STD_LOGIC;
b : in STD_LOGIC;
c : in STD_LOGIC;
sum : out STD_LOGIC;
ca : out STD_LOGIC
);
end component;
signal s : std_logic_vector (2 downto 0);
signal temp: std_logic;
begin
temp<='0';
u0 : fa port map (a(0),b(0),temp,sum(0),s(0));
u1 : fa port map (a(1),b(1),s(0),sum(1),s(1));
u2 : fa port map (a(2),b(2),s(1),sum(2),s(2));
ue : fa port map (a(3),b(3),s(2),sum(3),ca);
end vcgandhi;
Formes d'onde
Code VHDL - Vérificateur de parité 4 bits
library ieee;
use ieee.std_logic_1164.all;
entity parity_checker is
port (a0,a1,a2,a3 : in std_logic;
p : out std_logic);
end parity_checker;
architecture vcgandhi of parity_checker is
begin
p <= (((a0 xor a1) xor a2) xor a3);
end vcgandhi;
Formes d'onde
Code VHDL - Générateur de parité 4 bits
library ieee;
use ieee.std_logic_1164.all;
entity paritygen is
port (a0, a1, a2, a3: in std_logic; p_odd, p_even: out std_logic);
end paritygen;
architecture vcgandhi of paritygen is
begin
process (a0, a1, a2, a3)
if (a0 ='0' and a1 ='0' and a2 ='0' and a3 =’0’)
then odd_out <= "0";
even_out <= "0";
else
p_odd <= (((a0 xor a1) xor a2) xor a3);
p_even <= not(((a0 xor a1) xor a2) xor a3);
end vcgandhi
Formes d'onde
Ce chapitre explique comment effectuer la programmation VHDL pour les circuits séquentiels.
Code VHDL pour un SR Latch
library ieee;
use ieee.std_logic_1164.all;
entity srl is
port(r,s:in bit; q,qbar:buffer bit);
end srl;
architecture virat of srl is
signal s1,r1:bit;
begin
q<= s nand qbar;
qbar<= r nand q;
end virat;
Formes d'onde
Code VHDL pour un verrou D
library ieee;
use ieee.std_logic_1164.all;
entity Dl is
port(d:in bit; q,qbar:buffer bit);
end Dl;
architecture virat of Dl is
signal s1,r1:bit;
begin
q<= d nand qbar;
qbar<= d nand q;
end virat;
Formes d'onde
Code VHDL pour une bascule SR
library ieee;
use ieee.std_logic_1164.all;
entity srflip is
port(r,s,clk:in bit; q,qbar:buffer bit);
end srflip;
architecture virat of srflip is
signal s1,r1:bit;
begin
s1<=s nand clk;
r1<=r nand clk;
q<= s1 nand qbar;
qbar<= r1 nand q;
end virat;
Formes d'onde
Code VHDL pour une bascule JK
library IEEE;
use IEEE.STD_LOGIC_1164.all;
entity jk is
port(
j : in STD_LOGIC;
k : in STD_LOGIC;
clk : in STD_LOGIC;
reset : in STD_LOGIC;
q : out STD_LOGIC;
qb : out STD_LOGIC
);
end jk;
architecture virat of jk is
begin
jkff : process (j,k,clk,reset) is
variable m : std_logic := '0';
begin
if (reset = '1') then
m : = '0';
elsif (rising_edge (clk)) then
if (j/ = k) then
m : = j;
elsif (j = '1' and k = '1') then
m : = not m;
end if;
end if;
q <= m;
qb <= not m;
end process jkff;
end virat;
Formes d'onde
Code VHDL pour une bascule D
Library ieee;
use ieee.std_logic_1164.all;
entity dflip is
port(d,clk:in bit; q,qbar:buffer bit);
end dflip;
architecture virat of dflip is
signal d1,d2:bit;
begin
d1<=d nand clk;
d2<=(not d) nand clk;
q<= d1 nand qbar;
qbar<= d2 nand q;
end virat;
Formes d'onde
Code VHDL pour une bascule T
library IEEE;
use IEEE.STD_LOGIC_1164.all;
entity Toggle_flip_flop is
port(
t : in STD_LOGIC;
clk : in STD_LOGIC;
reset : in STD_LOGIC;
dout : out STD_LOGIC
);
end Toggle_flip_flop;
architecture virat of Toggle_flip_flop is
begin
tff : process (t,clk,reset) is
variable m : std_logic : = '0';
begin
if (reset = '1') then
m : = '0';
elsif (rising_edge (clk)) then
if (t = '1') then
m : = not m;
end if;
end if;
dout < = m;
end process tff;
end virat;
Formes d'onde
Code VHDL pour un compteur ascendant 4 bits
library IEEE;
use ieee.std_logic_1164.all;
use ieee.std_logic_unsigned.all;
entity counter is
port(Clock, CLR : in std_logic;
Q : out std_logic_vector(3 downto 0)
);
end counter;
architecture virat of counter is
signal tmp: std_logic_vector(3 downto 0);
begin
process (Clock, CLR)
begin
if (CLR = '1') then
tmp < = "0000";
elsif (Clock'event and Clock = '1') then
mp <= tmp + 1;
end if;
end process;
Q <= tmp;
end virat;
Formes d'onde
Code VHDL pour un décompteur 4 bits
library ieee;
use ieee.std_logic_1164.all;
use ieee.std_logic_unsigned.all;
entity dcounter is
port(Clock, CLR : in std_logic;
Q : out std_logic_vector(3 downto 0));
end dcounter;
architecture virat of dcounter is
signal tmp: std_logic_vector(3 downto 0);
begin
process (Clock, CLR)
begin
if (CLR = '1') then
tmp <= "1111";
elsif (Clock'event and Clock = '1') then
tmp <= tmp - 1;
end if;
end process;
Q <= tmp;
end virat;
Formes d'onde
Verilog est une LANGUE DE DESCRIPTION DU MATÉRIEL (HDL). C'est un langage utilisé pour décrire un système numérique comme un commutateur réseau ou un microprocesseur ou une mémoire ou une bascule. Cela signifie qu'en utilisant un HDL, nous pouvons décrire n'importe quel matériel numérique à n'importe quel niveau. Les conceptions, qui sont décrites en HDL sont indépendantes de la technologie, très faciles à concevoir et à déboguer, et sont normalement plus utiles que les schémas, en particulier pour les grands circuits.
Verilog prend en charge une conception à de nombreux niveaux d'abstraction. Les trois principaux sont -
- Niveau comportemental
- Niveau de transfert de registre
- Niveau de la porte
Niveau comportemental
Ce niveau décrit un système par algorithmes simultanés (comportemental). Chaque algorithme est séquentiel, ce qui signifie qu'il se compose d'un ensemble d'instructions qui sont exécutées une par une. Les fonctions, tâches et blocs sont les principaux éléments. Il n'y a aucun égard à la réalisation structurelle de la conception.
Registre − Niveau de transfert
Les conceptions utilisant le niveau de transfert de registre spécifient les caractéristiques d'un circuit utilisant des opérations et le transfert de données entre les registres. La définition moderne d'un code RTL est "Tout code synthétisable est appelé code RTL".
Niveau de la porte
Au niveau logique, les caractéristiques d'un système sont décrites par des liens logiques et leurs propriétés de synchronisation. Tous les signaux sont des signaux discrets. Ils ne peuvent avoir que des valeurs logiques définies («0», «1», «X», «Z»). Les opérations utilisables sont des primitives logiques prédéfinies (portes de base). La modélisation au niveau de la porte n'est peut-être pas une bonne idée pour la conception logique. Le code de niveau de porte est généré à l'aide d'outils tels que des outils de synthèse et sa netlist est utilisée pour la simulation de niveau de porte et pour le backend.
Jetons lexicaux
Les fichiers texte source du langage Verilog sont un flux de jetons lexicaux. Un jeton se compose d'un ou plusieurs caractères, et chaque caractère est exactement dans un jeton.
Les jetons lexicaux de base utilisés par le Verilog HDL sont similaires à ceux du langage de programmation C. Verilog est sensible à la casse. Tous les mots clés sont en minuscules.
Espace blanc
Les espaces blancs peuvent contenir des caractères pour les espaces, les tabulations, les nouvelles lignes et les flux de formulaires. Ces caractères sont ignorés sauf lorsqu'ils servent à séparer les jetons.
Les espaces blancs sont les espaces vides, les tabulations, les retours chariot, les nouvelles lignes et les sauts de formulaire.
commentaires
Il existe deux formes pour représenter les commentaires
- 1) Les commentaires sur une seule ligne commencent par le jeton // et se terminent par un retour chariot.
Ex .: // c'est une syntaxe sur une seule ligne
- 2) Les commentaires multilignes commencent par le jeton / * et se terminent par le jeton * /
Ex .: / * c'est une syntaxe multiligne * /
Nombres
Vous pouvez spécifier un nombre au format binaire, octal, décimal ou hexadécimal. Les nombres négatifs sont représentés dans les nombres de compliment de 2. Verilog autorise les entiers, les nombres réels et les nombres signés et non signés.
La syntaxe est donnée par - <size> <radix> <value>
La taille ou le nombre non dimensionné peut être défini dans <Size> et <radix> définit s'il est binaire, octal, hexadécimal ou décimal.
Identifiants
L'identifiant est le nom utilisé pour définir l'objet, tel qu'une fonction, un module ou un registre. Les identificateurs doivent commencer par des caractères alphabétiques ou des caractères de soulignement. Ex. A_Z, a_z, _
Les identificateurs sont une combinaison de caractères alphabétiques, numériques, de soulignement et $. Ils peuvent contenir jusqu'à 1024 caractères.
Les opérateurs
Les opérateurs sont des caractères spéciaux utilisés pour mettre des conditions ou pour faire fonctionner les variables. Un, deux et parfois trois caractères sont utilisés pour effectuer des opérations sur des variables.
Ex. >, +, ~ Et! =.
Mots-clés Verilog
Les mots qui ont une signification particulière dans Verilog sont appelés les mots-clés Verilog. Par exemple, affectez, case, while, wire, reg et, or, nand et module. Ils ne doivent pas être utilisés comme identifiants. Les mots-clés Verilog incluent également les directives du compilateur et les tâches et fonctions système.
Modélisation au niveau de la porte
Verilog a des primitives intégrées telles que des portes logiques, des portes de transmission et des commutateurs. Ceux-ci sont rarement utilisés pour les travaux de conception, mais ils sont utilisés dans le monde de la post-synthèse pour la modélisation des cellules ASIC / FPGA.
La modélisation au niveau de la porte présente deux propriétés -
Drive strength- La force des portes de sortie est définie par la force d'entraînement. La sortie est la plus puissante s'il existe une connexion directe à la source. La force diminue si la connexion se fait via un transistor conducteur et moins lorsqu'elle est connectée via un résistif pull-up / down. La force d'entraînement n'est généralement pas spécifiée, auquel cas les forces par défaut sont strong1 et strong0.
Delays- si les délais ne sont pas spécifiés, les portes n'ont pas de délais de propagation; si deux retards sont spécifiés, alors le premier représente le retard de montée et le second, le retard de chute; si un seul retard est spécifié, alors les deux, montée et descente sont égales. Les retards peuvent être ignorés dans la synthèse.
Primitives de porte
Les portes logiques de base utilisant une sortie et de nombreuses entrées sont utilisées dans Verilog. GATE utilise l'un des mots-clés - et, nand, ou, ni, xor, xnor à utiliser dans Verilog pour N nombre d'entrées et 1 sortie.
Example:
Module gate()
Wire ot0;
Wire ot1;
Wire ot2;
Reg in0,in1,in2,in3;
Not U1(ot0,in0);
Xor U2(ot1,in1,in2,in3);
And U3(ot2, in2,in3,in0)
Primitives de porte de transmission
Les primitives de porte de transmission comprennent à la fois des tampons et des onduleurs. Ils ont une seule entrée et une ou plusieurs sorties. Dans la syntaxe d'instanciation de porte présentée ci-dessous, GATE représente le mot clé buf ou NOT gate.
Exemple: Not, buf, bufif0, bufif1, notif0, notif1
Not - n outout onduleur
Tampon de sortie Buf - n
Bufifo - tampon à trois états, activation basse active
Bufif1 - tampon à trois états, activation haute active
Notifo - onduleur à trois états, activation basse active
Notif1 - onduleur à trois états, validation haute active
Example:
Module gate()
Wire out0;
Wire out1;
Reg in0,in1;
Not U1(out0,in0);
Buf U2(out0,in0);
Types de données
Ensemble de valeurs
Verilog se compose principalement de quatre valeurs de base. Tous les types de données Verilog, qui sont utilisés dans Verilog stockent ces valeurs -
0 (zéro logique ou condition fausse)
1 (logique un ou condition vraie)
x (valeur logique inconnue)
z (état haute impédance)
l'utilisation de x et z est très limitée pour la synthèse.
Câble
Un fil est utilisé pour représenter un fil physique dans un circuit et il est utilisé pour la connexion de portes ou de modules. La valeur d'un fil ne peut être lue et non affectée que dans une fonction ou un bloc. Un fil ne peut pas stocker de valeur mais est toujours piloté par une instruction d'affectation continue ou en connectant le fil à la sortie d'une porte / module. D'autres types spécifiques de fils sont -
Wand (wired-AND) - ici, la valeur de Wand dépend du ET logique de tous les pilotes de périphériques connectés.
Wor (wired-OR) - ici la valeur d'un Wor dépend du OU logique de tous les pilotes de périphériques qui y sont connectés.
Tri (three-state) - ici tous les pilotes connectés à un tri doivent être z, sauf un seul (qui détermine la valeur de tri).
Example:
Wire [msb:lsb] wire_variable_list;
Wirec // simple wire
Wand d;
Assign d = a; // value of d is the logical AND of
Assign d = b; // a and b
Wire [9:0] A; // a cable (vector) of 10 wires.
Wand [msb:lsb] wand_variable_list;
Wor [msb:lsb] wor_variable_list;
Tri [msb:lsb] tri_variable_list;
S'inscrire
Un reg (registre) est un objet de données, qui contient la valeur d'une affectation procédurale à l'autre et n'est utilisé que dans différentes fonctions et blocs procéduraux. Un reg est un simple registre de type variable Verilog et ne peut pas impliquer un registre physique. Dans les registres multi-bits, les données sont stockées sous la forme de nombres non signés et l'extension de signe n'est pas utilisée.
Exemple -
reg c; // variable de registre unique à 1 bit
reg [5: 0] gemme; // un vecteur 6 bits;
reg [6: 0] d, e; // deux variables 7 bits
Entrée, sortie, entrée
Ces mots clés sont utilisés pour déclarer les ports d'entrée, de sortie et bidirectionnels d'une tâche ou d'un module. Ici, les ports d'entrée et de sortie, qui sont de type filaire et le port de sortie est configuré pour être de type fil, reg, baguette, wor ou tri. Toujours, la valeur par défaut est le type de fil.
Example
Module sample(a, c, b, d);
Input c; // An input where wire is used.
Output a, b; // Two outputs where wire is used.
Output [2:0] d; /* A three-bit output. One must declare type in a separate statement. */
reg [1:0] a; // The above ‘a’ port is for declaration in reg.
Entier
Les entiers sont utilisés dans les variables à usage général. Ils sont principalement utilisés dans les indications de boucles, les constantes et les paramètres. Ils sont de type de données de type «reg». Ils stockent les données sous forme de nombres signés tandis que les types reg déclarés explicitement les stockent sous forme de données non signées. Si l'entier n'est pas défini au moment de la compilation, la taille par défaut serait de 32 bits.
Si un entier contient une constante, le synthétiseur les ajuste à la largeur minimale nécessaire au moment de la compilation.
Example
Integer c; // single 32-bit integer
Assign a = 63; // 63 defaults to a 7-bit variable.
Approvisionnement0, Approvisionnement1
Supply0 définit les fils liés à la logique 0 (masse) et alimentation1 définit les fils liés à la logique 1 (alimentation).
Example
supply0 logic_0_wires;
supply0 gnd1; // equivalent to a wire assigned as 0
supply1 logic_1_wires;
supply1 c, s;
Temps
Le temps est une quantité de 64 bits qui peut être utilisée conjointement avec la tâche système $ time pour conserver le temps de simulation. Le temps n'est pas pris en charge pour la synthèse et n'est donc utilisé qu'à des fins de simulation.
Example
time time_variable_list;
time c;
c = $time; //c = current simulation time
Paramètre
Un paramètre définit une constante qui peut être définie lorsque vous utilisez un module, ce qui permet la personnalisation du module pendant le processus d'instanciation.
Example
Parameter add = 3’b010, sub = 2’b11;
Parameter n = 3;
Parameter [2:0] param2 = 3’b110;
reg [n-1:0] jam; /* A 3-bit register with length of n or above. */
always @(z)
y = {{(add - sub){z}};
if (z)
begin
state = param2[1];
else
state = param2[2];
end
Les opérateurs
Opérateurs arithmétiques
Ces opérateurs effectuent des opérations arithmétiques. Les + et −sont utilisés comme opérateurs unaires (x) ou binaires (z − y).
Les opérateurs qui sont inclus dans l'opération arithmétique sont -
+ (addition), - (soustraction), * (multiplication), / (division),% (module)
Example -
parameter v = 5;
reg[3:0] b, d, h, i, count;
h = b + d;
i = d - v;
cnt = (cnt +1)%16; //Can count 0 thru 15.
Opérateurs relationnels
Ces opérateurs comparent deux opérandes et renvoient le résultat dans un seul bit, 1 ou 0.
Les variables Wire et Reg sont positives. Ainsi (−3'd001) = = 3'd111 et (−3b001)> 3b110.
Les Opérateurs qui sont inclus dans l'opération relationnelle sont -
- == (égal à)
- ! = (différent de)
- > (supérieur à)
- > = (supérieur ou égal à)
- < (Inférieur à)
- <= (inférieur ou égal à)
Example
if (z = = y) c = 1;
else c = 0; // Compare in 2’s compliment; d>b
reg [3:0] d,b;
if (d[3]= = b[3]) d[2:0] > b[2:0];
else b[3];
Equivalent Statement
e = (z == y);
Opérateurs bit par bit
Opérateurs bit par bit qui effectuent une comparaison bit par bit entre deux opérandes.
Les opérateurs qui sont inclus dans l'opération par bit sont -
- & (ET au niveau du bit)
- | (bit à bit OU)
- ~ (PAS au niveau du bit)
- ^ (XOR au niveau du bit)
- ~ ^ ou ^ ~ (XNOR au niveau du bit)
Example
module and2 (d, b, c);
input [1:0] d, b;
output [1:0] c;
assign c = d & b;
end module
Opérateurs logiques
Les opérateurs logiques sont des opérateurs de bits et ne sont utilisés que pour les opérandes à un seul bit. Ils renvoient une valeur de bit unique, 0 ou 1. Ils peuvent travailler sur des entiers ou des groupes de bits, des expressions et traiter toutes les valeurs non nulles comme 1. Les opérateurs logiques sont généralement utilisés dans les instructions conditionnelles car ils fonctionnent avec des expressions.
Les opérateurs inclus dans l'opération logique sont -
- ! (NON logique)
- && (ET logique)
- || (OU logique)
Example
wire[7:0] a, b, c; // a, b and c are multibit variables.
reg x;
if ((a == b) && (c)) x = 1; //x = 1 if a equals b, and c is nonzero.
else x = !a; // x =0 if a is anything but zero.
Opérateurs de réduction
Les opérateurs de réduction sont la forme unaire des opérateurs binaires et opèrent sur tous les bits d'un vecteur d'opérande. Ceux-ci renvoient également une valeur sur un seul bit.
Les opérateurs inclus dans l'opération de réduction sont:
- & (réduction ET)
- | (réduction OU)
- ~ & (réduction NAND)
- ~ | (réduction NOR)
- ^ (réduction XOR)
- ~ ^ ou ^ ~ (réduction XNOR)
Example
Module chk_zero (x, z);
Input [2:0] x;
Output z;
Assign z = & x; // Reduction AND
End module
Opérateurs de quart
Opérateurs de décalage, qui décalent le premier opérande du nombre de bits spécifié par le second opérande dans la syntaxe. Les postes vacants sont remplis de zéros pour les deux directions, les décalages gauche et droit (il n'y a pas d'extension de signe d'utilisation).
Les opérateurs qui sont inclus dans l'opération Shift sont -
- << (décalage vers la gauche)
- >> (décalage vers la droite)
Example
Assign z = c << 3; /* z = c shifted left 3 bits;
Les postes vacants sont pourvus de 0 * /
Opérateur de concaténation
L'opérateur de concaténation combine deux opérandes ou plus pour former un vecteur plus grand.
L'opérateur inclus dans l'opération de concaténation est - {} (concaténation)
Example
wire [1:0] a, h; wire [2:0] x; wire [3;0] y, Z;
assign x = {1’b0, a}; // x[2] = 0, x[1] = a[1], x[0] = a[0]
assign b = {a, h}; /* b[3] = a[1], b[2] = a[0], b[1] = h[1],
b[0] = h[0] */
assign {cout, b} = x + Z; // Concatenation of a result
Opérateur de réplication
L'opérateur de réplication effectue plusieurs copies d'un élément.
L'opérateur utilisé dans l'opération de réplication est - {n {item}} (réplication n fois d'un élément)
Example
Wire [1:0] a, f; wire [4:0] x;
Assign x = {2{1’f0}, a}; // Equivalent to x = {0,0,a }
Assign y = {2{a}, 3{f}}; //Equivalent to y = {a,a,f,f}
For synthesis, Synopsis did not like a zero replication.
For example:-
Parameter l = 5, k = 5;
Assign x = {(l-k){a}}
Opérateur conditionnel
L'opérateur conditionnel synthétise vers un multiplexeur. Il est du même type que celui utilisé en C / C ++ et évalue l'une des deux expressions en fonction de la condition.
L'opérateur utilisé en opération conditionnelle est -
(État) ? (Résultat si la condition est vraie) -
(résultat si la condition est fausse)
Example
Assign x = (g) ? a : b;
Assign x = (inc = = 2) ? x+1 : x-1;
/* if (inc), x = x+1, else x = x-1 */
Opérandes
Littéraux
Les littéraux sont des opérandes à valeur constante utilisés dans les expressions Verilog. Les deux littéraux Verilog couramment utilisés sont -
String - Un opérande littéral de chaîne est un tableau unidimensionnel de caractères, qui sont placés entre guillemets ("").
Numeric - Un opérande de nombre constant est spécifié en nombre binaire, octal, décimal ou hexadécimal.
Example
n - entier représentant le nombre de bits
F - l'un des quatre formats de base possibles -
b pour binaire, o pour octal, d pour décimal, h pour hexadécimal.
“time is” // string literal
267 // 32-bit decimal number
2’b01 // 2-bit binary
20’hB36F // 20-bit hexadecimal number
‘062 // 32-bit octal number
Fils, regs et paramètres
Les fils, regs et paramètres sont les types de données utilisés comme opérandes dans les expressions Verilog.
Sélection de bit "x [2]" et sélection de partie "x [4: 2]"
Les sélections de bits et les sélections partielles sont utilisées pour sélectionner respectivement un bit et plusieurs bits à partir d'un vecteur de fil, de reg ou de paramètre à l'aide de crochets «[]». Les sélections de bits et les sélections partielles sont également utilisées comme opérandes dans les expressions de la même manière que leurs principaux objets de données sont utilisés.
Example
reg [7:0] x, y;
reg [3:0] z;
reg a;
a = x[7] & y[7]; // bit-selects
z = x[7:4] + y[3:0]; // part-selects
Appels de fonction
Dans les appels de fonction, la valeur de retour d'une fonction est utilisée directement dans une expression sans qu'il soit nécessaire de l'affecter au préalable à un registre ou à un câble. Il suffit de placer l'appel de fonction comme l'un des types d'opérandes. Il est nécessaire de s'assurer que vous connaissez la largeur en bits de la valeur de retour de l'appel de fonction.
Example
Assign x = y & z & chk_yz(z, y); // chk_yz is a function
. . ./* Definition of the function */
Function chk_yz; // function definition
Input z,y;
chk_yz = y^z;
End function
Modules
Déclaration du module
Dans Verilog, un module est la principale entité de conception. Cela indique le nom et la liste des ports (arguments). Les quelques lignes suivantes qui spécifient le type d'entrée / sortie (entrée, sortie ou inout) et la largeur de chaque port. La largeur de port par défaut n'est que de 1 bit. Les variables de port doivent être déclarées par fil, baguette ,. . ., reg. La variable de port par défaut est wire. Normalement, les entrées sont câblées car leurs données sont verrouillées à l'extérieur du module. Les sorties sont de type reg si leurs signaux sont stockés à l'intérieur.
Example
module sub_add(add, in1, in2, out);
input add; // defaults to wire
input [7:0] in1, in2; wire in1, in2;
output [7:0] out; reg out;
... statements ...
End module
Affectation continue
L'affectation continue dans un module est utilisée pour attribuer une valeur à un fil, qui est l'affectation normale utilisée à l'extérieur des blocs toujours ou initiaux. Cette affectation se fait avec une instruction d'affectation explicite ou pour affecter une valeur à un fil lors de sa déclaration. Les affectations continues sont exécutées en continu au moment de la simulation. L'ordre des instructions d'assignation ne l'affecte pas. Si vous modifiez l'un des signaux d'entrée du côté droit, le signal de sortie du côté gauche sera modifié.
Example
Wire [1:0] x = 2’y01; // assigned on declaration
Assign y = c | d; // using assign statement
Assign d = a & b;
/* the order of the assign statements does not matter. */
Instanciations de module
Les déclarations de module sont des modèles pour créer des objets réels. Les modules sont instanciés à l'intérieur d'autres modules, et chaque instanciation crée un seul objet à partir de ce modèle. L'exception est le module de niveau supérieur qui est sa propre instanciation. Les ports du module doivent correspondre à ceux définis dans le modèle. Il est précisé -
By name, en utilisant un point «.template port name (nom du fil connecté au port)». Ou
By position, en plaçant les ports au même endroit dans les listes de ports du modèle et de l'instance.
Example
MODULE DEFINITION
Module and4 (x, y, z);
Input [3:0] x, y;
Output [3:0] z;
Assign z = x | y;
End module
Les modèles comportementaux de Verilog contiennent des instructions procédurales qui contrôlent la simulation et manipulent les variables des types de données. Ces toutes les déclarations sont contenues dans les procédures. Chaque procédure est associée à un flux d'activités.
Lors de la simulation du modèle comportemental, tous les flux définis par les instructions «toujours» et «initial» démarrent ensemble au temps de simulation «zéro». Les instructions initiales sont exécutées une fois, et les instructions always sont exécutées de manière répétitive. Dans ce modèle, les variables de registre a et b sont initialisées respectivement à 1 et 0 binaire au temps de simulation «zéro». L'instruction initiale est alors terminée et n'est plus exécutée pendant cette simulation. Cette instruction initiale contient un bloc de début-fin (également appelé bloc séquentiel) d'instructions. Dans ce bloc de type début-fin, a est initialisé en premier suivi de b.
Exemple de modélisation comportementale
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
Affectations procédurales
Les affectations procédurales permettent de mettre à jour les variables reg, integer, time et memory. Il existe une différence significative entre l'affectation procédurale et l'affectation continue, comme décrit ci-dessous -
Les affectations continues pilotent les variables réseau et sont évaluées et mises à jour chaque fois qu'un opérande d'entrée change de valeur.
Les affectations procédurales mettent à jour la valeur des variables de registre sous le contrôle des constructions de flux procédurales qui les entourent.
Le côté droit d'une affectation procédurale peut être n'importe quelle expression qui prend la valeur d'une valeur. Cependant, les sélections partielles sur le côté droit doivent avoir des indices constants. Le côté gauche indique la variable qui reçoit l'affectation du côté droit. Le côté gauche d'une affectation procédurale peut prendre l'une des formes suivantes:
register, integer, real ou time variable - Une affectation à la référence de nom de l'un de ces types de données.
sélection de bits d'un registre, d'un entier, d'une variable réelle ou temporelle - Une affectation à un seul bit qui laisse les autres bits intacts.
sélection partielle d'un registre, d'un entier, d'une variable réelle ou temporelle - Une sélection partielle de deux bits contigus ou plus qui laisse le reste des bits inchangé. Pour le formulaire de sélection de partie, seules les expressions constantes sont autorisées.
élément de mémoire - Un seul mot d'une mémoire. Notez que les sélections de bits et les sélections partielles sont illégales sur les références d'éléments de mémoire.
concaténation de l'un des éléments ci-dessus - Une concaténation de l'une des quatre formes précédentes peut être spécifiée, qui partitionne effectivement le résultat de l'expression de droite et attribue les parties de partition, dans l'ordre, aux différentes parties de la concaténation.
Retard dans l'attribution (pas pour la synthèse)
Dans une affectation retardée Δt, les unités de temps s'écoulent avant que l'instruction ne soit exécutée et que l'affectation de gauche soit effectuée. Avec un délai intra-affectation, le côté droit est évalué immédiatement mais il y a un délai de Δt avant que le résultat ne soit placé dans l'affectation de gauche. Si une autre procédure modifie un signal du côté droit pendant Δt, elle n'affecte pas la sortie. Les retards ne sont pas pris en charge par les outils de synthèse.
Syntaxe
Procedural Assignmentvariable = expression
Delayed assignment# Δt variable = expression;
Intra-assignment delayvariable = # Δt expression;
Exemple
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. */
Blocage des attributions
Une instruction d'affectation procédurale bloquante doit être exécutée avant l'exécution des instructions qui la suivent dans un bloc séquentiel. Une instruction d'affectation procédurale bloquante n'empêche pas l'exécution des instructions qui la suivent dans un bloc parallèle.
Syntaxe
La syntaxe pour une affectation procédurale bloquante est la suivante -
<lvalue> = <timing_control> <expression>
Où, lvalue est un type de données valide pour une instruction d'affectation de procédure, = est l'opérateur d'affectation et le contrôle de synchronisation est le délai intra-affectation facultatif. Le délai de commande de synchronisation peut être soit une commande de retard (par exemple, # 6) soit une commande d'événement (par exemple, @ (posedge clk)). L'expression est la valeur du côté droit que le simulateur attribue au côté gauche. L'opérateur d'affectation = utilisé pour bloquer les affectations procédurales est également utilisé par les affectations procédurales continues et les affectations continues.
Exemple
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
Affectations non bloquantes (RTL)
L'affectation procédurale non bloquante vous permet de planifier des affectations sans bloquer le flux procédural. Vous pouvez utiliser l'instruction procédurale non bloquante chaque fois que vous souhaitez effectuer plusieurs affectations de registre dans le même pas de temps sans tenir compte de l'ordre ou de la dépendance les uns des autres.
Syntaxe
La syntaxe pour une affectation procédurale non bloquante est la suivante -
<lvalue> <= <timing_control> <expression>
Où lvalue est un type de données valide pour une instruction d'affectation de procédure, <= est l'opérateur d'affectation non bloquant et le contrôle de synchronisation est le contrôle de synchronisation intra-affectation facultatif. Le délai de commande de synchronisation peut être soit une commande de retard, soit une commande d'événement (par exemple, @ (posedge clk)). L'expression est la valeur du côté droit que le simulateur attribue au côté gauche. L'opérateur d'affectation non bloquante est le même que celui que le simulateur utilise pour l'opérateur relationnel inférieur ou égal. Le simulateur interprète l'opérateur <= comme un opérateur relationnel lorsque vous l'utilisez dans une expression et interprète l'opérateur <= comme un opérateur d'affectation lorsque vous l'utilisez dans une construction d'affectation procédurale non bloquante.
Comment le simulateur évalue les affectations procédurales non bloquantes Lorsque le simulateur rencontre une affectation procédurale non bloquante, le simulateur évalue et exécute l'affectation procédurale non bloquante en deux étapes comme suit -
Le simulateur évalue le côté droit et planifie l'attribution de la nouvelle valeur pour qu'elle ait lieu à un moment spécifié par une commande de synchronisation procédurale. Le simulateur évalue le côté droit et planifie l'attribution de la nouvelle valeur pour qu'elle ait lieu à un moment spécifié par une commande de synchronisation procédurale.
A la fin du pas de temps, dans lequel le délai donné a expiré ou l'événement approprié s'est produit, le simulateur exécute l'affectation en attribuant la valeur au côté gauche.
Exemple
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
Conditions
L'instruction conditionnelle (ou l'instruction if-else) est utilisée pour décider si une instruction est exécutée ou non.
Formellement, la syntaxe est la suivante -
<statement>
::= if ( <expression> ) <statement_or_null>
||= if ( <expression> ) <statement_or_null>
else <statement_or_null>
<statement_or_null>
::= <statement>
||= ;
L '<expression> est évaluée; s'il est vrai (c'est-à-dire qu'il a une valeur connue différente de zéro), la première instruction s'exécute. S'il est faux (a une valeur nulle ou la valeur est x ou z), la première instruction ne s'exécute pas. S'il existe une instruction else et que <expression> est fausse, l'instruction else s'exécute. Étant donné que la valeur numérique de l'expression if est testée pour être égale à zéro, certains raccourcis sont possibles.
Par exemple, les deux instructions suivantes expriment la même logique -
if (expression)
if (expression != 0)
Comme la partie else d'un if-else est facultative, il peut y avoir confusion lorsqu'un else est omis d'une séquence if imbriquée. Ceci est résolu en associant toujours le else au précédent le plus proche s'il manque un else.
Exemple
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;
Construction de: if- else- if
La construction suivante se produit si souvent qu'elle mérite une brève discussion séparée.
Example
if (<expression>)
<statement>
else if (<expression>)
<statement>
else if (<expression>)
<statement>
else
<statement>
Cette séquence de if (connue sous le nom de construction if-else-if) est la manière la plus générale d'écrire une décision à plusieurs voies. Les expressions sont évaluées dans l'ordre; si une expression est vraie, l'instruction qui lui est associée est exécutée, ce qui met fin à toute la chaîne. Chaque instruction est soit une seule instruction, soit un bloc d'instructions.
La dernière partie else de la construction if-else-if gère le cas «rien de ce qui précède» ou le cas par défaut où aucune des autres conditions n'est satisfaite. Parfois, il n'y a pas d'action explicite pour la valeur par défaut; dans ce cas, la fin d'autre peut être omise ou elle peut être utilisée pour la vérification d'erreur afin de détecter une condition impossible.
Déclaration de cas
L'instruction case est une instruction de décision multidirectionnelle spéciale qui teste si une expression correspond à l'une des nombreuses autres expressions et se branche en conséquence. L'énoncé de cas est utile pour décrire, par exemple, le décodage d'une instruction de microprocesseur. L'instruction case a la syntaxe suivante -
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>
Les expressions de cas sont évaluées et comparées dans l'ordre exact dans lequel elles sont données. Lors de la recherche linéaire, si l'une des expressions d'élément de cas correspond à l'expression entre parenthèses, l'instruction associée à cet élément de cas est exécutée. Si toutes les comparaisons échouent et que l'élément par défaut est donné, l'instruction d'élément par défaut est exécutée. Si l'instruction par défaut n'est pas donnée et que toutes les comparaisons échouent, aucune des instructions d'élément de cas n'est exécutée.
Outre la syntaxe, l'instruction case diffère de la construction multi-voies if-else-if de deux manières importantes -
Les expressions conditionnelles de la construction if-else-if sont plus générales que la comparaison d'une expression avec plusieurs autres, comme dans l'instruction case.
L'instruction case fournit un résultat définitif lorsqu'il existe des valeurs x et z dans une expression.
Instructions en boucle
Il existe quatre types d'instructions en boucle. Ils fournissent un moyen de contrôler l'exécution d'une instruction zéro, une ou plusieurs fois.
forever exécute en permanence une instruction.
repeat exécute une instruction un nombre fixe de fois.
while exécute une instruction jusqu'à ce qu'une expression devienne fausse. Si l'expression commence par false, l'instruction n'est pas exécutée du tout.
pour contrôler l'exécution de ses instructions associées par un processus en trois étapes, comme suit -
Exécute une affectation normalement utilisée pour initialiser une variable qui contrôle le nombre de boucles exécutées
Évalue une expression: si le résultat est zéro, la boucle for se termine et si elle n'est pas nulle, la boucle for exécute sa ou ses instructions associées, puis exécute l'étape 3
Exécute une affectation normalement utilisée pour modifier la valeur de la variable de contrôle de boucle, puis répète l'étape 2
Voici les règles de syntaxe pour les instructions en boucle -
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
Commandes de retard
Contrôle du retard
L'exécution d'une instruction procédurale peut être contrôlée par délai en utilisant la syntaxe suivante -
<statement>
::= <delay_control> <statement_or_null>
<delay_control>
::= # <NUMBER>
||= # <identifier>
||= # ( <mintypmax_expression> )
L'exemple suivant retarde l'exécution de l'affectation de 10 unités de temps -
# 10 rega = regb;
Les trois exemples suivants fournissent une expression après le signe dièse (#). L'exécution des retards d'affectation de la durée de simulation spécifiée par la valeur de l'expression.
Contrôle des événements
L'exécution d'une instruction procédurale peut être synchronisée avec un changement de valeur sur un réseau ou un registre, ou avec l'occurrence d'un événement déclaré, en utilisant la syntaxe de contrôle d'événement suivante -
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> est une expression qui se résout en une valeur d'un bit.
Les changements de valeur sur les réseaux et les registres peuvent être utilisés comme événements pour déclencher l'exécution d'une instruction. C'est ce qu'on appelle la détection d'un événement implicite. La syntaxe Verilog vous permet également de détecter le changement en fonction de la direction du changement, c'est-à-dire vers la valeur 1 (posedge) ou vers la valeur 0 (negedge). Le comportement de posedge et negedge pour les valeurs d'expression inconnues est le suivant:
- un negedge est détecté lors du passage de 1 à inconnu et de inconnu à 0
- un posedge est détecté lors du passage de 0 à inconnu et de inconnu à 1
Procédures: blocs toujours et initiaux
Toutes les procédures de Verilog sont spécifiées dans l'un des quatre blocs suivants. 1) Blocs initiaux 2) Bloque toujours 3) Tâche 4) Fonction
Les instructions initial et always sont activées au début de la simulation. Les blocs initiaux ne s'exécutent qu'une seule fois et son activité meurt lorsque l'instruction est terminée. En revanche, le bloc always s'exécute à plusieurs reprises. Son activité ne meurt que lorsque la simulation est terminée. Il n'y a pas de limite au nombre de blocs initiaux et toujours qui peuvent être définis dans un module. Les tâches et les fonctions sont des procédures activées à partir d'un ou plusieurs endroits dans d'autres procédures.
Blocs initiaux
La syntaxe de l'instruction initiale est la suivante -
<initial_statement>
::= initial <statement>
L'exemple suivant illustre l'utilisation de l'instruction initiale pour l'initialisation des variables au début de la simulation.
Initial
Begin
Areg = 0; // initialize a register
For (index = 0; index < size; index = index + 1)
Memory [index] = 0; //initialize a memory
Word
End
Une autre utilisation typique des blocs initiaux est la spécification de descriptions de formes d'onde qui s'exécutent une fois pour fournir un stimulus à la partie principale du circuit simulé.
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
Bloque toujours
L'instruction «always» se répète continuellement tout au long de la simulation. La syntaxe de l'instruction always est donnée ci-dessous
<always_statement>
::= always <statement>
L'instruction 'always', en raison de sa nature en boucle, n'est utile que lorsqu'elle est utilisée conjointement avec une forme de contrôle de synchronisation. Si une instruction 'always' ne fournit aucun moyen d'avancer le temps, l'instruction 'always' crée une condition de blocage de simulation. Le code suivant, par exemple, crée une boucle infinie sans délai -
Always areg = ~areg;
Fournir un contrôle de synchronisation au code ci-dessus crée une description potentiellement utile - comme dans l'exemple suivant -
Always #half_period areg = ~areg;