From 04c6500af0716646c1a6661890e7a26850a4e2f8 Mon Sep 17 00:00:00 2001 From: Andrea Mazzai <97525105+andreamazzai@users.noreply.github.com> Date: Fri, 1 Nov 2024 12:21:30 +0100 Subject: [PATCH] DDER UPDATE --- docs/_docs/97-bin-math.md | 50 ++++++++++++++++++++------------------- 1 file changed, 26 insertions(+), 24 deletions(-) diff --git a/docs/_docs/97-bin-math.md b/docs/_docs/97-bin-math.md index 7c28122..4b1e4b9 100644 --- a/docs/_docs/97-bin-math.md +++ b/docs/_docs/97-bin-math.md @@ -718,58 +718,58 @@ WORK IN PROGRESS -- WORK IN PROGRESS -- WORK IN PROGRESS Un Adder di tipo Carry Look Ahead (CLA) affronta il problema del ritardo causato dalla lunga catena di livelli presente in un Ripple Carry Adder. -Se in un Ripple Carry Adder a 4 bit il ritardo può essere modesto, il ritardo in un Adder a 16 o più bit può ridurre significativamente la frequenza massima di lavoro. Infatti, ogni Full Adder di un RCA dipende dal risultato dello stage precedente; nell'ottica di doversi mettere nella situazione del caso peggiore, si può dire che il risultato dell'ultimo Full Adder dipende da tutti gli stage precedenti. +Se in un Ripple Carry Adder a 4 bit il ritardo può essere modesto, il ritardo in un Adder a 16 o più bit può ridurre significativamente la frequenza massima di lavoro. Infatti, ogni Full Adder di un RCA dipende dal risultato dell'Adder precedente; nell'ottica di doversi mettere nella situazione del caso peggiore, si può dire che il risultato dell'ultimo Full Adder dipende da tutti i Full Adder precedenti. -Per indirizzare la problematica appena esposta, si è cercato un metodo che permettesse a ogni Adder di calcolare indipendentemente il proprio Carry In, rendendolo indipendente dagli stadi precedenti. In questo modo, il risultato di ogni Adder attraversa un numero fisso e ridotto di livelli. +Per indirizzare la problematica appena esposta, è necessario che ogni Full Adder possa calcolare il proprio Carry Out in maniera indipendente dal Carry Out reso disponibile dallo stadio precedente. Perché questo accada, ad ogni Full Adder deve essere aggiunta circuiteria in grado di calcolare il proprio Carry In: ![Carry Look Ahead Adder](../../assets/math/carry-look-ahead-schema.png){:width="100%"} *Carry Look Ahead Adder.* -Per raggiungere questo obiettivo, è stato necessario trovare un compromesso: la velocità dell'Adder aumenta, ma anche il numero di porte logiche necessarie in ogni Adder - e dunque la sua complessità - cresce di conseguenza. +Tutto questo porta ad un compromesso: la velocità di risposta di ogni Full Adder aumenta, a discapito della aumentata complessità dovuta all'incremento di porte logice. -Come si vede nell'immagine precedente, gli ingressi dei vari stadi di un CLA dipendono solamente dai termini A e B e dal Carry C0; questo è stato possibile grazie a una serie di sostituzioni algebriche che permettono ad ogni singolo Adder(N) di poter computare la propria somma e il proprio Carry Out partendo dagli ingressi A(N) e B(N) e da C0. +L'immagine precedente mostra che gli ingressi dei vari stadi di un CLA dipendono solamente dai termini A e B e dal Carry C0; questo è possibile grazie a una serie di sostituzioni algebriche che permettono ad ogni singolo Full Adder(N) di poter computare la propria somma e il proprio Carry Out partendo dagli ingressi A(N) e B(N) e da C0. -In altre parole, si è creata una logica dipendente dai soli termini A e B e dal Carry C0. Come è stato possibile ottenere questo risultato? +In altre parole, si crea una logica dipendente dai soli termini A e B e dal Carry C0. Come si ottiene questo risultato? Riducendo questo concetto a espressioni logiche, due sono i casi da analizzare: -1. In quali situazioni un Adder, il cui Carry In è a 0, ***genera*** un Carry Out che viene passato al prossimo stadio? -2. In quali situazioni un Adder, il cui Carry In è a 1, ***propaga*** il proprio Carry In al prossimo stadio? +1. In quali situazioni un Full Adder(N-1), il cui Carry In è a 0, ***genera*** un Carry Out che viene passato al prossimo stadio? +2. In quali situazioni un Full Adder(N-1), il cui Carry In è a 1, ***propaga*** il proprio Carry In al prossimo stadio? -Valutando i due casi, si trova risposta alla domanda chiave: in quali situazioni un Adder(N) ritrova sicuramente un Carry In sul proprio ingresso? +Valutando i due casi, si trova risposta alla domanda chiave: in quali situazioni un Adder(N) ritrova un Carry In a 1 sul proprio ingresso? -Identificando le situazioni nelle quali un Carry viene *generato* o *propagato*, ogni Adder(N) può essere dotato di un circuito in grado di sapere se troverà un Carry in ingresso computandolo a partire *dagli ingressi* dell'Adder(N-1) precedente, e dunque senza dipendere da quanto presente *dall'uscita* dello stadio precedente. +**VERIFICARE** Identificando le situazioni nelle quali un Carry viene *generato* o *propagato*, ogni Adder(N) può essere dotato di un circuito in grado di sapere se troverà un Carry in ingresso computandolo a partire *dagli ingressi* dell'Adder(N-1) precedente, e dunque senza dipendere da quanto presente *dall'uscita* dello stadio precedente. Le due situazioni vengono tradotte in espressioni denominate **Generate** e **Propagate**. -Riprendendo la truth table di un Full Adder, troviamo che se CIN è a 0, il COUT è a 1 solo se entrambi A e B sono a 1, dunque per realizzare questo circuito possiamo utilizzare una porta AND. Questo comportamento viene descritto con l'espressione **g = A*B**. +In quali casi un Full Adder genera un Carry senza che sia presente un Carry al suo iongresso Carry In? Riprendendo la truth table di un Full Adder, troviamo che se CIN è a 0, il COUT è a 1 solo se entrambi A **e** B sono a 1, dunque per realizzare questo circuito possiamo utilizzare una porta AND. Questo comportamento viene descritto con l'espressione **g = A\*B** e si può leggere come "la logica Generate di ogni Full Adder corrisponde ad A*B". -Se CIN è invece a 1, il COUT è a 1 quando A o B sono a 1, dunque per realizzare questo circuito possiamo utilizzare una porta OR. Questo comportamento viene descritto con l'espressione **p = A+B**. +Quando, invece, un Full Adder propaga un Carry presente al suo ingresso Carry In? Se CIN è invece a 1, il COUT è a 1 quando A **o** B sono a 1, dunque per realizzare questo circuito possiamo utilizzare una porta OR. Questo comportamento viene descritto con l'espressione **p = A+B** e si può leggere come "la logica Propagate di ogni Full Adder corrisponde ad A+B". | CIN | A | B | Q | COUT | Generate / Propagate | | - | - | - | - | - | - | | 0 | 0 | 0 | 0 | 0 | | | 0 | 0 | 1 | 1 | 0 | | | 0 | 1 | 0 | 1 | 0 | | -| **0** | **1** | **1** | 0 | **1** | <-- **Generate** | +| **0** | **1** | **1** | 0 | **1** | **<-- Generate** | | 1 | 0 | 0 | 1 | 0 | | -| **1** | 0 | **1** | 0 | **1** | <-- **Propagate** | -| **1** | **1** | 0 | 0 | **1** | <-- **Propagate** | -| **1** | **1** | **1** | 1 | **1** | <-- **Propagate** | +| **1** | 0 | **1** | 0 | **1** | **<-- Propagate** | +| **1** | **1** | 0 | 0 | **1** | **<-- Propagate** | +| **1** | **1** | **1** | 1 | **1** | **<-- Propagate** | In altre parole: -- l'espressione Generate permette a un Adder "N" di identificare quando, in assenza di un Carry agli ingressi dello stadio precedente "N-1", questi produrrà ("genererà") un Carry in uscita; -- l'espressione Propagate permette a un Adder "N" di identificare quando, in presenza di un Carry agli ingressi dello stadio precedente "N-1", questi produrrà ("propagherà") un Carry in uscita. +- l'espressione Generate viene utilizzata da un Adder "N" per identificare quando, in assenza di un Carry agli ingressi dello stadio precedente "N-1", questi produrrà ("genererà") un Carry in uscita; +- l'espressione Propagate viene utilizzata da un Adder "N" per identificare quando, in presenza di un Carry agli ingressi dello stadio precedente "N-1", questi produrrà ("propagherà") un Carry in uscita. -La circuiteria Look Ahead considera se lo stage precedente introduce un Carry analizzando appunto due condizioni: in una truth table i cui ingressi siano A, B e CIN: +La circuiteria Look Ahead considera se lo stadio precedente introduce un Carry analizzando appunto due condizioni: in una truth table i cui ingressi siano A, B e CIN: -- che Genera un COUT solo se A*B = 1 -- che Propaga un COUT solo se CCIN è a 1 e A+B = 1 +- che **genera** un COUT solo se A*B = 1 +- che **propaga** un COUT solo se CIN è a 1 e A+B = 1 -per ogni stage, COUT = g + P * CIN -= A*B + (A+B)\*CIN che abbiamo già visto perché è il COUT di ogni Full Adder +per ogni stadio, COUT = g + P * CIN += A*B + (A+B)\*CIN che abbiamo già visto perché è il COUT di ogni Full Adder gi = Ai*Bi pi = Ai+ Bi @@ -782,13 +782,13 @@ C2 = g1 + p1 * g0 + p1 * p0 * C0 C3 = g2 + p2 * C2 C3 = g2 + p2 * (g1 + p1 * g0 + p1 * p0 * C0) C3 = g2 + p2 * g1 + p2 * p1 * g0 + p2 * p1 * p0 * C0 -notare che l'espressione tiene conto solo di g2, g1, g0 e p2, p1 e p0, cioè delle AND e degli OR degli stage precedenti +notare che l'espressione tiene conto solo di g2, g1, g0 e p2, p1 e p0, cioè delle AND e degli OR degli stadio precedenti C4 = g3 + p3 * C3 C4 = g3 + p3 * (g2 + p2 * (g1 + p1 * g0 + p1 * p0 * C0)) C4 = g3 + p3 * (g2 + p2 * g1 + p2 * p1 * g0 + p2 * p1 * p0 * C0) C4 = g3 + p3 * g2 + p3 * p2 * g1 + p3 * p2 * p1 * g0 + p3 * p2 * p1 * p0 * C0 -notare che l'espressione tiene conto solo di g2, g1, g0 e p2, p1 e p0, cioè delle AND e degli OR degli stage precedenti +notare che l'espressione tiene conto solo di g2, g1, g0 e p2, p1 e p0, cioè delle AND e degli OR degli stadi precedenti A questo punto realizzo un Modified Full Adder. Q è sempre A XOR B XOR CIN @@ -805,6 +805,8 @@ il lavoro per creare il carry è molto elevato... ma ho le prestazioni massime. un problema potrebbe essere il fan-in! a un certo dovrò spacchettare la OR che raccoglie le AND (ma perché non parla delle AND da spacchettare?) in effetti dunque ogni tot bit il livello salirà di due (Ma allora forse in questo momento sta tenendo in conto anche le porte AND?), Ma non cresce sicuramente in maniera così veloce come con il RCA. +DA CONTESTUALIZZARE: In questo modo, il risultato di ogni Adder attraversa un numero fisso e ridotto di livelli. + ## Le sottrazioni con gli Adder È per questo motivo che una XOR all'ingresso B permette di negare gli input e, aggiungendo un Carry in ingresso, diventa facile effettuare una sottrazione, così come esposto in precedenza parlando del complemento a 2.